mirror of
https://github.com/jstemmer/go-junit-report.git
synced 2025-04-05 13:08:07 -05:00
gtr,parser/gotest: Improve benchmark output matching
This commit is contained in:
parent
a52c1b921d
commit
c03c92d418
@ -43,4 +43,10 @@ type Event struct {
|
|||||||
// Code coverage
|
// Code coverage
|
||||||
CovPct float64
|
CovPct float64
|
||||||
CovPackages []string
|
CovPackages []string
|
||||||
|
|
||||||
|
// Benchmarks
|
||||||
|
Iterations int64
|
||||||
|
NsPerOp int64
|
||||||
|
BytesPerOp int64
|
||||||
|
AllocsPerOp int64
|
||||||
}
|
}
|
||||||
|
@ -14,7 +14,8 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
regexBenchmark = regexp.MustCompile(`^(Benchmark[^\s]+)\s(\d+|\d+\.\d+)\s((?:[^\s]+\s[^\s]+)+)`)
|
// regexBenchmark captures 3-5 groups: benchmark name, number of times ran, ns/op (with or without decimal), B/op (optional), and allocs/op (optional).
|
||||||
|
regexBenchmark = regexp.MustCompile(`^(Benchmark[^ -]+)(?:-\d+\s+|\s+)(\d+)\s+(\d+|\d+\.\d+)\sns\/op(?:\s+(\d+)\sB\/op)?(?:\s+(\d+)\\sallocs/op)?`)
|
||||||
regexCoverage = regexp.MustCompile(`^coverage:\s+(\d+|\d+\.\d+)%\s+of\s+statements(?:\sin\s(.+))?$`)
|
regexCoverage = regexp.MustCompile(`^coverage:\s+(\d+|\d+\.\d+)%\s+of\s+statements(?:\sin\s(.+))?$`)
|
||||||
regexEndTest = regexp.MustCompile(`((?: )*)--- (PASS|FAIL|SKIP): ([^ ]+) \((\d+\.\d+)(?: seconds|s)\)`)
|
regexEndTest = regexp.MustCompile(`((?: )*)--- (PASS|FAIL|SKIP): ([^ ]+) \((\d+\.\d+)(?: seconds|s)\)`)
|
||||||
regexStatus = regexp.MustCompile(`^(PASS|FAIL|SKIP)$`)
|
regexStatus = regexp.MustCompile(`^(PASS|FAIL|SKIP)$`)
|
||||||
@ -29,11 +30,7 @@ func Parse(r io.Reader) ([]gtr.Event, error) {
|
|||||||
for s.Scan() {
|
for s.Scan() {
|
||||||
p.parseLine(s.Text())
|
p.parseLine(s.Text())
|
||||||
}
|
}
|
||||||
if s.Err() != nil {
|
return p.events, s.Err()
|
||||||
return nil, s.Err()
|
|
||||||
}
|
|
||||||
|
|
||||||
return p.events, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type parser struct {
|
type parser struct {
|
||||||
@ -55,10 +52,8 @@ func (p *parser) parseLine(line string) {
|
|||||||
p.summary(matches[1], matches[2], matches[3], matches[4], matches[5], matches[6])
|
p.summary(matches[1], matches[2], matches[3], matches[4], matches[5], matches[6])
|
||||||
} else if matches := regexCoverage.FindStringSubmatch(line); len(matches) == 3 {
|
} else if matches := regexCoverage.FindStringSubmatch(line); len(matches) == 3 {
|
||||||
p.coverage(matches[1], matches[2])
|
p.coverage(matches[1], matches[2])
|
||||||
} else if matches := regexBenchmark.FindStringSubmatch(line); len(matches) == 4 {
|
} else if matches := regexBenchmark.FindStringSubmatch(line); len(matches) == 6 {
|
||||||
fields := strings.Fields(matches[3])
|
p.benchmark(matches[1], matches[2], matches[3], matches[4], matches[5])
|
||||||
//p.benchmark(matches[1], matches[2], fields)
|
|
||||||
p.benchmark(fields)
|
|
||||||
} else {
|
} else {
|
||||||
p.output(line)
|
p.output(line)
|
||||||
}
|
}
|
||||||
@ -118,10 +113,14 @@ func (p *parser) coverage(percent, packages string) {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *parser) benchmark(fields []string) {
|
func (p *parser) benchmark(name, iterations, timePerOp, bytesPerOp, allocsPerOp string) {
|
||||||
p.add(gtr.Event{
|
p.add(gtr.Event{
|
||||||
Type: "benchmark",
|
Type: "benchmark",
|
||||||
Name: fields[0],
|
Name: name,
|
||||||
|
Iterations: parseInt(iterations),
|
||||||
|
NsPerOp: parseInt(timePerOp),
|
||||||
|
BytesPerOp: parseInt(bytesPerOp),
|
||||||
|
AllocsPerOp: parseInt(allocsPerOp),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -154,6 +153,12 @@ func parsePackages(pkgList string) []string {
|
|||||||
return strings.Split(pkgList, ", ")
|
return strings.Split(pkgList, ", ")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func parseInt(s string) int64 {
|
||||||
|
// ignore error
|
||||||
|
n, _ := strconv.ParseInt(s, 10, 64)
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
func stripIndent(line string) (string, int) {
|
func stripIndent(line string) (string, int) {
|
||||||
var indent int
|
var indent int
|
||||||
for indent = 0; strings.HasPrefix(line, " "); indent++ {
|
for indent = 0; strings.HasPrefix(line, " "); indent++ {
|
||||||
|
@ -1,8 +1,10 @@
|
|||||||
package gotest
|
package gotest
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"flag"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
"regexp"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
@ -13,6 +15,8 @@ import (
|
|||||||
|
|
||||||
const testdataRoot = "../../../testdata/"
|
const testdataRoot = "../../../testdata/"
|
||||||
|
|
||||||
|
var matchTest = flag.String("match", "", "only test testdata matching this pattern")
|
||||||
|
|
||||||
var tests = []struct {
|
var tests = []struct {
|
||||||
in string
|
in string
|
||||||
expected []gtr.Event
|
expected []gtr.Event
|
||||||
@ -328,23 +332,116 @@ var tests = []struct {
|
|||||||
{Type: "status", Result: "PASS"},
|
{Type: "status", Result: "PASS"},
|
||||||
{Type: "summary", Result: "ok", Name: "package/one", Data: "(cached)"},
|
{Type: "summary", Result: "ok", Name: "package/one", Data: "(cached)"},
|
||||||
}},
|
}},
|
||||||
{"22-whitespace",
|
{"22-bench",
|
||||||
[]gtr.Event{}},
|
[]gtr.Event{
|
||||||
|
{Type: "output", Data: "goos: darwin"},
|
||||||
|
{Type: "output", Data: "goarch: amd64"},
|
||||||
|
{Type: "output", Data: "pkg: code.internal/state"},
|
||||||
|
{Type: "benchmark", Name: "BenchmarkParse", Iterations: 2000000, NsPerOp: 604},
|
||||||
|
{Type: "benchmark", Name: "BenchmarkReadingList", Iterations: 1000000, NsPerOp: 1425},
|
||||||
|
{Type: "status", Result: "PASS"},
|
||||||
|
{Type: "summary", Result: "ok", Name: "package/basic", Duration: 3212 * time.Millisecond},
|
||||||
|
}},
|
||||||
|
{"23-benchmem", []gtr.Event{}},
|
||||||
|
{"24-benchtests", []gtr.Event{}},
|
||||||
|
{"25-benchcount", []gtr.Event{}},
|
||||||
|
{"26-testbenchmultiple", []gtr.Event{}},
|
||||||
|
{"27-benchdecimal", []gtr.Event{}},
|
||||||
|
{"28-bench-1cpu", []gtr.Event{}},
|
||||||
|
{"29-bench-16cpu", []gtr.Event{}},
|
||||||
|
{"30-stdout", []gtr.Event{}},
|
||||||
|
{"31-syntax-error-test-binary", []gtr.Event{}},
|
||||||
|
{"32-failed-summary", []gtr.Event{}},
|
||||||
|
{"130-bench-mb", []gtr.Event{}},
|
||||||
|
{"131-whitespace",
|
||||||
|
[]gtr.Event{
|
||||||
|
{Type: "run_test", Name: "TestFlat"},
|
||||||
|
{Type: "output", Data: "printf 1"},
|
||||||
|
{Type: "output", Data: "printf 2"},
|
||||||
|
{Type: "end_test", Name: "TestFlat", Result: "PASS"},
|
||||||
|
{Type: "output", Data: "\twhitespace_test.go:9: log 1"},
|
||||||
|
{Type: "output", Data: "\twhitespace_test.go:10: log 2"},
|
||||||
|
{Type: "run_test", Name: "TestWithSpace"},
|
||||||
|
{Type: "output", Data: "no-space"},
|
||||||
|
{Type: "output", Data: " one-space"},
|
||||||
|
{Type: "output", Data: " two-space"},
|
||||||
|
{Type: "output", Data: " four-space"},
|
||||||
|
{Type: "output", Data: " eight-space"},
|
||||||
|
{Type: "output", Data: "no-space"},
|
||||||
|
{Type: "end_test", Name: "TestWithSpace", Result: "PASS"},
|
||||||
|
{Type: "output", Data: "\twhitespace_test.go:16: no-space"},
|
||||||
|
{Type: "output", Data: "\twhitespace_test.go:17: one-space"},
|
||||||
|
{Type: "output", Data: "\twhitespace_test.go:18: two-space"},
|
||||||
|
{Type: "output", Data: "\twhitespace_test.go:19: four-space"},
|
||||||
|
{Type: "output", Data: "\twhitespace_test.go:20: eight-space"},
|
||||||
|
{Type: "output", Data: "\twhitespace_test.go:21: no-space"},
|
||||||
|
{Type: "run_test", Name: "TestWithTab"},
|
||||||
|
{Type: "output", Data: "no-tab"},
|
||||||
|
{Type: "output", Data: "\tone-tab"},
|
||||||
|
{Type: "output", Data: "\ttwo-tab"},
|
||||||
|
{Type: "end_test", Name: "TestWithTab", Result: "PASS"},
|
||||||
|
{Type: "output", Data: "\twhitespace_test.go:31: no-tab"},
|
||||||
|
{Type: "output", Data: "\twhitespace_test.go:32: \tone-tab"},
|
||||||
|
{Type: "output", Data: "\twhitespace_test.go:33: \t\ttwo-tab"},
|
||||||
|
{Type: "run_test", Name: "TestWithNewlinesFlat"},
|
||||||
|
{Type: "output", Data: "no-newline"},
|
||||||
|
{Type: "output", Data: "one-newline"},
|
||||||
|
{Type: "output", Data: "one-newline"},
|
||||||
|
{Type: "output", Data: "two-newlines"},
|
||||||
|
{Type: "output", Data: "two-newlines"},
|
||||||
|
{Type: "output", Data: "two-newlines"},
|
||||||
|
{Type: "end_test", Name: "TestWithNewlinesFlat", Result: "PASS"},
|
||||||
|
{Type: "output", Data: " whitespace_test.go:40: no-newline"},
|
||||||
|
{Type: "output", Data: " whitespace_test.go:41: one-newline"},
|
||||||
|
{Type: "output", Data: " one-newline"},
|
||||||
|
{Type: "output", Data: " whitespace_test.go:42: two-newlines"},
|
||||||
|
{Type: "output", Data: " two-newlines"},
|
||||||
|
{Type: "output", Data: " two-newlines"},
|
||||||
|
{Type: "run_test", Name: "TestSubTests"},
|
||||||
|
{Type: "run_test", Name: "TestSubTests/TestFlat"},
|
||||||
|
{Type: "output", Data: "printf 1"},
|
||||||
|
{Type: "output", Data: "printf 2"},
|
||||||
|
{Type: "run_test", Name: "TestSubTests/TestWithSpace"},
|
||||||
|
{Type: "output", Data: "no-space"},
|
||||||
|
{Type: "output", Data: " one-space"},
|
||||||
|
{Type: "output", Data: " two-space"},
|
||||||
|
{Type: "output", Data: " four-space"},
|
||||||
|
{Type: "output", Data: " eight-space"},
|
||||||
|
{Type: "output", Data: "no-space"},
|
||||||
|
{Type: "run_test", Name: "TestSubTests/TestWithTab"},
|
||||||
|
{Type: "output", Data: "no-tab"},
|
||||||
|
{Type: "output", Data: " one-tab"},
|
||||||
|
{Type: "output", Data: " two-tab"},
|
||||||
|
{Type: "run_test", Name: "TestSubTests/TestWithNewlinesFlat"},
|
||||||
|
{Type: "output", Data: "no-newline"},
|
||||||
|
{Type: "output", Data: "one-newline"},
|
||||||
|
{Type: "output", Data: "one-newline"},
|
||||||
|
{Type: "output", Data: "two-newlines"},
|
||||||
|
{Type: "output", Data: "two-newlines"},
|
||||||
|
{Type: "output", Data: "two-newlines"},
|
||||||
|
{Type: "end_test", Name: "TestSubTests", Result: "PASS"},
|
||||||
|
{Type: "end_test", Name: "TestSubTests/TestFlat", Result: "PASS", Indent: 1},
|
||||||
|
{Type: "output", Data: " whitespace_test.go:9: log 1"},
|
||||||
|
{Type: "output", Data: " whitespace_test.go:10: log 2"},
|
||||||
|
{Type: "end_test", Name: "TestSubTests/TestWithSpace", Result: "PASS", Indent: 1},
|
||||||
|
{Type: "output", Data: " whitespace_test.go:16: no-space"},
|
||||||
|
{Type: "output", Data: " whitespace_test.go:17: one-space"},
|
||||||
|
}},
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestParse(t *testing.T) {
|
func TestParse(t *testing.T) {
|
||||||
|
matchRegex := compileMatch(t)
|
||||||
for _, test := range tests {
|
for _, test := range tests {
|
||||||
t.Run(test.in, func(t *testing.T) {
|
t.Run(test.in, func(t *testing.T) {
|
||||||
|
if !matchRegex.MatchString(test.in) || len(test.expected) == 0 {
|
||||||
|
t.SkipNow()
|
||||||
|
}
|
||||||
testParse(t, test.in, test.expected)
|
testParse(t, test.in, test.expected)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func testParse(t *testing.T, name string, expected []gtr.Event) {
|
func testParse(t *testing.T, name string, expected []gtr.Event) {
|
||||||
if len(expected) == 0 {
|
|
||||||
t.SkipNow()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
f, err := os.Open(filepath.Join(testdataRoot, name+".txt"))
|
f, err := os.Open(filepath.Join(testdataRoot, name+".txt"))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("error reading %s: %v", name, err)
|
t.Errorf("error reading %s: %v", name, err)
|
||||||
@ -362,3 +459,11 @@ func testParse(t *testing.T, name string, expected []gtr.Event) {
|
|||||||
t.Errorf("Parse %s returned unexpected events, diff (-got, +want):\n%v", name, diff)
|
t.Errorf("Parse %s returned unexpected events, diff (-got, +want):\n%v", name, diff)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func compileMatch(t *testing.T) *regexp.Regexp {
|
||||||
|
rx, err := regexp.Compile(*matchTest)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Error compiling -match flag %q: %v", *matchTest, err)
|
||||||
|
}
|
||||||
|
return rx
|
||||||
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user