mirror of
https://github.com/jstemmer/go-junit-report.git
synced 2025-04-05 21:18:08 -05:00

Mainly: * Moving the averaging/merging of benchmarks from the parser to the formatter package * Tightening up the regex so it only captures the numeric values (no more of trimming spaces and the ns/op) * Deleting the writing up in xml file the benchmark memory sections of B/op and Allocs/op Also added a test case for parseNanoseconds().
196 lines
5.2 KiB
Go
196 lines
5.2 KiB
Go
package formatter
|
|
|
|
import (
|
|
"bufio"
|
|
"encoding/xml"
|
|
"fmt"
|
|
"io"
|
|
"runtime"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/jstemmer/go-junit-report/parser"
|
|
)
|
|
|
|
// JUnitTestSuites is a collection of JUnit test suites.
|
|
type JUnitTestSuites struct {
|
|
XMLName xml.Name `xml:"testsuites"`
|
|
Suites []JUnitTestSuite
|
|
}
|
|
|
|
// JUnitTestSuite is a single JUnit test suite which may contain many
|
|
// testcases.
|
|
type JUnitTestSuite struct {
|
|
XMLName xml.Name `xml:"testsuite"`
|
|
Tests int `xml:"tests,attr"`
|
|
Failures int `xml:"failures,attr"`
|
|
Time string `xml:"time,attr"`
|
|
Name string `xml:"name,attr"`
|
|
Properties []JUnitProperty `xml:"properties>property,omitempty"`
|
|
TestCases []JUnitTestCase
|
|
}
|
|
|
|
// JUnitTestCase is a single test case with its result.
|
|
type JUnitTestCase struct {
|
|
XMLName xml.Name `xml:"testcase"`
|
|
Classname string `xml:"classname,attr"`
|
|
Name string `xml:"name,attr"`
|
|
Time string `xml:"time,attr"`
|
|
SkipMessage *JUnitSkipMessage `xml:"skipped,omitempty"`
|
|
Failure *JUnitFailure `xml:"failure,omitempty"`
|
|
}
|
|
|
|
// JUnitSkipMessage contains the reason why a testcase was skipped.
|
|
type JUnitSkipMessage struct {
|
|
Message string `xml:"message,attr"`
|
|
}
|
|
|
|
// JUnitProperty represents a key/value pair used to define properties.
|
|
type JUnitProperty struct {
|
|
Name string `xml:"name,attr"`
|
|
Value string `xml:"value,attr"`
|
|
}
|
|
|
|
// JUnitFailure contains data related to a failed test.
|
|
type JUnitFailure struct {
|
|
Message string `xml:"message,attr"`
|
|
Type string `xml:"type,attr"`
|
|
Contents string `xml:",chardata"`
|
|
}
|
|
|
|
// JUnitReportXML writes a JUnit xml representation of the given report to w
|
|
// in the format described at http://windyroad.org/dl/Open%20Source/JUnit.xsd
|
|
func JUnitReportXML(report *parser.Report, noXMLHeader bool, goVersion string, w io.Writer) error {
|
|
suites := JUnitTestSuites{}
|
|
|
|
// convert Report to JUnit test suites
|
|
for _, pkg := range report.Packages {
|
|
ts := JUnitTestSuite{
|
|
Tests: len(pkg.Tests) + len(pkg.Benchmarks),
|
|
Failures: 0,
|
|
Time: formatTime(pkg.Duration),
|
|
Name: pkg.Name,
|
|
Properties: []JUnitProperty{},
|
|
TestCases: []JUnitTestCase{},
|
|
}
|
|
|
|
classname := pkg.Name
|
|
if idx := strings.LastIndex(classname, "/"); idx > -1 && idx < len(pkg.Name) {
|
|
classname = pkg.Name[idx+1:]
|
|
}
|
|
|
|
// properties
|
|
if goVersion == "" {
|
|
// if goVersion was not specified as a flag, fall back to version reported by runtime
|
|
goVersion = runtime.Version()
|
|
}
|
|
ts.Properties = append(ts.Properties, JUnitProperty{"go.version", goVersion})
|
|
if pkg.CoveragePct != "" {
|
|
ts.Properties = append(ts.Properties, JUnitProperty{"coverage.statements.pct", pkg.CoveragePct})
|
|
}
|
|
|
|
// individual test cases
|
|
for _, test := range pkg.Tests {
|
|
testCase := JUnitTestCase{
|
|
Classname: classname,
|
|
Name: test.Name,
|
|
Time: formatTime(test.Duration),
|
|
Failure: nil,
|
|
}
|
|
|
|
if test.Result == parser.FAIL {
|
|
ts.Failures++
|
|
testCase.Failure = &JUnitFailure{
|
|
Message: "Failed",
|
|
Type: "",
|
|
Contents: strings.Join(test.Output, "\n"),
|
|
}
|
|
}
|
|
|
|
if test.Result == parser.SKIP {
|
|
testCase.SkipMessage = &JUnitSkipMessage{strings.Join(test.Output, "\n")}
|
|
}
|
|
|
|
ts.TestCases = append(ts.TestCases, testCase)
|
|
}
|
|
|
|
// individual benchmarks
|
|
for _, benchmark := range mergeBenchmarks(pkg.Benchmarks) {
|
|
benchmarkCase := JUnitTestCase{
|
|
Classname: classname,
|
|
Name: benchmark.Name,
|
|
Time: formatBenchmarkTime(benchmark.Duration),
|
|
Failure: nil,
|
|
}
|
|
|
|
ts.TestCases = append(ts.TestCases, benchmarkCase)
|
|
}
|
|
|
|
suites.Suites = append(suites.Suites, ts)
|
|
}
|
|
|
|
// to xml
|
|
bytes, err := xml.MarshalIndent(suites, "", "\t")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
writer := bufio.NewWriter(w)
|
|
|
|
if !noXMLHeader {
|
|
writer.WriteString(xml.Header)
|
|
}
|
|
|
|
writer.Write(bytes)
|
|
writer.WriteByte('\n')
|
|
writer.Flush()
|
|
|
|
return nil
|
|
}
|
|
|
|
func mergeBenchmarks(benchmarks []*parser.Benchmark) []*parser.Benchmark {
|
|
// calculate the cumulative moving average CMA for each benchmark.
|
|
// CMA(n + 1) = val(n+1) + n*CMA/(n+1)
|
|
alloc := "Allocs"
|
|
bytes := "Bytes"
|
|
dur := "Duration"
|
|
n := 1
|
|
var merged []*parser.Benchmark
|
|
averages := make(map[string] /*bench name*/ map[string] /* type */ int)
|
|
for _, b := range benchmarks {
|
|
if avg, found := averages[b.Name]; found {
|
|
// calculate CMAs
|
|
averages[b.Name][alloc] = (b.Allocs + n*averages[b.Name][alloc]) / (n + 1)
|
|
averages[b.Name][bytes] = (b.Bytes + n*avg[bytes]) / (n + 1)
|
|
averages[b.Name][dur] = (int(b.Duration.Nanoseconds()) + n*avg[dur]) / (n + 1)
|
|
|
|
n++
|
|
continue
|
|
}
|
|
n = 1 // reset duplicate counter
|
|
merged = append(merged, &parser.Benchmark{Name: b.Name})
|
|
averages[b.Name] = make(map[string]int)
|
|
averages[b.Name][alloc] = b.Allocs
|
|
averages[b.Name][bytes] = b.Bytes
|
|
averages[b.Name][dur] = int(b.Duration.Nanoseconds())
|
|
}
|
|
|
|
// fill out benchmarks
|
|
for i := range merged {
|
|
avgVals := averages[merged[i].Name]
|
|
merged[i].Allocs = avgVals[alloc]
|
|
merged[i].Bytes = avgVals[bytes]
|
|
merged[i].Duration = time.Duration(avgVals[dur])
|
|
}
|
|
|
|
return merged
|
|
}
|
|
|
|
func formatTime(d time.Duration) string {
|
|
return fmt.Sprintf("%.3f", d.Seconds())
|
|
}
|
|
|
|
func formatBenchmarkTime(d time.Duration) string {
|
|
return fmt.Sprintf("%.9f", d.Seconds())
|
|
}
|