| #!/usr/bin/env python3 |
| # |
| # [VPYTHON:BEGIN] |
| # python_version: "3.8" |
| # [VPYTHON:END] |
| # |
| # Copyright (C) 2021 The Android Open Source Project |
| # |
| # 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 sys, os, argparse, subprocess, shlex, re, concurrent.futures, multiprocessing |
| |
| def parse_args(): |
| parser = argparse.ArgumentParser(description="Run libcore tests using the vogar testing tool.") |
| parser.add_argument('--mode', choices=['device', 'host', 'jvm'], required=True, |
| help='Specify where tests should be run.') |
| parser.add_argument('--variant', choices=['X32', 'X64'], |
| help='Which dalvikvm variant to execute with.') |
| parser.add_argument('-j', '--jobs', type=int, |
| help='Number of tests to run simultaneously.') |
| parser.add_argument('--timeout', type=int, |
| help='How long to run the test before aborting (seconds).') |
| parser.add_argument('--debug', action='store_true', |
| help='Use debug version of ART (device|host only).') |
| parser.add_argument('--dry-run', action='store_true', |
| help='Print vogar command-line, but do not run.') |
| parser.add_argument('--no-getrandom', action='store_false', dest='getrandom', |
| help='Ignore failures from getrandom() (for kernel < 3.17).') |
| parser.add_argument('--no-jit', action='store_false', dest='jit', |
| help='Disable JIT (device|host only).') |
| parser.add_argument('--gcstress', action='store_true', |
| help='Enable GC stress configuration (device|host only).') |
| parser.add_argument('tests', nargs="*", |
| help='Name(s) of the test(s) to run') |
| parser.add_argument('--verbose', action='store_true', help='Print verbose output from vogar.') |
| return parser.parse_args() |
| |
| ART_TEST_ANDROID_ROOT = os.environ.get("ART_TEST_ANDROID_ROOT", "/system") |
| ART_TEST_CHROOT = os.environ.get("ART_TEST_CHROOT") |
| ANDROID_PRODUCT_OUT = os.environ.get("ANDROID_PRODUCT_OUT") |
| |
| LIBCORE_TEST_NAMES = [ |
| ### luni tests. ### |
| # Naive critical path optimization: Run the longest tests first. |
| "org.apache.harmony.tests.java.util", # 90min under gcstress |
| "libcore.java.lang", # 90min under gcstress |
| "jsr166", # 60min under gcstress |
| "libcore.java.util", # 60min under gcstress |
| "libcore.java.math", # 50min under gcstress |
| "org.apache.harmony.crypto", # 30min under gcstress |
| "org.apache.harmony.tests.java.io", # 30min under gcstress |
| "org.apache.harmony.tests.java.text", # 30min under gcstress |
| # Split highmemorytest to individual classes since it is too big. |
| "libcore.highmemorytest.java.text.DateFormatTest", |
| "libcore.highmemorytest.java.text.DecimalFormatTest", |
| "libcore.highmemorytest.java.text.SimpleDateFormatTest", |
| "libcore.highmemorytest.java.time.format.DateTimeFormatterTest", |
| "libcore.highmemorytest.java.util.CalendarTest", |
| "libcore.highmemorytest.java.util.CurrencyTest", |
| "libcore.highmemorytest.libcore.icu.SimpleDateFormatDataTest", |
| # All other luni tests in alphabetical order. |
| "libcore.android.system", |
| "libcore.build", |
| "libcore.dalvik.system", |
| "libcore.java.awt", |
| "libcore.java.text", |
| "libcore.javax.crypto", |
| "libcore.javax.net", |
| "libcore.javax.security", |
| "libcore.javax.sql", |
| "libcore.javax.xml", |
| "libcore.libcore.icu", |
| "libcore.libcore.internal", |
| "libcore.libcore.io", |
| "libcore.libcore.net", |
| "libcore.libcore.reflect", |
| "libcore.libcore.util", |
| "libcore.sun.invoke", |
| "libcore.sun.misc", |
| "libcore.sun.net", |
| "libcore.sun.security", |
| "libcore.sun.util", |
| "libcore.xml", |
| "org.apache.harmony.annotation", |
| "org.apache.harmony.luni.tests.internal.net.www.protocol.http.HttpURLConnection", |
| "org.apache.harmony.luni.tests.internal.net.www.protocol.https.HttpsURLConnection", |
| "org.apache.harmony.luni.tests.java.io", |
| "org.apache.harmony.luni.tests.java.net", |
| "org.apache.harmony.nio", |
| "org.apache.harmony.regex", |
| "org.apache.harmony.testframework", |
| "org.apache.harmony.tests.java.lang", |
| "org.apache.harmony.tests.java.math", |
| "org.apache.harmony.tests.javax.security", |
| "tests.java.lang.String", |
| ### OpenJDK upstream tests (ojluni). ### |
| # "test.java.awt", |
| "test.java.awt", |
| # test.java.io |
| "test.java.io.ByteArrayInputStream", |
| "test.java.io.ByteArrayOutputStream", |
| "test.java.io.FileReader", |
| "test.java.io.FileWriter", |
| "test.java.io.InputStream", |
| "test.java.io.OutputStream", |
| "test.java.io.PrintStream", |
| "test.java.io.PrintWriter", |
| "test.java.io.Reader", |
| "test.java.io.Writer", |
| # test.java.lang |
| "test.java.lang.Boolean", |
| "test.java.lang.ClassLoader", |
| "test.java.lang.Double", |
| "test.java.lang.Float", |
| "test.java.lang.Integer", |
| "test.java.lang.Long", |
| # Sharded test.java.lang.StrictMath |
| "test.java.lang.StrictMath.CubeRootTests", |
| # TODO: disable the test until b/248208762 is fixed. |
| # "test.java.lang.StrictMath.ExactArithTests", |
| "test.java.lang.StrictMath.Expm1Tests", |
| "test.java.lang.StrictMath.ExpTests", |
| "test.java.lang.StrictMath.HyperbolicTests", |
| "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard1", |
| "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard2", |
| "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard3", |
| "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard4", |
| "test.java.lang.StrictMath.HypotTests#testHypot", |
| "test.java.lang.StrictMath.Log1pTests", |
| "test.java.lang.StrictMath.Log10Tests", |
| "test.java.lang.StrictMath.MultiplicationTests", |
| "test.java.lang.StrictMath.PowTests", |
| "test.java.lang.String", |
| "test.java.lang.Thread", |
| # test.java.lang.invoke |
| "test.java.lang.invoke", |
| # test.java.lang.ref |
| "test.java.lang.ref.SoftReference", |
| "test.java.lang.ref.BasicTest", |
| "test.java.lang.ref.EnqueueNullRefTest", |
| "test.java.lang.ref.EnqueuePollRaceTest", |
| "test.java.lang.ref.ReferenceCloneTest", |
| "test.java.lang.ref.ReferenceEnqueuePendingTest", |
| # test.java.math |
| "test.java.math.BigDecimal", |
| # Sharded test.java.math.BigInteger |
| "test.java.math.BigInteger#testArithmetic", |
| "test.java.math.BigInteger#testBitCount", |
| "test.java.math.BigInteger#testBitLength", |
| "test.java.math.BigInteger#testbitOps", |
| "test.java.math.BigInteger#testBitwise", |
| "test.java.math.BigInteger#testByteArrayConv", |
| "test.java.math.BigInteger#testConstructor", |
| "test.java.math.BigInteger#testDivideAndReminder", |
| "test.java.math.BigInteger#testDivideLarge", |
| "test.java.math.BigInteger#testModExp", |
| "test.java.math.BigInteger#testMultiplyLarge", |
| "test.java.math.BigInteger#testNextProbablePrime", |
| "test.java.math.BigInteger#testPow", |
| "test.java.math.BigInteger#testSerialize", |
| "test.java.math.BigInteger#testShift", |
| "test.java.math.BigInteger#testSquare", |
| "test.java.math.BigInteger#testSquareLarge", |
| "test.java.math.BigInteger#testSquareRootAndReminder", |
| "test.java.math.BigInteger#testStringConv_generic", |
| "test.java.math.RoundingMode", |
| # test.java.net |
| "test.java.net.DatagramSocket", |
| "test.java.net.Socket", |
| "test.java.net.SocketOptions", |
| "test.java.net.URLDecoder", |
| "test.java.net.URLEncoder", |
| # test.java.nio |
| "test.java.nio.channels.Channels", |
| "test.java.nio.channels.SelectionKey", |
| "test.java.nio.channels.Selector", |
| "test.java.nio.file", |
| # test.java.security |
| "test.java.security.cert", |
| # Sharded test.java.security.KeyAgreement |
| "test.java.security.KeyAgreement.KeyAgreementTest", |
| "test.java.security.KeyAgreement.KeySizeTest#testECDHKeySize", |
| "test.java.security.KeyAgreement.KeySpecTest", |
| "test.java.security.KeyAgreement.MultiThreadTest", |
| "test.java.security.KeyAgreement.NegativeTest", |
| "test.java.security.KeyStore", |
| "test.java.security.Provider", |
| # test.java.time |
| "test.java.time", |
| # test.java.util |
| "test.java.util.Arrays", |
| "test.java.util.Collection", |
| "test.java.util.Collections", |
| "test.java.util.Date", |
| "test.java.util.EnumMap", |
| "test.java.util.EnumSet", |
| "test.java.util.GregorianCalendar", |
| "test.java.util.LinkedHashMap", |
| "test.java.util.LinkedHashSet", |
| "test.java.util.List", |
| "test.java.util.Map", |
| "test.java.util.Optional", |
| "test.java.util.TestFormatter", |
| "test.java.util.TimeZone", |
| # test.java.util.concurrent |
| "test.java.util.concurrent", |
| # test.java.util.function |
| "test.java.util.function", |
| # test.java.util.stream |
| "test.java.util.stream", |
| # test.java.util.zip |
| "test.java.util.zip.ZipFile", |
| # tck.java.time |
| "tck.java.time", |
| ] |
| # "org.apache.harmony.security", # We don't have rights to revert changes in case of failures. |
| |
| # Note: This must start with the CORE_IMG_JARS in Android.common_path.mk |
| # because that's what we use for compiling the boot.art image. |
| # It may contain additional modules from TEST_CORE_JARS. |
| BOOT_CLASSPATH = [ |
| "/apex/com.android.art/javalib/core-oj.jar", |
| "/apex/com.android.art/javalib/core-libart.jar", |
| "/apex/com.android.art/javalib/okhttp.jar", |
| "/apex/com.android.art/javalib/bouncycastle.jar", |
| "/apex/com.android.art/javalib/apache-xml.jar", |
| "/apex/com.android.i18n/javalib/core-icu4j.jar", |
| "/apex/com.android.conscrypt/javalib/conscrypt.jar", |
| ] |
| |
| CLASSPATH = ["core-tests", "core-ojtests", "jsr166-tests", "mockito-target"] |
| |
| SLOW_OJLUNI_TESTS = { |
| "test.java.awt", |
| "test.java.lang.String", |
| "test.java.lang.invoke", |
| "test.java.nio.channels.Selector", |
| "test.java.time", |
| "test.java.util.Arrays", |
| "test.java.util.Map", |
| "test.java.util.concurrent", |
| "test.java.util.stream", |
| "test.java.util.zip.ZipFile", |
| "tck.java.time", |
| } |
| |
| # Disabled to unblock art-buildbot |
| # These tests fail with "java.io.IOException: Stream closed", tracked in |
| # http://b/235566533 and http://b/208639267 |
| DISABLED_GCSTRESS_DEBUG_TESTS = { |
| "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard1", |
| "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard2", |
| "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard3", |
| "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard4", |
| "test.java.math.BigDecimal", |
| "test.java.math.BigInteger#testConstructor", |
| } |
| |
| DISABLED_FUGU_TESTS = { |
| "org.apache.harmony.luni.tests.internal.net.www.protocol.http.HttpURLConnection", |
| "org.apache.harmony.luni.tests.internal.net.www.protocol.https.HttpsURLConnection", |
| "test.java.awt", |
| "test.java.io.ByteArrayInputStream", |
| "test.java.io.ByteArrayOutputStream", |
| "test.java.io.InputStream", |
| "test.java.io.OutputStream", |
| "test.java.io.PrintStream", |
| "test.java.io.PrintWriter", |
| "test.java.io.Reader", |
| "test.java.io.Writer", |
| "test.java.lang.Boolean", |
| "test.java.lang.ClassLoader", |
| "test.java.lang.Double", |
| "test.java.lang.Float", |
| "test.java.lang.Integer", |
| "test.java.lang.Long", |
| "test.java.lang.StrictMath.CubeRootTests", |
| "test.java.lang.StrictMath.Expm1Tests", |
| "test.java.lang.StrictMath.ExpTests", |
| "test.java.lang.StrictMath.HyperbolicTests", |
| "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard1", |
| "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard2", |
| "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard3", |
| "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard4", |
| "test.java.lang.StrictMath.HypotTests#testHypot", |
| "test.java.lang.StrictMath.Log1pTests", |
| "test.java.lang.StrictMath.Log10Tests", |
| "test.java.lang.StrictMath.MultiplicationTests", |
| "test.java.lang.StrictMath.PowTests", |
| "test.java.lang.String", |
| "test.java.lang.Thread", |
| "test.java.lang.invoke", |
| "test.java.lang.ref.SoftReference", |
| "test.java.lang.ref.BasicTest", |
| "test.java.lang.ref.EnqueueNullRefTest", |
| "test.java.lang.ref.EnqueuePollRaceTest", |
| "test.java.lang.ref.ReferenceCloneTest", |
| "test.java.lang.ref.ReferenceEnqueuePendingTest", |
| "test.java.math.BigDecimal", |
| "test.java.math.BigInteger#testArithmetic", |
| "test.java.math.BigInteger#testBitCount", |
| "test.java.math.BigInteger#testBitLength", |
| "test.java.math.BigInteger#testbitOps", |
| "test.java.math.BigInteger#testBitwise", |
| "test.java.math.BigInteger#testByteArrayConv", |
| "test.java.math.BigInteger#testConstructor", |
| "test.java.math.BigInteger#testDivideAndReminder", |
| "test.java.math.BigInteger#testDivideLarge", |
| "test.java.math.BigInteger#testModExp", |
| "test.java.math.BigInteger#testMultiplyLarge", |
| "test.java.math.BigInteger#testNextProbablePrime", |
| "test.java.math.BigInteger#testPow", |
| "test.java.math.BigInteger#testSerialize", |
| "test.java.math.BigInteger#testShift", |
| "test.java.math.BigInteger#testSquare", |
| "test.java.math.BigInteger#testSquareLarge", |
| "test.java.math.BigInteger#testSquareRootAndReminder", |
| "test.java.math.BigInteger#testStringConv_generic", |
| "test.java.math.RoundingMode", |
| "test.java.net.DatagramSocket", |
| "test.java.net.Socket", |
| "test.java.net.SocketOptions", |
| "test.java.net.URLDecoder", |
| "test.java.net.URLEncoder", |
| "test.java.nio.channels.Channels", |
| "test.java.nio.channels.SelectionKey", |
| "test.java.nio.channels.Selector", |
| "test.java.nio.file", |
| "test.java.security.cert", |
| "test.java.security.KeyAgreement.KeyAgreementTest", |
| "test.java.security.KeyAgreement.KeySizeTest#testECDHKeySize", |
| "test.java.security.KeyAgreement.KeySpecTest", |
| "test.java.security.KeyAgreement.MultiThreadTest", |
| "test.java.security.KeyAgreement.NegativeTest", |
| "test.java.security.KeyStore", |
| "test.java.security.Provider", |
| "test.java.time", |
| "test.java.util.Arrays", |
| "test.java.util.Collection", |
| "test.java.util.Collections", |
| "test.java.util.Date", |
| "test.java.util.EnumMap", |
| "test.java.util.EnumSet", |
| "test.java.util.GregorianCalendar", |
| "test.java.util.LinkedHashMap", |
| "test.java.util.LinkedHashSet", |
| "test.java.util.List", |
| "test.java.util.Map", |
| "test.java.util.Optional", |
| "test.java.util.TestFormatter", |
| "test.java.util.TimeZone", |
| "test.java.util.function", |
| "test.java.util.stream", |
| "tck.java.time", |
| } |
| |
| def get_jar_filename(classpath): |
| base_path = (ANDROID_PRODUCT_OUT + "/../..") if ANDROID_PRODUCT_OUT else "out/target" |
| base_path = os.path.normpath(base_path) # Normalize ".." components for readability. |
| return f"{base_path}/common/obj/JAVA_LIBRARIES/{classpath}_intermediates/classes.jar" |
| |
| def get_timeout_secs(): |
| default_timeout_secs = 600 |
| if args.gcstress: |
| default_timeout_secs = 1200 |
| if args.debug: |
| default_timeout_secs = 1800 |
| return args.timeout or default_timeout_secs |
| |
| def get_expected_failures(): |
| failures = ["art/tools/libcore_failures.txt"] |
| if args.mode != "jvm": |
| if args.gcstress: |
| failures.append("art/tools/libcore_gcstress_failures.txt") |
| if args.gcstress and args.debug: |
| failures.append("art/tools/libcore_gcstress_debug_failures.txt") |
| if args.debug and not args.gcstress and args.getrandom: |
| failures.append("art/tools/libcore_debug_failures.txt") |
| if not args.getrandom: |
| failures.append("art/tools/libcore_fugu_failures.txt") |
| return failures |
| |
| def get_test_names(): |
| if args.tests: |
| return args.tests |
| test_names = list(LIBCORE_TEST_NAMES) |
| # See b/78228743 and b/178351808. |
| if args.gcstress or args.debug or args.mode == "jvm": |
| test_names = list(t for t in test_names if not t.startswith("libcore.highmemorytest")) |
| test_names = list(filter(lambda x: x not in SLOW_OJLUNI_TESTS, test_names)) |
| if args.gcstress and args.debug: |
| test_names = list(filter(lambda x: x not in DISABLED_GCSTRESS_DEBUG_TESTS, test_names)) |
| if not args.getrandom: |
| # Disable libcore.highmemorytest due to limited ram on fugu. http://b/258173036 |
| test_names = list(filter(lambda x: x not in DISABLED_FUGU_TESTS and |
| not x.startswith("libcore.highmemorytest"), test_names)) |
| return test_names |
| |
| def get_vogar_command(test_name): |
| cmd = ["vogar"] |
| if args.mode == "device": |
| cmd.append("--mode=device --vm-arg -Ximage:/system/framework/art_boot_images/boot.art") |
| cmd.append("--vm-arg -Xbootclasspath:" + ":".join(BOOT_CLASSPATH)) |
| |
| if args.mode == "host": |
| # We explicitly give a wrong path for the image, to ensure vogar |
| # will create a boot image with the default compiler. Note that |
| # giving an existing image on host does not work because of |
| # classpath/resources differences when compiling the boot image. |
| cmd.append("--mode=host --vm-arg -Ximage:/non/existent/vogar.art") |
| if args.mode == "jvm": |
| cmd.append("--mode=jvm") |
| if args.variant: |
| cmd.append("--variant=" + args.variant) |
| if args.gcstress: |
| cmd.append("--vm-arg -Xgc:gcstress") |
| cmd.append('--vm-arg -Djsr166.delay.factor="1.50"') |
| if args.debug: |
| cmd.append("--vm-arg -XXlib:libartd.so --vm-arg -XX:SlowDebug=true") |
| |
| # The only device in go/art-buildbot without getrandom is fugu. We limit the amount of memory |
| # per runtime for fugu to avoid low memory killer, fugu has 4-cores 1GB RAM (b/258171768). |
| if not args.getrandom: |
| cmd.append("--vm-arg -Xmx128M") |
| |
| if args.mode == "device": |
| if ART_TEST_CHROOT: |
| cmd.append(f"--chroot {ART_TEST_CHROOT} --device-dir=/tmp/vogar/test-{test_name}") |
| else: |
| cmd.append(f"--device-dir=/data/local/tmp/vogar/test-{test_name}") |
| cmd.append(f"--vm-command={ART_TEST_ANDROID_ROOT}/bin/art") |
| else: |
| cmd.append(f"--device-dir=/tmp/vogar/test-{test_name}") |
| |
| if args.mode != "jvm": |
| cmd.append("--timeout {}".format(get_timeout_secs())) |
| cmd.append("--toolchain d8 --language CUR") |
| if args.jit: |
| cmd.append("--vm-arg -Xcompiler-option --vm-arg --compiler-filter=quicken") |
| cmd.append("--vm-arg -Xusejit:{}".format(str(args.jit).lower())) |
| |
| if args.verbose: |
| cmd.append("--verbose") |
| |
| # Suppress color codes if not attached to a terminal |
| if not sys.stdout.isatty(): |
| cmd.append("--no-color") |
| |
| cmd.extend("--expectations " + f for f in get_expected_failures()) |
| cmd.extend("--classpath " + get_jar_filename(cp) for cp in CLASSPATH) |
| cmd.append(test_name) |
| return cmd |
| |
| def get_target_cpu_count(): |
| adb_command = 'adb shell cat /sys/devices/system/cpu/present' |
| with subprocess.Popen(adb_command.split(), |
| stderr=subprocess.STDOUT, |
| stdout=subprocess.PIPE, |
| universal_newlines=True) as proc: |
| assert(proc.wait() == 0) # Check the exit code. |
| match = re.match(r'\d*-(\d*)', proc.stdout.read()) |
| assert(match) |
| return int(match.group(1)) + 1 # Add one to convert from "last-index" to "count" |
| |
| def main(): |
| global args |
| args = parse_args() |
| |
| if not os.path.exists('build/envsetup.sh'): |
| raise AssertionError("Script needs to be run at the root of the android tree") |
| for jar in map(get_jar_filename, CLASSPATH): |
| if not os.path.exists(jar): |
| raise AssertionError(f"Missing {jar}. Run buildbot-build.sh first.") |
| |
| if not args.jobs: |
| if args.mode == "device": |
| args.jobs = get_target_cpu_count() |
| else: |
| args.jobs = multiprocessing.cpu_count() |
| if args.gcstress: |
| # TODO: Investigate and fix the underlying issues. |
| args.jobs = args.jobs // 2 |
| |
| def run_test(test_name): |
| cmd = " ".join(get_vogar_command(test_name)) |
| if args.dry_run: |
| return test_name, cmd, "Dry-run: skipping execution", 0 |
| with subprocess.Popen(shlex.split(cmd), |
| stderr=subprocess.STDOUT, |
| stdout=subprocess.PIPE, |
| universal_newlines=True) as proc: |
| return test_name, cmd, proc.communicate()[0], proc.wait() |
| |
| failed_regex = re.compile(r"^.* FAIL \((?:EXEC_FAILED|ERROR)\)$", re.MULTILINE) |
| failed_tests, max_exit_code = [], 0 |
| with concurrent.futures.ThreadPoolExecutor(max_workers=args.jobs) as pool: |
| futures = [pool.submit(run_test, test_name) for test_name in get_test_names()] |
| print(f"Running {len(futures)} tasks on {args.jobs} core(s)...\n") |
| for i, future in enumerate(concurrent.futures.as_completed(futures)): |
| test_name, cmd, stdout, exit_code = future.result() |
| if exit_code != 0 or args.dry_run or args.verbose: |
| print(cmd) |
| print(stdout.strip()) |
| else: |
| print(stdout.strip().split("\n")[-1]) # Vogar final summary line. |
| failed_match = failed_regex.findall(stdout) |
| failed_tests.extend(failed_match) |
| max_exit_code = max(max_exit_code, exit_code) |
| result = "PASSED" if exit_code == 0 else f"FAILED ({len(failed_match)} test(s) failed)" |
| print(f"[{i+1}/{len(futures)}] Test set {test_name} {result}\n") |
| print(f"Overall, {len(failed_tests)} test(s) failed:") |
| print("\n".join(failed_tests)) |
| sys.exit(max_exit_code) |
| |
| if __name__ == '__main__': |
| main() |