diff --git a/cmd/junit2jira/main.go b/cmd/junit2jira/main.go index ba7001d..1ebfb2c 100644 --- a/cmd/junit2jira/main.go +++ b/cmd/junit2jira/main.go @@ -109,7 +109,7 @@ func run(p params) error { jiraClient: jiraClient, } - testSuites, err := junit.IngestDir(p.junitReportsDir) + testSuites, err := testcase.LoadTestSuites(p.junitReportsDir) if err != nil { log.Fatalf("could not read files: %s", err) } diff --git a/cmd/junit2jira/main_test.go b/cmd/junit2jira/main_test.go index 52da7ae..198b382 100644 --- a/cmd/junit2jira/main_test.go +++ b/cmd/junit2jira/main_test.go @@ -3,18 +3,18 @@ package main import ( "bytes" _ "embed" + "github.com/stackrox/junit2jira/pkg/testcase" "net/url" "testing" "github.com/andygrunwald/go-jira" - "github.com/joshdk/go-junit" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" ) func TestParseJunitReport(t *testing.T) { t.Run("not existing", func(t *testing.T) { - tests, err := junit.IngestDir("not existing") + tests, err := testcase.LoadTestSuites("not existing") assert.Error(t, err) assert.Nil(t, tests) }) @@ -22,7 +22,7 @@ func TestParseJunitReport(t *testing.T) { j := junit2jira{ params: params{junitReportsDir: "testdata/jira/report.xml"}, } - testsSuites, err := junit.IngestDir(j.junitReportsDir) + testsSuites, err := testcase.LoadTestSuites(j.junitReportsDir) assert.NoError(t, err) tests, err := j.getMergedFailedTests(testsSuites) assert.NoError(t, err) @@ -47,7 +47,7 @@ func TestParseJunitReport(t *testing.T) { j := junit2jira{ params: params{junitReportsDir: "testdata/jira/report.xml", JobName: "job-name", threshold: 1}, } - testsSuites, err := junit.IngestDir(j.junitReportsDir) + testsSuites, err := testcase.LoadTestSuites(j.junitReportsDir) assert.NoError(t, err) tests, err := j.getMergedFailedTests(testsSuites) assert.NoError(t, err) @@ -65,7 +65,7 @@ github.com/stackrox/rox/sensor/kubernetes/localscanner / TestLocalScannerTLSIssu j := junit2jira{ params: params{junitReportsDir: "testdata/jira", JobName: "job-name", BuildId: "1", threshold: 3}, } - testsSuites, err := junit.IngestDir(j.junitReportsDir) + testsSuites, err := testcase.LoadTestSuites(j.junitReportsDir) assert.NoError(t, err) tests, err := j.getMergedFailedTests(testsSuites) assert.NoError(t, err) @@ -75,6 +75,7 @@ github.com/stackrox/rox/sensor/kubernetes/localscanner / TestLocalScannerTLSIssu []j2jTestCase{ { Message: `DefaultPoliciesTest / Verify policy Apache Struts CVE-2017-5638 is triggered FAILED +github.com/stackrox/rox/pkg/grpc / Test_APIServerSuite/Test_TwoTestsStartingAPIs FAILED central-basic / step 90-activate-scanner-v4 FAILED github.com/stackrox/rox/pkg/booleanpolicy/evaluator / TestDifferentBaseTypes FAILED github.com/stackrox/rox/sensor/kubernetes/localscanner / TestLocalScannerTLSIssuerIntegrationTests FAILED @@ -93,7 +94,7 @@ command-line-arguments / TestTimeout FAILED j := junit2jira{ params: params{junitReportsDir: "testdata/jira", BuildId: "1"}, } - testsSuites, err := junit.IngestDir(j.junitReportsDir) + testsSuites, err := testcase.LoadTestSuites(j.junitReportsDir) assert.NoError(t, err) tests, err := j.getMergedFailedTests(testsSuites) assert.NoError(t, err) @@ -135,6 +136,20 @@ command-line-arguments / TestTimeout FAILED "\n" + "\tat DefaultPoliciesTest.Verify policy #policyName is triggered(DefaultPoliciesTest.groovy:181)\n", }, + { + Name: "Test_APIServerSuite/Test_TwoTestsStartingAPIs", + Message: "No test result found", + Stdout: "", + Suite: "github.com/stackrox/rox/pkg/grpc", + BuildId: "1", + Error: ` testutils.go:94: Stopping [2] listeners + testutils.go:87: [http handler listener: stopped] + testutils.go:87: [gRPC server listener: not stopped in loop. Comparing with grpcServer pointer with listener.srv pointer (0xc002ab2e00 : 0xc002ab2e00)] + server_test.go:229: ----------------------------------------------- + server_test.go:230: STACK TRACE INFO + server_test.go:231: ----------------------------------------------- +`, + }, { Name: "TestDifferentBaseTypes", Suite: "github.com/stackrox/rox/pkg/booleanpolicy/evaluator", @@ -176,9 +191,9 @@ command-line-arguments / TestTimeout FAILED }) t.Run("gradle", func(t *testing.T) { j := junit2jira{ - params: params{junitReportsDir: "testdata/jira/TEST-DefaultPoliciesTest.xml", BuildId: "1"}, + params: params{junitReportsDir: "testdata/jira/csv/TEST-DefaultPoliciesTest.xml", BuildId: "1"}, } - testsSuites, err := junit.IngestDir(j.junitReportsDir) + testsSuites, err := testcase.LoadTestSuites(j.junitReportsDir) assert.NoError(t, err) tests, err := j.getMergedFailedTests(testsSuites) assert.NoError(t, err) @@ -304,19 +319,17 @@ waitForViolation(deploymentName, policyName, 60) func TestCsvOutput(t *testing.T) { p := params{ - BuildId: "1", - JobName: "comma ,", - Orchestrator: "test", - BuildTag: "0.0.0", - BaseLink: `quote "`, - BuildLink: "buildLink", - timestamp: "time", + BuildId: "1", + JobName: "comma ,", + Orchestrator: "test", + BuildTag: "0.0.0", + BaseLink: `quote "`, + BuildLink: "buildLink", + timestamp: "time", + junitReportsDir: "testdata/jira/csv", } buf := bytes.NewBufferString("") - testSuites, err := junit.IngestFiles([]string{ - "testdata/jira/TEST-DefaultPoliciesTest.xml", - "testdata/jira/kuttl-report.xml", - }) + testSuites, err := testcase.LoadTestSuites(p.junitReportsDir) assert.NoError(t, err) err = junit2csv(testSuites, p, buf) assert.NoError(t, err) @@ -337,6 +350,10 @@ func TestCsvOutput(t *testing.T) { 1,time,DefaultPoliciesTest,Verify that built-in services don't trigger unexpected alerts,0,skipped,"comma ,",0.0.0 1,time,DefaultPoliciesTest,Verify that alert counts API is consistent with alerts,0,skipped,"comma ,",0.0.0 1,time,DefaultPoliciesTest,Verify that alert groups API is consistent with alerts,0,skipped,"comma ,",0.0.0 +1,time,github.com/stackrox/rox/pkg/grpc,Test_APIServerSuite/Test_TwoTestsStartingAPIs,0,error,"comma ,",0.0.0 +1,time,github.com/stackrox/rox/pkg/grpc/authz/user,TestLogInternalErrorInterceptor,0,passed,"comma ,",0.0.0 +1,time,fallback-classname,TestNoClassname,0,passed,"comma ,",0.0.0 +1,time,TestNoName,fallback-name,0,passed,"comma ,",0.0.0 1,time,central-basic,setup,0,passed,"comma ,",0.0.0 1,time,central-basic,step 0-image-pull-secrets,0,passed,"comma ,",0.0.0 1,time,central-basic,step 10-central-cr,0,passed,"comma ,",0.0.0 diff --git a/cmd/junit2jira/testdata/jira/TEST-DefaultPoliciesTest.xml b/cmd/junit2jira/testdata/jira/csv/TEST-DefaultPoliciesTest.xml similarity index 100% rename from cmd/junit2jira/testdata/jira/TEST-DefaultPoliciesTest.xml rename to cmd/junit2jira/testdata/jira/csv/TEST-DefaultPoliciesTest.xml diff --git a/cmd/junit2jira/testdata/jira/csv/golang-junit-report1.xml b/cmd/junit2jira/testdata/jira/csv/golang-junit-report1.xml new file mode 100644 index 0000000..5171955 --- /dev/null +++ b/cmd/junit2jira/testdata/jira/csv/golang-junit-report1.xml @@ -0,0 +1,17 @@ + + + + + + + + + + + diff --git a/cmd/junit2jira/testdata/jira/csv/golang-junit-report2-bad.xml b/cmd/junit2jira/testdata/jira/csv/golang-junit-report2-bad.xml new file mode 100644 index 0000000..80bbf9b --- /dev/null +++ b/cmd/junit2jira/testdata/jira/csv/golang-junit-report2-bad.xml @@ -0,0 +1,19 @@ + + + + + + + + + + + + + + diff --git a/cmd/junit2jira/testdata/jira/kuttl-report.xml b/cmd/junit2jira/testdata/jira/csv/kuttl-report.xml similarity index 100% rename from cmd/junit2jira/testdata/jira/kuttl-report.xml rename to cmd/junit2jira/testdata/jira/csv/kuttl-report.xml diff --git a/pkg/testcase/testcase.go b/pkg/testcase/testcase.go index 007bb33..a48fa16 100644 --- a/pkg/testcase/testcase.go +++ b/pkg/testcase/testcase.go @@ -3,10 +3,13 @@ package testcase import ( "fmt" "github.com/joshdk/go-junit" + "slices" "strings" ) const subTestFormat = "\nSub test %s: %s" +const fallbackName = "fallback-name" +const fallbackClassname = "fallback-classname" type TestCase struct { Name string @@ -19,6 +22,77 @@ type TestCase struct { IsSubtest bool } +type ignoreTestCase struct { + Name string + Classname string +} + +var ignoreList = []ignoreTestCase{ + // Go unit test crashes include stack traces of all threads, as well as some memory stats. + // We use go-junit-report which ingests plaintext-but-sort-of-machine-readable go test output + // and produces junit XML files. This tool seems to get confused by the crash dump, + // and thinks there is a failure in there from a (non-existent) go package runtime.MemStats, + // with an empty test case name. + {Name: "", Classname: "runtime.MemStats"}, +} + +// LoadTestSuites loads all reports in provided directory. +// It omits certain reports which are known to be useless, and fills in empty class and test case names. +func LoadTestSuites(reportDir string) ([]junit.Suite, error) { + testSuites, err := junit.IngestDir(reportDir) + if err != nil { + return nil, err + } + + return getClearedSuites(testSuites), nil +} + +func deleteHelperTest(testElem junit.Test) bool { + for _, ignore := range ignoreList { + if testElem.Name == ignore.Name && testElem.Classname == ignore.Classname { + return true + } + } + + return false +} + +// Makes sure the passed tests all have class and test names set to a non-empty value. +func addFallbacks(tests []junit.Test) []junit.Test { + testsWithFallback := make([]junit.Test, len(tests)) + + for i, test := range tests { + if test.Classname == "" { + test.Classname = fallbackClassname + } + + if test.Name == "" { + test.Name = fallbackName + } + + testsWithFallback[i] = test + } + + return testsWithFallback +} + +// getClearedSuites recursively removes ignored tests. +func getClearedSuites(suites []junit.Suite) []junit.Suite { + resSuites := make([]junit.Suite, 0, len(suites)) + for _, suite := range suites { + suite.Suites = getClearedSuites(suite.Suites) + suite.Tests = addFallbacks(slices.DeleteFunc(suite.Tests, deleteHelperTest)) + + // Filter out empty suites. + if len(suite.Suites) == 0 && len(suite.Tests) == 0 { + continue + } + resSuites = append(resSuites, suite) + } + + return resSuites +} + func (tc *TestCase) addSubTest(subTest junit.Test) { if subTest.Message != "" { tc.Message += fmt.Sprintf(subTestFormat, subTest.Name, subTest.Message) diff --git a/pkg/testcase/testcase_test.go b/pkg/testcase/testcase_test.go new file mode 100644 index 0000000..48a50e5 --- /dev/null +++ b/pkg/testcase/testcase_test.go @@ -0,0 +1,410 @@ +package testcase + +import ( + "github.com/joshdk/go-junit" + "github.com/stretchr/testify/assert" + "testing" +) + +func Test_getClearedSuites(t *testing.T) { + tests := map[string]struct { + ignoreList []ignoreTestCase + suites []junit.Suite + expectedSuites []junit.Suite + }{ + "empty suites": {}, + "simple no match": { + ignoreList: []ignoreTestCase{ + {Name: "match", Classname: "me"}, + }, + suites: []junit.Suite{ + { + Suites: []junit.Suite{}, + Tests: []junit.Test{ + { + Name: "do not match", + Classname: "me", + }, + }, + }, + }, + expectedSuites: []junit.Suite{ + { + Suites: []junit.Suite{}, + Tests: []junit.Test{ + { + Name: "do not match", + Classname: "me", + }, + }, + }, + }, + }, + "simple match": { + ignoreList: []ignoreTestCase{ + {Name: "match", Classname: "me"}, + }, + suites: []junit.Suite{ + { + Suites: []junit.Suite{}, + Tests: []junit.Test{ + { + Name: "match", + Classname: "me", + }, + }, + }, + }, + expectedSuites: []junit.Suite{}, + }, + "nested suites only": { + ignoreList: []ignoreTestCase{ + {Name: "match", Classname: "me"}, + }, + suites: []junit.Suite{ + { + Suites: []junit.Suite{ + { + Suites: []junit.Suite{}, + Tests: []junit.Test{ + { + Name: "skip", + Classname: "me", + }, + { + Name: "match", + Classname: "me", + }, + { + Name: "match", + Classname: "other", + }, + }, + }, + }, + Tests: []junit.Test{}, + }, + }, + expectedSuites: []junit.Suite{ + { + Suites: []junit.Suite{ + { + Suites: []junit.Suite{}, + Tests: []junit.Test{ + { + Name: "skip", + Classname: "me", + }, + { + Name: "match", + Classname: "other", + }, + }, + }, + }, + Tests: []junit.Test{}, + }, + }, + }, + "nested suites and tests": { + ignoreList: []ignoreTestCase{ + {Name: "match", Classname: "me"}, + }, + suites: []junit.Suite{ + { + Suites: []junit.Suite{ + { + Suites: []junit.Suite{}, + Tests: []junit.Test{ + { + Name: "skip", + Classname: "me", + }, + { + Name: "match", + Classname: "me", + }, + { + Name: "match", + Classname: "other", + }, + }, + }, + }, + Tests: []junit.Test{ + { + Name: "skip", + Classname: "me", + }, + { + Name: "match", + Classname: "me", + }, + { + Name: "match", + Classname: "other", + }, + }, + }, + }, + expectedSuites: []junit.Suite{ + { + Suites: []junit.Suite{ + { + Suites: []junit.Suite{}, + Tests: []junit.Test{ + { + Name: "skip", + Classname: "me", + }, + { + Name: "match", + Classname: "other", + }, + }, + }, + }, + Tests: []junit.Test{ + { + Name: "skip", + Classname: "me", + }, + { + Name: "match", + Classname: "other", + }, + }, + }, + }, + }, + "match all nested suites only": { + ignoreList: []ignoreTestCase{ + {Name: "match", Classname: "me"}, + {Name: "me", Classname: "including"}, + }, + suites: []junit.Suite{ + { + Suites: []junit.Suite{ + { + Suites: []junit.Suite{}, + Tests: []junit.Test{ + { + Name: "match", + Classname: "me", + }, + { + Name: "me", + Classname: "including", + }, + }, + }, + }, + Tests: []junit.Test{}, + }, + }, + expectedSuites: []junit.Suite{}, + }, + "match all test only": { + ignoreList: []ignoreTestCase{ + {Name: "match", Classname: "me"}, + }, + suites: []junit.Suite{ + { + Suites: []junit.Suite{ + { + Suites: []junit.Suite{}, + Tests: []junit.Test{ + { + Name: "skip", + Classname: "me", + }, + }, + }, + }, + Tests: []junit.Test{ + { + Name: "match", + Classname: "me", + }, + }, + }, + }, + expectedSuites: []junit.Suite{ + { + Suites: []junit.Suite{ + { + Suites: []junit.Suite{}, + Tests: []junit.Test{ + { + Name: "skip", + Classname: "me", + }, + }, + }, + }, + Tests: []junit.Test{}, + }, + }, + }, + "match all suites and tests": { + ignoreList: []ignoreTestCase{ + {Name: "remove", Classname: "it"}, + {Name: "why", Classname: "me"}, + }, + suites: []junit.Suite{ + { + Suites: []junit.Suite{ + { + Suites: []junit.Suite{}, + Tests: []junit.Test{ + { + Name: "remove", + Classname: "it", + }, + }, + }, + { + Suites: []junit.Suite{}, + Tests: []junit.Test{ + { + Name: "why", + Classname: "me", + }, + }, + }, + }, + Tests: []junit.Test{ + { + Name: "remove", + Classname: "it", + }, + { + Name: "why", + Classname: "me", + }, + }, + }, + }, + expectedSuites: []junit.Suite{}, + }, + "match empty name and classname": { + ignoreList: []ignoreTestCase{ + {Name: "only-name", Classname: ""}, + {Name: "", Classname: "only-class"}, + }, + suites: []junit.Suite{ + { + Suites: []junit.Suite{}, + Tests: []junit.Test{ + { + Name: "only-name", + Classname: "", + }, + { + Name: "only-name", + Classname: "only-class", + }, + { + Name: "", + Classname: "only-class", + }, + }, + }, + }, + expectedSuites: []junit.Suite{ + { + Suites: []junit.Suite{}, + Tests: []junit.Test{ + { + Name: "only-name", + Classname: "only-class", + }, + }, + }, + }, + }, + } + + for testName, tt := range tests { + t.Run(testName, func(t *testing.T) { + ignoreList = tt.ignoreList + actual := getClearedSuites(tt.suites) + assert.ElementsMatch(t, tt.expectedSuites, actual) + }) + } +} + +func Test_addFallbacks(t *testing.T) { + tests := map[string]struct { + tests []junit.Test + expectedTest []junit.Test + }{ + "empty": { + tests: []junit.Test{}, + expectedTest: []junit.Test{}, + }, + "mysterious test case": { + tests: []junit.Test{ + { + Name: "", + Classname: "", + }, + }, + expectedTest: []junit.Test{ + { + Name: fallbackName, + Classname: fallbackClassname, + }, + }, + }, + "add name": { + tests: []junit.Test{ + { + Name: "", + Classname: "no name", + }, + }, + expectedTest: []junit.Test{ + { + Name: fallbackName, + Classname: "no name", + }, + }, + }, + "add classname": { + tests: []junit.Test{ + { + Name: "no class", + Classname: "", + }, + }, + expectedTest: []junit.Test{ + { + Name: "no class", + Classname: fallbackClassname, + }, + }, + }, + "all good": { + tests: []junit.Test{ + { + Name: "with name", + Classname: "and class", + }, + }, + expectedTest: []junit.Test{ + { + Name: "with name", + Classname: "and class", + }, + }, + }, + } + + for testName, tt := range tests { + t.Run(testName, func(t *testing.T) { + actual := addFallbacks(tt.tests) + assert.ElementsMatch(t, tt.expectedTest, actual) + }) + } +}