From aea868c9d01f93f2030c4624e97910dffc5bc090 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Sun, 7 Jul 2024 20:04:09 +0000 Subject: [PATCH] Fix `invalid-name` regression for class attributes in subclasses (#9772) (#9775) (cherry picked from commit b9a42e8c350fc92034d1e9e3eebd66e47a96197e) Co-authored-by: Jacob Walls --- doc/whatsnew/fragments/9765.false_positive | 4 ++++ pylint/checkers/base/name_checker/checker.py | 4 +++- tests/functional/i/invalid/invalid_name.py | 7 +++++++ tests/functional/i/invalid/invalid_name.rc | 2 ++ tests/functional/i/invalid/invalid_name.txt | 1 + 5 files changed, 17 insertions(+), 1 deletion(-) create mode 100644 doc/whatsnew/fragments/9765.false_positive create mode 100644 tests/functional/i/invalid/invalid_name.rc diff --git a/doc/whatsnew/fragments/9765.false_positive b/doc/whatsnew/fragments/9765.false_positive new file mode 100644 index 0000000000..c85df15eee --- /dev/null +++ b/doc/whatsnew/fragments/9765.false_positive @@ -0,0 +1,4 @@ +Fix a regression that raised ``invalid-name`` on class attributes merely +overriding invalid names from an ancestor. + +Closes #9765 diff --git a/pylint/checkers/base/name_checker/checker.py b/pylint/checkers/base/name_checker/checker.py index 68f5767206..3514829fb1 100644 --- a/pylint/checkers/base/name_checker/checker.py +++ b/pylint/checkers/base/name_checker/checker.py @@ -491,7 +491,9 @@ def visit_assignname( # pylint: disable=too-many-branches self._check_name("variable", node.name, node) # Check names defined in class scopes - elif isinstance(frame, nodes.ClassDef): + elif isinstance(frame, nodes.ClassDef) and not any( + frame.local_attr_ancestors(node.name) + ): if utils.is_enum_member(node) or utils.is_assign_name_annotated_with( node, "Final" ): diff --git a/tests/functional/i/invalid/invalid_name.py b/tests/functional/i/invalid/invalid_name.py index 66ad1e77cd..28638bcb4c 100644 --- a/tests/functional/i/invalid/invalid_name.py +++ b/tests/functional/i/invalid/invalid_name.py @@ -102,3 +102,10 @@ def test_disable_mixed( """Invalid-name will still be raised for other arguments.""" self.foo_bar = fooBar self.foo_bar2 = fooBar2 + + def tearDown(self): ... # pylint: disable=invalid-name + + +class FooBarSubclass(FooBar): + tearDown = FooBar.tearDown + tearDownNotInAncestor = None # [invalid-name] diff --git a/tests/functional/i/invalid/invalid_name.rc b/tests/functional/i/invalid/invalid_name.rc new file mode 100644 index 0000000000..b2444c2d15 --- /dev/null +++ b/tests/functional/i/invalid/invalid_name.rc @@ -0,0 +1,2 @@ +[MAIN] +class-attribute-naming-style=snake_case diff --git a/tests/functional/i/invalid/invalid_name.txt b/tests/functional/i/invalid/invalid_name.txt index 72f8fcd906..e8622f864a 100644 --- a/tests/functional/i/invalid/invalid_name.txt +++ b/tests/functional/i/invalid/invalid_name.txt @@ -6,3 +6,4 @@ invalid-name:66:0:66:68:a_very_very_very_long_function_name_WithCamelCase_to_mak invalid-name:74:23:74:29:FooBar.__init__:"Argument name ""fooBar"" doesn't conform to snake_case naming style":HIGH invalid-name:80:8:80:14:FooBar.func1:"Argument name ""fooBar"" doesn't conform to snake_case naming style":HIGH invalid-name:100:8:100:15:FooBar.test_disable_mixed:"Argument name ""fooBar2"" doesn't conform to snake_case naming style":HIGH +invalid-name:111:4:111:25:FooBarSubclass:"Class attribute name ""tearDownNotInAncestor"" doesn't conform to snake_case naming style":HIGH