mirror of
https://github.com/jstemmer/go-junit-report.git
synced 2025-04-05 05:00:15 -05:00
Refactor and simplify tests
For most testcases we shouldn't need to specify the input and report files, instead just look at the testdata/ dir and process each input file. For the few testcases that need a non-standard config, an entry can be added to `testConfigs` map (indexed by the test number). The testdata files must follow a certain naming pattern to be valid. They should start with a number, followed by a dash, then any text describing the test and finally end with `.txt`. The corresponding report file must start with the same number, a dash, followed by `report.xml`.
This commit is contained in:
parent
752d873a6e
commit
6e3153dd44
@ -1,13 +1,12 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/xml"
|
"bytes"
|
||||||
"flag"
|
"flag"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
|
||||||
"os"
|
"os"
|
||||||
|
"path/filepath"
|
||||||
"regexp"
|
"regexp"
|
||||||
"runtime"
|
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
@ -19,276 +18,99 @@ import (
|
|||||||
"github.com/google/go-cmp/cmp"
|
"github.com/google/go-cmp/cmp"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const testDataDir = "testdata/"
|
||||||
|
|
||||||
var matchTest = flag.String("match", "", "only test testdata matching this pattern")
|
var matchTest = flag.String("match", "", "only test testdata matching this pattern")
|
||||||
|
|
||||||
type TestCase struct {
|
type TestConfig struct {
|
||||||
name string
|
|
||||||
reportName string
|
|
||||||
noXMLHeader bool
|
noXMLHeader bool
|
||||||
packageName string
|
packageName string
|
||||||
}
|
}
|
||||||
|
|
||||||
var testCases = []TestCase{
|
var testConfigs = map[int]TestConfig{
|
||||||
{
|
5: {noXMLHeader: true},
|
||||||
name: "001-pass.txt",
|
6: {noXMLHeader: true},
|
||||||
reportName: "001-report.xml",
|
7: {packageName: "test/package"},
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "002-fail.txt",
|
|
||||||
reportName: "002-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "003-skip.txt",
|
|
||||||
reportName: "003-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "004-go_1_4.txt",
|
|
||||||
reportName: "004-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "005-no_xml_header.txt",
|
|
||||||
reportName: "005-report.xml",
|
|
||||||
noXMLHeader: true,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "006-mixed.txt",
|
|
||||||
reportName: "006-report.xml",
|
|
||||||
noXMLHeader: true,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "007-compiled_test.txt",
|
|
||||||
reportName: "007-report.xml",
|
|
||||||
packageName: "test/package",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "008-parallel.txt",
|
|
||||||
reportName: "008-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "009-coverage.txt",
|
|
||||||
reportName: "009-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "010-multipkg-coverage.txt",
|
|
||||||
reportName: "010-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "011-go_1_5.txt",
|
|
||||||
reportName: "011-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "012-go_1_7.txt",
|
|
||||||
reportName: "012-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "013-syntax-error.txt",
|
|
||||||
reportName: "013-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "014-panic.txt",
|
|
||||||
reportName: "014-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "015-empty.txt",
|
|
||||||
reportName: "015-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "016-repeated-names.txt",
|
|
||||||
reportName: "016-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "017-race.txt",
|
|
||||||
reportName: "017-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "018-coverpkg.txt",
|
|
||||||
reportName: "018-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "019-pass.txt",
|
|
||||||
reportName: "019-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "020-parallel.txt",
|
|
||||||
reportName: "020-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "021-cached.txt",
|
|
||||||
reportName: "021-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "022-bench.txt",
|
|
||||||
reportName: "022-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "023-benchmem.txt",
|
|
||||||
reportName: "023-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "024-benchtests.txt",
|
|
||||||
reportName: "024-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "025-benchcount.txt",
|
|
||||||
reportName: "025-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "026-testbenchmultiple.txt",
|
|
||||||
reportName: "026-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "027-benchdecimal.txt",
|
|
||||||
reportName: "027-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "028-bench-1cpu.txt",
|
|
||||||
reportName: "028-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "029-bench-16cpu.txt",
|
|
||||||
reportName: "029-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
// generated by running go test on https://gist.github.com/liggitt/09a021ccec988b19917e0c2d60a18ee9
|
|
||||||
name: "030-stdout.txt",
|
|
||||||
reportName: "030-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "031-syntax-error-test-binary.txt",
|
|
||||||
reportName: "031-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "032-failed-summary.txt",
|
|
||||||
reportName: "032-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "033-bench-mb.txt",
|
|
||||||
reportName: "033-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "034-notest.txt",
|
|
||||||
reportName: "034-report.xml",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "035-whitespace.txt",
|
|
||||||
reportName: "035-report.xml",
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestNewOutput(t *testing.T) {
|
func TestRun(t *testing.T) {
|
||||||
matchRegex := compileMatch(t)
|
matchRegex := compileMatch(t)
|
||||||
for _, testCase := range testCases {
|
|
||||||
if !matchRegex.MatchString(testCase.name) {
|
files, err := filepath.Glob(testDataDir + "*.txt")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("error finding files in testdata: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, file := range files {
|
||||||
|
if !matchRegex.MatchString(file) {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
t.Run(testCase.name, func(t *testing.T) {
|
conf, reportFile, err := testFileConfig(strings.TrimPrefix(file, testDataDir))
|
||||||
testReport(testCase.name, testCase.reportName, testCase.packageName, t)
|
if err != nil {
|
||||||
|
t.Errorf("testFileConfig error: %v", err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
t.Run(file, func(t *testing.T) {
|
||||||
|
testRun(file, testDataDir+reportFile, conf, t)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func testReport(input, reportFile, packageName string, t *testing.T) {
|
func testRun(inputFile, reportFile string, config TestConfig, t *testing.T) {
|
||||||
file, err := os.Open("testdata/" + input)
|
input, err := os.Open(inputFile)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatalf("error opening input file: %v", err)
|
||||||
|
}
|
||||||
|
defer input.Close()
|
||||||
|
|
||||||
|
wantReport, err := os.ReadFile(reportFile)
|
||||||
|
if os.IsNotExist(err) {
|
||||||
|
t.Skipf("Skipping test with missing report file: %s", reportFile)
|
||||||
|
} else if err != nil {
|
||||||
|
t.Fatalf("error loading report file: %v", err)
|
||||||
}
|
}
|
||||||
defer file.Close()
|
|
||||||
|
|
||||||
parser := gotest.New(
|
parser := gotest.New(
|
||||||
gotest.PackageName(packageName),
|
gotest.PackageName(config.packageName),
|
||||||
gotest.TimestampFunc(func() time.Time {
|
gotest.TimestampFunc(func() time.Time {
|
||||||
return time.Date(2022, 1, 1, 0, 0, 0, 0, time.UTC)
|
return time.Date(2022, 1, 1, 0, 0, 0, 0, time.UTC)
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
|
|
||||||
report, err := parser.Parse(file)
|
report, err := parser.Parse(input)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if *printEvents {
|
for i := range report.Packages {
|
||||||
for _, event := range parser.Events() {
|
report.Packages[i].SetProperty("go.version", "1.0")
|
||||||
t.Logf("Event: %+v", event)
|
}
|
||||||
}
|
testsuites := junit.CreateFromReport(report, "hostname")
|
||||||
|
|
||||||
|
var output bytes.Buffer
|
||||||
|
if err := writeXML(&output, testsuites, config.noXMLHeader); err != nil {
|
||||||
|
t.Fatalf("error writing XML: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
actual := junit.CreateFromReport(report, "hostname")
|
if diff := cmp.Diff(output.String(), string(wantReport)); diff != "" {
|
||||||
|
|
||||||
expectedXML, err := loadTestReport(reportFile, "")
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
var expected junit.Testsuites
|
|
||||||
if err := xml.Unmarshal([]byte(expectedXML), &expected); err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
expected = modifyForBackwardsCompat(expected)
|
|
||||||
if diff := cmp.Diff(actual, expected); diff != "" {
|
|
||||||
t.Errorf("Unexpected report diff (-got, +want):\n%v", diff)
|
t.Errorf("Unexpected report diff (-got, +want):\n%v", diff)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func modifyForBackwardsCompat(testsuites junit.Testsuites) junit.Testsuites {
|
func testFileConfig(filename string) (conf TestConfig, reportFile string, err error) {
|
||||||
testsuites.XMLName.Local = ""
|
var prefix string
|
||||||
for i, suite := range testsuites.Suites {
|
if idx := strings.IndexByte(filename, '-'); idx < 0 {
|
||||||
for j := range suite.Testcases {
|
return conf, "", fmt.Errorf("testdata file does not contain a dash (-); expected name `{id}-{name}.txt` got `%s`", filename)
|
||||||
testsuites.Suites[i].Testcases[j].Classname = suite.Name
|
} else {
|
||||||
}
|
prefix = filename[:idx]
|
||||||
|
|
||||||
if suite.Properties != nil {
|
|
||||||
if covIdx, covProp := getProperty("coverage.statements.pct", *suite.Properties); covIdx > -1 {
|
|
||||||
pct, _ := strconv.ParseFloat(covProp.Value, 64)
|
|
||||||
(*testsuites.Suites[i].Properties)[covIdx].Value = fmt.Sprintf("%.2f", pct)
|
|
||||||
}
|
|
||||||
testsuites.Suites[i].Properties = dropProperty("go.version", suite.Properties)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return testsuites
|
id, err := strconv.Atoi(prefix)
|
||||||
}
|
|
||||||
|
|
||||||
func dropProperty(name string, properties *[]junit.Property) *[]junit.Property {
|
|
||||||
if properties == nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
var props []junit.Property
|
|
||||||
for _, prop := range *properties {
|
|
||||||
if prop.Name != name {
|
|
||||||
props = append(props, prop)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if len(props) == 0 {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return &props
|
|
||||||
}
|
|
||||||
|
|
||||||
func getProperty(name string, properties []junit.Property) (int, junit.Property) {
|
|
||||||
for i, prop := range properties {
|
|
||||||
if prop.Name == name {
|
|
||||||
return i, prop
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return -1, junit.Property{}
|
|
||||||
}
|
|
||||||
|
|
||||||
func loadTestReport(name, goVersion string) (string, error) {
|
|
||||||
contents, err := ioutil.ReadFile("testdata/" + name)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", err
|
return conf, "", fmt.Errorf("testdata file did not start with a valid number: %w", err)
|
||||||
}
|
}
|
||||||
|
return testConfigs[id], fmt.Sprintf("%s-report.xml", prefix), nil
|
||||||
if goVersion == "" {
|
|
||||||
// if goVersion is not specified, default to runtime version
|
|
||||||
goVersion = runtime.Version()
|
|
||||||
}
|
|
||||||
|
|
||||||
// replace value="1.0" With actual version
|
|
||||||
report := strings.Replace(string(contents), `value="1.0"`, fmt.Sprintf(`value="%s"`, goVersion), -1)
|
|
||||||
|
|
||||||
return report, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func compileMatch(t *testing.T) *regexp.Regexp {
|
func compileMatch(t *testing.T) *regexp.Regexp {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user