gtr,parser/gotest: move Event and building a report to parser/gotest

The Parse method now directly returns a report, rather than a list of
events that then need to be converted into a report. As part of this
change, the Event struct has also been moved to the gotest package. It's
now the responsibility of the parser to construct a gtr.Report.
This commit is contained in:
Joël Stemmer 2022-03-14 23:17:57 +00:00
parent 832cc97037
commit c78e04707f
9 changed files with 272 additions and 252 deletions

View File

@ -9,7 +9,6 @@ import (
"strings" "strings"
"time" "time"
"github.com/jstemmer/go-junit-report/v2/pkg/gtr"
"github.com/jstemmer/go-junit-report/v2/pkg/junit" "github.com/jstemmer/go-junit-report/v2/pkg/junit"
"github.com/jstemmer/go-junit-report/v2/pkg/parser/gotest" "github.com/jstemmer/go-junit-report/v2/pkg/parser/gotest"
) )
@ -76,18 +75,18 @@ func main() {
in = io.TeeReader(in, os.Stdout) in = io.TeeReader(in, os.Stdout)
} }
parser := gotest.New() parser := gotest.New(gotest.PackageName(*packageName))
events, err := parser.Parse(in)
report, err := parser.Parse(in)
if err != nil { if err != nil {
exitf("error reading input: %s\n", err) exitf("error parsing input: %s\n", err)
} }
if *printEvents { if *printEvents {
for i, ev := range events { for i, ev := range parser.Events() {
fmt.Printf("%02d: %#v\n", i, ev) fmt.Printf("%02d: %#v\n", i, ev)
} }
} }
report := gtr.FromEvents(events, *packageName)
for i := range report.Packages { for i := range report.Packages {
for k, v := range properties { for k, v := range properties {
report.Packages[i].SetProperty(k, v) report.Packages[i].SetProperty(k, v)

View File

@ -13,7 +13,6 @@ import (
"testing" "testing"
"time" "time"
"github.com/jstemmer/go-junit-report/v2/pkg/gtr"
"github.com/jstemmer/go-junit-report/v2/pkg/junit" "github.com/jstemmer/go-junit-report/v2/pkg/junit"
"github.com/jstemmer/go-junit-report/v2/pkg/parser/gotest" "github.com/jstemmer/go-junit-report/v2/pkg/parser/gotest"
@ -192,20 +191,20 @@ func testReport(input, reportFile, packageName string, t *testing.T) {
} }
defer file.Close() defer file.Close()
parser := gotest.New() parser := gotest.New(gotest.PackageName(packageName))
events, err := parser.Parse(file)
report, err := parser.Parse(file)
if err != nil { if err != nil {
t.Fatal(err) t.Fatal(err)
} }
if *printEvents { if *printEvents {
for _, event := range events { for _, event := range parser.Events() {
t.Logf("Event: %+v", event) t.Logf("Event: %+v", event)
} }
} }
testTime := time.Date(2022, 1, 1, 0, 0, 0, 0, time.UTC) testTime := time.Date(2022, 1, 1, 0, 0, 0, 0, time.UTC)
report := gtr.FromEvents(events, packageName)
actual := junit.CreateFromReport(report, "hostname", testTime) actual := junit.CreateFromReport(report, "hostname", testTime)
expectedXML, err := loadTestReport(reportFile, "") expectedXML, err := loadTestReport(reportFile, "")

View File

@ -26,18 +26,17 @@ type ReportBuilder struct {
coverage float64 // coverage percentage coverage float64 // coverage percentage
// default values // default values
packageName string PackageName string
} }
// NewReportBuilder creates a new ReportBuilder. // NewReportBuilder creates a new ReportBuilder.
func NewReportBuilder(packageName string) *ReportBuilder { func NewReportBuilder() *ReportBuilder {
return &ReportBuilder{ return &ReportBuilder{
tests: make(map[int]Test), tests: make(map[int]Test),
benchmarks: make(map[int]Benchmark), benchmarks: make(map[int]Benchmark),
buildErrors: make(map[int]Error), buildErrors: make(map[int]Error),
runErrors: make(map[int]Error), runErrors: make(map[int]Error),
nextId: 1, nextId: 1,
packageName: packageName,
} }
} }
@ -54,7 +53,7 @@ func (b *ReportBuilder) newId() int {
// benchmark did not end with a summary. // benchmark did not end with a summary.
func (b *ReportBuilder) flush() { func (b *ReportBuilder) flush() {
if len(b.tests) > 0 || len(b.benchmarks) > 0 { if len(b.tests) > 0 || len(b.benchmarks) > 0 {
b.CreatePackage(b.packageName, "", 0, "") b.CreatePackage(b.PackageName, "", 0, "")
} }
} }

View File

@ -1,53 +0,0 @@
package gtr
import "time"
type Result int
const (
Unknown Result = iota
Pass
Fail
Skip
)
func (r Result) String() string {
switch r {
case Unknown:
return "UNKNOWN"
case Pass:
return "PASS"
case Fail:
return "FAIL"
case Skip:
return "SKIP"
default:
panic("invalid Result")
}
}
// TODO: provide some common types, or have a custom type (e.g.
// identifier:type, where identifier is a unique identifier for a particular
// parser.)
// Event is a single event in a test or benchmark.
type Event struct {
Type string
Name string
Result string
Duration time.Duration
Data string
Indent int
// Code coverage
CovPct float64
CovPackages []string
// Benchmarks
Iterations int64
NsPerOp float64
MBPerSec float64
BytesPerOp int64
AllocsPerOp int64
}

View File

@ -3,11 +3,35 @@
package gtr package gtr
import ( import (
"fmt"
"strings" "strings"
"time" "time"
) )
// Result is the result of a test or benchmark.
type Result int
const (
Unknown Result = iota
Pass
Fail
Skip
)
func (r Result) String() string {
switch r {
case Unknown:
return "UNKNOWN"
case Pass:
return "PASS"
case Fail:
return "FAIL"
case Skip:
return "SKIP"
default:
panic("invalid Result")
}
}
// Report contains the build, test and/or benchmark results of a collection of // Report contains the build, test and/or benchmark results of a collection of
// packages. // packages.
type Report struct { type Report struct {
@ -84,39 +108,6 @@ type Error struct {
Output []string Output []string
} }
// FromEvents creates a Report from the given list of events.
// TODO: make packageName optional option
func FromEvents(events []Event, packageName string) Report {
report := NewReportBuilder(packageName)
for _, ev := range events {
switch ev.Type {
case "run_test":
report.CreateTest(ev.Name)
case "pause_test":
report.PauseTest(ev.Name)
case "cont_test":
report.ContinueTest(ev.Name)
case "end_test":
report.EndTest(ev.Name, ev.Result, ev.Duration, ev.Indent)
case "benchmark":
report.Benchmark(ev.Name, ev.Iterations, ev.NsPerOp, ev.MBPerSec, ev.BytesPerOp, ev.AllocsPerOp)
case "status":
report.End()
case "summary":
report.CreatePackage(ev.Name, ev.Result, ev.Duration, ev.Data)
case "coverage":
report.Coverage(ev.CovPct, ev.CovPackages)
case "build_output":
report.CreateBuildError(ev.Name)
case "output":
report.AppendOutput(ev.Data)
default:
fmt.Printf("unhandled event type: %v\n", ev.Type)
}
}
return report.Build()
}
// TrimPrefixSpaces trims the leading whitespace of the given line using the // TrimPrefixSpaces trims the leading whitespace of the given line using the
// indentation level of the test. Printing logs in a Go test is typically // indentation level of the test. Printing logs in a Go test is typically
// prepended by blocks of 4 spaces to align it with the rest of the test // prepended by blocks of 4 spaces to align it with the rest of the test

View File

@ -1,100 +1,26 @@
package gtr package gtr
import ( import "testing"
"testing"
"time"
"github.com/google/go-cmp/cmp" func TestTrimPrefixSpaces(t *testing.T) {
) tests := []struct {
input string
func TestFromEvents(t *testing.T) { indent int
events := []Event{ want string
{Type: "run_test", Name: "TestOne"}, }{
{Type: "output", Data: "\tHello"}, {"", 0, ""},
{Type: "end_test", Name: "TestOne", Result: "PASS", Duration: 1 * time.Millisecond}, {"\ttest", 0, "test"}, // prefixed tabs are always trimmed
{Type: "status", Result: "PASS"}, {" test", 0, "test"},
{Type: "run_test", Name: "TestSkip"}, {" test", 0, " test"},
{Type: "end_test", Name: "TestSkip", Result: "SKIP", Duration: 1 * time.Millisecond}, {" test", 2, "test"},
{Type: "summary", Result: "ok", Name: "package/name", Duration: 1 * time.Millisecond}, {" test", 1, " test"}, // prefix is not a multiple of 4
{Type: "run_test", Name: "TestOne"}, {" \t test", 3, " test"},
{Type: "output", Data: "\tfile_test.go:10: error"},
{Type: "end_test", Name: "TestOne", Result: "FAIL", Duration: 1 * time.Millisecond},
{Type: "status", Result: "FAIL"},
{Type: "summary", Result: "FAIL", Name: "package/name2", Duration: 1 * time.Millisecond},
{Type: "output", Data: "goarch: amd64"},
{Type: "benchmark", Name: "BenchmarkOne", NsPerOp: 100},
{Type: "benchmark", Name: "BenchmarkOne", NsPerOp: 300},
{Type: "status", Result: "PASS"},
{Type: "summary", Result: "ok", Name: "package/name3", Duration: 1234 * time.Millisecond},
{Type: "build_output", Name: "package/failing1"},
{Type: "output", Data: "error message"},
{Type: "summary", Result: "FAIL", Name: "package/failing1", Data: "[build failed]"},
}
expected := Report{
Packages: []Package{
{
Name: "package/name",
Duration: 1 * time.Millisecond,
Tests: []Test{
{
Name: "TestOne",
Duration: 1 * time.Millisecond,
Result: Pass,
Output: []string{
"\tHello", // TODO: strip tabs?
},
},
{
Name: "TestSkip",
Duration: 1 * time.Millisecond,
Result: Skip,
},
},
},
{
Name: "package/name2",
Duration: 1 * time.Millisecond,
Tests: []Test{
{
Name: "TestOne",
Duration: 1 * time.Millisecond,
Result: Fail,
Output: []string{
"\tfile_test.go:10: error",
},
},
},
},
{
Name: "package/name3",
Duration: 1234 * time.Millisecond,
Benchmarks: []Benchmark{
{
Name: "BenchmarkOne",
Result: Pass,
NsPerOp: 100,
},
{
Name: "BenchmarkOne",
Result: Pass,
NsPerOp: 300,
},
},
Output: []string{"goarch: amd64"},
},
{
Name: "package/failing1",
BuildError: Error{
Name: "package/failing1",
Cause: "[build failed]",
Output: []string{"error message"},
},
},
},
} }
actual := FromEvents(events, "") for _, test := range tests {
if diff := cmp.Diff(actual, expected); diff != "" { got := TrimPrefixSpaces(test.input, test.indent)
t.Errorf("FromEvents report incorrect, diff (-got, +want):\n%v", diff) if got != test.want {
t.Errorf("TrimPrefixSpaces(%q, %d) incorrect, got %q, want %q", test.input, test.indent, got, test.want)
}
} }
} }

View File

@ -0,0 +1,25 @@
package gotest
import "time"
// Event is a single event in a test or benchmark.
type Event struct {
Type string
Name string
Result string
Duration time.Duration
Data string
Indent int
// Code coverage
CovPct float64
CovPackages []string
// Benchmarks
Iterations int64
NsPerOp float64
MBPerSec float64
BytesPerOp int64
AllocsPerOp int64
}

View File

@ -59,16 +59,58 @@ func New(options ...Option) *Parser {
type Parser struct { type Parser struct {
packageName string packageName string
events []gtr.Event events []Event
} }
// Parse parses Go test output from the given io.Reader r. // Parse parses Go test output from the given io.Reader r and returns
func (p *Parser) Parse(r io.Reader) ([]gtr.Event, error) { // gtr.Report.
func (p *Parser) Parse(r io.Reader) (gtr.Report, error) {
p.events = nil
s := bufio.NewScanner(r) s := bufio.NewScanner(r)
for s.Scan() { for s.Scan() {
p.parseLine(s.Text()) p.parseLine(s.Text())
} }
return p.events, s.Err() return p.report(p.events), s.Err()
}
// report generates a gtr.Report from the given list of events.
func (p *Parser) report(events []Event) gtr.Report {
rb := gtr.NewReportBuilder()
rb.PackageName = p.packageName
for _, ev := range events {
switch ev.Type {
case "run_test":
rb.CreateTest(ev.Name)
case "pause_test":
rb.PauseTest(ev.Name)
case "cont_test":
rb.ContinueTest(ev.Name)
case "end_test":
rb.EndTest(ev.Name, ev.Result, ev.Duration, ev.Indent)
case "benchmark":
rb.Benchmark(ev.Name, ev.Iterations, ev.NsPerOp, ev.MBPerSec, ev.BytesPerOp, ev.AllocsPerOp)
case "status":
rb.End()
case "summary":
rb.CreatePackage(ev.Name, ev.Result, ev.Duration, ev.Data)
case "coverage":
rb.Coverage(ev.CovPct, ev.CovPackages)
case "build_output":
rb.CreateBuildError(ev.Name)
case "output":
rb.AppendOutput(ev.Data)
default:
fmt.Printf("unhandled event type: %v\n", ev.Type)
}
}
return rb.Build()
}
// Events returns the events created by the parser.
func (p *Parser) Events() []Event {
events := make([]Event, len(p.events))
copy(events, p.events)
return events
} }
func (p *Parser) parseLine(line string) { func (p *Parser) parseLine(line string) {
@ -101,20 +143,20 @@ func (p *Parser) parseLine(line string) {
} }
} }
func (p *Parser) add(event gtr.Event) { func (p *Parser) add(event Event) {
p.events = append(p.events, event) p.events = append(p.events, event)
} }
func (p *Parser) runTest(name string) { func (p *Parser) runTest(name string) {
p.add(gtr.Event{Type: "run_test", Name: name}) p.add(Event{Type: "run_test", Name: name})
} }
func (p *Parser) pauseTest(name string) { func (p *Parser) pauseTest(name string) {
p.add(gtr.Event{Type: "pause_test", Name: name}) p.add(Event{Type: "pause_test", Name: name})
} }
func (p *Parser) contTest(name string) { func (p *Parser) contTest(name string) {
p.add(gtr.Event{Type: "cont_test", Name: name}) p.add(Event{Type: "cont_test", Name: name})
} }
func (p *Parser) endTest(line, indent, result, name, duration string) { func (p *Parser) endTest(line, indent, result, name, duration string) {
@ -122,7 +164,7 @@ func (p *Parser) endTest(line, indent, result, name, duration string) {
p.output(line[:idx]) p.output(line[:idx])
} }
_, n := stripIndent(indent) _, n := stripIndent(indent)
p.add(gtr.Event{ p.add(Event{
Type: "end_test", Type: "end_test",
Name: name, Name: name,
Result: result, Result: result,
@ -132,11 +174,11 @@ func (p *Parser) endTest(line, indent, result, name, duration string) {
} }
func (p *Parser) status(result string) { func (p *Parser) status(result string) {
p.add(gtr.Event{Type: "status", Result: result}) p.add(Event{Type: "status", Result: result})
} }
func (p *Parser) summary(result, name, duration, cached, status, covpct, packages string) { func (p *Parser) summary(result, name, duration, cached, status, covpct, packages string) {
p.add(gtr.Event{ p.add(Event{
Type: "summary", Type: "summary",
Result: result, Result: result,
Name: name, Name: name,
@ -148,7 +190,7 @@ func (p *Parser) summary(result, name, duration, cached, status, covpct, package
} }
func (p *Parser) coverage(percent, packages string) { func (p *Parser) coverage(percent, packages string) {
p.add(gtr.Event{ p.add(Event{
Type: "coverage", Type: "coverage",
CovPct: parseFloat(percent), CovPct: parseFloat(percent),
CovPackages: parsePackages(packages), CovPackages: parsePackages(packages),
@ -156,7 +198,7 @@ func (p *Parser) coverage(percent, packages string) {
} }
func (p *Parser) benchmark(name, iterations, nsPerOp, mbPerSec, bytesPerOp, allocsPerOp string) { func (p *Parser) benchmark(name, iterations, nsPerOp, mbPerSec, bytesPerOp, allocsPerOp string) {
p.add(gtr.Event{ p.add(Event{
Type: "benchmark", Type: "benchmark",
Name: name, Name: name,
Iterations: parseInt(iterations), Iterations: parseInt(iterations),
@ -168,14 +210,14 @@ func (p *Parser) benchmark(name, iterations, nsPerOp, mbPerSec, bytesPerOp, allo
} }
func (p *Parser) buildOutput(packageName string) { func (p *Parser) buildOutput(packageName string) {
p.add(gtr.Event{ p.add(Event{
Type: "build_output", Type: "build_output",
Name: packageName, Name: packageName,
}) })
} }
func (p *Parser) output(line string) { func (p *Parser) output(line string) {
p.add(gtr.Event{Type: "output", Data: line}) p.add(Event{Type: "output", Data: line})
} }
func parseSeconds(s string) time.Duration { func parseSeconds(s string) time.Duration {

View File

@ -6,9 +6,8 @@ import (
"testing" "testing"
"time" "time"
"github.com/jstemmer/go-junit-report/v2/pkg/gtr"
"github.com/google/go-cmp/cmp" "github.com/google/go-cmp/cmp"
"github.com/jstemmer/go-junit-report/v2/pkg/gtr"
) )
type parseLineTest struct { type parseLineTest struct {
@ -19,160 +18,160 @@ type parseLineTest struct {
var parseLineTests = []parseLineTest{ var parseLineTests = []parseLineTest{
{ {
"=== RUN TestOne", "=== RUN TestOne",
gtr.Event{Type: "run_test", Name: "TestOne"}, Event{Type: "run_test", Name: "TestOne"},
}, },
{ {
"=== RUN TestTwo/Subtest", "=== RUN TestTwo/Subtest",
gtr.Event{Type: "run_test", Name: "TestTwo/Subtest"}, Event{Type: "run_test", Name: "TestTwo/Subtest"},
}, },
{ {
"=== PAUSE TestOne", "=== PAUSE TestOne",
gtr.Event{Type: "pause_test", Name: "TestOne"}, Event{Type: "pause_test", Name: "TestOne"},
}, },
{ {
"=== CONT TestOne", "=== CONT TestOne",
gtr.Event{Type: "cont_test", Name: "TestOne"}, Event{Type: "cont_test", Name: "TestOne"},
}, },
{ {
"--- PASS: TestOne (12.34 seconds)", "--- PASS: TestOne (12.34 seconds)",
gtr.Event{Type: "end_test", Name: "TestOne", Result: "PASS", Duration: 12_340 * time.Millisecond}, Event{Type: "end_test", Name: "TestOne", Result: "PASS", Duration: 12_340 * time.Millisecond},
}, },
{ {
" --- SKIP: TestOne/Subtest (0.00s)", " --- SKIP: TestOne/Subtest (0.00s)",
gtr.Event{Type: "end_test", Name: "TestOne/Subtest", Result: "SKIP", Indent: 1}, Event{Type: "end_test", Name: "TestOne/Subtest", Result: "SKIP", Indent: 1},
}, },
{ {
" --- FAIL: TestOne/Subtest/#01 (0.35s)", " --- FAIL: TestOne/Subtest/#01 (0.35s)",
gtr.Event{Type: "end_test", Name: "TestOne/Subtest/#01", Result: "FAIL", Duration: 350 * time.Millisecond, Indent: 2}, Event{Type: "end_test", Name: "TestOne/Subtest/#01", Result: "FAIL", Duration: 350 * time.Millisecond, Indent: 2},
}, },
{ {
"some text--- PASS: TestTwo (0.06 seconds)", "some text--- PASS: TestTwo (0.06 seconds)",
[]gtr.Event{ []Event{
{Type: "output", Data: "some text"}, {Type: "output", Data: "some text"},
{Type: "end_test", Name: "TestTwo", Result: "PASS", Duration: 60 * time.Millisecond}, {Type: "end_test", Name: "TestTwo", Result: "PASS", Duration: 60 * time.Millisecond},
}, },
}, },
{ {
"PASS", "PASS",
gtr.Event{Type: "status", Result: "PASS"}, Event{Type: "status", Result: "PASS"},
}, },
{ {
"FAIL", "FAIL",
gtr.Event{Type: "status", Result: "FAIL"}, Event{Type: "status", Result: "FAIL"},
}, },
{ {
"SKIP", "SKIP",
gtr.Event{Type: "status", Result: "SKIP"}, Event{Type: "status", Result: "SKIP"},
}, },
{ {
"ok package/name/ok 0.100s", "ok package/name/ok 0.100s",
gtr.Event{Type: "summary", Name: "package/name/ok", Result: "ok", Duration: 100 * time.Millisecond}, Event{Type: "summary", Name: "package/name/ok", Result: "ok", Duration: 100 * time.Millisecond},
}, },
{ {
"FAIL package/name/failing [build failed]", "FAIL package/name/failing [build failed]",
gtr.Event{Type: "summary", Name: "package/name/failing", Result: "FAIL", Data: "[build failed]"}, Event{Type: "summary", Name: "package/name/failing", Result: "FAIL", Data: "[build failed]"},
}, },
{ {
"FAIL package/other/failing [setup failed]", "FAIL package/other/failing [setup failed]",
gtr.Event{Type: "summary", Name: "package/other/failing", Result: "FAIL", Data: "[setup failed]"}, Event{Type: "summary", Name: "package/other/failing", Result: "FAIL", Data: "[setup failed]"},
}, },
{ {
"ok package/other (cached)", "ok package/other (cached)",
gtr.Event{Type: "summary", Name: "package/other", Result: "ok", Data: "(cached)"}, Event{Type: "summary", Name: "package/other", Result: "ok", Data: "(cached)"},
}, },
{ {
"ok package/name 0.400s coverage: 10.0% of statements", "ok package/name 0.400s coverage: 10.0% of statements",
gtr.Event{Type: "summary", Name: "package/name", Result: "ok", Duration: 400 * time.Millisecond, CovPct: 10}, Event{Type: "summary", Name: "package/name", Result: "ok", Duration: 400 * time.Millisecond, CovPct: 10},
}, },
{ {
"ok package/name 4.200s coverage: 99.8% of statements in fmt, encoding/xml", "ok package/name 4.200s coverage: 99.8% of statements in fmt, encoding/xml",
gtr.Event{Type: "summary", Name: "package/name", Result: "ok", Duration: 4200 * time.Millisecond, CovPct: 99.8, CovPackages: []string{"fmt", "encoding/xml"}}, Event{Type: "summary", Name: "package/name", Result: "ok", Duration: 4200 * time.Millisecond, CovPct: 99.8, CovPackages: []string{"fmt", "encoding/xml"}},
}, },
{ {
"? package/name [no test files]", "? package/name [no test files]",
gtr.Event{Type: "summary", Name: "package/name", Result: "?", Data: "[no test files]"}, Event{Type: "summary", Name: "package/name", Result: "?", Data: "[no test files]"},
}, },
{ {
"ok package/name 0.001s [no tests to run]", "ok package/name 0.001s [no tests to run]",
gtr.Event{Type: "summary", Name: "package/name", Result: "ok", Duration: 1 * time.Millisecond, Data: "[no tests to run]"}, Event{Type: "summary", Name: "package/name", Result: "ok", Duration: 1 * time.Millisecond, Data: "[no tests to run]"},
}, },
{ {
"ok package/name (cached) [no tests to run]", "ok package/name (cached) [no tests to run]",
gtr.Event{Type: "summary", Name: "package/name", Result: "ok", Data: "(cached) [no tests to run]"}, Event{Type: "summary", Name: "package/name", Result: "ok", Data: "(cached) [no tests to run]"},
}, },
{ {
"coverage: 10% of statements", "coverage: 10% of statements",
gtr.Event{Type: "coverage", CovPct: 10}, Event{Type: "coverage", CovPct: 10},
}, },
{ {
"coverage: 10% of statements in fmt, encoding/xml", "coverage: 10% of statements in fmt, encoding/xml",
gtr.Event{Type: "coverage", CovPct: 10, CovPackages: []string{"fmt", "encoding/xml"}}, Event{Type: "coverage", CovPct: 10, CovPackages: []string{"fmt", "encoding/xml"}},
}, },
{ {
"coverage: 13.37% of statements", "coverage: 13.37% of statements",
gtr.Event{Type: "coverage", CovPct: 13.37}, Event{Type: "coverage", CovPct: 13.37},
}, },
{ {
"coverage: 99.8% of statements in fmt, encoding/xml", "coverage: 99.8% of statements in fmt, encoding/xml",
gtr.Event{Type: "coverage", CovPct: 99.8, CovPackages: []string{"fmt", "encoding/xml"}}, Event{Type: "coverage", CovPct: 99.8, CovPackages: []string{"fmt", "encoding/xml"}},
}, },
{ {
"BenchmarkOne-8 2000000 604 ns/op", "BenchmarkOne-8 2000000 604 ns/op",
gtr.Event{Type: "benchmark", Name: "BenchmarkOne", Iterations: 2_000_000, NsPerOp: 604}, Event{Type: "benchmark", Name: "BenchmarkOne", Iterations: 2_000_000, NsPerOp: 604},
}, },
{ {
"BenchmarkTwo-16 30000 52568 ns/op 24879 B/op 494 allocs/op", "BenchmarkTwo-16 30000 52568 ns/op 24879 B/op 494 allocs/op",
gtr.Event{Type: "benchmark", Name: "BenchmarkTwo", Iterations: 30_000, NsPerOp: 52_568, BytesPerOp: 24_879, AllocsPerOp: 494}, Event{Type: "benchmark", Name: "BenchmarkTwo", Iterations: 30_000, NsPerOp: 52_568, BytesPerOp: 24_879, AllocsPerOp: 494},
}, },
{ {
"BenchmarkThree 2000000000 0.26 ns/op", "BenchmarkThree 2000000000 0.26 ns/op",
gtr.Event{Type: "benchmark", Name: "BenchmarkThree", Iterations: 2_000_000_000, NsPerOp: 0.26}, Event{Type: "benchmark", Name: "BenchmarkThree", Iterations: 2_000_000_000, NsPerOp: 0.26},
}, },
{ {
"BenchmarkFour-8 10000 104427 ns/op 95.76 MB/s 40629 B/op 5 allocs/op", "BenchmarkFour-8 10000 104427 ns/op 95.76 MB/s 40629 B/op 5 allocs/op",
gtr.Event{Type: "benchmark", Name: "BenchmarkFour", Iterations: 10_000, NsPerOp: 104_427, MBPerSec: 95.76, BytesPerOp: 40_629, AllocsPerOp: 5}, Event{Type: "benchmark", Name: "BenchmarkFour", Iterations: 10_000, NsPerOp: 104_427, MBPerSec: 95.76, BytesPerOp: 40_629, AllocsPerOp: 5},
}, },
{ {
"# package/name/failing1", "# package/name/failing1",
gtr.Event{Type: "build_output", Name: "package/name/failing1"}, Event{Type: "build_output", Name: "package/name/failing1"},
}, },
{ {
"# package/name/failing2 [package/name/failing2.test]", "# package/name/failing2 [package/name/failing2.test]",
gtr.Event{Type: "build_output", Name: "package/name/failing2"}, Event{Type: "build_output", Name: "package/name/failing2"},
}, },
{ {
"single line stdout", "single line stdout",
gtr.Event{Type: "output", Data: "single line stdout"}, Event{Type: "output", Data: "single line stdout"},
}, },
{ {
"# some more output", "# some more output",
gtr.Event{Type: "output", Data: "# some more output"}, Event{Type: "output", Data: "# some more output"},
}, },
{ {
"\tfile_test.go:11: Error message", "\tfile_test.go:11: Error message",
gtr.Event{Type: "output", Data: "\tfile_test.go:11: Error message"}, Event{Type: "output", Data: "\tfile_test.go:11: Error message"},
}, },
{ {
"\tfile_test.go:12: Longer", "\tfile_test.go:12: Longer",
gtr.Event{Type: "output", Data: "\tfile_test.go:12: Longer"}, Event{Type: "output", Data: "\tfile_test.go:12: Longer"},
}, },
{ {
"\t\terror", "\t\terror",
gtr.Event{Type: "output", Data: "\t\terror"}, Event{Type: "output", Data: "\t\terror"},
}, },
{ {
"\t\tmessage.", "\t\tmessage.",
gtr.Event{Type: "output", Data: "\t\tmessage."}, Event{Type: "output", Data: "\t\tmessage."},
}, },
} }
func TestParseLine(t *testing.T) { func TestParseLine(t *testing.T) {
for i, test := range parseLineTests { for i, test := range parseLineTests {
var want []gtr.Event var want []Event
switch e := test.events.(type) { switch e := test.events.(type) {
case gtr.Event: case Event:
want = []gtr.Event{e} want = []Event{e}
case []gtr.Event: case []Event:
want = e want = e
default: default:
panic("invalid events type") panic("invalid events type")
@ -194,3 +193,96 @@ func TestParseLine(t *testing.T) {
}) })
} }
} }
func TestReport(t *testing.T) {
events := []Event{
{Type: "run_test", Name: "TestOne"},
{Type: "output", Data: "\tHello"},
{Type: "end_test", Name: "TestOne", Result: "PASS", Duration: 1 * time.Millisecond},
{Type: "status", Result: "PASS"},
{Type: "run_test", Name: "TestSkip"},
{Type: "end_test", Name: "TestSkip", Result: "SKIP", Duration: 1 * time.Millisecond},
{Type: "summary", Result: "ok", Name: "package/name", Duration: 1 * time.Millisecond},
{Type: "run_test", Name: "TestOne"},
{Type: "output", Data: "\tfile_test.go:10: error"},
{Type: "end_test", Name: "TestOne", Result: "FAIL", Duration: 1 * time.Millisecond},
{Type: "status", Result: "FAIL"},
{Type: "summary", Result: "FAIL", Name: "package/name2", Duration: 1 * time.Millisecond},
{Type: "output", Data: "goarch: amd64"},
{Type: "benchmark", Name: "BenchmarkOne", NsPerOp: 100},
{Type: "benchmark", Name: "BenchmarkOne", NsPerOp: 300},
{Type: "status", Result: "PASS"},
{Type: "summary", Result: "ok", Name: "package/name3", Duration: 1234 * time.Millisecond},
{Type: "build_output", Name: "package/failing1"},
{Type: "output", Data: "error message"},
{Type: "summary", Result: "FAIL", Name: "package/failing1", Data: "[build failed]"},
}
expected := gtr.Report{
Packages: []gtr.Package{
{
Name: "package/name",
Duration: 1 * time.Millisecond,
Tests: []gtr.Test{
{
Name: "TestOne",
Duration: 1 * time.Millisecond,
Result: gtr.Pass,
Output: []string{
"\tHello", // TODO: strip tabs?
},
},
{
Name: "TestSkip",
Duration: 1 * time.Millisecond,
Result: gtr.Skip,
},
},
},
{
Name: "package/name2",
Duration: 1 * time.Millisecond,
Tests: []gtr.Test{
{
Name: "TestOne",
Duration: 1 * time.Millisecond,
Result: gtr.Fail,
Output: []string{
"\tfile_test.go:10: error",
},
},
},
},
{
Name: "package/name3",
Duration: 1234 * time.Millisecond,
Benchmarks: []gtr.Benchmark{
{
Name: "BenchmarkOne",
Result: gtr.Pass,
NsPerOp: 100,
},
{
Name: "BenchmarkOne",
Result: gtr.Pass,
NsPerOp: 300,
},
},
Output: []string{"goarch: amd64"},
},
{
Name: "package/failing1",
BuildError: gtr.Error{
Name: "package/failing1",
Cause: "[build failed]",
Output: []string{"error message"},
},
},
},
}
parser := &Parser{}
actual := parser.report(events)
if diff := cmp.Diff(actual, expected); diff != "" {
t.Errorf("FromEvents report incorrect, diff (-got, +want):\n%v", diff)
}
}