316 lines
8.6 KiB
Go
316 lines
8.6 KiB
Go
package main
|
|
|
|
import (
|
|
"flag"
|
|
"fmt"
|
|
"math/rand"
|
|
"os"
|
|
"regexp"
|
|
"runtime"
|
|
"strings"
|
|
"time"
|
|
|
|
"io/ioutil"
|
|
"path/filepath"
|
|
|
|
"github.com/onsi/ginkgo/config"
|
|
"github.com/onsi/ginkgo/ginkgo/interrupthandler"
|
|
"github.com/onsi/ginkgo/ginkgo/testrunner"
|
|
colorable "github.com/onsi/ginkgo/reporters/stenographer/support/go-colorable"
|
|
"github.com/onsi/ginkgo/types"
|
|
)
|
|
|
|
func BuildRunCommand() *Command {
|
|
commandFlags := NewRunCommandFlags(flag.NewFlagSet("ginkgo", flag.ExitOnError))
|
|
notifier := NewNotifier(commandFlags)
|
|
interruptHandler := interrupthandler.NewInterruptHandler()
|
|
runner := &SpecRunner{
|
|
commandFlags: commandFlags,
|
|
notifier: notifier,
|
|
interruptHandler: interruptHandler,
|
|
suiteRunner: NewSuiteRunner(notifier, interruptHandler),
|
|
}
|
|
|
|
return &Command{
|
|
Name: "",
|
|
FlagSet: commandFlags.FlagSet,
|
|
UsageCommand: "ginkgo <FLAGS> <PACKAGES> -- <PASS-THROUGHS>",
|
|
Usage: []string{
|
|
"Run the tests in the passed in <PACKAGES> (or the package in the current directory if left blank).",
|
|
"Any arguments after -- will be passed to the test.",
|
|
"Accepts the following flags:",
|
|
},
|
|
Command: runner.RunSpecs,
|
|
}
|
|
}
|
|
|
|
type SpecRunner struct {
|
|
commandFlags *RunWatchAndBuildCommandFlags
|
|
notifier *Notifier
|
|
interruptHandler *interrupthandler.InterruptHandler
|
|
suiteRunner *SuiteRunner
|
|
}
|
|
|
|
func (r *SpecRunner) RunSpecs(args []string, additionalArgs []string) {
|
|
r.commandFlags.computeNodes()
|
|
r.notifier.VerifyNotificationsAreAvailable()
|
|
|
|
deprecationTracker := types.NewDeprecationTracker()
|
|
|
|
if r.commandFlags.ParallelStream && (runtime.GOOS != "windows") {
|
|
deprecationTracker.TrackDeprecation(types.Deprecation{
|
|
Message: "--stream is deprecated and will be removed in Ginkgo 2.0",
|
|
DocLink: "removed--stream",
|
|
Version: "1.16.0",
|
|
})
|
|
}
|
|
|
|
if r.commandFlags.Notify {
|
|
deprecationTracker.TrackDeprecation(types.Deprecation{
|
|
Message: "--notify is deprecated and will be removed in Ginkgo 2.0",
|
|
DocLink: "removed--notify",
|
|
Version: "1.16.0",
|
|
})
|
|
}
|
|
|
|
if deprecationTracker.DidTrackDeprecations() {
|
|
fmt.Fprintln(colorable.NewColorableStderr(), deprecationTracker.DeprecationsReport())
|
|
}
|
|
|
|
suites, skippedPackages := findSuites(args, r.commandFlags.Recurse, r.commandFlags.SkipPackage, true)
|
|
if len(skippedPackages) > 0 {
|
|
fmt.Println("Will skip:")
|
|
for _, skippedPackage := range skippedPackages {
|
|
fmt.Println(" " + skippedPackage)
|
|
}
|
|
}
|
|
|
|
if len(skippedPackages) > 0 && len(suites) == 0 {
|
|
fmt.Println("All tests skipped! Exiting...")
|
|
os.Exit(0)
|
|
}
|
|
|
|
if len(suites) == 0 {
|
|
complainAndQuit("Found no test suites")
|
|
}
|
|
|
|
r.ComputeSuccinctMode(len(suites))
|
|
|
|
t := time.Now()
|
|
|
|
runners := []*testrunner.TestRunner{}
|
|
for _, suite := range suites {
|
|
runners = append(runners, testrunner.New(suite, r.commandFlags.NumCPU, r.commandFlags.ParallelStream, r.commandFlags.Timeout, r.commandFlags.GoOpts, additionalArgs))
|
|
}
|
|
|
|
numSuites := 0
|
|
runResult := testrunner.PassingRunResult()
|
|
if r.commandFlags.UntilItFails {
|
|
iteration := 0
|
|
for {
|
|
r.UpdateSeed()
|
|
randomizedRunners := r.randomizeOrder(runners)
|
|
runResult, numSuites = r.suiteRunner.RunSuites(randomizedRunners, r.commandFlags.NumCompilers, r.commandFlags.KeepGoing, nil)
|
|
iteration++
|
|
|
|
if r.interruptHandler.WasInterrupted() {
|
|
break
|
|
}
|
|
|
|
if runResult.Passed {
|
|
fmt.Printf("\nAll tests passed...\nWill keep running them until they fail.\nThis was attempt #%d\n%s\n", iteration, orcMessage(iteration))
|
|
} else {
|
|
fmt.Printf("\nTests failed on attempt #%d\n\n", iteration)
|
|
break
|
|
}
|
|
}
|
|
} else {
|
|
randomizedRunners := r.randomizeOrder(runners)
|
|
runResult, numSuites = r.suiteRunner.RunSuites(randomizedRunners, r.commandFlags.NumCompilers, r.commandFlags.KeepGoing, nil)
|
|
}
|
|
|
|
for _, runner := range runners {
|
|
runner.CleanUp()
|
|
}
|
|
|
|
if r.isInCoverageMode() {
|
|
if r.getOutputDir() != "" {
|
|
// If coverprofile is set, combine coverages
|
|
if r.getCoverprofile() != "" {
|
|
if err := r.combineCoverprofiles(runners); err != nil {
|
|
fmt.Println(err.Error())
|
|
os.Exit(1)
|
|
}
|
|
} else {
|
|
// Just move them
|
|
r.moveCoverprofiles(runners)
|
|
}
|
|
}
|
|
}
|
|
|
|
fmt.Printf("\nGinkgo ran %d %s in %s\n", numSuites, pluralizedWord("suite", "suites", numSuites), time.Since(t))
|
|
|
|
if runResult.Passed {
|
|
if runResult.HasProgrammaticFocus && strings.TrimSpace(os.Getenv("GINKGO_EDITOR_INTEGRATION")) == "" {
|
|
fmt.Printf("Test Suite Passed\n")
|
|
fmt.Printf("Detected Programmatic Focus - setting exit status to %d\n", types.GINKGO_FOCUS_EXIT_CODE)
|
|
os.Exit(types.GINKGO_FOCUS_EXIT_CODE)
|
|
} else {
|
|
fmt.Printf("Test Suite Passed\n")
|
|
os.Exit(0)
|
|
}
|
|
} else {
|
|
fmt.Printf("Test Suite Failed\n")
|
|
os.Exit(1)
|
|
}
|
|
}
|
|
|
|
// Moves all generated profiles to specified directory
|
|
func (r *SpecRunner) moveCoverprofiles(runners []*testrunner.TestRunner) {
|
|
for _, runner := range runners {
|
|
_, filename := filepath.Split(runner.CoverageFile)
|
|
err := os.Rename(runner.CoverageFile, filepath.Join(r.getOutputDir(), filename))
|
|
|
|
if err != nil {
|
|
fmt.Printf("Unable to move coverprofile %s, %v\n", runner.CoverageFile, err)
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
// Combines all generated profiles in the specified directory
|
|
func (r *SpecRunner) combineCoverprofiles(runners []*testrunner.TestRunner) error {
|
|
|
|
path, _ := filepath.Abs(r.getOutputDir())
|
|
if !fileExists(path) {
|
|
return fmt.Errorf("Unable to create combined profile, outputdir does not exist: %s", r.getOutputDir())
|
|
}
|
|
|
|
fmt.Println("path is " + path)
|
|
|
|
combined, err := os.OpenFile(
|
|
filepath.Join(path, r.getCoverprofile()),
|
|
os.O_WRONLY|os.O_CREATE,
|
|
0666,
|
|
)
|
|
|
|
if err != nil {
|
|
fmt.Printf("Unable to create combined profile, %v\n", err)
|
|
return nil // non-fatal error
|
|
}
|
|
|
|
modeRegex := regexp.MustCompile(`^mode: .*\n`)
|
|
for index, runner := range runners {
|
|
contents, err := ioutil.ReadFile(runner.CoverageFile)
|
|
|
|
if err != nil {
|
|
fmt.Printf("Unable to read coverage file %s to combine, %v\n", runner.CoverageFile, err)
|
|
return nil // non-fatal error
|
|
}
|
|
|
|
// remove the cover mode line from every file
|
|
// except the first one
|
|
if index > 0 {
|
|
contents = modeRegex.ReplaceAll(contents, []byte{})
|
|
}
|
|
|
|
_, err = combined.Write(contents)
|
|
|
|
// Add a newline to the end of every file if missing.
|
|
if err == nil && len(contents) > 0 && contents[len(contents)-1] != '\n' {
|
|
_, err = combined.Write([]byte("\n"))
|
|
}
|
|
|
|
if err != nil {
|
|
fmt.Printf("Unable to append to coverprofile, %v\n", err)
|
|
return nil // non-fatal error
|
|
}
|
|
}
|
|
|
|
fmt.Println("All profiles combined")
|
|
return nil
|
|
}
|
|
|
|
func (r *SpecRunner) isInCoverageMode() bool {
|
|
opts := r.commandFlags.GoOpts
|
|
return *opts["cover"].(*bool) || *opts["coverpkg"].(*string) != "" || *opts["covermode"].(*string) != ""
|
|
}
|
|
|
|
func (r *SpecRunner) getCoverprofile() string {
|
|
return *r.commandFlags.GoOpts["coverprofile"].(*string)
|
|
}
|
|
|
|
func (r *SpecRunner) getOutputDir() string {
|
|
return *r.commandFlags.GoOpts["outputdir"].(*string)
|
|
}
|
|
|
|
func (r *SpecRunner) ComputeSuccinctMode(numSuites int) {
|
|
if config.DefaultReporterConfig.Verbose {
|
|
config.DefaultReporterConfig.Succinct = false
|
|
return
|
|
}
|
|
|
|
if numSuites == 1 {
|
|
return
|
|
}
|
|
|
|
if numSuites > 1 && !r.commandFlags.wasSet("succinct") {
|
|
config.DefaultReporterConfig.Succinct = true
|
|
}
|
|
}
|
|
|
|
func (r *SpecRunner) UpdateSeed() {
|
|
if !r.commandFlags.wasSet("seed") {
|
|
config.GinkgoConfig.RandomSeed = time.Now().Unix()
|
|
}
|
|
}
|
|
|
|
func (r *SpecRunner) randomizeOrder(runners []*testrunner.TestRunner) []*testrunner.TestRunner {
|
|
if !r.commandFlags.RandomizeSuites {
|
|
return runners
|
|
}
|
|
|
|
if len(runners) <= 1 {
|
|
return runners
|
|
}
|
|
|
|
randomizedRunners := make([]*testrunner.TestRunner, len(runners))
|
|
randomizer := rand.New(rand.NewSource(config.GinkgoConfig.RandomSeed))
|
|
permutation := randomizer.Perm(len(runners))
|
|
for i, j := range permutation {
|
|
randomizedRunners[i] = runners[j]
|
|
}
|
|
return randomizedRunners
|
|
}
|
|
|
|
func orcMessage(iteration int) string {
|
|
if iteration < 10 {
|
|
return ""
|
|
} else if iteration < 30 {
|
|
return []string{
|
|
"If at first you succeed...",
|
|
"...try, try again.",
|
|
"Looking good!",
|
|
"Still good...",
|
|
"I think your tests are fine....",
|
|
"Yep, still passing",
|
|
"Oh boy, here I go testin' again!",
|
|
"Even the gophers are getting bored",
|
|
"Did you try -race?",
|
|
"Maybe you should stop now?",
|
|
"I'm getting tired...",
|
|
"What if I just made you a sandwich?",
|
|
"Hit ^C, hit ^C, please hit ^C",
|
|
"Make it stop. Please!",
|
|
"Come on! Enough is enough!",
|
|
"Dave, this conversation can serve no purpose anymore. Goodbye.",
|
|
"Just what do you think you're doing, Dave? ",
|
|
"I, Sisyphus",
|
|
"Insanity: doing the same thing over and over again and expecting different results. -Einstein",
|
|
"I guess Einstein never tried to churn butter",
|
|
}[iteration-10] + "\n"
|
|
} else {
|
|
return "No, seriously... you can probably stop now.\n"
|
|
}
|
|
}
|