mirror of https://github.com/golang/go.git
150 lines
4.1 KiB
Go
150 lines
4.1 KiB
Go
// Copyright 2020 The Go 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 regtest
|
|
|
|
import (
|
|
"context"
|
|
"flag"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"runtime"
|
|
"testing"
|
|
"time"
|
|
|
|
"golang.org/x/tools/internal/lsp/cmd"
|
|
"golang.org/x/tools/internal/lsp/source"
|
|
"golang.org/x/tools/internal/testenv"
|
|
"golang.org/x/tools/internal/tool"
|
|
)
|
|
|
|
var (
|
|
runSubprocessTests = flag.Bool("enable_gopls_subprocess_tests", false, "run regtests against a gopls subprocess")
|
|
goplsBinaryPath = flag.String("gopls_test_binary", "", "path to the gopls binary for use as a remote, for use with the -enable_gopls_subprocess_tests flag")
|
|
regtestTimeout = flag.Duration("regtest_timeout", defaultRegtestTimeout(), "if nonzero, default timeout for each regtest; defaults to GOPLS_REGTEST_TIMEOUT")
|
|
skipCleanup = flag.Bool("regtest_skip_cleanup", false, "whether to skip cleaning up temp directories")
|
|
printGoroutinesOnFailure = flag.Bool("regtest_print_goroutines", false, "whether to print goroutines info on failure")
|
|
printLogs = flag.Bool("regtest_print_logs", false, "whether to print LSP logs")
|
|
)
|
|
|
|
func defaultRegtestTimeout() time.Duration {
|
|
s := os.Getenv("GOPLS_REGTEST_TIMEOUT")
|
|
if s == "" {
|
|
return 0
|
|
}
|
|
d, err := time.ParseDuration(s)
|
|
if err != nil {
|
|
fmt.Fprintf(os.Stderr, "invalid GOPLS_REGTEST_TIMEOUT %q: %v\n", s, err)
|
|
os.Exit(2)
|
|
}
|
|
return d
|
|
}
|
|
|
|
var runner *Runner
|
|
|
|
type regtestRunner interface {
|
|
Run(t *testing.T, files string, f TestFunc)
|
|
}
|
|
|
|
func Run(t *testing.T, files string, f TestFunc) {
|
|
runner.Run(t, files, f)
|
|
}
|
|
|
|
func WithOptions(opts ...RunOption) configuredRunner {
|
|
return configuredRunner{opts: opts}
|
|
}
|
|
|
|
type configuredRunner struct {
|
|
opts []RunOption
|
|
}
|
|
|
|
func (r configuredRunner) Run(t *testing.T, files string, f TestFunc) {
|
|
runner.Run(t, files, f, r.opts...)
|
|
}
|
|
|
|
type RunMultiple []struct {
|
|
Name string
|
|
Runner regtestRunner
|
|
}
|
|
|
|
func (r RunMultiple) Run(t *testing.T, files string, f TestFunc) {
|
|
for _, runner := range r {
|
|
t.Run(runner.Name, func(t *testing.T) {
|
|
runner.Runner.Run(t, files, f)
|
|
})
|
|
}
|
|
}
|
|
|
|
// The regtests run significantly slower on these operating systems, due to (we
|
|
// believe) kernel locking behavior. Only run in singleton mode on these
|
|
// operating system when using -short.
|
|
var slowGOOS = map[string]bool{
|
|
"darwin": true,
|
|
"openbsd": true,
|
|
"plan9": true,
|
|
}
|
|
|
|
func DefaultModes() Mode {
|
|
normal := Singleton | Experimental
|
|
if slowGOOS[runtime.GOOS] && testing.Short() {
|
|
normal = Singleton
|
|
}
|
|
if *runSubprocessTests {
|
|
return normal | SeparateProcess
|
|
}
|
|
return normal
|
|
}
|
|
|
|
// Main sets up and tears down the shared regtest state.
|
|
func Main(m *testing.M, hook func(*source.Options)) {
|
|
testenv.ExitIfSmallMachine()
|
|
|
|
// Disable GOPACKAGESDRIVER, as it can cause spurious test failures.
|
|
os.Setenv("GOPACKAGESDRIVER", "off")
|
|
|
|
flag.Parse()
|
|
if os.Getenv("_GOPLS_TEST_BINARY_RUN_AS_GOPLS") == "true" {
|
|
tool.Main(context.Background(), cmd.New("gopls", "", nil, nil), os.Args[1:])
|
|
os.Exit(0)
|
|
}
|
|
|
|
runner = &Runner{
|
|
DefaultModes: DefaultModes(),
|
|
Timeout: *regtestTimeout,
|
|
PrintGoroutinesOnFailure: *printGoroutinesOnFailure,
|
|
SkipCleanup: *skipCleanup,
|
|
OptionsHook: hook,
|
|
}
|
|
if *runSubprocessTests {
|
|
goplsPath := *goplsBinaryPath
|
|
if goplsPath == "" {
|
|
var err error
|
|
goplsPath, err = os.Executable()
|
|
if err != nil {
|
|
panic(fmt.Sprintf("finding test binary path: %v", err))
|
|
}
|
|
}
|
|
runner.GoplsPath = goplsPath
|
|
}
|
|
dir, err := ioutil.TempDir("", "gopls-regtest-")
|
|
if err != nil {
|
|
panic(fmt.Errorf("creating regtest temp directory: %v", err))
|
|
}
|
|
runner.TempDir = dir
|
|
|
|
code := m.Run()
|
|
if err := runner.Close(); err != nil {
|
|
fmt.Fprintf(os.Stderr, "closing test runner: %v\n", err)
|
|
// Regtest cleanup is broken in go1.12 and earlier, and sometimes flakes on
|
|
// Windows due to file locking, but this is OK for our CI.
|
|
//
|
|
// Fail on go1.13+, except for windows and android which have shutdown problems.
|
|
if testenv.Go1Point() >= 13 && runtime.GOOS != "windows" && runtime.GOOS != "android" {
|
|
os.Exit(1)
|
|
}
|
|
}
|
|
os.Exit(code)
|
|
}
|