diff --git a/ddtrace/contrib/pytest/instrument.py b/ddtrace/contrib/pytest/instrument.py index d4428cf9c0d..bf8d947f809 100644 --- a/ddtrace/contrib/pytest/instrument.py +++ b/ddtrace/contrib/pytest/instrument.py @@ -59,17 +59,20 @@ def after_import(self, module): @classmethod def _trace(cls, f, args, kwargs): + # We may not have a tracer if code is in stdlib: + if cls._instance is None: + return f(*args, **kwargs) with cls._instance._tracer.trace( - name=f.__code__.co_name, resource=f.__code__.co_qualname, service=f.__module__ + name=f.__code__.co_name, resource=f.__code__.co_name, service=f.__module__ ) as span: collector = DynamicInstrumentation._instance._collector - message = "test" + message = "%s %s %s" % (f.__code__.co_name, f.__code__.co_name, f.__module__) probe = LogFunctionProbe( probe_id=str(uuid.uuid4()), version=0, tags={}, module=f.__module__, - func_qname=f.__code__.co_qualname, + func_qname=f.__code__.co_name, template=message, segments=[LiteralTemplateSegment(message)], take_snapshot=True, diff --git a/ddtrace/contrib/pytest/plugin.py b/ddtrace/contrib/pytest/plugin.py index 84353533675..5c7e30f5c91 100644 --- a/ddtrace/contrib/pytest/plugin.py +++ b/ddtrace/contrib/pytest/plugin.py @@ -59,6 +59,9 @@ DD_SPANS_STASH_KEY = StashKey[Dict] DD_FAILED_TESTS_STASH_KEY = StashKey[Set] +# Hack: this is to get around conftest importing modules (eg Flask) and getting recorded too late +ModuleCollector.install() + def encode_test_parameter(parameter): param_repr = repr(parameter) @@ -323,7 +326,10 @@ def pytest_configure(config): from ddtrace.debugging._exception import auto_instrument auto_instrument.GLOBAL_RATE_LIMITER = RateLimiter(limit_rate=float(math.inf), raise_on_exceed=False) - _DynamicInstrumentation.enable() + _DynamicInstrumentation.enable(tracer=_CIVisibility._instance.tracer) + # Moved to module-level to collect earlier in Flask's case + # if config.getoption("ddtrace-instrument-tests"): + # ModuleCollector.install() def pytest_sessionstart(session): @@ -352,9 +358,6 @@ def pytest_sessionstart(session): session.stash[DD_FAILED_TESTS_STASH_KEY] = set() _store_span(session, test_session_span) - if session.config.getoption("ddtrace-instrument-tests"): - ModuleCollector.install() - def pytest_sessionfinish(session, exitstatus): if _CIVisibility.enabled: @@ -367,28 +370,37 @@ def pytest_sessionfinish(session, exitstatus): ) test_session_span.set_metric(test.ITR_TEST_SKIPPING_COUNT, _global_skipped_elements) _mark_test_status(session, test_session_span) - from pprint import pprint - - pprint(session.stash[DD_SPANS_STASH_KEY]) - pprint(session.stash[DD_FAILED_TESTS_STASH_KEY]) - # breakpoint() - if session.stash[DD_FAILED_TESTS_STASH_KEY]: - print("ROMAIN IS RERUNNING TESTS") - print("INSTRUMENTING") - # ModuleCollector.instrument(_CIVisibility._instance.tracer) - print("INSTRUMENTED") - for failed_test in session.stash[DD_FAILED_TESTS_STASH_KEY]: - from _pytest.runner import runtestprotocol - - failed_test.ihook.pytest_runtest_logstart(nodeid=failed_test.nodeid, location=failed_test.location) - reports = runtestprotocol(failed_test, failed_test) + + # Hack: retry failed tests if instrumentation is on: + if session.config.getoption("ddtrace-instrument-tests"): + if session.stash[DD_FAILED_TESTS_STASH_KEY]: + ModuleCollector.instrument(tracer=_CIVisibility._instance.tracer) + for failed_test in session.stash[DD_FAILED_TESTS_STASH_KEY]: + from _pytest.runner import runtestprotocol + + with _CIVisibility._instance.tracer._start_span( + ddtrace.config.pytest.operation_name, + service=_CIVisibility._instance._service, + resource=failed_test.nodeid, + span_type=SpanTypes.TEST, + activate=True, + ) as span: + failed_span = _extract_span(failed_test) + span.set_tags(failed_span.get_tags()) + span.set_tag(test.NAME, failed_test.name + "_rerun") + # More hack: don't add logs, but this doesn't really improve output at all + # failed_test.ihook.pytest_runtest_logstart(nodeid=failed_test.nodeid, location=failed_test.location) + reports = runtestprotocol(failed_test, failed_test) + # failed_test.ihook.pytest_runtest_logfinish(nodeid=failed_test.nodeid, location=failed_test.location) test_session_span.finish() - _CIVisibility.disable() - _DynamicInstrumentation.disable() - if session.config.getoption("ddtrace-instrument-tests"): - ModuleCollector.uninstall() + # Turn off extra bits that were turned on + if session.config.getoption("ddtrace-instrument-tests"): + ModuleCollector.uninstall() + _DynamicInstrumentation.disable() + _CIVisibility._instance.tracer.flush() + _CIVisibility.disable() @pytest.fixture(scope="function") @@ -449,7 +461,6 @@ def pytest_collection_modifyitems(session, config, items): @pytest.hookimpl(tryfirst=True, hookwrapper=True) def pytest_runtest_protocol(item, nextitem): - print("ROMAIN SAYS WE RAN OUR PROTOCOL") if not _CIVisibility.enabled: yield return @@ -676,8 +687,3 @@ def pytest_runtest_makereport(item, call): span.set_tag_str(test.RESULT, test.Status.XPASS.value) if call.excinfo: span.set_exc_info(call.excinfo.type, call.excinfo.value, call.excinfo.tb) - - -def pytest_collection_finish(session): - if session.config.getoption("ddtrace-instrument-tests"): - ModuleCollector.instrument(_CIVisibility._instance.tracer) diff --git a/ddtrace/debugging/_debugger.py b/ddtrace/debugging/_debugger.py index 22f9a32cd20..7b8823c6bef 100644 --- a/ddtrace/debugging/_debugger.py +++ b/ddtrace/debugging/_debugger.py @@ -173,8 +173,8 @@ class Debugger(Service): __logger__ = ProbeStatusLogger @classmethod - def enable(cls, run_module=False): - # type: (bool) -> None + def enable(cls, run_module=False, tracer=None): + # type: (bool, Optional[Tracer] ) -> None """Enable dynamic instrumentation This class method is idempotent. Dynamic instrumentation will be @@ -191,7 +191,7 @@ def enable(cls, run_module=False): if di_config.metrics: metrics.enable() - cls._instance = debugger = cls() + cls._instance = debugger = cls(tracer=tracer) debugger.start()