2016-04-10 04:00:00 +00:00
|
|
|
#!/usr/bin/python3
|
2014-04-07 12:38:09 +00:00
|
|
|
""" Run the Anaconda pylint tests on the files changed in this commit
|
|
|
|
|
|
|
|
Set NOPYLINT env variable to skip this. eg. NOPYLINT= git commit
|
|
|
|
|
|
|
|
|
|
|
|
"""
|
2016-04-10 04:00:00 +00:00
|
|
|
|
2014-04-07 12:38:09 +00:00
|
|
|
import os
|
|
|
|
import sys
|
|
|
|
from subprocess import check_output, CalledProcessError
|
2015-05-30 11:20:59 +00:00
|
|
|
import tempfile
|
2014-04-07 12:38:09 +00:00
|
|
|
|
|
|
|
def is_python(filename):
|
|
|
|
if filename.endswith(".py"):
|
|
|
|
return True
|
|
|
|
|
|
|
|
try:
|
|
|
|
with open(filename) as testfile:
|
|
|
|
if testfile.readline().startswith("#!/usr/bin/python"):
|
|
|
|
return True
|
|
|
|
except IOError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
return False
|
|
|
|
|
2017-01-09 02:09:07 +00:00
|
|
|
OTHER_MODULES_PATH = ".:./tests/lib:../blivet/:../pykickstart/"
|
2014-04-07 12:38:09 +00:00
|
|
|
|
|
|
|
if "NOPYLINT" in os.environ:
|
2015-03-23 11:36:12 +00:00
|
|
|
print("Skipping pre-commit pylint run")
|
2014-04-07 12:38:09 +00:00
|
|
|
sys.exit(0)
|
|
|
|
|
|
|
|
# run pylint on all the python files changed by this commit
|
|
|
|
try:
|
2015-05-30 11:20:59 +00:00
|
|
|
git_files = check_output(["git", "status", "--porcelain", "-z"])
|
2014-04-07 12:38:09 +00:00
|
|
|
except CalledProcessError:
|
|
|
|
sys.exit(1)
|
|
|
|
|
2015-05-30 11:20:59 +00:00
|
|
|
# If git status returned nothing, return
|
|
|
|
if not git_files:
|
|
|
|
sys.exit(0)
|
|
|
|
|
2016-04-10 04:00:00 +00:00
|
|
|
git_files = git_files.decode('utf-8').strip('\0').split('\0')
|
2015-05-30 11:20:59 +00:00
|
|
|
|
|
|
|
pylint_names = []
|
2014-04-07 12:38:09 +00:00
|
|
|
pylint_files = []
|
|
|
|
# Lines look like: MM tests/pylint/runpylint.sh
|
|
|
|
# The first character is the status in the index (or our side of a merge),
|
|
|
|
# the second character is the status in the tree (or their side of a merge).
|
2015-05-30 11:20:59 +00:00
|
|
|
# Use an iterator so we can skip lines in the case of renames
|
|
|
|
git_iter = iter(git_files)
|
|
|
|
for gf in git_iter:
|
2015-03-23 11:36:12 +00:00
|
|
|
# If the file is being removed, or is not in git, or is not part of this
|
|
|
|
# commit, ignore it
|
|
|
|
if gf[0] in ('D', '?', ' '):
|
2014-04-07 12:38:09 +00:00
|
|
|
continue
|
2015-05-30 11:20:59 +00:00
|
|
|
|
|
|
|
# If the file is being renamed, the target filename (which we want) is on
|
|
|
|
# the current line, and the next line is the source, which we want to skip
|
2015-03-23 11:36:12 +00:00
|
|
|
if gf[0] == 'R':
|
2017-01-09 02:09:07 +00:00
|
|
|
next(git_iter)
|
2015-05-30 11:20:59 +00:00
|
|
|
|
|
|
|
if is_python(gf[3:]):
|
|
|
|
pylint_names.append(gf[3:])
|
|
|
|
|
|
|
|
# If the file is unmerged, changed, or deleted locally, create a
|
|
|
|
# temporary file with the actual content that will be committed
|
|
|
|
if gf[0] == 'U' or gf[1] in ('U', 'M', 'D'):
|
|
|
|
tmpfile = tempfile.NamedTemporaryFile(prefix=gf[3:].replace('/', '.'))
|
|
|
|
tmpfile.write(check_output(["git", "show", ":%s" % gf[3:]]))
|
|
|
|
tmpfile.flush()
|
|
|
|
pylint_files.append(tmpfile.name)
|
|
|
|
else:
|
|
|
|
pylint_files.append(gf[3:])
|
2014-04-07 12:38:09 +00:00
|
|
|
|
|
|
|
if not pylint_files:
|
|
|
|
sys.exit(0)
|
|
|
|
|
|
|
|
# Make sure pykickstart and blivet can be found
|
|
|
|
# Note that if the checked out versions are too far off pylint may fail
|
|
|
|
env = os.environ.copy()
|
|
|
|
env["PYTHONPATH"] = OTHER_MODULES_PATH
|
|
|
|
|
2015-05-30 11:20:59 +00:00
|
|
|
print("Running pylint on %s" % " ".join(pylint_names))
|
2014-04-07 12:38:09 +00:00
|
|
|
try:
|
2016-04-10 04:00:00 +00:00
|
|
|
check_output(["./tests/pylint/runpylint.py"] + pylint_files, env=env)
|
2014-04-07 12:38:09 +00:00
|
|
|
except CalledProcessError as e:
|
2016-04-10 04:00:00 +00:00
|
|
|
print(e.output.decode('utf-8'))
|
2014-04-07 12:38:09 +00:00
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
sys.exit(0)
|