mirror of
https://fuchsia.googlesource.com/third_party/github.com/pylint-dev/pylint
synced 2024-09-21 07:58:04 +00:00
fa7274a884
Close #2527
528 lines
17 KiB
Python
528 lines
17 KiB
Python
# -*- coding: utf-8 -*-
|
|
# Copyright (c) 2013-2015 LOGILAB S.A. (Paris, FRANCE) <contact@logilab.fr>
|
|
# Copyright (c) 2013-2014 Google, Inc.
|
|
# Copyright (c) 2015-2018 Claudiu Popa <pcmanticore@gmail.com>
|
|
# Copyright (c) 2015 Dmitry Pribysh <dmand@yandex.ru>
|
|
# Copyright (c) 2015 Ionel Cristian Maries <contact@ionelmc.ro>
|
|
# Copyright (c) 2016 Derek Gustafson <degustaf@gmail.com>
|
|
# Copyright (c) 2016 Yannack <yannack@users.noreply.github.com>
|
|
# Copyright (c) 2017 ttenhoeve-aa <ttenhoeve@appannie.com>
|
|
# Copyright (c) 2017 Łukasz Rogalski <rogalski.91@gmail.com>
|
|
# Copyright (c) 2017 Ville Skyttä <ville.skytta@iki.fi>
|
|
# Copyright (c) 2018 Fureigh <fureigh@users.noreply.github.com>
|
|
# Copyright (c) 2018 glmdgrielson <32415403+glmdgrielson@users.noreply.github.com>
|
|
|
|
# Licensed under the GPL: https://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
|
# For details: https://github.com/PyCQA/pylint/blob/master/COPYING
|
|
|
|
"""Unittest for the base checker."""
|
|
|
|
import re
|
|
import sys
|
|
import unittest
|
|
|
|
import astroid
|
|
from pylint.checkers import base
|
|
from pylint.testutils import CheckerTestCase, Message, set_config
|
|
|
|
|
|
class TestDocstring(CheckerTestCase):
|
|
CHECKER_CLASS = base.DocStringChecker
|
|
|
|
def test_missing_docstring_module(self):
|
|
module = astroid.parse("something")
|
|
message = Message("missing-docstring", node=module, args=("module",))
|
|
with self.assertAddsMessages(message):
|
|
self.checker.visit_module(module)
|
|
|
|
def test_missing_docstring_empty_module(self):
|
|
module = astroid.parse("")
|
|
with self.assertNoMessages():
|
|
self.checker.visit_module(module)
|
|
|
|
def test_empty_docstring_module(self):
|
|
module = astroid.parse("''''''")
|
|
message = Message("empty-docstring", node=module, args=("module",))
|
|
with self.assertAddsMessages(message):
|
|
self.checker.visit_module(module)
|
|
|
|
def test_empty_docstring_function(self):
|
|
func = astroid.extract_node(
|
|
"""
|
|
def func(tion):
|
|
pass"""
|
|
)
|
|
message = Message("missing-docstring", node=func, args=("function",))
|
|
with self.assertAddsMessages(message):
|
|
self.checker.visit_functiondef(func)
|
|
|
|
@set_config(docstring_min_length=2)
|
|
def test_short_function_no_docstring(self):
|
|
func = astroid.extract_node(
|
|
"""
|
|
def func(tion):
|
|
pass"""
|
|
)
|
|
with self.assertNoMessages():
|
|
self.checker.visit_functiondef(func)
|
|
|
|
@set_config(docstring_min_length=2)
|
|
def test_long_function_no_docstring(self):
|
|
func = astroid.extract_node(
|
|
"""
|
|
def func(tion):
|
|
pass
|
|
pass
|
|
"""
|
|
)
|
|
message = Message("missing-docstring", node=func, args=("function",))
|
|
with self.assertAddsMessages(message):
|
|
self.checker.visit_functiondef(func)
|
|
|
|
@set_config(docstring_min_length=2)
|
|
def test_long_function_nested_statements_no_docstring(self):
|
|
func = astroid.extract_node(
|
|
"""
|
|
def func(tion):
|
|
try:
|
|
pass
|
|
except:
|
|
pass
|
|
"""
|
|
)
|
|
message = Message("missing-docstring", node=func, args=("function",))
|
|
with self.assertAddsMessages(message):
|
|
self.checker.visit_functiondef(func)
|
|
|
|
@set_config(docstring_min_length=2)
|
|
def test_function_no_docstring_by_name(self):
|
|
func = astroid.extract_node(
|
|
"""
|
|
def __fun__(tion):
|
|
pass"""
|
|
)
|
|
with self.assertNoMessages():
|
|
self.checker.visit_functiondef(func)
|
|
|
|
def test_class_no_docstring(self):
|
|
klass = astroid.extract_node(
|
|
"""
|
|
class Klass(object):
|
|
pass"""
|
|
)
|
|
message = Message("missing-docstring", node=klass, args=("class",))
|
|
with self.assertAddsMessages(message):
|
|
self.checker.visit_classdef(klass)
|
|
|
|
def test_inner_function_no_docstring(self):
|
|
func = astroid.extract_node(
|
|
"""
|
|
def func(tion):
|
|
\"""Documented\"""
|
|
def inner(fun):
|
|
# Not documented
|
|
pass
|
|
"""
|
|
)
|
|
with self.assertNoMessages():
|
|
self.checker.visit_functiondef(func)
|
|
|
|
|
|
class TestNameChecker(CheckerTestCase):
|
|
CHECKER_CLASS = base.NameChecker
|
|
CONFIG = {"bad_names": set()}
|
|
|
|
@set_config(
|
|
attr_rgx=re.compile("[A-Z]+"),
|
|
property_classes=("abc.abstractproperty", ".custom_prop"),
|
|
)
|
|
def test_property_names(self):
|
|
# If a method is annotated with @property, its name should
|
|
# match the attr regex. Since by default the attribute regex is the same
|
|
# as the method regex, we override it here.
|
|
methods = astroid.extract_node(
|
|
"""
|
|
import abc
|
|
|
|
def custom_prop(f):
|
|
return property(f)
|
|
|
|
class FooClass(object):
|
|
@property
|
|
def FOO(self): #@
|
|
pass
|
|
|
|
@property
|
|
def bar(self): #@
|
|
pass
|
|
|
|
@abc.abstractproperty
|
|
def BAZ(self): #@
|
|
pass
|
|
|
|
@custom_prop
|
|
def QUX(self): #@
|
|
pass
|
|
"""
|
|
)
|
|
with self.assertNoMessages():
|
|
self.checker.visit_functiondef(methods[0])
|
|
self.checker.visit_functiondef(methods[2])
|
|
self.checker.visit_functiondef(methods[3])
|
|
with self.assertAddsMessages(
|
|
Message(
|
|
"invalid-name",
|
|
node=methods[1],
|
|
args=("Attribute", "bar", "'[A-Z]+' pattern"),
|
|
)
|
|
):
|
|
self.checker.visit_functiondef(methods[1])
|
|
|
|
@set_config(attr_rgx=re.compile("[A-Z]+"))
|
|
def test_property_setters(self):
|
|
method = astroid.extract_node(
|
|
"""
|
|
class FooClass(object):
|
|
@property
|
|
def foo(self): pass
|
|
|
|
@foo.setter
|
|
def FOOSETTER(self): #@
|
|
pass
|
|
"""
|
|
)
|
|
with self.assertNoMessages():
|
|
self.checker.visit_functiondef(method)
|
|
|
|
def test_module_level_names(self):
|
|
assign = astroid.extract_node(
|
|
"""
|
|
import collections
|
|
Class = collections.namedtuple("a", ("b", "c")) #@
|
|
"""
|
|
)
|
|
with self.assertNoMessages():
|
|
self.checker.visit_assignname(assign.targets[0])
|
|
|
|
assign = astroid.extract_node(
|
|
"""
|
|
class ClassA(object):
|
|
pass
|
|
ClassB = ClassA
|
|
"""
|
|
)
|
|
with self.assertNoMessages():
|
|
self.checker.visit_assignname(assign.targets[0])
|
|
|
|
module = astroid.parse(
|
|
"""
|
|
def A():
|
|
return 1, 2, 3
|
|
CONSTA, CONSTB, CONSTC = A()
|
|
CONSTD = A()"""
|
|
)
|
|
with self.assertNoMessages():
|
|
self.checker.visit_assignname(module.body[1].targets[0].elts[0])
|
|
self.checker.visit_assignname(module.body[2].targets[0])
|
|
|
|
assign = astroid.extract_node(
|
|
"""
|
|
CONST = "12 34 ".rstrip().split()"""
|
|
)
|
|
with self.assertNoMessages():
|
|
self.checker.visit_assignname(assign.targets[0])
|
|
|
|
@unittest.skipIf(sys.version_info >= (3, 7), reason="Needs Python 3.6 or earlier")
|
|
@set_config(const_rgx=re.compile(".+"))
|
|
@set_config(function_rgx=re.compile(".+"))
|
|
@set_config(class_rgx=re.compile(".+"))
|
|
def test_assign_to_new_keyword_py3(self):
|
|
ast = astroid.extract_node(
|
|
"""
|
|
async = "foo" #@
|
|
await = "bar" #@
|
|
def async(): #@
|
|
pass
|
|
class async: #@
|
|
pass
|
|
"""
|
|
)
|
|
with self.assertAddsMessages(
|
|
Message(
|
|
msg_id="assign-to-new-keyword",
|
|
node=ast[0].targets[0],
|
|
args=("async", "3.7"),
|
|
)
|
|
):
|
|
self.checker.visit_assignname(ast[0].targets[0])
|
|
with self.assertAddsMessages(
|
|
Message(
|
|
msg_id="assign-to-new-keyword",
|
|
node=ast[1].targets[0],
|
|
args=("await", "3.7"),
|
|
)
|
|
):
|
|
self.checker.visit_assignname(ast[1].targets[0])
|
|
with self.assertAddsMessages(
|
|
Message(msg_id="assign-to-new-keyword", node=ast[2], args=("async", "3.7"))
|
|
):
|
|
self.checker.visit_functiondef(ast[2])
|
|
with self.assertAddsMessages(
|
|
Message(msg_id="assign-to-new-keyword", node=ast[3], args=("async", "3.7"))
|
|
):
|
|
self.checker.visit_classdef(ast[3])
|
|
|
|
|
|
class TestMultiNamingStyle(CheckerTestCase):
|
|
CHECKER_CLASS = base.NameChecker
|
|
|
|
MULTI_STYLE_RE = re.compile("(?:(?P<UP>[A-Z]+)|(?P<down>[a-z]+))$")
|
|
|
|
@set_config(class_rgx=MULTI_STYLE_RE)
|
|
def test_multi_name_detection_majority(self):
|
|
classes = astroid.extract_node(
|
|
"""
|
|
class classb(object): #@
|
|
pass
|
|
class CLASSA(object): #@
|
|
pass
|
|
class CLASSC(object): #@
|
|
pass
|
|
"""
|
|
)
|
|
message = Message(
|
|
"invalid-name",
|
|
node=classes[0],
|
|
args=("Class", "classb", "'(?:(?P<UP>[A-Z]+)|(?P<down>[a-z]+))$' pattern"),
|
|
)
|
|
with self.assertAddsMessages(message):
|
|
for cls in classes:
|
|
self.checker.visit_classdef(cls)
|
|
self.checker.leave_module(cls.root)
|
|
|
|
@set_config(class_rgx=MULTI_STYLE_RE)
|
|
def test_multi_name_detection_first_invalid(self):
|
|
classes = astroid.extract_node(
|
|
"""
|
|
class class_a(object): #@
|
|
pass
|
|
class classb(object): #@
|
|
pass
|
|
class CLASSC(object): #@
|
|
pass
|
|
"""
|
|
)
|
|
messages = [
|
|
Message(
|
|
"invalid-name",
|
|
node=classes[0],
|
|
args=(
|
|
"Class",
|
|
"class_a",
|
|
"'(?:(?P<UP>[A-Z]+)|(?P<down>[a-z]+))$' pattern",
|
|
),
|
|
),
|
|
Message(
|
|
"invalid-name",
|
|
node=classes[2],
|
|
args=(
|
|
"Class",
|
|
"CLASSC",
|
|
"'(?:(?P<UP>[A-Z]+)|(?P<down>[a-z]+))$' pattern",
|
|
),
|
|
),
|
|
]
|
|
with self.assertAddsMessages(*messages):
|
|
for cls in classes:
|
|
self.checker.visit_classdef(cls)
|
|
self.checker.leave_module(cls.root)
|
|
|
|
@set_config(
|
|
method_rgx=MULTI_STYLE_RE,
|
|
function_rgx=MULTI_STYLE_RE,
|
|
name_group=("function:method",),
|
|
)
|
|
def test_multi_name_detection_group(self):
|
|
function_defs = astroid.extract_node(
|
|
"""
|
|
class First(object):
|
|
def func(self): #@
|
|
pass
|
|
|
|
def FUNC(): #@
|
|
pass
|
|
""",
|
|
module_name="test",
|
|
)
|
|
message = Message(
|
|
"invalid-name",
|
|
node=function_defs[1],
|
|
args=("Function", "FUNC", "'(?:(?P<UP>[A-Z]+)|(?P<down>[a-z]+))$' pattern"),
|
|
)
|
|
with self.assertAddsMessages(message):
|
|
for func in function_defs:
|
|
self.checker.visit_functiondef(func)
|
|
self.checker.leave_module(func.root)
|
|
|
|
@set_config(
|
|
function_rgx=re.compile("(?:(?P<ignore>FOO)|(?P<UP>[A-Z]+)|(?P<down>[a-z]+))$")
|
|
)
|
|
def test_multi_name_detection_exempt(self):
|
|
function_defs = astroid.extract_node(
|
|
"""
|
|
def FOO(): #@
|
|
pass
|
|
def lower(): #@
|
|
pass
|
|
def FOO(): #@
|
|
pass
|
|
def UPPER(): #@
|
|
pass
|
|
"""
|
|
)
|
|
message = Message(
|
|
"invalid-name",
|
|
node=function_defs[3],
|
|
args=(
|
|
"Function",
|
|
"UPPER",
|
|
"'(?:(?P<ignore>FOO)|(?P<UP>[A-Z]+)|(?P<down>[a-z]+))$' pattern",
|
|
),
|
|
)
|
|
with self.assertAddsMessages(message):
|
|
for func in function_defs:
|
|
self.checker.visit_functiondef(func)
|
|
self.checker.leave_module(func.root)
|
|
|
|
|
|
class TestComparison(CheckerTestCase):
|
|
CHECKER_CLASS = base.ComparisonChecker
|
|
|
|
def test_comparison(self):
|
|
node = astroid.extract_node("foo == True")
|
|
message = Message("singleton-comparison", node=node, args=(True, "just 'expr'"))
|
|
with self.assertAddsMessages(message):
|
|
self.checker.visit_compare(node)
|
|
|
|
node = astroid.extract_node("foo == False")
|
|
message = Message("singleton-comparison", node=node, args=(False, "'not expr'"))
|
|
with self.assertAddsMessages(message):
|
|
self.checker.visit_compare(node)
|
|
|
|
node = astroid.extract_node("foo == None")
|
|
message = Message(
|
|
"singleton-comparison", node=node, args=(None, "'expr is None'")
|
|
)
|
|
with self.assertAddsMessages(message):
|
|
self.checker.visit_compare(node)
|
|
|
|
node = astroid.extract_node("True == foo")
|
|
messages = (
|
|
Message("misplaced-comparison-constant", node=node, args=("foo == True",)),
|
|
Message("singleton-comparison", node=node, args=(True, "just 'expr'")),
|
|
)
|
|
with self.assertAddsMessages(*messages):
|
|
self.checker.visit_compare(node)
|
|
|
|
node = astroid.extract_node("False == foo")
|
|
messages = (
|
|
Message("misplaced-comparison-constant", node=node, args=("foo == False",)),
|
|
Message("singleton-comparison", node=node, args=(False, "'not expr'")),
|
|
)
|
|
with self.assertAddsMessages(*messages):
|
|
self.checker.visit_compare(node)
|
|
|
|
node = astroid.extract_node("None == foo")
|
|
messages = (
|
|
Message("misplaced-comparison-constant", node=node, args=("foo == None",)),
|
|
Message("singleton-comparison", node=node, args=(None, "'expr is None'")),
|
|
)
|
|
with self.assertAddsMessages(*messages):
|
|
self.checker.visit_compare(node)
|
|
|
|
|
|
class TestNamePresets(unittest.TestCase):
|
|
SNAKE_CASE_NAMES = {"test_snake_case", "test_snake_case11", "test_https_200"}
|
|
CAMEL_CASE_NAMES = {"testCamelCase", "testCamelCase11", "testHTTP200"}
|
|
UPPER_CASE_NAMES = {"TEST_UPPER_CASE", "TEST_UPPER_CASE11", "TEST_HTTP_200"}
|
|
PASCAL_CASE_NAMES = {"TestPascalCase", "TestPascalCase11", "TestHTTP200"}
|
|
ALL_NAMES = (
|
|
SNAKE_CASE_NAMES | CAMEL_CASE_NAMES | UPPER_CASE_NAMES | PASCAL_CASE_NAMES
|
|
)
|
|
|
|
def _test_name_is_correct_for_all_name_types(self, naming_style, name):
|
|
for name_type in base.KNOWN_NAME_TYPES:
|
|
self._test_is_correct(naming_style, name, name_type)
|
|
|
|
def _test_name_is_incorrect_for_all_name_types(self, naming_style, name):
|
|
for name_type in base.KNOWN_NAME_TYPES:
|
|
self._test_is_incorrect(naming_style, name, name_type)
|
|
|
|
def _test_should_always_pass(self, naming_style):
|
|
always_pass_data = [
|
|
("__add__", "method"),
|
|
("__set_name__", "method"),
|
|
("__version__", "const"),
|
|
("__author__", "const"),
|
|
]
|
|
for name, name_type in always_pass_data:
|
|
self._test_is_correct(naming_style, name, name_type)
|
|
|
|
def _test_is_correct(self, naming_style, name, name_type):
|
|
rgx = naming_style.get_regex(name_type)
|
|
self.assertTrue(
|
|
rgx.match(name),
|
|
"{!r} does not match pattern {!r} (style: {}, type: {})".format(
|
|
name, rgx, naming_style, name_type
|
|
),
|
|
)
|
|
|
|
def _test_is_incorrect(self, naming_style, name, name_type):
|
|
rgx = naming_style.get_regex(name_type)
|
|
self.assertFalse(
|
|
rgx.match(name),
|
|
"{!r} match pattern {!r} but shouldn't (style: {}, type: {})".format(
|
|
name, rgx, naming_style, name_type
|
|
),
|
|
)
|
|
|
|
def test_snake_case(self):
|
|
naming_style = base.SnakeCaseStyle
|
|
|
|
for name in self.SNAKE_CASE_NAMES:
|
|
self._test_name_is_correct_for_all_name_types(naming_style, name)
|
|
for name in self.ALL_NAMES - self.SNAKE_CASE_NAMES:
|
|
self._test_name_is_incorrect_for_all_name_types(naming_style, name)
|
|
|
|
self._test_should_always_pass(naming_style)
|
|
|
|
def test_camel_case(self):
|
|
naming_style = base.CamelCaseStyle
|
|
|
|
for name in self.CAMEL_CASE_NAMES:
|
|
self._test_name_is_correct_for_all_name_types(naming_style, name)
|
|
for name in self.ALL_NAMES - self.CAMEL_CASE_NAMES:
|
|
self._test_name_is_incorrect_for_all_name_types(naming_style, name)
|
|
|
|
self._test_should_always_pass(naming_style)
|
|
|
|
def test_upper_case(self):
|
|
naming_style = base.UpperCaseStyle
|
|
|
|
for name in self.UPPER_CASE_NAMES:
|
|
self._test_name_is_correct_for_all_name_types(naming_style, name)
|
|
for name in self.ALL_NAMES - self.UPPER_CASE_NAMES:
|
|
self._test_name_is_incorrect_for_all_name_types(naming_style, name)
|
|
|
|
self._test_should_always_pass(naming_style)
|
|
|
|
def test_pascal_case(self):
|
|
naming_style = base.PascalCaseStyle
|
|
|
|
for name in self.PASCAL_CASE_NAMES:
|
|
self._test_name_is_correct_for_all_name_types(naming_style, name)
|
|
for name in self.ALL_NAMES - self.PASCAL_CASE_NAMES:
|
|
self._test_name_is_incorrect_for_all_name_types(naming_style, name)
|
|
|
|
self._test_should_always_pass(naming_style)
|