go-junit-report/parser/gotest/report_builder.go
Joël Stemmer a4c479dbbb gtr: Reset builder state after the end of a test or benchmark
The consequence of this is that any output emitted after the test or
benchmark result, will no longer belong to the last test. Instead it
will be appended to the global log. This is necessary to correctly
attribute output to the correct test or benchmark, for example in cases
where both tests and benchmarks appear.

In the past, Go test output printed test logs after the test result.
However, from Go 1.14 this behaviour was changed and test logs are now
printed before the test result.

With this commit, we effectively no longer support the old behaviour in
Go1.13 and earlier. The output will still appear in the generated
report, however it will no longer belong to the correct test.
2022-05-14 23:25:41 +01:00

338 lines
9.8 KiB
Go

package gotest
import (
"time"
"github.com/jstemmer/go-junit-report/v2/gtr"
)
// reportBuilder helps build a test Report from a collection of events.
//
// The reportBuilder keeps track of the active context whenever a test,
// benchmark or build error is created. This is necessary because the test
// parser do not contain any state themselves and simply just emit an event for
// every line that is read. By tracking the active context, any output that is
// appended to the reportBuilder gets attributed to the correct test, benchmark
// or build error.
type reportBuilder struct {
packages []gtr.Package
tests map[int]gtr.Test
benchmarks map[int]gtr.Benchmark
buildErrors map[int]gtr.Error
runErrors map[int]gtr.Error
// state
nextId int // next free unused id
lastId int // most recently created id
output []string // output that does not belong to any test
coverage float64 // coverage percentage
// default values
PackageName string
TimestampFunc func() time.Time
}
// newReportBuilder creates a new reportBuilder.
func newReportBuilder() *reportBuilder {
return &reportBuilder{
tests: make(map[int]gtr.Test),
benchmarks: make(map[int]gtr.Benchmark),
buildErrors: make(map[int]gtr.Error),
runErrors: make(map[int]gtr.Error),
nextId: 1,
TimestampFunc: time.Now,
}
}
// newId returns a new unique id and sets the active context this id.
func (b *reportBuilder) newId() int {
id := b.nextId
b.lastId = id
b.nextId += 1
return id
}
// flush creates a new package in this report containing any tests or
// benchmarks we've collected so far. This is necessary when a test or
// benchmark did not end with a summary.
func (b *reportBuilder) flush() {
if len(b.tests) > 0 || len(b.benchmarks) > 0 {
b.CreatePackage(b.PackageName, "", 0, "")
}
}
// Build returns the new Report containing all the tests, benchmarks and output
// created so far.
func (b *reportBuilder) Build() gtr.Report {
b.flush()
return gtr.Report{Packages: b.packages}
}
// CreateTest adds a test with the given name to the report, and marks it as
// active.
func (b *reportBuilder) CreateTest(name string) {
b.tests[b.newId()] = gtr.Test{Name: name}
}
// PauseTest marks the active context as no longer active. Any results or
// output added to the report after calling PauseTest will no longer be assumed
// to belong to this test.
func (b *reportBuilder) PauseTest(name string) {
b.lastId = 0
}
// ContinueTest finds the test with the given name and marks it as active. If
// more than one test exist with this name, the most recently created test will
// be used.
func (b *reportBuilder) ContinueTest(name string) {
b.lastId = b.findTest(name)
}
// EndTest finds the test with the given name, sets the result, duration and
// level. If more than one test exists with this name, the most recently
// created test will be used. If no test exists with this name, a new test is
// created.
func (b *reportBuilder) EndTest(name, result string, duration time.Duration, level int) {
b.lastId = b.findTest(name)
if b.lastId < 0 {
// test did not exist, create one
// TODO: Likely reason is that the user ran go test without the -v
// flag, should we report this somewhere?
b.CreateTest(name)
}
t := b.tests[b.lastId]
t.Result = parseResult(result)
t.Duration = duration
t.Level = level
b.tests[b.lastId] = t
b.lastId = 0
}
// End marks the active context as no longer active.
func (b *reportBuilder) End() {
b.lastId = 0
}
// CreateBenchmark adds a benchmark with the given name to the report, and
// marks it as active. If more than one benchmark exists with this name, the
// most recently created benchmark will be updated. If no benchmark exists with
// this name, a new benchmark is created.
func (b *reportBuilder) CreateBenchmark(name string) {
b.benchmarks[b.newId()] = gtr.Benchmark{
Name: name,
}
}
// BenchmarkResult updates an existing or adds a new benchmark with the given
// results and marks it as active. If an existing benchmark with this name
// exists but without result, then that one is updated. Otherwise a new one is
// added to the report.
func (b *reportBuilder) BenchmarkResult(name string, iterations int64, nsPerOp, mbPerSec float64, bytesPerOp, allocsPerOp int64) {
b.lastId = b.findBenchmark(name)
if b.lastId < 0 || b.benchmarks[b.lastId].Result != gtr.Unknown {
b.CreateBenchmark(name)
}
b.benchmarks[b.lastId] = gtr.Benchmark{
Name: name,
Result: gtr.Pass,
Iterations: iterations,
NsPerOp: nsPerOp,
MBPerSec: mbPerSec,
BytesPerOp: bytesPerOp,
AllocsPerOp: allocsPerOp,
}
}
// EndBenchmark finds the benchmark with the given name and sets the result. If
// more than one benchmark exists with this name, the most recently created
// benchmark will be used. If no benchmark exists with this name, a new
// benchmark is created.
func (b *reportBuilder) EndBenchmark(name, result string) {
b.lastId = b.findBenchmark(name)
if b.lastId < 0 {
b.CreateBenchmark(name)
}
bm := b.benchmarks[b.lastId]
bm.Result = parseResult(result)
b.benchmarks[b.lastId] = bm
b.lastId = 0
}
// CreateBuildError creates a new build error and marks it as active.
func (b *reportBuilder) CreateBuildError(packageName string) {
b.buildErrors[b.newId()] = gtr.Error{Name: packageName}
}
// CreatePackage adds a new package with the given name to the Report. This
// package contains all the build errors, output, tests and benchmarks created
// so far. Afterwards all state is reset.
func (b *reportBuilder) CreatePackage(name, result string, duration time.Duration, data string) {
pkg := gtr.Package{
Name: name,
Duration: duration,
}
if b.TimestampFunc != nil {
pkg.Timestamp = b.TimestampFunc()
}
// Build errors are treated somewhat differently. Rather than having a
// single package with all build errors collected so far, we only care
// about the build errors for this particular package.
for id, buildErr := range b.buildErrors {
if buildErr.Name == name {
if len(b.tests) > 0 || len(b.benchmarks) > 0 {
panic("unexpected tests and/or benchmarks found in build error package")
}
buildErr.Duration = duration
buildErr.Cause = data
pkg.BuildError = buildErr
b.packages = append(b.packages, pkg)
delete(b.buildErrors, id)
// TODO: reset state
// TODO: buildErrors shouldn't reset/use nextId/lastId, they're more like a global cache
return
}
}
// If we've collected output, but there were no tests or benchmarks then
// either there were no tests, or there was some other non-build error.
if len(b.output) > 0 && len(b.tests) == 0 && len(b.benchmarks) == 0 {
if parseResult(result) == gtr.Fail {
pkg.RunError = gtr.Error{
Name: name,
Output: b.output,
}
}
b.packages = append(b.packages, pkg)
// TODO: reset state
b.output = nil
return
}
// If the summary result says we failed, but there were no failing tests
// then something else must have failed.
if parseResult(result) == gtr.Fail && (len(b.tests) > 0 || len(b.benchmarks) > 0) && !b.containsFailingTest() {
pkg.RunError = gtr.Error{
Name: name,
Output: b.output,
}
b.output = nil
}
// Collect tests and benchmarks for this package, maintaining insertion order.
var tests []gtr.Test
var benchmarks []gtr.Benchmark
for id := 1; id < b.nextId; id++ {
if t, ok := b.tests[id]; ok {
tests = append(tests, t)
}
if bm, ok := b.benchmarks[id]; ok {
benchmarks = append(benchmarks, bm)
}
}
pkg.Coverage = b.coverage
pkg.Output = b.output
pkg.Tests = tests
pkg.Benchmarks = benchmarks
b.packages = append(b.packages, pkg)
// reset state
b.nextId = 1
b.lastId = 0
b.output = nil
b.coverage = 0
b.tests = make(map[int]gtr.Test)
b.benchmarks = make(map[int]gtr.Benchmark)
}
// Coverage sets the code coverage percentage.
func (b *reportBuilder) Coverage(pct float64, packages []string) {
b.coverage = pct
}
// AppendOutput appends the given line to the currently active context. If no
// active context exists, the output is assumed to belong to the package.
func (b *reportBuilder) AppendOutput(line string) {
if b.lastId <= 0 {
b.output = append(b.output, line)
return
}
if t, ok := b.tests[b.lastId]; ok {
t.Output = append(t.Output, line)
b.tests[b.lastId] = t
} else if bm, ok := b.benchmarks[b.lastId]; ok {
bm.Output = append(bm.Output, line)
b.benchmarks[b.lastId] = bm
} else if be, ok := b.buildErrors[b.lastId]; ok {
be.Output = append(be.Output, line)
b.buildErrors[b.lastId] = be
} else {
b.output = append(b.output, line)
}
}
// findTest returns the id of the most recently created test with the given
// name, or -1 if no such test exists.
func (b *reportBuilder) findTest(name string) int {
// check if this test was lastId
if t, ok := b.tests[b.lastId]; ok && t.Name == name {
return b.lastId
}
for id := len(b.tests); id >= 0; id-- {
if b.tests[id].Name == name {
return id
}
}
return -1
}
// findBenchmark returns the id of the most recently created benchmark with the
// given name, or -1 if no such benchmark exists.
func (b *reportBuilder) findBenchmark(name string) int {
// check if this benchmark was lastId
if bm, ok := b.benchmarks[b.lastId]; ok && bm.Name == name {
return b.lastId
}
for id := len(b.benchmarks); id >= 0; id-- {
if b.benchmarks[id].Name == name {
return id
}
}
return -1
}
// containsFailingTest return true if the current list of tests contains at
// least one failing test or an unknown result.
func (b *reportBuilder) containsFailingTest() bool {
for _, test := range b.tests {
if test.Result == gtr.Fail || test.Result == gtr.Unknown {
return true
}
}
return false
}
// parseResult returns a Result for the given string r.
func parseResult(r string) gtr.Result {
switch r {
case "PASS":
return gtr.Pass
case "FAIL":
return gtr.Fail
case "SKIP":
return gtr.Skip
case "BENCH":
return gtr.Pass
default:
return gtr.Unknown
}
}