Signed-off-by: Andrew Dunham <andrew@du.nham.ca> Change-Id: Ibab5860f5797b3db151d3c27855333e43a9088a4main
parent
2df38b1feb
commit
aea251d42a
@ -0,0 +1,46 @@ |
||||
// Copyright (c) 2023 Tailscale Inc & AUTHORS All rights reserved.
|
||||
// Use of this source code is governed by a BSD-style
|
||||
// license that can be found in the LICENSE file.
|
||||
|
||||
// Package flakytest contains test helpers for marking a test as flaky. For
|
||||
// tests run using cmd/testwrapper, a failed flaky test will cause tests to be
|
||||
// re-run a few time until they succeed or exceed our iteration limit.
|
||||
package flakytest |
||||
|
||||
import ( |
||||
"os" |
||||
"regexp" |
||||
"testing" |
||||
) |
||||
|
||||
// InTestWrapper returns whether or not this binary is running under our test
|
||||
// wrapper.
|
||||
func InTestWrapper() bool { |
||||
return os.Getenv("TS_IN_TESTWRAPPER") != "" |
||||
} |
||||
|
||||
var issueRegexp = regexp.MustCompile(`\Ahttps://github\.com/tailscale/[a-zA-Z0-9_.-]+/issues/\d+\z`) |
||||
|
||||
// Mark sets the current test as a flaky test, such that if it fails, it will
|
||||
// be retried a few times on failure. issue must be a GitHub issue that tracks
|
||||
// the status of the flaky test being marked, of the format:
|
||||
//
|
||||
// https://github.com/tailscale/myRepo-H3re/issues/12345
|
||||
func Mark(t *testing.T, issue string) { |
||||
if !issueRegexp.MatchString(issue) { |
||||
t.Fatalf("bad issue format: %q", issue) |
||||
} |
||||
|
||||
if !InTestWrapper() { |
||||
return |
||||
} |
||||
|
||||
t.Cleanup(func() { |
||||
if t.Failed() { |
||||
t.Logf("flakytest: signaling test wrapper to retry test") |
||||
|
||||
// Signal to test wrapper that we should restart.
|
||||
os.Exit(123) |
||||
} |
||||
}) |
||||
} |
||||
@ -0,0 +1,27 @@ |
||||
// Copyright (c) 2023 Tailscale Inc & AUTHORS All rights reserved.
|
||||
// Use of this source code is governed by a BSD-style
|
||||
// license that can be found in the LICENSE file.
|
||||
|
||||
package flakytest |
||||
|
||||
import "testing" |
||||
|
||||
func TestIssueFormat(t *testing.T) { |
||||
testCases := []struct { |
||||
issue string |
||||
want bool |
||||
}{ |
||||
{"https://github.com/tailscale/cOrp/issues/1234", true}, |
||||
{"https://github.com/otherproject/corp/issues/1234", false}, |
||||
{"https://github.com/tailscale/corp/issues/", false}, |
||||
} |
||||
for _, testCase := range testCases { |
||||
if issueRegexp.MatchString(testCase.issue) != testCase.want { |
||||
ss := "" |
||||
if !testCase.want { |
||||
ss = " not" |
||||
} |
||||
t.Errorf("expected issueRegexp to%s match %q", ss, testCase.issue) |
||||
} |
||||
} |
||||
} |
||||
@ -0,0 +1,63 @@ |
||||
// Copyright (c) 2023 Tailscale Inc & AUTHORS All rights reserved.
|
||||
// Use of this source code is governed by a BSD-style
|
||||
// license that can be found in the LICENSE file.
|
||||
|
||||
// testwrapper is a wrapper for retrying flaky tests, using the -exec flag of
|
||||
// 'go test'. Tests that are flaky can use the 'flakytest' subpackage to mark
|
||||
// themselves as flaky and be retried on failure.
|
||||
package main |
||||
|
||||
import ( |
||||
"context" |
||||
"errors" |
||||
"log" |
||||
"os" |
||||
"os/exec" |
||||
) |
||||
|
||||
const ( |
||||
retryStatus = 123 |
||||
maxIterations = 3 |
||||
) |
||||
|
||||
func main() { |
||||
ctx := context.Background() |
||||
debug := os.Getenv("TS_TESTWRAPPER_DEBUG") != "" |
||||
|
||||
log.SetPrefix("testwrapper: ") |
||||
if !debug { |
||||
log.SetFlags(0) |
||||
} |
||||
|
||||
for i := 1; i <= maxIterations; i++ { |
||||
if i > 1 { |
||||
log.Printf("retrying flaky tests (%d of %d)", i, maxIterations) |
||||
} |
||||
cmd := exec.CommandContext(ctx, os.Args[1], os.Args[2:]...) |
||||
cmd.Stdout = os.Stdout |
||||
cmd.Stderr = os.Stderr |
||||
cmd.Env = append(os.Environ(), "TS_IN_TESTWRAPPER=1") |
||||
err := cmd.Run() |
||||
if err == nil { |
||||
return |
||||
} |
||||
|
||||
var exitErr *exec.ExitError |
||||
if !errors.As(err, &exitErr) { |
||||
if debug { |
||||
log.Printf("error isn't an ExitError") |
||||
} |
||||
os.Exit(1) |
||||
} |
||||
|
||||
if code := exitErr.ExitCode(); code != retryStatus { |
||||
if debug { |
||||
log.Printf("code (%d) != retryStatus (%d)", code, retryStatus) |
||||
} |
||||
os.Exit(code) |
||||
} |
||||
} |
||||
|
||||
log.Printf("test did not pass in %d iterations", maxIterations) |
||||
os.Exit(1) |
||||
} |
||||
Loading…
Reference in new issue