mirror of
https://github.com/jstemmer/go-junit-report.git
synced 2025-04-05 05:00:15 -05:00
gtr: Add benchmarks to Report
This commit is contained in:
parent
050e22e86b
commit
a100cfaa5f
@ -7,8 +7,9 @@ import (
|
|||||||
|
|
||||||
// ReportBuilder builds Reports.
|
// ReportBuilder builds Reports.
|
||||||
type ReportBuilder struct {
|
type ReportBuilder struct {
|
||||||
packages []Package
|
packages []Package
|
||||||
tests map[int]Test
|
tests map[int]Test
|
||||||
|
benchmarks map[int]Benchmark
|
||||||
|
|
||||||
// state
|
// state
|
||||||
nextId int // next free id
|
nextId int // next free id
|
||||||
@ -18,8 +19,9 @@ type ReportBuilder struct {
|
|||||||
|
|
||||||
func NewReportBuilder() *ReportBuilder {
|
func NewReportBuilder() *ReportBuilder {
|
||||||
return &ReportBuilder{
|
return &ReportBuilder{
|
||||||
tests: make(map[int]Test),
|
tests: make(map[int]Test),
|
||||||
nextId: 1,
|
benchmarks: make(map[int]Benchmark),
|
||||||
|
nextId: 1,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -52,19 +54,46 @@ func (b *ReportBuilder) EndTest(name, result string, duration time.Duration) {
|
|||||||
b.tests[id] = t
|
b.tests[id] = t
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (b *ReportBuilder) Benchmark(name string, iterations int64, nsPerOp, mbPerSec float64, bytesPerOp, allocsPerOp int64) {
|
||||||
|
id := b.nextId
|
||||||
|
b.lastId = id
|
||||||
|
|
||||||
|
b.nextId += 1
|
||||||
|
b.benchmarks[id] = Benchmark{
|
||||||
|
Name: name,
|
||||||
|
Result: Pass,
|
||||||
|
Iterations: iterations,
|
||||||
|
NsPerOp: nsPerOp,
|
||||||
|
MBPerSec: mbPerSec,
|
||||||
|
BytesPerOp: bytesPerOp,
|
||||||
|
AllocsPerOp: allocsPerOp,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (b *ReportBuilder) CreatePackage(name string, duration time.Duration) {
|
func (b *ReportBuilder) CreatePackage(name string, duration time.Duration) {
|
||||||
var tests []Test
|
var tests []Test
|
||||||
|
var benchmarks []Benchmark
|
||||||
|
|
||||||
|
// Iterate by id to maintain original test order
|
||||||
for id := 1; id < b.nextId; id++ {
|
for id := 1; id < b.nextId; id++ {
|
||||||
tests = append(tests, b.tests[id])
|
if t, ok := b.tests[id]; ok {
|
||||||
|
tests = append(tests, t)
|
||||||
|
}
|
||||||
|
if bm, ok := b.benchmarks[id]; ok {
|
||||||
|
benchmarks = append(benchmarks, bm)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
b.packages = append(b.packages, Package{
|
b.packages = append(b.packages, Package{
|
||||||
Name: name,
|
Name: name,
|
||||||
Duration: duration,
|
Duration: duration,
|
||||||
Tests: tests,
|
Tests: tests,
|
||||||
Output: b.output,
|
Benchmarks: benchmarks,
|
||||||
|
Output: b.output,
|
||||||
})
|
})
|
||||||
|
|
||||||
b.tests = make(map[int]Test)
|
b.tests = make(map[int]Test)
|
||||||
|
b.benchmarks = make(map[int]Benchmark)
|
||||||
b.output = nil
|
b.output = nil
|
||||||
b.nextId = 1
|
b.nextId = 1
|
||||||
b.lastId = 0
|
b.lastId = 0
|
||||||
@ -93,6 +122,19 @@ func (b *ReportBuilder) findTest(name string) int {
|
|||||||
return -1
|
return -1
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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
|
||||||
|
}
|
||||||
|
|
||||||
func parseResult(r string) Result {
|
func parseResult(r string) Result {
|
||||||
switch r {
|
switch r {
|
||||||
case "PASS":
|
case "PASS":
|
||||||
|
@ -10,6 +10,11 @@ import (
|
|||||||
"github.com/jstemmer/go-junit-report/v2/pkg/junit"
|
"github.com/jstemmer/go-junit-report/v2/pkg/junit"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
propPrefixes = map[string]bool{"goos": true, "goarch": true, "pkg": true}
|
||||||
|
propFieldsFunc = func(r rune) bool { return r == ':' || r == ' ' }
|
||||||
|
)
|
||||||
|
|
||||||
type Report struct {
|
type Report struct {
|
||||||
Packages []Package
|
Packages []Package
|
||||||
}
|
}
|
||||||
@ -31,7 +36,8 @@ type Package struct {
|
|||||||
Coverage float64
|
Coverage float64
|
||||||
Output []string
|
Output []string
|
||||||
|
|
||||||
Tests []Test
|
Tests []Test
|
||||||
|
Benchmarks []Benchmark
|
||||||
}
|
}
|
||||||
|
|
||||||
type Test struct {
|
type Test struct {
|
||||||
@ -41,6 +47,17 @@ type Test struct {
|
|||||||
Output []string
|
Output []string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type Benchmark struct {
|
||||||
|
Name string
|
||||||
|
Result Result
|
||||||
|
Output []string
|
||||||
|
Iterations int64
|
||||||
|
NsPerOp float64
|
||||||
|
MBPerSec float64
|
||||||
|
BytesPerOp int64
|
||||||
|
AllocsPerOp int64
|
||||||
|
}
|
||||||
|
|
||||||
// FromEvents creates a Report from the given list of events.
|
// FromEvents creates a Report from the given list of events.
|
||||||
func FromEvents(events []Event) Report {
|
func FromEvents(events []Event) Report {
|
||||||
report := NewReportBuilder()
|
report := NewReportBuilder()
|
||||||
@ -50,6 +67,8 @@ func FromEvents(events []Event) Report {
|
|||||||
report.CreateTest(ev.Name)
|
report.CreateTest(ev.Name)
|
||||||
case "end_test":
|
case "end_test":
|
||||||
report.EndTest(ev.Name, ev.Result, ev.Duration)
|
report.EndTest(ev.Name, ev.Result, ev.Duration)
|
||||||
|
case "benchmark":
|
||||||
|
report.Benchmark(ev.Name, ev.Iterations, ev.NsPerOp, ev.MBPerSec, ev.BytesPerOp, ev.AllocsPerOp)
|
||||||
case "status": // ignore for now
|
case "status": // ignore for now
|
||||||
case "summary":
|
case "summary":
|
||||||
report.CreatePackage(ev.Name, ev.Duration)
|
report.CreatePackage(ev.Name, ev.Duration)
|
||||||
@ -72,6 +91,12 @@ func JUnit(report Report) junit.Testsuites {
|
|||||||
Time: junit.FormatDuration(pkg.Duration),
|
Time: junit.FormatDuration(pkg.Duration),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for _, line := range pkg.Output {
|
||||||
|
if fields := strings.FieldsFunc(line, propFieldsFunc); len(fields) == 2 && propPrefixes[fields[0]] {
|
||||||
|
suite.AddProperty(fields[0], fields[1])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
for _, test := range pkg.Tests {
|
for _, test := range pkg.Tests {
|
||||||
tc := junit.Testcase{
|
tc := junit.Testcase{
|
||||||
Classname: pkg.Name,
|
Classname: pkg.Name,
|
||||||
@ -90,7 +115,50 @@ func JUnit(report Report) junit.Testsuites {
|
|||||||
}
|
}
|
||||||
suite.AddTestcase(tc)
|
suite.AddTestcase(tc)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for _, bm := range mergeBenchmarks(pkg.Benchmarks) {
|
||||||
|
tc := junit.Testcase{
|
||||||
|
Classname: pkg.Name,
|
||||||
|
Name: bm.Name,
|
||||||
|
Time: junit.FormatBenchmarkTime(time.Duration(bm.NsPerOp)),
|
||||||
|
}
|
||||||
|
if bm.Result == Fail {
|
||||||
|
tc.Failure = &junit.Result{
|
||||||
|
Message: "Failed",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
suite.AddTestcase(tc)
|
||||||
|
}
|
||||||
|
|
||||||
suites.AddSuite(suite)
|
suites.AddSuite(suite)
|
||||||
}
|
}
|
||||||
return suites
|
return suites
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func mergeBenchmarks(benchmarks []Benchmark) []Benchmark {
|
||||||
|
var merged []Benchmark
|
||||||
|
|
||||||
|
benchmap := make(map[string][]Benchmark)
|
||||||
|
for _, bm := range benchmarks {
|
||||||
|
if _, ok := benchmap[bm.Name]; !ok {
|
||||||
|
merged = append(merged, Benchmark{Name: bm.Name})
|
||||||
|
}
|
||||||
|
benchmap[bm.Name] = append(benchmap[bm.Name], bm)
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, bm := range merged {
|
||||||
|
for _, b := range benchmap[bm.Name] {
|
||||||
|
bm.NsPerOp += b.NsPerOp
|
||||||
|
bm.MBPerSec += b.MBPerSec
|
||||||
|
bm.BytesPerOp += b.BytesPerOp
|
||||||
|
bm.AllocsPerOp += b.AllocsPerOp
|
||||||
|
}
|
||||||
|
n := len(benchmap[bm.Name])
|
||||||
|
merged[i].NsPerOp = bm.NsPerOp / float64(n)
|
||||||
|
merged[i].MBPerSec = bm.MBPerSec / float64(n)
|
||||||
|
merged[i].BytesPerOp = bm.BytesPerOp / int64(n)
|
||||||
|
merged[i].AllocsPerOp = bm.AllocsPerOp / int64(n)
|
||||||
|
}
|
||||||
|
|
||||||
|
return merged
|
||||||
|
}
|
||||||
|
@ -21,6 +21,11 @@ func TestFromEvents(t *testing.T) {
|
|||||||
{Type: "end_test", Name: "TestOne", Result: "FAIL", Duration: 1 * time.Millisecond},
|
{Type: "end_test", Name: "TestOne", Result: "FAIL", Duration: 1 * time.Millisecond},
|
||||||
{Type: "status", Result: "FAIL"},
|
{Type: "status", Result: "FAIL"},
|
||||||
{Type: "summary", Result: "FAIL", Name: "package/name2", Duration: 1 * time.Millisecond},
|
{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},
|
||||||
}
|
}
|
||||||
expected := Report{
|
expected := Report{
|
||||||
Packages: []Package{
|
Packages: []Package{
|
||||||
@ -31,7 +36,7 @@ func TestFromEvents(t *testing.T) {
|
|||||||
{
|
{
|
||||||
Name: "TestOne",
|
Name: "TestOne",
|
||||||
Duration: 1 * time.Millisecond,
|
Duration: 1 * time.Millisecond,
|
||||||
Result: PASS,
|
Result: Pass,
|
||||||
Output: []string{
|
Output: []string{
|
||||||
"\tHello", // TODO: strip tabs?
|
"\tHello", // TODO: strip tabs?
|
||||||
},
|
},
|
||||||
@ -39,7 +44,7 @@ func TestFromEvents(t *testing.T) {
|
|||||||
{
|
{
|
||||||
Name: "TestSkip",
|
Name: "TestSkip",
|
||||||
Duration: 1 * time.Millisecond,
|
Duration: 1 * time.Millisecond,
|
||||||
Result: SKIP,
|
Result: Skip,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@ -50,13 +55,30 @@ func TestFromEvents(t *testing.T) {
|
|||||||
{
|
{
|
||||||
Name: "TestOne",
|
Name: "TestOne",
|
||||||
Duration: 1 * time.Millisecond,
|
Duration: 1 * time.Millisecond,
|
||||||
Result: FAIL,
|
Result: Fail,
|
||||||
Output: []string{
|
Output: []string{
|
||||||
"\tfile_test.go:10: error",
|
"\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"},
|
||||||
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user