Skip to content

Commit 3f06c46

Browse files
authored
Merge pull request #45 from buildkite/pie-3532-failed-test-reported-as-passed
Fix the hook handling order that caused `failed` test to be reported as `passed`
2 parents 2fe6710 + 30dde3a commit 3f06c46

File tree

3 files changed

+32
-20
lines changed

3 files changed

+32
-20
lines changed

src/buildkite_test_collector/collector/payload.py

+3-1
Original file line numberDiff line numberDiff line change
@@ -141,7 +141,9 @@ def tag_execution(self, key: str, val: str) -> 'TestData':
141141
if not isinstance(key, str) or not isinstance(val, str):
142142
raise TypeError("Expected string for key and value")
143143

144-
self.tags[key] = val
144+
new_tags = self.tags.copy()
145+
new_tags[key] = val
146+
return replace(self, tags=new_tags)
145147

146148
def finish(self) -> 'TestData':
147149
"""Set the end_at and duration on this test"""

src/buildkite_test_collector/pytest_plugin/buildkite_plugin.py

+25-15
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
11
"""Buildkite test collector plugin for Pytest"""
22
import json
3-
43
from uuid import uuid4
54

65
from ..collector.payload import TestData
@@ -29,26 +28,21 @@ def pytest_runtest_logstart(self, nodeid, location):
2928
)
3029
self.in_flight[nodeid] = test_data
3130

32-
def pytest_runtest_teardown(self, item):
33-
"""pytest_runtest_hook hook callback to collect execution_tag"""
34-
test_data = self.in_flight.get(item.nodeid)
35-
36-
if test_data:
37-
tags = item.iter_markers("execution_tag")
38-
for tag in tags:
39-
test_data.tag_execution(tag.args[0], tag.args[1])
40-
4131
def pytest_runtest_logreport(self, report):
42-
"""pytest_runtest_logreport hook callback"""
43-
if report.when != 'teardown':
32+
"""pytest_runtest_logreport hook callback to get test outcome after test call"""
33+
34+
# This hook is called three times during the lifecycle of a test:
35+
# after the setup phase, the call phase, and the teardown phase.
36+
# Since we want to capture the outcome from the call phase,
37+
# we only proceed when this hook is triggered following the call phase.
38+
# See: https://github.com/buildkite/test-collector-python/pull/45
39+
if report.when != 'call':
4440
return
4541

4642
nodeid = report.nodeid
4743
test_data = self.in_flight.get(nodeid)
4844

4945
if test_data:
50-
test_data = test_data.finish()
51-
5246
if report.passed:
5347
test_data = test_data.passed()
5448

@@ -58,7 +52,23 @@ def pytest_runtest_logreport(self, report):
5852
if report.skipped:
5953
test_data = test_data.skipped()
6054

61-
del self.in_flight[nodeid]
55+
# TestData is immutable.
56+
# We need to replace the test_data in `in_flight` with updated test_data,
57+
# so we can get the correct result when we process it during the teardown hook.
58+
self.in_flight[nodeid] = test_data
59+
60+
def pytest_runtest_teardown(self, item):
61+
"""pytest_runtest_hook hook callback to mark test as finished and add it to the payload"""
62+
test_data = self.in_flight.get(item.nodeid)
63+
64+
if test_data:
65+
test_data = test_data.finish()
66+
67+
tags = item.iter_markers("execution_tag")
68+
for tag in tags:
69+
test_data = test_data.tag_execution(tag.args[0], tag.args[1])
70+
71+
del self.in_flight[item.nodeid]
6272
self.payload = self.payload.push_test_data(test_data)
6373

6474
def save_payload_as_json(self, path):

tests/buildkite_test_collector/collector/test_payload.py

+4-4
Original file line numberDiff line numberDiff line change
@@ -171,14 +171,14 @@ def test_test_data_as_json_when_skipped(skipped_test):
171171

172172
class TestTestDataTagExecution:
173173
def test_test_data_tag_execution(self, successful_test):
174-
successful_test.tag_execution("owner", "test-engine")
175-
successful_test.tag_execution("python.version", "3.12.3")
174+
test_data = successful_test.tag_execution("owner", "test-engine")
175+
test_data = test_data.tag_execution("python.version", "3.12.3")
176176

177177
expected_tags = {"owner": "test-engine", "python.version": "3.12.3"}
178178

179-
assert successful_test.tags == expected_tags
179+
assert test_data.tags == expected_tags
180180

181-
json = successful_test.as_json(Instant.now())
181+
json = test_data.as_json(Instant.now())
182182
assert json["tags"] == {"owner": "test-engine", "python.version": "3.12.3"}
183183

184184
def test_test_data_tag_execution_non_string(self, successful_test):

0 commit comments

Comments
 (0)