diff --git a/ChangeLog b/ChangeLog index a49bb9e300..12e3c5599d 100644 --- a/ChangeLog +++ b/ChangeLog @@ -41,6 +41,10 @@ Release date: TBA Closes #6539 +* Fix a false positive for ``undefined-loop-variable`` when using ``enumerate()``. + + Closes #6593 + What's New in Pylint 2.13.8? ============================ diff --git a/doc/whatsnew/2.13.rst b/doc/whatsnew/2.13.rst index 1ae1ca7f9a..e5a628d8ab 100644 --- a/doc/whatsnew/2.13.rst +++ b/doc/whatsnew/2.13.rst @@ -657,3 +657,7 @@ Other Changes * Fix a crash in ``unnecessary-dict-index-lookup`` when subscripting an attribute. Closes #6557 + +* Fix a false positive for ``undefined-loop-variable`` when using ``enumerate()``. + + Closes #6593 diff --git a/pylint/checkers/variables.py b/pylint/checkers/variables.py index 807e031e60..fb50f4501b 100644 --- a/pylint/checkers/variables.py +++ b/pylint/checkers/variables.py @@ -2265,6 +2265,13 @@ def _loopvar_name(self, node: astroid.Name) -> None: # For functions we can do more by inferring the length of the itered object try: inferred = next(assign.iter.infer()) + # Prefer the target of enumerate() rather than the enumerate object itself + if ( + isinstance(inferred, astroid.Instance) + and inferred.qname() == "builtins.enumerate" + and assign.iter.args + ): + inferred = next(assign.iter.args[0].infer()) except astroid.InferenceError: self.add_message("undefined-loop-variable", args=node.name, node=node) else: diff --git a/tests/functional/u/undefined/undefined_loop_variable.py b/tests/functional/u/undefined/undefined_loop_variable.py index 6bcf6103d9..d66bd5a05a 100644 --- a/tests/functional/u/undefined/undefined_loop_variable.py +++ b/tests/functional/u/undefined/undefined_loop_variable.py @@ -139,3 +139,10 @@ def variable_name_assigned_in_body_of_second_loop(): alias = True if alias: print(alias) + + +def use_enumerate(): + """https://github.com/PyCQA/pylint/issues/6593""" + for i, num in enumerate(range(3)): + pass + print(i, num)