Skip to content

Commit 2fe6710

Browse files
authored
Merge pull request #44 from buildkite/pie-3500-add-execution-level-tagging-for-pytest-test-collector
Add execution level tagging
2 parents 7feab6e + c819fe4 commit 2fe6710

File tree

4 files changed

+46
-2
lines changed

4 files changed

+46
-2
lines changed

src/buildkite_test_collector/collector/payload.py

+14-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
"""Buildkite Test Analytics payload"""
22

3-
from dataclasses import dataclass, replace
3+
from dataclasses import dataclass, replace, field
44
from typing import Dict, Tuple, Optional, Union, Literal
55
from datetime import timedelta
66
from uuid import UUID
@@ -107,12 +107,15 @@ def as_json(self, started_at: Instant) -> JsonDict:
107107
@dataclass(frozen=True)
108108
class TestData:
109109
"""An individual test execution"""
110+
# 8 attributes for this class seems reasonable
111+
# pylint: disable=too-many-instance-attributes
110112
id: UUID
111113
scope: str
112114
name: str
113115
history: TestHistory
114116
location: Optional[str] = None
115117
file_name: Optional[str] = None
118+
tags: Dict[str,str] = field(default_factory=dict)
116119
result: Union[TestResultPassed, TestResultFailed,
117120
TestResultSkipped, None] = None
118121

@@ -133,6 +136,13 @@ def start(cls, id: UUID,
133136
history=TestHistory(start_at=Instant.now())
134137
)
135138

139+
def tag_execution(self, key: str, val: str) -> 'TestData':
140+
"""Set tag to test execution"""
141+
if not isinstance(key, str) or not isinstance(val, str):
142+
raise TypeError("Expected string for key and value")
143+
144+
self.tags[key] = val
145+
136146
def finish(self) -> 'TestData':
137147
"""Set the end_at and duration on this test"""
138148
if self.is_finished():
@@ -175,6 +185,9 @@ def as_json(self, started_at: Instant) -> JsonDict:
175185
"history": self.history.as_json(started_at)
176186
}
177187

188+
if len(self.tags) > 0:
189+
attrs["tags"] = self.tags
190+
178191
if isinstance(self.result, TestResultPassed):
179192
attrs["result"] = "passed"
180193

src/buildkite_test_collector/pytest_plugin/__init__.py

+3
Original file line numberDiff line numberDiff line change
@@ -28,10 +28,13 @@ def pytest_configure(config):
2828
env = detect_env()
2929
debug = environ.get("BUILDKITE_ANALYTICS_DEBUG_ENABLED")
3030

31+
config.addinivalue_line("markers", "execution_tag(key, value): add tag to test execution for Buildkite Test Collector. Both key and value must be a string.")
32+
3133
if env:
3234
plugin = BuildkitePlugin(Payload.init(env))
3335
setattr(config, '_buildkite', plugin)
3436
config.pluginmanager.register(plugin)
37+
3538
elif debug:
3639
warning("Unable to detect CI environment. No test analytics will be sent.")
3740

src/buildkite_test_collector/pytest_plugin/buildkite_plugin.py

+10-1
Original file line numberDiff line numberDiff line change
@@ -29,9 +29,18 @@ def pytest_runtest_logstart(self, nodeid, location):
2929
)
3030
self.in_flight[nodeid] = test_data
3131

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+
3241
def pytest_runtest_logreport(self, report):
3342
"""pytest_runtest_logreport hook callback"""
34-
if report.when != 'call':
43+
if report.when != 'teardown':
3544
return
3645

3746
nodeid = report.nodeid

tests/buildkite_test_collector/collector/test_payload.py

+19
Original file line numberDiff line numberDiff line change
@@ -168,3 +168,22 @@ def test_test_data_as_json_when_skipped(skipped_test):
168168
json = skipped_test.as_json(Instant.now())
169169

170170
assert json["result"] == "skipped"
171+
172+
class TestTestDataTagExecution:
173+
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")
176+
177+
expected_tags = {"owner": "test-engine", "python.version": "3.12.3"}
178+
179+
assert successful_test.tags == expected_tags
180+
181+
json = successful_test.as_json(Instant.now())
182+
assert json["tags"] == {"owner": "test-engine", "python.version": "3.12.3"}
183+
184+
def test_test_data_tag_execution_non_string(self, successful_test):
185+
with pytest.raises(TypeError):
186+
successful_test.tag_execution("feature", True)
187+
188+
with pytest.raises(TypeError):
189+
successful_test.tag_execution(777, "lucky")

0 commit comments

Comments
 (0)