diff --git a/bugbear.py b/bugbear.py index ba0294f..09124d6 100644 --- a/bugbear.py +++ b/bugbear.py @@ -16,7 +16,19 @@ __version__ = "21.9.2" LOG = logging.getLogger("flake8.bugbear") +CONTEXTFUL_NODES = ( + ast.Module, + ast.ClassDef, + ast.AsyncFunctionDef, + ast.FunctionDef, + ast.Lambda, + ast.ListComp, + ast.SetComp, + ast.DictComp, + ast.GeneratorExp, +) +Context = namedtuple("Context", ["node", "stack"]) @attr.s(hash=False) class BugBearChecker: @@ -147,10 +159,11 @@ def _typesafe_issubclass(cls, class_or_tuple): class BugBearVisitor(ast.NodeVisitor): filename = attr.ib() lines = attr.ib() - node_stack = attr.ib(default=attr.Factory(list)) + contexts = attr.ib(default=attr.Factory(list)) node_window = attr.ib(default=attr.Factory(list)) errors = attr.ib(default=attr.Factory(list)) futures = attr.ib(default=attr.Factory(set)) + context = attr.ib(default=attr.Factory(list)) NODE_WINDOW_SIZE = 4 @@ -161,13 +174,30 @@ def __getattr__(self, name): print(name) return self.__getattribute__(name) + @property + def node_stack(self): + if len(self.contexts) == 0: + return [] + + context, stack = self.contexts[-1] + return stack + def visit(self, node): + is_contextful = isinstance(node, CONTEXTFUL_NODES) + + if is_contextful: + context = Context(node, []) + self.contexts.append(context) + self.node_stack.append(node) self.node_window.append(node) self.node_window = self.node_window[-self.NODE_WINDOW_SIZE :] super().visit(node) self.node_stack.pop() + if is_contextful: + self.contexts.pop() + def visit_ExceptHandler(self, node): if node.type is None: self.errors.append( @@ -479,9 +509,13 @@ def check_for_b901(self, node): break def check_for_b902(self, node): - if not isinstance(self.node_stack[-2], ast.ClassDef): + if ( + not len(self.contexts) >= 2 + or not isinstance(self.contexts[-2].node, ast.ClassDef) + ): return + cls = self.contexts[-2].node decorators = NameFinder() decorators.visit(node.decorator_list) @@ -490,7 +524,7 @@ def check_for_b902(self, node): # `cls`? return - bases = {b.id for b in self.node_stack[-2].bases if isinstance(b, ast.Name)} + bases = {b.id for b in cls.bases if isinstance(b, ast.Name)} if "type" in bases: if ( "classmethod" in decorators.names diff --git a/tests/b904.py b/tests/b904.py index 104fda8..3967f00 100644 --- a/tests/b904.py +++ b/tests/b904.py @@ -19,3 +19,10 @@ raise base_err finally: raise Exception("Nothing to chain from, so no warning here") + +try: + raise ValueError +except ValueError: + # should not emit, since we are not raising something + def proxy(): + raise NameError