package gtr import ( "time" ) // 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 []Package tests map[int]Test benchmarks map[int]Benchmark buildErrors map[int]Error runErrors map[int]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]Test), benchmarks: make(map[int]Benchmark), buildErrors: make(map[int]Error), runErrors: make(map[int]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() Report { b.flush() return 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()] = 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, and marks it as active. 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 } // End marks the active context as no longer active. func (b *ReportBuilder) End() { b.lastId = 0 } // Benchmark adds a new Benchmark to the report and marks it as active. func (b *ReportBuilder) Benchmark(name string, iterations int64, nsPerOp, mbPerSec float64, bytesPerOp, allocsPerOp int64) { b.benchmarks[b.newId()] = Benchmark{ Name: name, Result: Pass, Iterations: iterations, NsPerOp: nsPerOp, MBPerSec: mbPerSec, BytesPerOp: bytesPerOp, AllocsPerOp: allocsPerOp, } } // CreateBuildError creates a new build error and marks it as active. func (b *ReportBuilder) CreateBuildError(packageName string) { b.buildErrors[b.newId()] = 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 := 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) == Fail { pkg.RunError = 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) == Fail && (len(b.tests) > 0 || len(b.benchmarks) > 0) && !b.containsFailingTest() { pkg.RunError = Error{ Name: name, Output: b.output, } b.output = nil } // Collect tests and benchmarks for this package, maintaining insertion order. var tests []Test var benchmarks []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]Test) b.benchmarks = make(map[int]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 } // 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 == Fail || test.Result == Unknown { return true } } return false } // parseResult returns a Result for the given string r. func parseResult(r string) Result { switch r { case "PASS": return Pass case "FAIL": return Fail case "SKIP": return Skip default: return Unknown } }