Skip to content

Commit 73171de

Browse files
Merge pull request #20179 from smarterclayton/update_tools
Switch origin to use gotest2junit
2 parents ef616a7 + 5bd42aa commit 73171de

File tree

16 files changed

+690
-48
lines changed

16 files changed

+690
-48
lines changed

hack/lib/build/environment.sh

-2
Original file line numberDiff line numberDiff line change
@@ -110,8 +110,6 @@ function os::build::environment::cleanup() {
110110
local container=$1
111111
local volume=$2
112112
local tmp_volume=$3
113-
os::log::debug "Recording container information to ${LOG_DIR}/${container}.json"
114-
docker inspect "${container}" > "${LOG_DIR}/${container}.json"
115113
os::log::debug "Stopping container ${container}"
116114
docker stop --time=0 "${container}" > /dev/null || true
117115
if [[ -z "${OS_BUILD_ENV_LEAVE_CONTAINER:-}" ]]; then

hack/lib/cmd.sh

+1-1
Original file line numberDiff line numberDiff line change
@@ -157,7 +157,7 @@ readonly -f os::cmd::try_until_text
157157

158158
# In order to harvest stderr and stdout at the same time into different buckets, we need to stick them into files
159159
# in an intermediate step
160-
os_cmd_internal_tmpdir="${TMPDIR:-"/tmp"}/openshift"
160+
os_cmd_internal_tmpdir="${TMPDIR:-"/tmp"}/cmd"
161161
os_cmd_internal_tmpout="${os_cmd_internal_tmpdir}/tmp_stdout.log"
162162
os_cmd_internal_tmperr="${os_cmd_internal_tmpdir}/tmp_stderr.log"
163163

hack/lib/init.sh

+3-7
Original file line numberDiff line numberDiff line change
@@ -51,14 +51,10 @@ os::log::stacktrace::install
5151
os::util::environment::update_path_var
5252

5353
if [[ -z "${OS_TMP_ENV_SET-}" ]]; then
54-
if [[ "${0}" =~ .*\.sh ]]; then
55-
os::util::environment::setup_tmpdir_vars "$( basename "${0}" ".sh" )"
56-
else
57-
os::util::environment::setup_tmpdir_vars "shell"
58-
fi
54+
os::util::environment::setup_tmpdir_vars "$( basename "$0" ".sh" )"
5955
fi
6056

6157
# Allow setting $JUNIT_REPORT to toggle output behavior
6258
if [[ -n "${JUNIT_REPORT:-}" ]]; then
63-
export JUNIT_REPORT_OUTPUT="${LOG_DIR}/raw_test_output.log"
64-
fi
59+
export JUNIT_REPORT_OUTPUT="${LOG_DIR}/raw_test_output.log"
60+
fi

hack/lib/log/output.sh

+5-5
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@
88
# Arguments:
99
# - all: message to write
1010
function os::log::info() {
11-
local message; message="$( os::log::internal::prefix_lines "[INFO] [$( date +%H:%M:%S%z )]" "$*" )"
11+
local message; message="$( os::log::internal::prefix_lines "[INFO]" "$*" )"
1212
os::log::internal::to_logfile "${message}"
1313
echo "${message}"
1414
}
@@ -21,7 +21,7 @@ readonly -f os::log::info
2121
# Arguments:
2222
# - all: message to write
2323
function os::log::warning() {
24-
local message; message="$( os::log::internal::prefix_lines "[WARNING] [$( date +%H:%M:%S%z )]" "$*" )"
24+
local message; message="$( os::log::internal::prefix_lines "[WARNING]" "$*" )"
2525
os::log::internal::to_logfile "${message}"
2626
os::text::print_yellow "${message}" 1>&2
2727
}
@@ -34,7 +34,7 @@ readonly -f os::log::warning
3434
# Arguments:
3535
# - all: message to write
3636
function os::log::error() {
37-
local message; message="$( os::log::internal::prefix_lines "[ERROR] [$( date +%H:%M:%S%z )]" "$*" )"
37+
local message; message="$( os::log::internal::prefix_lines "[ERROR]" "$*" )"
3838
os::log::internal::to_logfile "${message}"
3939
os::text::print_red "${message}" 1>&2
4040
}
@@ -48,7 +48,7 @@ readonly -f os::log::error
4848
# Arguments:
4949
# - all: message to write
5050
function os::log::fatal() {
51-
local message; message="$( os::log::internal::prefix_lines "[FATAL] [$( date +%H:%M:%S%z )]" "$*" )"
51+
local message; message="$( os::log::internal::prefix_lines "[FATAL]" "$*" )"
5252
os::log::internal::to_logfile "${message}"
5353
os::text::print_red "${message}" 1>&2
5454
exit 1
@@ -63,7 +63,7 @@ readonly -f os::log::fatal
6363
# Arguments:
6464
# - all: message to write
6565
function os::log::debug() {
66-
local message; message="$( os::log::internal::prefix_lines "[DEBUG] [$( date +%H:%M:%S%z )]" "$*" )"
66+
local message; message="$( os::log::internal::prefix_lines "[DEBUG]" "$*" )"
6767
os::log::internal::to_logfile "${message}"
6868
if [[ -n "${OS_DEBUG:-}" ]]; then
6969
os::text::print_blue "${message}" 1>&2

hack/lib/test/junit.sh

-2
Original file line numberDiff line numberDiff line change
@@ -165,8 +165,6 @@ function os::test::junit::generate_report() {
165165
os::test::junit::reconcile_output
166166
os::test::junit::check_test_counters
167167
os::test::junit::internal::generate_report "oscmd"
168-
else
169-
os::test::junit::internal::generate_report "gotest"
170168
fi
171169
}
172170

hack/push-release.sh

-14
Original file line numberDiff line numberDiff line change
@@ -70,11 +70,6 @@ if [[ "${OS_PUSH_BASE_REGISTRY-}" != "" || "${tag}" != "" ]]; then
7070
docker tag "openshift/${image}:${source_tag}" "${OS_PUSH_BASE_REGISTRY-}${OS_PUSH_BASE_REPO}${image}${tag}"
7171
done
7272
done
73-
# TODO: remove in 3.11
74-
for tag in "${tags[@]}"; do
75-
docker tag "openshift/origin-control-plane:${source_tag}" "${OS_PUSH_BASE_REGISTRY-}${OS_PUSH_BASE_REPO}origin${tag}"
76-
docker tag "openshift/origin-node:${source_tag}" "${OS_PUSH_BASE_REGISTRY-}${OS_PUSH_BASE_REPO}node${tag}"
77-
done
7873
fi
7974

8075
for image in "${images[@]}"; do
@@ -83,14 +78,5 @@ for image in "${images[@]}"; do
8378
docker push ${PUSH_OPTS} "${OS_PUSH_BASE_REGISTRY-}${OS_PUSH_BASE_REPO}${image}${tag}"
8479
done
8580
done
86-
# TODO: remove in 3.11
87-
for tag in "${tags[@]}"; do
88-
os::log::info "Pushing ${OS_PUSH_BASE_REGISTRY-}${OS_PUSH_BASE_REPO}origin${tag}..."
89-
docker push ${PUSH_OPTS} "${OS_PUSH_BASE_REGISTRY-}${OS_PUSH_BASE_REPO}origin${tag}"
90-
done
91-
for tag in "${tags[@]}"; do
92-
os::log::info "Pushing ${OS_PUSH_BASE_REGISTRY-}${OS_PUSH_BASE_REPO}node${tag}..."
93-
docker push ${PUSH_OPTS} "${OS_PUSH_BASE_REGISTRY-}${OS_PUSH_BASE_REPO}node${tag}"
94-
done
9581

9682
ret=$?; ENDTIME=$(date +%s); echo "$0 took $(($ENDTIME - $STARTTIME)) seconds"; exit "$ret"

hack/test-go.sh

+4-12
Original file line numberDiff line numberDiff line change
@@ -22,14 +22,6 @@
2222
function cleanup() {
2323
return_code=$?
2424

25-
os::test::junit::generate_report
26-
if [[ "${JUNIT_REPORT_NUM_FAILED:-}" == "0 failed" ]]; then
27-
if [[ "${return_code}" -ne "0" ]]; then
28-
os::log::warning "While the jUnit report found no failed tests, the \`go test\` process failed."
29-
os::log::warning "This usually means that the unit test suite failed to compile."
30-
fi
31-
fi
32-
3325
os::util::describe_return_code "${return_code}"
3426
exit "${return_code}"
3527
}
@@ -136,8 +128,10 @@ if [[ -n "${junit_report}" ]]; then
136128
# we don't care if the `go test` fails in this pipe, as we want to generate the report and summarize the output anyway
137129
set +o pipefail
138130

139-
go test -i ${gotest_flags} ${test_packages}
140-
go test ${gotest_flags} ${test_packages} 2>"${test_error_file}" | tee "${JUNIT_REPORT_OUTPUT}"
131+
os::util::ensure::built_binary_exists 'gotest2junit'
132+
report_file="$( mktemp "${ARTIFACT_DIR}/unit_report_XXXXX" ).xml"
133+
134+
go test -json ${gotest_flags} ${test_packages} 2>"${test_error_file}" | tee "${JUNIT_REPORT_OUTPUT}" | gotest2junit > "${report_file}"
141135

142136
test_return_code="${PIPESTATUS[0]}"
143137

@@ -163,7 +157,6 @@ $( cat "${test_error_file}") "
163157

164158
elif [[ -n "${coverage_output_dir}" ]]; then
165159
# we need to generate coverage reports
166-
go test -i ${gotest_flags} ${test_packages}
167160
for test_package in ${test_packages}; do
168161
mkdir -p "${coverage_output_dir}/${test_package}"
169162
local_gotest_flags="${gotest_flags} -coverprofile=${coverage_output_dir}/${test_package}/profile.out"
@@ -188,6 +181,5 @@ elif [[ -n "${dlv_debug}" ]]; then
188181
dlv test ${test_packages}
189182
else
190183
# we need to generate neither jUnit XML nor coverage reports
191-
go test -i ${gotest_flags} ${test_packages}
192184
go test ${gotest_flags} ${test_packages}
193185
fi

tools/gotest2junit/gotest2junit.go

+211
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,211 @@
1+
package main
2+
3+
import (
4+
"bufio"
5+
"encoding/json"
6+
"encoding/xml"
7+
"flag"
8+
"fmt"
9+
"io"
10+
"os"
11+
"sort"
12+
"strings"
13+
"time"
14+
15+
"github.com/openshift/origin/tools/gotest2junit/pkg/api"
16+
)
17+
18+
type Record struct {
19+
Package string
20+
Test string
21+
22+
Time time.Time
23+
Action string
24+
Output string
25+
Elapsed float64
26+
}
27+
28+
type testSuite struct {
29+
suite *api.TestSuite
30+
tests map[string]*api.TestCase
31+
}
32+
33+
func main() {
34+
summarize := false
35+
verbose := false
36+
flag.BoolVar(&summarize, "summary", true, "display a summary as items are processed")
37+
flag.BoolVar(&verbose, "v", false, "display passing results")
38+
flag.Parse()
39+
40+
if err := process(os.Stdin, summarize, verbose); err != nil {
41+
fmt.Fprintf(os.Stderr, "error: %v\n", err)
42+
os.Exit(1)
43+
}
44+
}
45+
46+
func process(r io.Reader, summarize, verbose bool) error {
47+
suites, err := stream(r, summarize, verbose)
48+
if err != nil {
49+
return err
50+
}
51+
obj := newTestSuites(suites)
52+
out, err := xml.MarshalIndent(obj, "", " ")
53+
if err != nil {
54+
return err
55+
}
56+
fmt.Fprintf(os.Stdout, "%s\n", string(out))
57+
return nil
58+
}
59+
60+
func newTestSuites(suites map[string]*testSuite) *api.TestSuites {
61+
all := &api.TestSuites{}
62+
for _, suite := range suites {
63+
for _, test := range suite.suite.TestCases {
64+
suite.suite.NumTests++
65+
if test.SkipMessage != nil {
66+
suite.suite.NumSkipped++
67+
continue
68+
}
69+
if test.FailureOutput != nil {
70+
suite.suite.NumFailed++
71+
continue
72+
}
73+
}
74+
// suites with no tests are usually empty packages, ignore them
75+
if suite.suite.NumTests == 0 {
76+
continue
77+
}
78+
// always return the test cases in consistent order
79+
sort.Slice(suite.suite.TestCases, func(i, j int) bool {
80+
return suite.suite.TestCases[i].Name < suite.suite.TestCases[j].Name
81+
})
82+
all.Suites = append(all.Suites, suite.suite)
83+
}
84+
// always return the test suites in consistent order
85+
sort.Slice(all.Suites, func(i, j int) bool {
86+
return all.Suites[i].Name < all.Suites[j].Name
87+
})
88+
return all
89+
}
90+
91+
func stream(r io.Reader, summarize, verbose bool) (map[string]*testSuite, error) {
92+
suites := make(map[string]*testSuite)
93+
defaultTest := &api.TestCase{
94+
Name: "build and execution",
95+
}
96+
defaultSuite := &testSuite{
97+
suite: &api.TestSuite{Name: "go test", TestCases: []*api.TestCase{defaultTest}},
98+
}
99+
suites[""] = defaultSuite
100+
101+
rdr := bufio.NewReader(r)
102+
for {
103+
// some output from go test -json is not valid JSON - read the line to see whether it
104+
// starts with { - if not, just mirror it to stderr and continue.
105+
line, err := rdr.ReadString('\n')
106+
if err != nil {
107+
if err != io.EOF {
108+
return suites, err
109+
}
110+
break
111+
}
112+
if len(line) == 0 || line[0] != '{' {
113+
defaultTest.SystemOut += line
114+
if strings.HasPrefix(line, "FAIL") {
115+
defaultTest.FailureOutput = &api.FailureOutput{}
116+
}
117+
fmt.Fprint(os.Stderr, line)
118+
continue
119+
}
120+
var r Record
121+
if err := json.Unmarshal([]byte(line), &r); err != nil {
122+
if err == io.EOF {
123+
return suites, nil
124+
}
125+
fmt.Fprintf(os.Stderr, "error: Unable to parse remainder of output %v\n", err)
126+
return suites, nil
127+
}
128+
129+
suite, ok := suites[r.Package]
130+
if !ok {
131+
suite = &testSuite{
132+
suite: &api.TestSuite{
133+
Name: r.Package,
134+
},
135+
tests: make(map[string]*api.TestCase),
136+
}
137+
suites[r.Package] = suite
138+
}
139+
140+
// if this is package level output, we only care about pass/fail duration
141+
if len(r.Test) == 0 {
142+
switch r.Action {
143+
case "pass", "fail":
144+
suite.suite.Duration = r.Elapsed
145+
}
146+
continue
147+
}
148+
149+
test, ok := suite.tests[r.Test]
150+
if !ok {
151+
test = &api.TestCase{
152+
Name: r.Test,
153+
}
154+
suite.suite.TestCases = append(suite.suite.TestCases, test)
155+
suite.tests[r.Test] = test
156+
}
157+
158+
switch r.Action {
159+
case "run":
160+
case "pause":
161+
case "cont":
162+
case "bench":
163+
case "skip":
164+
if summarize {
165+
fmt.Fprintf(os.Stderr, "SKIP: %s %s\n", r.Package, r.Test)
166+
}
167+
test.SkipMessage = &api.SkipMessage{
168+
Message: r.Output,
169+
}
170+
case "pass":
171+
if summarize && verbose {
172+
fmt.Fprintf(os.Stderr, "PASS: %s %s %s\n", r.Package, r.Test, time.Duration(r.Elapsed*float64(time.Second)))
173+
}
174+
test.SystemOut = ""
175+
test.Duration = r.Elapsed
176+
case "fail":
177+
if summarize {
178+
fmt.Fprintf(os.Stderr, "FAIL: %s %s %s\n", r.Package, r.Test, time.Duration(r.Elapsed*float64(time.Second)))
179+
}
180+
test.Duration = r.Elapsed
181+
if len(r.Output) == 0 {
182+
r.Output = test.SystemOut
183+
if len(r.Output) > 50 {
184+
r.Output = r.Output[:50] + " ..."
185+
}
186+
}
187+
test.FailureOutput = &api.FailureOutput{
188+
Message: r.Output,
189+
Output: r.Output,
190+
}
191+
case "output":
192+
test.SystemOut += r.Output
193+
default:
194+
// usually a bug in go test -json
195+
out := fmt.Sprintf("error: Unrecognized go test action %s: %#v\n", r.Action, r)
196+
defaultTest.SystemOut += line
197+
defaultTest.SystemOut += out
198+
defaultTest.FailureOutput = &api.FailureOutput{}
199+
fmt.Fprintf(os.Stderr, out)
200+
}
201+
}
202+
203+
// if we recorded any failure output
204+
if defaultTest.FailureOutput != nil {
205+
defaultTest.FailureOutput.Message = "Some packages failed during test execution"
206+
defaultTest.FailureOutput.Output = defaultTest.SystemOut
207+
defaultTest.SystemOut = ""
208+
}
209+
210+
return suites, nil
211+
}

0 commit comments

Comments
 (0)