Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gitlab-ce
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
1
Merge Requests
1
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
gitlab-ce
Commits
c41492dd
Commit
c41492dd
authored
Sep 04, 2018
by
Shinya Maeda
Committed by
Kamil Trzciński
Sep 04, 2018
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Fix bugs/edge cases of JUnitParser
parent
f7c1a5e1
Changes
4
Hide whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
93 additions
and
29 deletions
+93
-29
changelogs/unreleased/fix-junit-parser.yml
changelogs/unreleased/fix-junit-parser.yml
+5
-0
doc/ci/junit_test_reports.md
doc/ci/junit_test_reports.md
+8
-0
lib/gitlab/ci/parsers/junit.rb
lib/gitlab/ci/parsers/junit.rb
+21
-24
spec/lib/gitlab/ci/parsers/junit_spec.rb
spec/lib/gitlab/ci/parsers/junit_spec.rb
+59
-5
No files found.
changelogs/unreleased/fix-junit-parser.yml
0 → 100644
View file @
c41492dd
---
title
:
Fix edge cases of JUnitParser
merge_request
:
21469
author
:
type
:
fixed
doc/ci/junit_test_reports.md
View file @
c41492dd
...
@@ -139,3 +139,11 @@ java:
...
@@ -139,3 +139,11 @@ java:
-
target/surefire-reports/TEST-*.xml
-
target/surefire-reports/TEST-*.xml
-
target/failsafe-reports/TEST-*.xml
-
target/failsafe-reports/TEST-*.xml
```
```
## Limitations
Currently, the following tools might not work because their XML formats are unsupported in GitLab.
|Case|Tool|Issue|
|---|---|---|
|
`<testcase>`
does not have
`classname`
attribute|ESlint, sass-lint|https://gitlab.com/gitlab-org/gitlab-ce/issues/50964|
lib/gitlab/ci/parsers/junit.rb
View file @
c41492dd
...
@@ -2,18 +2,14 @@ module Gitlab
...
@@ -2,18 +2,14 @@ module Gitlab
module
Ci
module
Ci
module
Parsers
module
Parsers
class
Junit
class
Junit
attr_reader
:data
JunitParserError
=
Class
.
new
(
StandardError
)
JunitParserError
=
Class
.
new
(
StandardError
)
def
parse!
(
xml_data
,
test_suite
)
def
parse!
(
xml_data
,
test_suite
)
@data
=
Hash
.
from_xml
(
xml_data
)
root
=
Hash
.
from_xml
(
xml_data
)
each_suite
do
|
testcases
|
all_cases
(
root
)
do
|
test_case
|
testcases
.
each
do
|
testcase
|
test_case
=
create_test_case
(
test_case
)
test_case
=
create_test_case
(
testcase
)
test_suite
.
add_test_case
(
test_case
)
test_suite
.
add_test_case
(
test_case
)
end
end
end
rescue
REXML
::
ParseException
=>
e
rescue
REXML
::
ParseException
=>
e
raise
JunitParserError
,
"XML parsing failed:
#{
e
.
message
}
"
raise
JunitParserError
,
"XML parsing failed:
#{
e
.
message
}
"
...
@@ -23,26 +19,27 @@ module Gitlab
...
@@ -23,26 +19,27 @@ module Gitlab
private
private
def
each_suite
def
all_cases
(
root
,
parent
=
nil
,
&
blk
)
testsuites
.
each
do
|
testsuite
|
return
unless
root
.
present?
yield
testcases
(
testsuite
)
end
end
def
testsuites
[
root
].
flatten
.
compact
.
map
do
|
node
|
if
data
[
'testsuites'
]
next
unless
node
.
is_a?
(
Hash
)
data
[
'testsuites'
][
'testsuite'
]
else
# we allow only one top-level 'testsuites'
[
data
[
'testsuite'
]]
all_cases
(
node
[
'testsuites'
],
root
,
&
blk
)
unless
parent
# we require at least one level of testsuites or testsuite
each_case
(
node
[
'testcase'
],
&
blk
)
if
parent
# we allow multiple nested 'testsuite' (eg. PHPUnit)
all_cases
(
node
[
'testsuite'
],
root
,
&
blk
)
end
end
end
end
def
testcases
(
testsuite
)
def
each_case
(
testcase
,
&
blk
)
if
testsuite
[
'testcase'
].
is_a?
(
Array
)
return
unless
testcase
.
present?
testsuite
[
'testcase'
]
else
[
testcase
].
flatten
.
compact
.
map
(
&
blk
)
[
testsuite
[
'testcase'
]]
end
end
end
def
create_test_case
(
data
)
def
create_test_case
(
data
)
...
...
spec/lib/gitlab/ci/parsers/junit_spec.rb
View file @
c41492dd
require
'spec_helper'
require
'
fast_
spec_helper'
describe
Gitlab
::
Ci
::
Parsers
::
Junit
do
describe
Gitlab
::
Ci
::
Parsers
::
Junit
do
describe
'#parse!'
do
describe
'#parse!'
do
...
@@ -8,21 +8,35 @@ describe Gitlab::Ci::Parsers::Junit do
...
@@ -8,21 +8,35 @@ describe Gitlab::Ci::Parsers::Junit do
let
(
:test_cases
)
{
flattened_test_cases
(
test_suite
)
}
let
(
:test_cases
)
{
flattened_test_cases
(
test_suite
)
}
context
'when data is JUnit style XML'
do
context
'when data is JUnit style XML'
do
context
'when there are no
test cases
'
do
context
'when there are no
<testcases> in <testsuite>
'
do
let
(
:junit
)
do
let
(
:junit
)
do
<<-
EOF
.
strip_heredoc
<<-
EOF
.
strip_heredoc
<testsuite></testsuite>
<testsuite></testsuite>
EOF
EOF
end
end
it
'raises an error and does not add any test cases'
do
it
'ignores the case'
do
expect
{
subject
}.
to
raise_error
(
described_class
::
JunitParserError
)
expect
{
subject
}.
not_to
raise_error
expect
(
test_cases
.
count
).
to
eq
(
0
)
end
end
context
'when there are no <testcases> in <testsuites>'
do
let
(
:junit
)
do
<<-
EOF
.
strip_heredoc
<testsuites><testsuite /></testsuites>
EOF
end
it
'ignores the case'
do
expect
{
subject
}.
not_to
raise_error
expect
(
test_cases
.
count
).
to
eq
(
0
)
expect
(
test_cases
.
count
).
to
eq
(
0
)
end
end
end
end
context
'when there is
a test case
'
do
context
'when there is
only one <testcase> in <testsuite>
'
do
let
(
:junit
)
do
let
(
:junit
)
do
<<-
EOF
.
strip_heredoc
<<-
EOF
.
strip_heredoc
<testsuite>
<testsuite>
...
@@ -40,6 +54,46 @@ describe Gitlab::Ci::Parsers::Junit do
...
@@ -40,6 +54,46 @@ describe Gitlab::Ci::Parsers::Junit do
end
end
end
end
context
'when there is only one <testsuite> in <testsuites>'
do
let
(
:junit
)
do
<<-
EOF
.
strip_heredoc
<testsuites>
<testsuite>
<testcase classname='Calculator' name='sumTest1' time='0.01'></testcase>
</testsuite>
</testsuites>
EOF
end
it
'parses XML and adds a test case to a suite'
do
expect
{
subject
}.
not_to
raise_error
expect
(
test_cases
[
0
].
classname
).
to
eq
(
'Calculator'
)
expect
(
test_cases
[
0
].
name
).
to
eq
(
'sumTest1'
)
expect
(
test_cases
[
0
].
execution_time
).
to
eq
(
0.01
)
end
end
context
'PHPUnit'
do
let
(
:junit
)
do
<<-
EOF
.
strip_heredoc
<testsuites>
<testsuite name="Project Test Suite" tests="1" assertions="1" failures="0" errors="0" time="1.376748">
<testsuite name="XXX
\\
FrontEnd
\\
WebBundle
\\
Tests
\\
Controller
\\
LogControllerTest" file="/Users/mcfedr/projects/xxx/server/tests/XXX/FrontEnd/WebBundle/Tests/Controller/LogControllerTest.php" tests="1" assertions="1" failures="0" errors="0" time="1.376748">
<testcase name="testIndexAction" class="XXX
\\
FrontEnd
\\
WebBundle
\\
Tests
\\
Controller
\\
LogControllerTest" file="/Users/mcfedr/projects/xxx/server/tests/XXX/FrontEnd/WebBundle/Tests/Controller/LogControllerTest.php" line="9" assertions="1" time="1.376748"/>
</testsuite>
</testsuite>
</testsuites>
EOF
end
it
'parses XML and adds a test case to a suite'
do
expect
{
subject
}.
not_to
raise_error
expect
(
test_cases
.
count
).
to
eq
(
1
)
end
end
context
'when there are two test cases'
do
context
'when there are two test cases'
do
let
(
:junit
)
do
let
(
:junit
)
do
<<-
EOF
.
strip_heredoc
<<-
EOF
.
strip_heredoc
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment