#!/usr/bin/env python # This file is part of rippled: https://github.com/ripple/rippled # Copyright (c) 2012 - 2017 Ripple Labs Inc. # # Permission to use, copy, modify, and/or distribute this software for any # purpose with or without fee is hereby granted, provided that the above # copyright notice and this permission notice appear in all copies. # # THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES # WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF # MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR # ANY SPECIAL , DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES # WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN # ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF # OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. """ Invocation: ./Builds/Test.py - builds and tests all configurations The build must succeed without shell aliases for this to work. To pass flags to scons, put them at the very end of the command line, after the -- flag - like this: ./Builds/Test.py -- -j4 # Pass -j4 to scons. To build with CMake, use the --cmake flag, or any of the specific configuration flags ./Builds/Test.py --cmake -- -j4 # Pass -j4 to cmake --build Common problems: 1) Boost not found. Solution: export BOOST_ROOT=[path to boost folder] 2) OpenSSL not found. Solution: export OPENSSL_ROOT=[path to OpenSSL folder] 3) scons is an alias. Solution: Create a script named "scons" somewhere in your $PATH (eg. ~/bin/scons will often work). #!/bin/sh python /C/Python27/Scripts/scons.py "${@}" """ from __future__ import absolute_import, division, print_function, unicode_literals import argparse import itertools import os import platform import re import shutil import sys import subprocess def powerset(iterable): """powerset([1,2,3]) --> () (1,) (2,) (3,) (1,2) (1,3) (2,3) (1,2,3)""" s = list(iterable) return itertools.chain.from_iterable(itertools.combinations(s, r) for r in range(len(s) + 1)) IS_WINDOWS = platform.system().lower() == 'windows' IS_OS_X = platform.system().lower() == 'darwin' # CMake if IS_WINDOWS: CMAKE_UNITY_CONFIGS = ['Debug', 'Release'] CMAKE_NONUNITY_CONFIGS = ['DebugClassic', 'ReleaseClassic'] else: CMAKE_UNITY_CONFIGS = [] CMAKE_NONUNITY_CONFIGS = [] CMAKE_UNITY_COMBOS = { '' : [['rippled', 'rippled_classic'], CMAKE_UNITY_CONFIGS], '.nounity' : [['rippled', 'rippled_unity'], CMAKE_NONUNITY_CONFIGS] } if IS_WINDOWS: CMAKE_DIR_TARGETS = { ('msvc' + unity,) : targets for unity, targets in CMAKE_UNITY_COMBOS.items() } elif IS_OS_X: CMAKE_DIR_TARGETS = { (build + unity,) : targets for build in ['debug', 'release'] for unity, targets in CMAKE_UNITY_COMBOS.items() } else: CMAKE_DIR_TARGETS = { (cc + "." + build + unity,) : targets for cc in ['gcc', 'clang'] for build in ['debug', 'release', 'coverage', 'profile'] for unity, targets in CMAKE_UNITY_COMBOS.items() } # list of tuples of all possible options if IS_WINDOWS or IS_OS_X: CMAKE_ALL_GENERATE_OPTIONS = [tuple(x) for x in powerset(['-GNinja', '-Dassert=true'])] else: CMAKE_ALL_GENERATE_OPTIONS = list(set( [tuple(x) for x in powerset(['-GNinja', '-Dstatic=true', '-Dassert=true', '-Dsan=address'])] + [tuple(x) for x in powerset(['-GNinja', '-Dstatic=true', '-Dassert=true', '-Dsan=thread'])])) # Scons if IS_WINDOWS or IS_OS_X: ALL_TARGETS = [('debug',), ('release',)] else: ALL_TARGETS = [(cc + "." + target,) for cc in ['gcc', 'clang'] for target in ['debug', 'release', 'coverage', 'profile', 'debug.nounity', 'release.nounity', 'coverage.nounity', 'profile.nounity']] # list of tuples of all possible options if IS_WINDOWS or IS_OS_X: ALL_OPTIONS = [tuple(x) for x in powerset(['--ninja', '--assert'])] else: ALL_OPTIONS = list(set( [tuple(x) for x in powerset(['--ninja', '--static', '--assert', '--sanitize=address'])] + [tuple(x) for x in powerset(['--ninja', '--static', '--assert', '--sanitize=thread'])])) # list of tuples of all possible options + all possible targets ALL_BUILDS = [options + target for target in ALL_TARGETS for options in ALL_OPTIONS] parser = argparse.ArgumentParser( description='Test.py - run ripple tests' ) parser.add_argument( '--all', '-a', action='store_true', help='Build all configurations.', ) parser.add_argument( '--keep_going', '-k', action='store_true', help='Keep going after one configuration has failed.', ) parser.add_argument( '--silent', '-s', action='store_true', help='Silence all messages except errors', ) parser.add_argument( '--verbose', '-v', action='store_true', help=('Report more information about which commands are executed and the ' 'results.'), ) parser.add_argument( '--test', '-t', default='', help='Add a prefix for unit tests', ) parser.add_argument( '--testjobs', default='0', type=int, help='Run tests in parallel' ) parser.add_argument( '--clean', '-c', action='store_true', help='delete all build artifacts after testing', ) parser.add_argument( '--quiet', '-q', action='store_true', help='Reduce output where possible (unit tests)', ) # Scons and CMake parameters are too different to run # both side-by-side pgroup = parser.add_mutually_exclusive_group() pgroup.add_argument( '--cmake', action='store_true', help='Build using CMake.', ) pgroup.add_argument( '--scons', action='store_true', help='Build using Scons. Default behavior.') parser.add_argument( '--dir', '-d', default=(), nargs='*', help='Specify one or more CMake dir names. Implies --cmake. ' 'Will also be used as -Dtarget= running cmake.' ) parser.add_argument( '--target', default=(), nargs='*', help='Specify one or more CMake build targets. Implies --cmake. ' 'Will be used as --target running cmake --build.' ) parser.add_argument( '--config', default=(), nargs='*', help='Specify one or more CMake build configs. Implies --cmake. ' 'Will be used as --config running cmake --build.' ) parser.add_argument( '--generator_option', action='append', help='Specify a CMake generator option. Repeat for multiple options. ' 'Implies --cmake. Will be passed to the cmake generator. ' 'Due to limits of the argument parser, arguments starting with \'-\' ' 'must be attached to this option. e.g. --generator_option=-GNinja.') parser.add_argument( '--build_option', action='append', help='Specify a build option. Repeat for multiple options. Implies --cmake. ' 'Will be passed to the build tool via cmake --build. ' 'Due to limits of the argument parser, arguments starting with \'-\' ' 'must be attached to this option. e.g. --build_option=-j8.') parser.add_argument( 'extra_args', default=(), nargs='*', help='Extra arguments are passed through to the tools' ) ARGS = parser.parse_args() def decodeString(line): # Python 2 vs. Python 3 if isinstance(line, str): return line else: return line.decode() def shell(cmd, args=(), silent=False): """"Execute a shell command and return the output.""" silent = ARGS.silent or silent verbose = not silent and ARGS.verbose if verbose: print('$' + cmd, *args) command = (cmd,) + args # shell is needed in Windows to find scons in the path process = subprocess.Popen( command, stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.STDOUT, shell=IS_WINDOWS) lines = [] count = 0 # readline returns '' at EOF for line in iter(process.stdout.readline, ''): if process.poll() is None: decoded = decodeString(line) lines.append(decoded) if verbose: print(decoded, end='') elif not silent: count += 1 if count >= 80: print() count = 0 else: print('.', end='') else: break if not verbose and count: print() process.wait() return process.returncode, lines def run_tests(args): failed = [] if IS_WINDOWS: binary_re = re.compile(r'build\\([^\\]+)\\rippled.exe') else: binary_re = re.compile(r'build/([^/]+)/rippled') _, lines = shell('scons', ('-n', '--tree=derived',) + args, silent=True) for line in lines: match = binary_re.search(line) if match: executable, target = match.group(0, 1) print('Unit tests for', target) testflag = '--unittest' quiet = '' if ARGS.test: testflag += ('=' + ARGS.test) if ARGS.quiet: quiet = '-q' if ARGS.testjobs: testjobs = ('--unittest-jobs=' + str(ARGS.testjobs)) resultcode, lines = shell(executable, (testflag, quiet, testjobs,)) if resultcode: if not ARGS.verbose: print('ERROR:', *lines, sep='') failed.append([target, 'unittest']) if not ARGS.keep_going: break return failed def run_build(args=None): print('Building:', *args or ('(default)',)) resultcode, lines = shell('scons', args) if resultcode: print('Build FAILED:') if not ARGS.verbose: print(*lines, sep='') sys.exit(1) if '--ninja' in args: resultcode, lines = shell('ninja') if resultcode: print('Ninja build FAILED:') if not ARGS.verbose: print(*lines, sep='') sys.exit(1) def get_cmake_dir(cmake_dir): return os.path.join('build' , 'cmake' , cmake_dir) def run_cmake(directory, cmake_dir, args): print('Generating build in', directory, 'with', *args or ('default options',)) old_dir = os.getcwd() if not os.path.exists(directory): os.makedirs(directory) os.chdir(directory) if IS_WINDOWS and not any(arg.startswith("-G") for arg in args) and not os.path.exists("CMakeCache.txt"): if '--ninja' in args: args += ( '-GNinja', ) else: args += ( '-GVisual Studio 14 2015 Win64', ) args += ( '-Dtarget=' + cmake_dir, os.path.join('..', '..', '..'), ) resultcode, lines = shell('cmake', args) if resultcode: print('Generating FAILED:') if not ARGS.verbose: print(*lines, sep='') sys.exit(1) os.chdir(old_dir) def run_cmake_build(directory, target, config, args): print('Building', target, config, 'in', directory, 'with', *args or ('default options',)) build_args=('--build', directory) if target: build_args += ('--target', target) if config: build_args += ('--config', config) if args: build_args += ('--',) build_args += tuple(args) resultcode, lines = shell('cmake', build_args) if resultcode: print('Build FAILED:') if not ARGS.verbose: print(*lines, sep='') sys.exit(1) def run_cmake_tests(directory, target, config): failed = [] if IS_WINDOWS: target += '.exe' executable = os.path.join(directory, config if config else 'Debug', target) if(not os.path.exists(executable)): executable = os.path.join(directory, target) print('Unit tests for', executable) testflag = '--unittest' quiet = '' testjobs = '' if ARGS.test: testflag += ('=' + ARGS.test) if ARGS.quiet: quiet = '-q' if ARGS.testjobs: testjobs = ('--unittest-jobs=' + str(ARGS.testjobs)) resultcode, lines = shell(executable, (testflag, quiet, testjobs,)) if resultcode: if not ARGS.verbose: print('ERROR:', *lines, sep='') failed.append([target, 'unittest']) return failed def main(): all_failed = [] if ARGS.dir or ARGS.target or ARGS.config or ARGS.build_option or ARGS.generator_option: ARGS.cmake=True if not ARGS.cmake: if ARGS.all: to_build = ALL_BUILDS else: to_build = [tuple(ARGS.extra_args)] for build in to_build: args = () # additional arguments come first for arg in list(ARGS.extra_args): if arg not in build: args += (arg,) args += build run_build(args) failed = run_tests(args) if failed: print('FAILED:', *(':'.join(f) for f in failed)) if not ARGS.keep_going: sys.exit(1) else: all_failed.extend([','.join(build), ':'.join(f)] for f in failed) else: print('Success') if ARGS.clean: shutil.rmtree('build') if '--ninja' in args: os.remove('build.ninja') os.remove('.ninja_deps') os.remove('.ninja_log') else: if ARGS.all: build_dir_targets = CMAKE_DIR_TARGETS generator_options = CMAKE_ALL_GENERATE_OPTIONS else: build_dir_targets = { tuple(ARGS.dir) : [ARGS.target, ARGS.config] } if ARGS.generator_option: generator_options = [tuple(ARGS.generator_option)] else: generator_options = [tuple()] if not build_dir_targets: # Let CMake choose the build tool. build_dir_targets = { () : [] } if ARGS.build_option: ARGS.build_option = ARGS.build_option + list(ARGS.extra_args) else: ARGS.build_option = list(ARGS.extra_args) for args in generator_options: for build_dirs, (build_targets, build_configs) in build_dir_targets.items(): if not build_dirs: build_dirs = ('default',) if not build_targets: build_targets = ('rippled',) if not build_configs: build_configs = ('',) for cmake_dir in build_dirs: cmake_full_dir = get_cmake_dir(cmake_dir) run_cmake(cmake_full_dir, cmake_dir, args) for target in build_targets: for config in build_configs: run_cmake_build(cmake_full_dir, target, config, ARGS.build_option) failed = run_cmake_tests(cmake_full_dir, target, config) if failed: print('FAILED:', *(':'.join(f) for f in failed)) if not ARGS.keep_going: sys.exit(1) else: all_failed.extend([decodeString(cmake_dir + "." + target + "." + config), ':'.join(f)] for f in failed) else: print('Success') if ARGS.clean: shutil.rmtree(cmake_full_dir) if all_failed: if len(all_failed) > 1: print() print('FAILED:', *(':'.join(f) for f in all_failed)) sys.exit(1) if __name__ == '__main__': main() sys.exit(0)