mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-17 05:12:57 +01:00
Merge pull request #3403 from phiros/buildtest_verbose
compile_test.py / buildtest: display errors and warnings
This commit is contained in:
commit
8850eb4035
@ -45,52 +45,57 @@ buildtest:
|
||||
APP_RETRY=0; \
|
||||
rm -rf "$$BINDIRBASE"; \
|
||||
for BOARD in $$($(MAKE) -s info-boards-supported); do \
|
||||
RIOTNOLINK=$$(echo $(BOARD_INSUFFICIENT_RAM) | grep $${BOARD} 2>&1 >/dev/null && echo 1); \
|
||||
${COLOR_ECHO} -n "Building for $${BOARD} "; \
|
||||
[ -n "$${RIOTNOLINK}" ] && ${COLOR_ECHO} -n "(no linking) "; \
|
||||
for NTH_TRY in 1 2 3; do \
|
||||
${COLOR_ECHO} -n ".. "; \
|
||||
LOG=$$(env -i \
|
||||
HOME=$${HOME} \
|
||||
PATH=$${PATH} \
|
||||
BOARD=$${BOARD} \
|
||||
CCACHE=$${CCACHE} \
|
||||
CCACHE_DIR=$${CCACHE_DIR} \
|
||||
CCACHE_BASEDIR=$${CCACHE_BASEDIR} \
|
||||
RIOTBASE=$${RIOTBASE} \
|
||||
RIOTBOARD=$${RIOTBOARD} \
|
||||
RIOTCPU=$${RIOTCPU} \
|
||||
BINDIRBASE=$${BINDIRBASE} \
|
||||
RIOTNOLINK=$${RIOTNOLINK} \
|
||||
RIOT_VERSION=$${RIOT_VERSION} \
|
||||
$(MAKE) -j$(NPROC) 2>&1) ; \
|
||||
if [ "$${?}" = "0" ]; then \
|
||||
${COLOR_ECHO} "${COLOR_GREEN}success${COLOR_RESET}"; \
|
||||
elif [ -n "$${RIOT_DO_RETRY}" ] && [ "$${APP_RETRY}" -lt "3" ] && [ $${NTH_TRY} != 3 ]; then \
|
||||
${COLOR_ECHO} -n "${COLOR_PURPLE}retrying${COLOR_RESET} "; \
|
||||
continue; \
|
||||
else \
|
||||
${COLOR_ECHO} "${COLOR_RED}failed${COLOR_RESET}"; \
|
||||
echo "$${LOG}" | grep -v -E '^make(\[[[:digit:]]])?:'; \
|
||||
APP_RETRY=`expr $${APP_RETRY} + 1`; \
|
||||
BUILDTESTOK=false; \
|
||||
fi; \
|
||||
break; \
|
||||
done; \
|
||||
env -i \
|
||||
HOME=$${HOME} \
|
||||
PATH=$${PATH} \
|
||||
BOARD=$${BOARD} \
|
||||
CCACHE=$${CCACHE} \
|
||||
CCACHE_DIR=$${CCACHE_DIR} \
|
||||
CCACHE_BASEDIR=$${CCACHE_BASEDIR} \
|
||||
RIOTBASE=$${RIOTBASE} \
|
||||
RIOTBOARD=$${RIOTBOARD} \
|
||||
RIOTCPU=$${RIOTCPU} \
|
||||
BINDIRBASE=$${BINDIRBASE} \
|
||||
RIOTNOLINK=$${RIOTNOLINK} \
|
||||
RIOT_VERSION=$${RIOT_VERSION} \
|
||||
$(MAKE) clean-intermediates 2>&1 >/dev/null || true; \
|
||||
RIOTNOLINK=$$(echo $(BOARD_INSUFFICIENT_RAM) | grep $${BOARD} 2>&1 >/dev/null && echo 1); \
|
||||
${COLOR_ECHO} -n "Building for $${BOARD} "; \
|
||||
[ -n "$${RIOTNOLINK}" ] && ${COLOR_ECHO} -n "(no linking) "; \
|
||||
for NTH_TRY in 1 2 3; do \
|
||||
${COLOR_ECHO} -n ".. "; \
|
||||
LOG=$$(env -i \
|
||||
HOME=$${HOME} \
|
||||
PATH=$${PATH} \
|
||||
BOARD=$${BOARD} \
|
||||
CCACHE=$${CCACHE} \
|
||||
CCACHE_DIR=$${CCACHE_DIR} \
|
||||
CCACHE_BASEDIR=$${CCACHE_BASEDIR} \
|
||||
RIOTBASE=$${RIOTBASE} \
|
||||
RIOTBOARD=$${RIOTBOARD} \
|
||||
RIOTCPU=$${RIOTCPU} \
|
||||
BINDIRBASE=$${BINDIRBASE} \
|
||||
RIOTNOLINK=$${RIOTNOLINK} \
|
||||
RIOT_VERSION=$${RIOT_VERSION} \
|
||||
$(MAKE) -j$(NPROC) 2>&1) ; \
|
||||
if [ "$${?}" = "0" ]; then \
|
||||
${COLOR_ECHO} "${COLOR_GREEN}success${COLOR_RESET}"; \
|
||||
if [ -n "$${BUILDTEST_VERBOSE}" ]; then \
|
||||
echo "$${LOG}" | tail -n +2 | head -n -2 | grep -v -E '^Building application|^\"make|^patching' | awk 'NF'; \
|
||||
fi; \
|
||||
elif [ -n "$${RIOT_DO_RETRY}" ] && [ "$${APP_RETRY}" -lt "3" ] && [ $${NTH_TRY} != 3 ]; then \
|
||||
${COLOR_ECHO} -n "${COLOR_PURPLE}retrying${COLOR_RESET} "; \
|
||||
continue; \
|
||||
else \
|
||||
${COLOR_ECHO} "${COLOR_RED}failed${COLOR_RESET}"; \
|
||||
if [ -n "$${BUILDTEST_VERBOSE}" ]; then \
|
||||
echo "$${LOG}" | grep -v -E '^\"make'; \
|
||||
fi; \
|
||||
APP_RETRY=`expr $${APP_RETRY} + 1`; \
|
||||
BUILDTESTOK=false; \
|
||||
fi; \
|
||||
break; \
|
||||
done; \
|
||||
env -i \
|
||||
HOME=$${HOME} \
|
||||
PATH=$${PATH} \
|
||||
BOARD=$${BOARD} \
|
||||
CCACHE=$${CCACHE} \
|
||||
CCACHE_DIR=$${CCACHE_DIR} \
|
||||
CCACHE_BASEDIR=$${CCACHE_BASEDIR} \
|
||||
RIOTBASE=$${RIOTBASE} \
|
||||
RIOTBOARD=$${RIOTBOARD} \
|
||||
RIOTCPU=$${RIOTCPU} \
|
||||
BINDIRBASE=$${BINDIRBASE} \
|
||||
RIOTNOLINK=$${RIOTNOLINK} \
|
||||
RIOT_VERSION=$${RIOT_VERSION} \
|
||||
$(MAKE) clean-intermediates 2>&1 >/dev/null || true; \
|
||||
done; \
|
||||
$${BUILDTESTOK}
|
||||
endif # BUILD_IN_DOCKER
|
||||
|
@ -42,6 +42,7 @@ export DOCKER_ENV_VARS = \
|
||||
SIZE \
|
||||
UNDEF \
|
||||
BUILDTEST_MCU_GROUP \
|
||||
BUILDTEST_VERBOSE \
|
||||
#
|
||||
|
||||
# Find which variables were set using the command line or the environment and
|
||||
|
251
dist/tools/compile_test/compile_test.py
vendored
251
dist/tools/compile_test/compile_test.py
vendored
@ -2,6 +2,7 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
# Copyright (C) 2014 René Kijewski <rene.kijewski@fu-berlin.de>
|
||||
# Copyright (C) 2015 Philipp Rosenkranz <philipp.rosenkranz@fu-berlin.de>
|
||||
#
|
||||
# This library is free software; you can redistribute it and/or
|
||||
# modify it under the terms of the GNU Lesser General Public
|
||||
@ -24,23 +25,17 @@ from itertools import groupby
|
||||
from os import devnull, environ, listdir
|
||||
from os.path import abspath, dirname, isfile, join
|
||||
from subprocess import CalledProcessError, check_call, check_output, PIPE, Popen
|
||||
from sys import exit, stdout, argv
|
||||
from sys import exit, stdout, argv, exc_info
|
||||
from StringIO import StringIO
|
||||
from itertools import tee
|
||||
|
||||
riotbase = environ.get('RIOTBASE') or abspath(join(dirname(abspath(__file__)), '../' * 3))
|
||||
|
||||
if len(argv) > 1:
|
||||
base_branch = argv[1]
|
||||
diff_files = check_output(('git', 'diff', '--name-only', base_branch, 'HEAD'))
|
||||
diff_files = set(diff_files.split())
|
||||
else:
|
||||
base_branch = ''
|
||||
|
||||
null = open(devnull, 'w', 0)
|
||||
|
||||
success = []
|
||||
failed = []
|
||||
skipped = []
|
||||
exceptions = []
|
||||
class Termcolor:
|
||||
red = '\033[1;31m'
|
||||
green = '\033[1;32m'
|
||||
yellow = '\033[1;33m'
|
||||
blue = '\033[1;34m'
|
||||
purple = '\033[1;35m'
|
||||
end = '\033[0m'
|
||||
|
||||
def is_tracked(application_folder):
|
||||
if not isfile(join(application_folder, 'Makefile')):
|
||||
@ -53,33 +48,47 @@ def is_tracked(application_folder):
|
||||
else:
|
||||
return True
|
||||
|
||||
def get_lines(readline, prefix):
|
||||
def get_results_and_output_from(fd):
|
||||
results_prefix = 'Building for '
|
||||
output_prefix = 'Building application '
|
||||
prev_results = False
|
||||
result = ['']
|
||||
output = StringIO()
|
||||
while 1:
|
||||
result = readline()
|
||||
if not result:
|
||||
line = fd.readline()
|
||||
if not line:
|
||||
if prev_results:
|
||||
yield (' .. '.join(result[:-1]), result[-1], output)
|
||||
break
|
||||
elif not result.startswith(prefix):
|
||||
continue
|
||||
|
||||
result = result[len(prefix):].rstrip().split(' .. ')[::-1]
|
||||
if (len(result) > 1) and ('success' in result[0] or 'failed' in result[0]):
|
||||
stdout.write('.')
|
||||
stdout.flush()
|
||||
yield (' .. '.join(result[:-1]), result[-1])
|
||||
elif line.startswith(results_prefix):
|
||||
read_more_output = False
|
||||
if prev_results:
|
||||
yield (' .. '.join(result[:-1]), result[-1], output)
|
||||
prev_results = True
|
||||
result = line[len(results_prefix):].rstrip().split(' .. ')[::-1]
|
||||
if (len(result) > 1) and ('success' in result[0] or 'failed' in result[0]):
|
||||
stdout.write('.')
|
||||
stdout.flush()
|
||||
elif line.startswith(output_prefix):
|
||||
output = StringIO()
|
||||
output.write(line)
|
||||
read_more_output = True
|
||||
elif read_more_output:
|
||||
output.write(line)
|
||||
|
||||
def _get_common_user(common):
|
||||
return [f for f in check_output(r'grep -l "{}" cpu/*/Makefile* boards/*/Makefile*'.format(common),
|
||||
shell=True).split() if "common" not in f]
|
||||
shell=True).split() if 'common' not in f]
|
||||
|
||||
def _get_boards_from_files(files):
|
||||
boards = set()
|
||||
if any("boards/" in s for s in files):
|
||||
if any('boards/' in s for s in files):
|
||||
for f in files:
|
||||
if "boards/" not in f:
|
||||
if 'boards/' not in f:
|
||||
continue
|
||||
board = re.sub(r"^boards/([^/]+)/.*$", r"\1", f)
|
||||
board = re.sub(r'^boards/([^/]+)/.*$', r'\1', f)
|
||||
|
||||
if "common" in board:
|
||||
if 'common' in board:
|
||||
boards |= _get_boards_from_files(_get_common_user(board))
|
||||
else:
|
||||
boards |= { board }
|
||||
@ -89,14 +98,14 @@ def _get_boards_from_files(files):
|
||||
def _get_cpus_from_files(files):
|
||||
cpus = set()
|
||||
|
||||
if any("cpu/" in s for s in files):
|
||||
if any('cpu/' in s for s in files):
|
||||
for f in files:
|
||||
if "cpu/" not in f:
|
||||
if 'cpu/' not in f:
|
||||
continue
|
||||
|
||||
cpu = re.sub(r"^cpu/([^/]+)/.*", r"\1", f)
|
||||
cpu = re.sub(r'^cpu/([^/]+)/.*', r'\1', f)
|
||||
|
||||
if "common" in cpu:
|
||||
if 'common' in cpu:
|
||||
cpus |= _get_cpus_from_files(_get_common_user(cpu))
|
||||
else:
|
||||
cpus |= { cpu }
|
||||
@ -108,8 +117,8 @@ def is_updated(application_folder, subprocess_env):
|
||||
if base_branch == '':
|
||||
return True
|
||||
|
||||
if ".travis.yml" in diff_files or \
|
||||
any("dist/" in s for s in diff_files):
|
||||
if '.travis.yml' in diff_files or \
|
||||
any('dist/' in s for s in diff_files):
|
||||
return True
|
||||
|
||||
boards_changes = set()
|
||||
@ -125,7 +134,7 @@ def is_updated(application_folder, subprocess_env):
|
||||
app_files = set()
|
||||
|
||||
for board in boards_changes:
|
||||
env = { "BOARD": board }
|
||||
env = { 'BOARD': board }
|
||||
env.update(subprocess_env)
|
||||
tmp = check_output(('make', 'info-files'), stderr=null,
|
||||
cwd=application_folder, env=env)
|
||||
@ -142,59 +151,123 @@ def is_updated(application_folder, subprocess_env):
|
||||
except CalledProcessError as e:
|
||||
return True
|
||||
|
||||
for folder in ('examples', 'tests'):
|
||||
print('Building all applications in: \033[1;34m{}\033[0m'.format(folder))
|
||||
def build_all():
|
||||
riotbase = environ.get('RIOTBASE') or abspath(join(dirname(abspath(__file__)), '../' * 3))
|
||||
for folder in ('examples', 'tests'):
|
||||
print('Building all applications in: {}'.format(colorize_str(folder, Termcolor.blue)))
|
||||
|
||||
applications = listdir(join(riotbase, folder))
|
||||
applications = filter(lambda app: is_tracked(join(riotbase, folder, app)), applications)
|
||||
applications = sorted(applications)
|
||||
applications = listdir(join(riotbase, folder))
|
||||
applications = filter(lambda app: is_tracked(join(riotbase, folder, app)), applications)
|
||||
applications = sorted(applications)
|
||||
|
||||
subprocess_env = environ.copy()
|
||||
subprocess_env['RIOT_DO_RETRY'] = '1'
|
||||
subprocess_env = environ.copy()
|
||||
subprocess_env['RIOT_DO_RETRY'] = '1'
|
||||
subprocess_env['BUILDTEST_VERBOSE'] = '1'
|
||||
|
||||
for nth, application in enumerate(applications, 1):
|
||||
stdout.write('\tBuilding application: \033[1;34m{}\033[0m ({}/{}) '.format(application, nth, len(applications)))
|
||||
stdout.flush()
|
||||
try:
|
||||
if not is_updated(join(riotbase, folder, application), subprocess_env):
|
||||
print("\033[1;33m(skipped)\033[0m")
|
||||
skipped.append(application)
|
||||
continue
|
||||
subprocess = Popen(('make', 'buildtest'),
|
||||
bufsize=1, stdin=null, stdout=PIPE, stderr=null,
|
||||
cwd=join(riotbase, folder, application),
|
||||
env=subprocess_env)
|
||||
|
||||
lines = get_lines(subprocess.stdout.readline, 'Building for ')
|
||||
lines = groupby(sorted(lines), lambda (outcome, board): outcome)
|
||||
|
||||
print()
|
||||
for group, results in lines:
|
||||
print('\t\t{}: {}'.format(group, ', '.join(sorted(board for outcome, board in results))))
|
||||
|
||||
returncode = subprocess.wait()
|
||||
if returncode == 0:
|
||||
success.append(application)
|
||||
else:
|
||||
failed.append(application)
|
||||
except Exception, e:
|
||||
print('\n\t\tException: {}'.format(e))
|
||||
exceptions.append(application)
|
||||
finally:
|
||||
for nth, application in enumerate(applications, 1):
|
||||
stdout.write('\tBuilding application: {} ({}/{}) '.format(colorize_str(application, Termcolor.blue), nth, len(applications)))
|
||||
stdout.flush()
|
||||
try:
|
||||
subprocess.kill()
|
||||
except:
|
||||
pass
|
||||
if not is_updated(join(riotbase, folder, application), subprocess_env):
|
||||
print(colorize_str('(skipped)', Termcolor.yellow))
|
||||
skipped.append(application)
|
||||
continue
|
||||
subprocess = Popen(('make', 'buildtest'),
|
||||
bufsize=1, stdin=null, stdout=PIPE, stderr=null,
|
||||
cwd=join(riotbase, folder, application),
|
||||
env=subprocess_env)
|
||||
|
||||
print('Outcome:')
|
||||
for color, group in (('3', 'skipped'), ('2', 'success'), ('1', 'failed'), ('4', 'exceptions')):
|
||||
applications = locals()[group]
|
||||
if applications:
|
||||
print('\t\033[1;3{}m{}\033[0m: {}'.format(color, group, ', '.join(applications)))
|
||||
results, results_with_output = tee(get_results_and_output_from(subprocess.stdout))
|
||||
results = groupby(sorted(results), lambda (outcome, board, output): outcome)
|
||||
results_with_output = filter(lambda (outcome, board, output): output.getvalue(), results_with_output)
|
||||
failed_with_output = filter(lambda (outcome, board, output): 'failed' in outcome, results_with_output)
|
||||
success_with_output = filter(lambda (outcome, board, output): 'success' in outcome, results_with_output)
|
||||
print()
|
||||
for group, results in results:
|
||||
print('\t\t{}: {}'.format(group, ', '.join(sorted(board for outcome, board, output in results))))
|
||||
returncode = subprocess.wait()
|
||||
if success_with_output:
|
||||
warnings.append((application, success_with_output))
|
||||
if returncode == 0:
|
||||
success.append(application)
|
||||
else:
|
||||
failed.append(application)
|
||||
errors.append((application, failed_with_output))
|
||||
except Exception, e:
|
||||
print('\n\t\tException: {}'.format(e))
|
||||
exceptions.append(application)
|
||||
finally:
|
||||
try:
|
||||
subprocess.kill()
|
||||
except:
|
||||
pass
|
||||
|
||||
if exceptions:
|
||||
exit(2)
|
||||
elif failed:
|
||||
exit(1)
|
||||
else:
|
||||
exit(0)
|
||||
def colorize_str(string, color):
|
||||
return '%s%s%s' % (color, string, Termcolor.end)
|
||||
|
||||
def print_output_for(buf, name, color):
|
||||
if buf:
|
||||
print('%s:' % name)
|
||||
for application, details in buf:
|
||||
for outcome, board, output in details:
|
||||
print()
|
||||
print(colorize_str('%s:%s:' % (application, board), color))
|
||||
print('%s' % output.getvalue())
|
||||
|
||||
def print_outcome(outputListDescription):
|
||||
print()
|
||||
print('Outcome:')
|
||||
for color, group, name in outputListDescription:
|
||||
applications = group
|
||||
if applications:
|
||||
print('\t{}{}{}: {}'.format(color, name, Termcolor.end, ', '.join(applications)))
|
||||
|
||||
def print_num_of_errors_and_warnings():
|
||||
stdout.write('Errors: ')
|
||||
if errors:
|
||||
num_of_errors = sum(map(lambda x: len(x[1]), errors))
|
||||
stdout.write('%s' % colorize_str(str(num_of_errors), Termcolor.red))
|
||||
else:
|
||||
stdout.write('0')
|
||||
stdout.write(' Warnings: ')
|
||||
if warnings:
|
||||
num_of_warnings = sum(map(lambda x: len(x[1]), warnings))
|
||||
stdout.write('%s' % colorize_str(str(num_of_warnings), Termcolor.yellow))
|
||||
else:
|
||||
stdout.write('0')
|
||||
stdout.write('\n')
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
success = []
|
||||
failed = []
|
||||
skipped = []
|
||||
exceptions = []
|
||||
warnings = []
|
||||
errors = []
|
||||
null = open(devnull, 'w', 0)
|
||||
|
||||
if len(argv) > 1:
|
||||
base_branch = argv[1]
|
||||
diff_files = check_output(('git', 'diff', '--name-only', base_branch, 'HEAD'))
|
||||
diff_files = set(diff_files.split())
|
||||
else:
|
||||
base_branch = ''
|
||||
|
||||
build_all()
|
||||
|
||||
print_output_for(warnings, 'Warnings', Termcolor.yellow)
|
||||
print_output_for(errors, 'Errors', Termcolor.red)
|
||||
|
||||
outputListDescription = [(Termcolor.yellow, skipped, 'skipped'), (Termcolor.green, success, 'success'),
|
||||
(Termcolor.red, failed, 'failed'), (Termcolor.blue, exceptions, 'exceptions')]
|
||||
print_outcome(outputListDescription)
|
||||
|
||||
print_num_of_errors_and_warnings()
|
||||
|
||||
if exceptions:
|
||||
exit(2)
|
||||
elif failed:
|
||||
exit(1)
|
||||
else:
|
||||
exit(0)
|
||||
|
Loading…
Reference in New Issue
Block a user