7787fefa30
hacking 3.0.x is too old. This also removes the note about old behavior of pip which has been changed in recent versions. Notes: - A few errors newly detected are also fixed. - This also fixes the compatibility with flake8 >= 3.0 because flake8 is also bumped when we bump hacking. Change-Id: I7fa87ed77bb39945ee5464692b7ef3d25c046127
187 lines
6.6 KiB
Python
187 lines
6.6 KiB
Python
# Copyright 2014 Hewlett-Packard Development Company, L.P.
|
|
# Copyright (c) 2012, Cloudscaling
|
|
#
|
|
# 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.
|
|
import re
|
|
|
|
from hacking import core
|
|
|
|
# D701: Default parameter value is a mutable type
|
|
# D702: Log messages require translation
|
|
# D703: Found use of _() without explicit import of _!
|
|
# D704: Found import of %s. This oslo library has been graduated!
|
|
# D705: timeutils.utcnow() must be used instead of datetime.%s()
|
|
# D706: Don't translate debug level logs
|
|
# D707: basestring is not Python3-compatible, use str instead.
|
|
# D708: Do not use xrange. Use range for large loops.
|
|
# D709: LOG.audit is deprecated, please use LOG.info!
|
|
# D710: LOG.warn() is not allowed. Use LOG.warning()
|
|
# D711: Don't use backslashes for line continuation.
|
|
|
|
UNDERSCORE_IMPORT_FILES = []
|
|
|
|
|
|
mutable_default_argument_check = re.compile(
|
|
r"^\s*def .+\((.+=\{\}|.+=\[\])")
|
|
string_translation = re.compile(r"[^_]*_\(\s*('|\")")
|
|
translated_log = re.compile(
|
|
r"(.)*LOG\.(audit|error|info|warn|warning|critical|exception)"
|
|
r"\(\s*_\(\s*('|\")")
|
|
underscore_import_check = re.compile(r"(.)*import _(.)*")
|
|
# We need this for cases where they have created their own _ function.
|
|
custom_underscore_check = re.compile(r"(.)*_\s*=\s*(.)*")
|
|
graduated_oslo_libraries_import_re = re.compile(
|
|
r"^\s*(?:import|from) designate\.openstack\.common\.?.*?"
|
|
r"(gettextutils|rpc)"
|
|
r".*?")
|
|
no_line_continuation_backslash_re = re.compile(r'.*(\\)\n')
|
|
|
|
|
|
@core.flake8ext
|
|
def mutable_default_arguments(logical_line, filename, noqa):
|
|
if noqa:
|
|
return
|
|
|
|
if mutable_default_argument_check.match(logical_line):
|
|
yield (0, "D701: Default parameter value is a mutable type")
|
|
|
|
|
|
@core.flake8ext
|
|
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.
|
|
N319
|
|
"""
|
|
if logical_line.startswith("LOG.debug(_("):
|
|
yield (0, "D706: Don't translate debug level logs")
|
|
|
|
|
|
@core.flake8ext
|
|
def check_explicit_underscore_import(logical_line, filename):
|
|
"""Check for explicit import of the _ function
|
|
|
|
We need to ensure that any files that are using the _() function
|
|
to translate logs are explicitly importing the _ function. We
|
|
can't trust unit test to catch whether the import has been
|
|
added so we need to check for it here.
|
|
"""
|
|
# Build a list of the files that have _ imported. No further
|
|
# checking needed once it is found.
|
|
if filename in UNDERSCORE_IMPORT_FILES:
|
|
pass
|
|
elif (underscore_import_check.match(logical_line) or
|
|
custom_underscore_check.match(logical_line)):
|
|
UNDERSCORE_IMPORT_FILES.append(filename)
|
|
elif (translated_log.match(logical_line) or
|
|
string_translation.match(logical_line)):
|
|
yield (0, "D703: Found use of _() without explicit import of _!")
|
|
|
|
|
|
@core.flake8ext
|
|
def no_import_graduated_oslo_libraries(logical_line, filename):
|
|
"""Check that we don't continue to use o.c. oslo libraries after graduation
|
|
|
|
After a library graduates from oslo-incubator, as we make the switch, we
|
|
should ensure we don't continue to use the oslo-incubator versions.
|
|
|
|
In many cases, it's not possible to immediately remove the code from the
|
|
openstack/common folder due to dependency issues.
|
|
"""
|
|
# We can't modify oslo-incubator code, so ignore it here.
|
|
if "designate/openstack/common" in filename:
|
|
return
|
|
|
|
matches = graduated_oslo_libraries_import_re.match(logical_line)
|
|
if matches:
|
|
yield (0, "D704: Found import of %s. This oslo library has been "
|
|
"graduated!" % matches.group(1))
|
|
|
|
|
|
@core.flake8ext
|
|
def use_timeutils_utcnow(logical_line, filename):
|
|
# tools are OK to use the standard datetime module
|
|
if "/tools/" in filename:
|
|
return
|
|
|
|
msg = "D705: timeutils.utcnow() must be used instead of datetime.%s()"
|
|
|
|
datetime_funcs = ['now', 'utcnow']
|
|
for f in datetime_funcs:
|
|
pos = logical_line.find('datetime.%s' % f)
|
|
if pos != -1:
|
|
yield (pos, msg % f)
|
|
|
|
|
|
@core.flake8ext
|
|
def check_no_basestring(logical_line):
|
|
if re.search(r"\bbasestring\b", logical_line):
|
|
msg = ("D707: basestring is not Python3-compatible, use "
|
|
"str instead.")
|
|
yield (0, msg)
|
|
|
|
|
|
@core.flake8ext
|
|
def check_python3_xrange(logical_line):
|
|
if re.search(r"\bxrange\s*\(", logical_line):
|
|
yield (0, "D708: Do not use xrange. Use range for "
|
|
"large loops.")
|
|
|
|
|
|
@core.flake8ext
|
|
def check_no_log_audit(logical_line):
|
|
"""Ensure that we are not using LOG.audit messages
|
|
Plans are in place going forward as discussed in the following
|
|
spec (https://review.opendev.org/#/c/132552/) to take out
|
|
LOG.audit messages. Given that audit was a concept invented
|
|
for OpenStack we can enforce not using it.
|
|
"""
|
|
if "LOG.audit(" in logical_line:
|
|
yield (0, "D709: LOG.audit is deprecated, please use LOG.info!")
|
|
|
|
|
|
@core.flake8ext
|
|
def check_no_log_warn(logical_line):
|
|
"""Disallow 'LOG.warn('
|
|
|
|
D710
|
|
"""
|
|
if logical_line.startswith('LOG.warn('):
|
|
yield (0, "D710:Use LOG.warning() rather than LOG.warn()")
|
|
|
|
|
|
@core.flake8ext
|
|
def check_line_continuation_no_backslash(logical_line, tokens):
|
|
"""D711 - Don't use backslashes for line continuation.
|
|
|
|
:param logical_line: The logical line to check. Not actually used.
|
|
:param tokens: List of tokens to check.
|
|
:returns: None if the tokens don't contain any issues, otherwise a tuple
|
|
is yielded that contains the offending index in the logical
|
|
line and a message describe the check validation failure.
|
|
"""
|
|
backslash = None
|
|
for token_type, text, start, end, orig_line in tokens:
|
|
m = no_line_continuation_backslash_re.match(orig_line)
|
|
if m:
|
|
backslash = (start[0], m.start(1))
|
|
break
|
|
|
|
if backslash is not None:
|
|
msg = 'D711 Backslash line continuations not allowed'
|
|
yield backslash, msg
|