mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
If a Lit test fails, re-run it 5 times to increase our odds of detecting non-determinism early. If the test turns out to be flaky, it will fail and be reported as a FLAKYFAIL to stress the encountered non-determinism. Long tests and tests in `test/Interop/Cxx` are deliberately exempted from retries to avoid timeouts and unreasonable spikes in turnaround time.
89 lines
3.6 KiB
Python
89 lines
3.6 KiB
Python
# swift/test/swift_test.py - SwiftTest format for lit tests -*- python -*-
|
|
#
|
|
# This source file is part of the Swift.org open source project
|
|
#
|
|
# Copyright (c) 2014 - 2017 Apple Inc. and the Swift project authors
|
|
# Licensed under Apache License v2.0 with Runtime Library Exception
|
|
#
|
|
# See https://swift.org/LICENSE.txt for license information
|
|
# See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
|
#
|
|
# -----------------------------------------------------------------------------
|
|
#
|
|
# This is a test format file for the 'lit' test runner.
|
|
#
|
|
# -----------------------------------------------------------------------------
|
|
|
|
import os
|
|
|
|
import lit
|
|
import lit.formats
|
|
import lit.util
|
|
import lit.Test
|
|
|
|
# Like FLAKYPASS, but this one is a failure.
|
|
FLAKYFAIL = lit.Test.ResultCode("FLAKYFAIL", "Flaky", True)
|
|
|
|
class SwiftTest(lit.formats.ShTest, object):
|
|
def __init__(self, coverage_mode=None, execute_external=True):
|
|
super(SwiftTest, self).__init__(execute_external=execute_external)
|
|
if coverage_mode == "FALSE":
|
|
self.coverage_mode = None
|
|
else:
|
|
self.coverage_mode = coverage_mode
|
|
self.skipped_tests = set()
|
|
|
|
def before_test(self, test, litConfig):
|
|
if self.coverage_mode:
|
|
# FIXME: The compiler crashers run so fast they fill up the
|
|
# merger's queue (and therefore the build bot's disk)
|
|
if 'crasher' in test.getSourcePath():
|
|
test.config.environment["LLVM_PROFILE_FILE"] = os.devnull
|
|
self.skipped_tests.add(test.getSourcePath())
|
|
return
|
|
|
|
if self.coverage_mode == "NOT_MERGED":
|
|
execpath = test.getExecPath()
|
|
profdir = os.path.join(os.path.dirname(execpath), "Output",
|
|
os.path.basename(execpath) + '.profdir')
|
|
if not os.path.exists(profdir):
|
|
os.makedirs(profdir)
|
|
|
|
test.config.environment["LLVM_PROFILE_FILE"] = \
|
|
os.path.join(profdir, "swift-%p.profraw")
|
|
else:
|
|
test.config.environment["LLVM_PROFILE_FILE"] = \
|
|
os.path.join(test.config.swift_test_results_dir,
|
|
"swift-%4m.profraw")
|
|
|
|
# If long tests are not run, and this is not a test known to be quite
|
|
# slow, tell Lit to re-run it at most 5 times upon failure to increase
|
|
# our odds detecting non-determinism early. If a test turns out to
|
|
# be flaky, it will fail and be reported as a FLAKYFAIL.
|
|
#
|
|
# NB: Unfortunately, we cannot base this condition on whether a
|
|
# particular test is a long test without hacks because the test is not
|
|
# parsed until it is executed.
|
|
if (
|
|
"long_test" not in test.config.available_features
|
|
# Some of these tests are notoriously slow.
|
|
and not ("Interop" in test.path_in_suite and "Cxx" in test.path_in_suite)
|
|
):
|
|
test.allowed_retries = 5
|
|
|
|
def after_test(self, test, litConfig, result):
|
|
# Intercept FLAKYPASS results and rewrite them into flaky failures. The
|
|
# goal here is to catch non-determinism and complain about it rather
|
|
# than to give a test more chances to succeed.
|
|
if result.code == lit.Test.FLAKYPASS:
|
|
result.code = FLAKYFAIL
|
|
|
|
if test.getSourcePath() in self.skipped_tests:
|
|
self.skipped_tests.remove(test.getSourcePath())
|
|
return result
|
|
|
|
def execute(self, test, litConfig):
|
|
self.before_test(test, litConfig)
|
|
result = super(SwiftTest, self).execute(test, litConfig)
|
|
return self.after_test(test, litConfig, result)
|