blob: facba6cfceff2aeb6feea9b92e969a5e197d76c1 [file] [log] [blame]
#!/usr/bin/env bash
# Licensed to the Apache Software Foundation (ASF) under one or more
# contributor license agreements. See the NOTICE file distributed with
# this work for additional information regarding copyright ownership.
# The ASF licenses this file to You 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.
add_test_type pylint
PYLINT_TIMER=0
PYLINT=${PYLINT:-$(which pylint 2>/dev/null)}
PYLINT_OPTIONS=${PYLINT_OPTIONS:-}
function pylint_usage
{
yetus_add_option "--pylint=<path>" "path to pylint executable"
yetus_add_option "--pylint-options=<path>" "pylint options other than output-format and reports"
}
function pylint_parse_args
{
local i
for i in "$@"; do
case ${i} in
--pylint=*)
PYLINT=${i#*=}
;;
--pylint-options=*)
PYLINT_OPTIONS=${i#*=}
;;
esac
done
}
function pylint_filefilter
{
local filename=$1
if [[ ${filename} =~ \.py$ ]]; then
add_test pylint
fi
}
function pylint_precheck
{
if ! verify_command "Pylint" "${PYLINT}"; then
add_vote_table 0 pylint "Pylint was not available."
delete_test pylint
fi
}
function pylint_preapply
{
local i
local count
local pylintStderr=branch-pylint-stderr.txt
if ! verify_needed_test pylint; then
return 0
fi
big_console_header "pylint plugin: ${PATCH_BRANCH}"
start_clock
echo "Running pylint against identified python scripts."
pushd "${BASEDIR}" >/dev/null
for i in "${CHANGED_FILES[@]}"; do
if [[ ${i} =~ \.py$ && -f ${i} ]]; then
# shellcheck disable=SC2086
eval "${PYLINT} ${PYLINT_OPTIONS} --msg-template='{path}:{line}: [{msg_id}({symbol}), {obj}] {msg}' --reports=n ${i}" \
2>>${PATCH_DIR}/${pylintStderr} | ${AWK} '1<NR' >> "${PATCH_DIR}/branch-pylint-result.txt"
fi
done
if [[ -f ${PATCH_DIR}/${pylintStderr} ]]; then
count=$(${GREP} -Evc "^(No config file found|Using config file)" "${PATCH_DIR}/${pylintStderr}")
if [[ ${count} -gt 0 ]]; then
add_footer_table pylint "${PATCH_BRANCH} stderr: @@BASE@@/${pylintStderr}"
return 1
fi
fi
rm "${PATCH_DIR}/${pylintStderr}" 2>/dev/null
popd >/dev/null
# keep track of how much as elapsed for us already
PYLINT_TIMER=$(stop_clock)
return 0
}
function pylint_postapply
{
declare i
declare count
declare numPrepatch
declare numPostpatch
declare diffPostpatch
declare pylintStderr=patch-pylint-stderr.txt
declare fixedpatch
declare statstring
if ! verify_needed_test pylint; then
return 0
fi
big_console_header "pylint plugin: ${BUILDMODE}"
start_clock
# add our previous elapsed to our new timer
# by setting the clock back
offset_clock "${PYLINT_TIMER}"
echo "Running pylint against identified python scripts."
# we re-check this in case one has been added
pushd "${BASEDIR}" >/dev/null
for i in "${CHANGED_FILES[@]}"; do
if [[ ${i} =~ \.py$ && -f ${i} ]]; then
# shellcheck disable=SC2086
eval "${PYLINT} ${PYLINT_OPTIONS} --msg-template='{path}:{line}: [{msg_id}({symbol}), {obj}] {msg}' --reports=n ${i}" \
2>>${PATCH_DIR}/${pylintStderr} | ${AWK} '1<NR' >> "${PATCH_DIR}/patch-pylint-result.txt"
fi
done
if [[ -f ${PATCH_DIR}/${pylintStderr} ]]; then
count=$(${GREP} -Evc "^(No config file found|Using config file)" "${PATCH_DIR}/${pylintStderr}")
if [[ ${count} -gt 0 ]]; then
add_vote_table -1 pylint "Something bad seems to have happened in running pylint. Please check pylint stderr files."
add_footer_table pylint "${BUILDMODEMSG} stderr: @@BASE@@/${pylintStderr}"
return 1
fi
fi
rm "${PATCH_DIR}/${pylintStderr}" 2>/dev/null
popd >/dev/null
# shellcheck disable=SC2016
PYLINT_VERSION=$(${PYLINT} --version 2>/dev/null | ${GREP} pylint | ${AWK} '{print $NF}')
add_footer_table pylint "v${PYLINT_VERSION%,}"
calcdiffs "${PATCH_DIR}/branch-pylint-result.txt" \
"${PATCH_DIR}/patch-pylint-result.txt" \
pylint > "${PATCH_DIR}/diff-patch-pylint.txt"
numPrepatch=$(${GREP} -c "^.*:.*: \[.*\] " "${PATCH_DIR}/branch-pylint-result.txt")
numPostpatch=$(${GREP} -c "^.*:.*: \[.*\] " "${PATCH_DIR}/patch-pylint-result.txt")
# Exclude Pylint messages from the information category to avoid false positives (see YETUS-309).
diffPostpatch=$(${GREP} -c "^.*:.*: \[[^I].*\] " "${PATCH_DIR}/diff-patch-pylint.txt")
((fixedpatch=numPrepatch-numPostpatch+diffPostpatch))
statstring=$(generic_calcdiff_status "${numPrepatch}" "${numPostpatch}" "${diffPostpatch}" )
if [[ ${diffPostpatch} -gt 0 ]] ; then
add_vote_table -1 pylint "${BUILDMODEMSG} ${statstring}"
add_footer_table pylint "@@BASE@@/diff-patch-pylint.txt"
return 1
elif [[ ${fixedpatch} -gt 0 ]]; then
add_vote_table +1 pylint "${BUILDMODEMSG} ${statstring}"
return 0
fi
add_vote_table +1 pylint "There were no new pylint issues."
return 0
}
function pylint_postcompile
{
declare repostatus=$1
if [[ "${repostatus}" = branch ]]; then
pylint_preapply
else
pylint_postapply
fi
}