148459e49d
The correct format is: :raises Exception: conditions This patch fix all docstrings across the project and add hacking rule to check it. Change-Id: I40cc20f8d555082e284c1683522932aaa68f575c Closes-Bug: 1470901
477 lines
16 KiB
Python
477 lines
16 KiB
Python
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
# not use this file except in compliance with the License. You may obtain
|
|
# a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
# License for the specific language governing permissions and limitations
|
|
# under the License.
|
|
|
|
"""
|
|
Guidelines for writing new hacking checks
|
|
|
|
- Use only for Rally specific tests. OpenStack general tests
|
|
should be submitted to the common 'hacking' module.
|
|
- Pick numbers in the range N3xx. Find the current test with
|
|
the highest allocated number and then pick the next value.
|
|
- Keep the test method code in the source file ordered based
|
|
on the N3xx value.
|
|
- List the new rule in the top level HACKING.rst file
|
|
- Add test cases for each new rule to tests/unit/test_hacking.py
|
|
|
|
"""
|
|
|
|
import functools
|
|
import re
|
|
import tokenize
|
|
|
|
re_assert_true_instance = re.compile(
|
|
r"(.)*assertTrue\(isinstance\((\w|\.|\'|\"|\[|\])+, "
|
|
r"(\w|\.|\'|\"|\[|\])+\)\)")
|
|
re_assert_equal_type = re.compile(
|
|
r"(.)*assertEqual\(type\((\w|\.|\'|\"|\[|\])+\), "
|
|
r"(\w|\.|\'|\"|\[|\])+\)")
|
|
re_assert_equal_end_with_none = re.compile(r"assertEqual\(.*?,\s+None\)$")
|
|
re_assert_equal_start_with_none = re.compile(r"assertEqual\(None,")
|
|
re_assert_true_false_with_in_or_not_in = re.compile(
|
|
r"assert(True|False)\("
|
|
r"(\w|[][.'\"])+( not)? in (\w|[][.'\",])+(, .*)?\)")
|
|
re_assert_true_false_with_in_or_not_in_spaces = re.compile(
|
|
r"assert(True|False)\((\w|[][.'\"])+( not)? in [\[|'|\"](\w|[][.'\", ])+"
|
|
r"[\[|'|\"](, .*)?\)")
|
|
re_assert_equal_in_end_with_true_or_false = re.compile(
|
|
r"assertEqual\((\w|[][.'\"])+( not)? in (\w|[][.'\", ])+, (True|False)\)")
|
|
re_assert_equal_in_start_with_true_or_false = re.compile(
|
|
r"assertEqual\((True|False), (\w|[][.'\"])+( not)? in (\w|[][.'\", ])+\)")
|
|
re_no_construct_dict = re.compile(
|
|
r"\sdict\(\)")
|
|
re_no_construct_list = re.compile(
|
|
r"\slist\(\)")
|
|
re_str_format = re.compile(r"""
|
|
% # start of specifier
|
|
\(([^)]+)\) # mapping key, in group 1
|
|
[#0 +\-]? # optional conversion flag
|
|
(?:-?\d*)? # optional minimum field width
|
|
(?:\.\d*)? # optional precision
|
|
[hLl]? # optional length modifier
|
|
[A-z%] # conversion modifier
|
|
""", re.X)
|
|
re_raises = re.compile(
|
|
r"\s:raise[^s] *.*$|\s:raises *:.*$|\s:raises *[^:]+$")
|
|
|
|
|
|
def skip_ignored_lines(func):
|
|
|
|
@functools.wraps(func)
|
|
def wrapper(logical_line, filename):
|
|
line = logical_line.strip()
|
|
if not line or line.startswith("#") or line.endswith("# noqa"):
|
|
return
|
|
yield next(func(logical_line, filename))
|
|
|
|
return wrapper
|
|
|
|
|
|
def _parse_assert_mock_str(line):
|
|
point = line.find(".assert_")
|
|
|
|
if point != -1:
|
|
end_pos = line[point:].find("(") + point
|
|
return point, line[point + 1: end_pos], line[: point]
|
|
else:
|
|
return None, None, None
|
|
|
|
|
|
@skip_ignored_lines
|
|
def check_assert_methods_from_mock(logical_line, filename):
|
|
"""Ensure that ``assert_*`` methods from ``mock`` library is used correctly
|
|
|
|
N301 - base error number
|
|
N302 - related to nonexistent "assert_called"
|
|
N303 - related to nonexistent "assert_called_once"
|
|
"""
|
|
|
|
correct_names = ["assert_any_call", "assert_called_once_with",
|
|
"assert_called_with", "assert_has_calls"]
|
|
ignored_files = ["./tests/unit/test_hacking.py"]
|
|
|
|
if filename.startswith("./tests") and filename not in ignored_files:
|
|
pos, method_name, obj_name = _parse_assert_mock_str(logical_line)
|
|
|
|
if pos:
|
|
if method_name not in correct_names:
|
|
error_number = "N301"
|
|
msg = ("%(error_number)s:'%(method)s' is not present in `mock`"
|
|
" library. %(custom_msg)s For more details, visit "
|
|
"http://www.voidspace.org.uk/python/mock/ .")
|
|
|
|
if method_name == "assert_called":
|
|
error_number = "N302"
|
|
custom_msg = ("Maybe, you should try to use "
|
|
"'assertTrue(%s.called)' instead." %
|
|
obj_name)
|
|
elif method_name == "assert_called_once":
|
|
# For more details, see a bug in Rally:
|
|
# https://bugs.launchpad.net/rally/+bug/1305991
|
|
error_number = "N303"
|
|
custom_msg = ("Maybe, you should try to use "
|
|
"'assertEqual(1, %s.call_count)' "
|
|
"or '%s.assert_called_once_with()'"
|
|
" instead." % (obj_name, obj_name))
|
|
else:
|
|
custom_msg = ("Correct 'assert_*' methods: '%s'."
|
|
% "', '".join(correct_names))
|
|
|
|
yield (pos, msg % {
|
|
"error_number": error_number,
|
|
"method": method_name,
|
|
"custom_msg": custom_msg})
|
|
|
|
|
|
@skip_ignored_lines
|
|
def check_import_of_logging(logical_line, filename):
|
|
"""Check correctness import of logging module
|
|
|
|
N310
|
|
"""
|
|
|
|
excluded_files = ["./rally/common/log.py", "./tests/unit/test_log.py"]
|
|
|
|
forbidden_imports = ["from oslo_log",
|
|
"import oslo_log",
|
|
"import logging"]
|
|
|
|
if filename not in excluded_files:
|
|
for forbidden_import in forbidden_imports:
|
|
if logical_line.startswith(forbidden_import):
|
|
yield (0, "N310 Wrong module for logging is imported. Please "
|
|
"use `rally.common.log` instead.")
|
|
|
|
|
|
@skip_ignored_lines
|
|
def no_translate_debug_logs(logical_line, filename):
|
|
"""Check for "LOG.debug(_("
|
|
|
|
As per our translation policy,
|
|
https://wiki.openstack.org/wiki/LoggingStandards#Log_Translation
|
|
we shouldn't translate debug level logs.
|
|
|
|
* This check assumes that 'LOG' is a logger.
|
|
* Use filename so we can start enforcing this in specific folders instead
|
|
of needing to do so all at once.
|
|
|
|
N311
|
|
"""
|
|
if logical_line.startswith("LOG.debug(_("):
|
|
yield(0, "N311 Don't translate debug level logs")
|
|
|
|
|
|
@skip_ignored_lines
|
|
def no_use_conf_debug_check(logical_line, filename):
|
|
"""Check for "cfg.CONF.debug"
|
|
|
|
Rally has two DEBUG level:
|
|
- Full DEBUG, which include all debug-messages from all OpenStack services
|
|
- Rally DEBUG, which include only Rally debug-messages
|
|
so we should use custom check to know debug-mode, instead of CONF.debug
|
|
|
|
N312
|
|
"""
|
|
excluded_files = ["./rally/common/log.py"]
|
|
|
|
point = logical_line.find("CONF.debug")
|
|
if point != -1 and filename not in excluded_files:
|
|
yield(point, "N312 Don't use `CONF.debug`. "
|
|
"Function `rally.common.log.is_debug` "
|
|
"should be used instead.")
|
|
|
|
|
|
@skip_ignored_lines
|
|
def assert_true_instance(logical_line, filename):
|
|
"""Check for assertTrue(isinstance(a, b)) sentences
|
|
|
|
N320
|
|
"""
|
|
if re_assert_true_instance.match(logical_line):
|
|
yield (0, "N320 assertTrue(isinstance(a, b)) sentences not allowed, "
|
|
"you should use assertIsInstance(a, b) instead.")
|
|
|
|
|
|
@skip_ignored_lines
|
|
def assert_equal_type(logical_line, filename):
|
|
"""Check for assertEqual(type(A), B) sentences
|
|
|
|
N321
|
|
"""
|
|
if re_assert_equal_type.match(logical_line):
|
|
yield (0, "N321 assertEqual(type(A), B) sentences not allowed, "
|
|
"you should use assertIsInstance(a, b) instead.")
|
|
|
|
|
|
@skip_ignored_lines
|
|
def assert_equal_none(logical_line, filename):
|
|
"""Check for assertEqual(A, None) or assertEqual(None, A) sentences
|
|
|
|
N322
|
|
"""
|
|
res = (re_assert_equal_start_with_none.search(logical_line) or
|
|
re_assert_equal_end_with_none.search(logical_line))
|
|
if res:
|
|
yield (0, "N322 assertEqual(A, None) or assertEqual(None, A) "
|
|
"sentences not allowed, you should use assertIsNone(A) "
|
|
"instead.")
|
|
|
|
|
|
@skip_ignored_lines
|
|
def assert_true_or_false_with_in(logical_line, filename):
|
|
"""Check assertTrue/False(A in/not in B) with collection contents
|
|
|
|
Check for assertTrue/False(A in B), assertTrue/False(A not in B),
|
|
assertTrue/False(A in B, message) or assertTrue/False(A not in B, message)
|
|
sentences.
|
|
|
|
N323
|
|
"""
|
|
res = (re_assert_true_false_with_in_or_not_in.search(logical_line) or
|
|
re_assert_true_false_with_in_or_not_in_spaces.search(logical_line))
|
|
if res:
|
|
yield (0, "N323 assertTrue/assertFalse(A in/not in B)sentences not "
|
|
"allowed, you should use assertIn(A, B) or assertNotIn(A, B)"
|
|
" instead.")
|
|
|
|
|
|
@skip_ignored_lines
|
|
def assert_equal_in(logical_line, filename):
|
|
"""Check assertEqual(A in/not in B, True/False) with collection contents
|
|
|
|
Check for assertEqual(A in B, True/False), assertEqual(True/False, A in B),
|
|
assertEqual(A not in B, True/False) or assertEqual(True/False, A not in B)
|
|
sentences.
|
|
|
|
N324
|
|
"""
|
|
res = (re_assert_equal_in_end_with_true_or_false.search(logical_line) or
|
|
re_assert_equal_in_start_with_true_or_false.search(logical_line))
|
|
if res:
|
|
yield (0, "N324: Use assertIn/NotIn(A, B) rather than "
|
|
"assertEqual(A in/not in B, True/False) when checking "
|
|
"collection contents.")
|
|
|
|
|
|
@skip_ignored_lines
|
|
def check_no_direct_rally_objects_import(logical_line, filename):
|
|
"""Check if rally.common.objects are properly imported.
|
|
|
|
If you import "from rally.common import objects" you are able to use
|
|
objects directly like objects.Task.
|
|
|
|
N340
|
|
"""
|
|
if filename == "./rally/common/objects/__init__.py":
|
|
return
|
|
|
|
if (logical_line.startswith("from rally.common.objects")
|
|
or logical_line.startswith("import rally.common.objects.")):
|
|
yield (0, "N340: Import objects module:"
|
|
"`from rally.common import objects`. "
|
|
"After that you can use directly objects e.g. objects.Task")
|
|
|
|
|
|
@skip_ignored_lines
|
|
def check_no_oslo_deprecated_import(logical_line, filename):
|
|
"""Check if oslo.foo packages are not imported instead of oslo_foo ones.
|
|
|
|
Libraries from oslo.foo namespace are deprecated because of namespace
|
|
problems.
|
|
|
|
N341
|
|
"""
|
|
if (logical_line.startswith("from oslo.")
|
|
or logical_line.startswith("import oslo.")):
|
|
yield (0, "N341: Import oslo module: `from oslo_xyz import ...`. "
|
|
"The oslo.xyz namespace was deprecated, use oslo_xyz "
|
|
"instead")
|
|
|
|
|
|
@skip_ignored_lines
|
|
def check_quotes(logical_line, filename):
|
|
"""Check that single quotation marks are not used
|
|
|
|
N350
|
|
"""
|
|
|
|
in_string = False
|
|
in_multiline_string = False
|
|
single_quotas_are_used = False
|
|
|
|
check_tripple = (
|
|
lambda line, i, char: (
|
|
i + 2 < len(line) and
|
|
(char == line[i] == line[i + 1] == line[i + 2])
|
|
)
|
|
)
|
|
|
|
i = 0
|
|
while i < len(logical_line):
|
|
char = logical_line[i]
|
|
|
|
if in_string:
|
|
if char == "\"":
|
|
in_string = False
|
|
if char == "\\":
|
|
i += 1 # ignore next char
|
|
|
|
elif in_multiline_string:
|
|
if check_tripple(logical_line, i, "\""):
|
|
i += 2 # skip next 2 chars
|
|
in_multiline_string = False
|
|
|
|
elif char == "#":
|
|
break
|
|
|
|
elif char == "'":
|
|
single_quotas_are_used = True
|
|
break
|
|
|
|
elif char == "\"":
|
|
if check_tripple(logical_line, i, "\""):
|
|
in_multiline_string = True
|
|
i += 3
|
|
continue
|
|
in_string = True
|
|
|
|
i += 1
|
|
|
|
if single_quotas_are_used:
|
|
yield (i, "N350 Remove Single quotes")
|
|
|
|
|
|
@skip_ignored_lines
|
|
def check_no_constructor_data_struct(logical_line, filename):
|
|
"""Check that data structs (lists, dicts) are declared using literals
|
|
|
|
N351
|
|
"""
|
|
|
|
match = re_no_construct_dict.search(logical_line)
|
|
if match:
|
|
yield (0, "N351 Remove dict() construct and use literal {}")
|
|
match = re_no_construct_list.search(logical_line)
|
|
if match:
|
|
yield (0, "N351 Remove list() construct and use literal []")
|
|
|
|
|
|
def check_dict_formatting_in_string(logical_line, tokens):
|
|
"""Check that strings do not use dict-formatting with a single replacement
|
|
|
|
N352
|
|
"""
|
|
# NOTE(stpierre): Can't use @skip_ignored_lines here because it's
|
|
# a stupid decorator that only works on functions that take
|
|
# (logical_line, filename) as arguments.
|
|
if (not logical_line or
|
|
logical_line.startswith("#") or
|
|
logical_line.endswith("# noqa")):
|
|
return
|
|
|
|
current_string = ""
|
|
in_string = False
|
|
for token_type, text, start, end, line in tokens:
|
|
if token_type == tokenize.STRING:
|
|
if not in_string:
|
|
current_string = ""
|
|
in_string = True
|
|
current_string += text.strip("\"")
|
|
elif token_type == tokenize.OP:
|
|
if not current_string:
|
|
continue
|
|
# NOTE(stpierre): The string formatting operator % has
|
|
# lower precedence than +, so we assume that the logical
|
|
# string has concluded whenever we hit an operator of any
|
|
# sort. (Most operators don't work for strings anyway.)
|
|
# Some string operators do have higher precedence than %,
|
|
# though, so you can technically trick this check by doing
|
|
# things like:
|
|
#
|
|
# "%(foo)s" * 1 % {"foo": 1}
|
|
# "%(foo)s"[:] % {"foo": 1}
|
|
#
|
|
# It also will produce false positives if you use explicit
|
|
# parenthesized addition for two strings instead of
|
|
# concatenation by juxtaposition, e.g.:
|
|
#
|
|
# ("%(foo)s" + "%(bar)s") % vals
|
|
#
|
|
# But if you do any of those things, then you deserve all
|
|
# of the horrible things that happen to you, and probably
|
|
# many more.
|
|
in_string = False
|
|
if text == "%":
|
|
format_keys = set()
|
|
for match in re_str_format.finditer(current_string):
|
|
format_keys.add(match.group(1))
|
|
if len(format_keys) == 1:
|
|
yield (0,
|
|
"N353 Do not use mapping key string formatting "
|
|
"with a single key")
|
|
if text != ")":
|
|
# NOTE(stpierre): You can have a parenthesized string
|
|
# followed by %, so a closing paren doesn't obviate
|
|
# the possibility for a substitution operator like
|
|
# every other operator does.
|
|
current_string = ""
|
|
elif token_type in (tokenize.NL, tokenize.COMMENT):
|
|
continue
|
|
else:
|
|
in_string = False
|
|
if token_type == tokenize.NEWLINE:
|
|
current_string = ""
|
|
|
|
|
|
@skip_ignored_lines
|
|
def check_using_unicode(logical_line, filename):
|
|
"""Check crosspython unicode usage
|
|
|
|
N353
|
|
"""
|
|
|
|
if re.search(r"\bunicode\(", logical_line):
|
|
yield (0, "N353 'unicode' function is absent in python3. Please "
|
|
"use 'six.text_type' instead.")
|
|
|
|
|
|
def check_raises(physical_line, filename):
|
|
"""Check raises usage
|
|
|
|
N354
|
|
"""
|
|
|
|
ignored_files = ["./tests/unit/test_hacking.py",
|
|
"./tests/hacking/checks.py"]
|
|
if filename not in ignored_files:
|
|
if re_raises.search(physical_line):
|
|
return (0, "N354 ':Please use ':raises Exception: conditions' "
|
|
"in docstrings.")
|
|
|
|
|
|
def factory(register):
|
|
register(check_assert_methods_from_mock)
|
|
register(check_import_of_logging)
|
|
register(no_translate_debug_logs)
|
|
register(no_use_conf_debug_check)
|
|
register(assert_true_instance)
|
|
register(assert_equal_type)
|
|
register(assert_equal_none)
|
|
register(assert_true_or_false_with_in)
|
|
register(assert_equal_in)
|
|
register(check_no_direct_rally_objects_import)
|
|
register(check_no_oslo_deprecated_import)
|
|
register(check_quotes)
|
|
register(check_no_constructor_data_struct)
|
|
register(check_dict_formatting_in_string)
|
|
register(check_using_unicode)
|
|
register(check_raises)
|