Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

chore(tracing): fix inferred base service for pytest command ran with --ddtrace optional argument #11394

Merged
merged 24 commits into from
Nov 19, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
24 commits
Select commit Hold shift + click to select a range
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
The table of contents is too big for display.
Diff view
Diff view
  •  
  •  
  •  
7 changes: 5 additions & 2 deletions ddtrace/contrib/internal/mako/patch.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@

from ddtrace import config
from ddtrace.constants import SPAN_MEASURED_KEY
from ddtrace.contrib.trace_utils import int_service
from ddtrace.contrib.trace_utils import unwrap as _u
from ddtrace.contrib.trace_utils import wrap as _w
from ddtrace.ext import SpanTypes
Expand All @@ -26,7 +27,7 @@ def patch():
return
mako.__datadog_patch = True

Pin(service=config.service or schematize_service_name("mako")).onto(Template)
Pin().onto(Template)
wconti27 marked this conversation as resolved.
Show resolved Hide resolved

_w(mako, "template.Template.render", _wrap_render)
_w(mako, "template.Template.render_unicode", _wrap_render)
Expand Down Expand Up @@ -57,7 +58,9 @@ def _wrap_render(wrapped, instance, args, kwargs):
template_name = getattr(instance, "filename", None)
template_name = template_name or DEFAULT_TEMPLATE_NAME

with pin.tracer.trace(func_name(wrapped), pin.service, span_type=SpanTypes.TEMPLATE) as span:
with pin.tracer.trace(
func_name(wrapped), int_service(pin, config.mako, schematize_service_name("mako")), span_type=SpanTypes.TEMPLATE
) as span:
span.set_tag_str(COMPONENT, "mako")

span.set_tag(SPAN_MEASURED_KEY)
Expand Down
11 changes: 9 additions & 2 deletions ddtrace/sampler.py
Original file line number Diff line number Diff line change
Expand Up @@ -134,10 +134,17 @@ def __init__(self, sample_rate=1.0):
def set_sample_rate(
self,
sample_rate, # type: float
service="", # type: str
env="", # type: str
service=None, # type: Optional[str]
env=None, # type: Optional[str]
):
# type: (...) -> None

# if we have a blank service, we need to match it to the config.service
if service is None:
service = config.service
if env is None:
env = config.env

self._by_service_samplers[self._key(service, env)] = _AgentRateSampler(sample_rate)

def sample(self, span):
Expand Down
4 changes: 3 additions & 1 deletion ddtrace/settings/_inferred_base_service.py
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,9 @@ def detect(self, args: List[str]) -> Optional[ServiceMetadata]:
module_flag = False

for arg in args:
has_flag_prefix = arg.startswith("-")
# we support the --ddtrace option for pytest, and shouldn't skip the following arg
# since it's usually the test location argument.
has_flag_prefix = arg.startswith("-") and not arg.startswith("--ddtrace")
is_env_variable = "=" in arg

should_skip_arg = prev_arg_is_flag or has_flag_prefix or is_env_variable
Expand Down
7 changes: 4 additions & 3 deletions ddtrace/settings/config.py
Original file line number Diff line number Diff line change
Expand Up @@ -474,15 +474,16 @@ def __init__(self):
self._propagation_http_baggage_enabled = _get_config("DD_TRACE_PROPAGATION_HTTP_BAGGAGE_ENABLED", False, asbool)

self.env = _get_config("DD_ENV", self.tags.get("env"))
self.service = _get_config("DD_SERVICE", self.tags.get("service", DEFAULT_SPAN_SERVICE_NAME))
self.service = _get_config("DD_SERVICE", self.tags.get("service", None))

self._inferred_base_service = detect_service(sys.argv)

if self.service is None and in_gcp_function():
self.service = _get_config(["K_SERVICE", "FUNCTION_NAME"], DEFAULT_SPAN_SERVICE_NAME)
if self.service is None and in_azure_function():
self.service = _get_config("WEBSITE_SITE_NAME", DEFAULT_SPAN_SERVICE_NAME)
if self.service is None and self._inferred_base_service:
self.service = self._inferred_base_service
if self.service is None and DEFAULT_SPAN_SERVICE_NAME:
self.service = _get_config("DD_SERVICE", DEFAULT_SPAN_SERVICE_NAME)

self._extra_services = set()
self._extra_services_queue = None if in_aws_lambda() or not self._remote_config_enabled else File_Queue()
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
---
fixes:
- |
ci_visibility: Updates the inferred base service name algorithm to ensure that arguments following ``--ddtrace`` are no
longer skipped when executing tests with pytest. Previously, the algorithm misinterpreted these arguments
as standard flags, overlooking possible test paths that may contribute to the inferred service name.
28 changes: 14 additions & 14 deletions tests/contrib/anthropic/test_anthropic_llmobs.py
Original file line number Diff line number Diff line change
Expand Up @@ -68,7 +68,7 @@ def test_completion(self, anthropic, ddtrace_global_config, mock_llmobs_writer,
output_messages=[{"content": 'THE BEST-SELLING BOOK OF ALL TIME IS "DON', "role": "assistant"}],
metadata={"temperature": 0.8, "max_tokens": 15.0},
token_metrics={"input_tokens": 32, "output_tokens": 15, "total_tokens": 47},
tags={"ml_app": "<ml-app-name>"},
tags={"ml_app": "<ml-app-name>", "service": "tests.contrib.anthropic"},
)
)

Expand Down Expand Up @@ -113,7 +113,7 @@ def test_error(self, anthropic, ddtrace_global_config, mock_llmobs_writer, mock_
error_message=span.get_tag("error.message"),
error_stack=span.get_tag("error.stack"),
metadata={"temperature": 0.8, "max_tokens": 15.0},
tags={"ml_app": "<ml-app-name>"},
tags={"ml_app": "<ml-app-name>", "service": "tests.contrib.anthropic"},
)
)

Expand Down Expand Up @@ -142,7 +142,7 @@ def test_error_unserializable_arg(
error_message=span.get_tag("error.message"),
error_stack=span.get_tag("error.stack"),
metadata={"temperature": 0.8, "max_tokens": mock.ANY},
tags={"ml_app": "<ml-app-name>"},
tags={"ml_app": "<ml-app-name>", "service": "tests.contrib.anthropic"},
)
mock_llmobs_writer.enqueue.assert_called_with(expected_span)
actual_span = mock_llmobs_writer.enqueue.call_args[0][0]
Expand Down Expand Up @@ -196,7 +196,7 @@ def test_stream(self, anthropic, ddtrace_global_config, mock_llmobs_writer, mock
],
metadata={"temperature": 0.8, "max_tokens": 15.0},
token_metrics={"input_tokens": 27, "output_tokens": 15, "total_tokens": 42},
tags={"ml_app": "<ml-app-name>"},
tags={"ml_app": "<ml-app-name>", "service": "tests.contrib.anthropic"},
)
)

Expand Down Expand Up @@ -253,7 +253,7 @@ def test_stream_helper(self, anthropic, ddtrace_global_config, mock_llmobs_write
],
metadata={"temperature": 0.8, "max_tokens": 15.0},
token_metrics={"input_tokens": 27, "output_tokens": 15, "total_tokens": 42},
tags={"ml_app": "<ml-app-name>"},
tags={"ml_app": "<ml-app-name>", "service": "tests.contrib.anthropic"},
)
)

Expand Down Expand Up @@ -308,7 +308,7 @@ def test_image(self, anthropic, ddtrace_global_config, mock_llmobs_writer, mock_
],
metadata={"temperature": 0.8, "max_tokens": 15.0},
token_metrics={"input_tokens": 246, "output_tokens": 15, "total_tokens": 261},
tags={"ml_app": "<ml-app-name>"},
tags={"ml_app": "<ml-app-name>", "service": "tests.contrib.anthropic"},
)
)

Expand Down Expand Up @@ -350,7 +350,7 @@ def test_tools_sync(self, anthropic, ddtrace_global_config, mock_llmobs_writer,
],
metadata={"max_tokens": 200.0},
token_metrics={"input_tokens": 599, "output_tokens": 152, "total_tokens": 751},
tags={"ml_app": "<ml-app-name>"},
tags={"ml_app": "<ml-app-name>", "service": "tests.contrib.anthropic"},
)
)

Expand Down Expand Up @@ -403,7 +403,7 @@ def test_tools_sync(self, anthropic, ddtrace_global_config, mock_llmobs_writer,
],
metadata={"max_tokens": 500.0},
token_metrics={"input_tokens": 768, "output_tokens": 29, "total_tokens": 797},
tags={"ml_app": "<ml-app-name>"},
tags={"ml_app": "<ml-app-name>", "service": "tests.contrib.anthropic"},
)
)

Expand Down Expand Up @@ -446,7 +446,7 @@ async def test_tools_async(self, anthropic, ddtrace_global_config, mock_llmobs_w
],
metadata={"max_tokens": 200.0},
token_metrics={"input_tokens": 599, "output_tokens": 152, "total_tokens": 751},
tags={"ml_app": "<ml-app-name>"},
tags={"ml_app": "<ml-app-name>", "service": "tests.contrib.anthropic"},
)
)

Expand Down Expand Up @@ -499,7 +499,7 @@ async def test_tools_async(self, anthropic, ddtrace_global_config, mock_llmobs_w
],
metadata={"max_tokens": 500.0},
token_metrics={"input_tokens": 768, "output_tokens": 29, "total_tokens": 797},
tags={"ml_app": "<ml-app-name>"},
tags={"ml_app": "<ml-app-name>", "service": "tests.contrib.anthropic"},
)
)

Expand Down Expand Up @@ -559,7 +559,7 @@ def test_tools_sync_stream(self, anthropic, ddtrace_global_config, mock_llmobs_w
],
metadata={"max_tokens": 200.0},
token_metrics={"input_tokens": 599, "output_tokens": 135, "total_tokens": 734},
tags={"ml_app": "<ml-app-name>"},
tags={"ml_app": "<ml-app-name>", "service": "tests.contrib.anthropic"},
)
)

Expand Down Expand Up @@ -609,7 +609,7 @@ def test_tools_sync_stream(self, anthropic, ddtrace_global_config, mock_llmobs_w
],
metadata={"max_tokens": 500.0},
token_metrics={"input_tokens": 762, "output_tokens": 33, "total_tokens": 795},
tags={"ml_app": "<ml-app-name>"},
tags={"ml_app": "<ml-app-name>", "service": "tests.contrib.anthropic"},
)
)

Expand Down Expand Up @@ -664,7 +664,7 @@ async def test_tools_async_stream_helper(
],
metadata={"max_tokens": 200.0},
token_metrics={"input_tokens": 599, "output_tokens": 146, "total_tokens": 745},
tags={"ml_app": "<ml-app-name>"},
tags={"ml_app": "<ml-app-name>", "service": "tests.contrib.anthropic"},
)
)

Expand Down Expand Up @@ -719,6 +719,6 @@ async def test_tools_async_stream_helper(
],
metadata={"max_tokens": 500.0},
token_metrics={"input_tokens": 762, "output_tokens": 18, "total_tokens": 780},
tags={"ml_app": "<ml-app-name>"},
tags={"ml_app": "<ml-app-name>", "service": "tests.contrib.anthropic"},
)
)
4 changes: 2 additions & 2 deletions tests/contrib/django/test_django.py
Original file line number Diff line number Diff line change
Expand Up @@ -1425,14 +1425,14 @@ def test_cached_view(client, test_spans):
"django.cache.key": (
"views.decorators.cache.cache_page..GET.03cdc1cc4aab71b038a6764e5fcabb82.d41d8cd98f00b204e9800998ecf8..."
),
"_dd.base_service": "",
"_dd.base_service": "tests.contrib.django",
}

expected_meta_header = {
"component": "django",
"django.cache.backend": "django.core.cache.backends.locmem.LocMemCache",
"django.cache.key": "views.decorators.cache.cache_header..03cdc1cc4aab71b038a6764e5fcabb82.en-us",
"_dd.base_service": "",
"_dd.base_service": "tests.contrib.django",
}

assert span_view.get_tags() == expected_meta_view
Expand Down
4 changes: 2 additions & 2 deletions tests/contrib/flask/test_blueprint.py
Original file line number Diff line number Diff line change
Expand Up @@ -104,7 +104,7 @@ def test():
self.assertEqual(span.name, "bp.test")
self.assertEqual(span.resource, "/")
self.assertNotEqual(span.parent_id, 0)
self.assertEqual(span.get_tags(), {"component": "flask", "_dd.base_service": ""})
self.assertEqual(span.get_tags(), {"component": "flask", "_dd.base_service": "tests.contrib.flask"})

def test_blueprint_request_pin_override(self):
"""
Expand Down Expand Up @@ -132,7 +132,7 @@ def test():
self.assertEqual(span.name, "bp.test")
self.assertEqual(span.resource, "/")
self.assertNotEqual(span.parent_id, 0)
self.assertEqual(span.get_tags(), {"component": "flask", "_dd.base_service": ""})
self.assertEqual(span.get_tags(), {"component": "flask", "_dd.base_service": "tests.contrib.flask"})

def test_blueprint_request_pin_disabled(self):
"""
Expand Down
2 changes: 1 addition & 1 deletion tests/contrib/flask/test_errorhandler.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@
from . import BaseFlaskTestCase


EXPECTED_METADATA = {"component": "flask", "_dd.base_service": ""}
EXPECTED_METADATA = {"component": "flask", "_dd.base_service": "tests.contrib.flask"}


class FlaskErrorhandlerTestCase(BaseFlaskTestCase):
Expand Down
2 changes: 1 addition & 1 deletion tests/contrib/flask/test_flask_helpers.py
Original file line number Diff line number Diff line change
Expand Up @@ -49,7 +49,7 @@ def test_jsonify(self):
spans = self.get_spans()
self.assertEqual(len(spans), 3)

self.assertIsNone(spans[0].service)
self.assertEqual(spans[0].service, "tests.contrib.flask")
self.assertEqual(spans[0].name, "flask.jsonify")
self.assertEqual(spans[0].resource, "flask.jsonify")
assert set(spans[0].get_tags().keys()) == {"runtime-id", "_dd.p.dm", "_dd.p.tid", "component", "language"}
Expand Down
4 changes: 2 additions & 2 deletions tests/contrib/flask/test_template.py
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,7 @@ def test_render_template(self):
spans = self.get_spans()
self.assertEqual(len(spans), 3)

self.assertIsNone(spans[0].service)
self.assertEqual(spans[0].service, "tests.contrib.flask")
self.assertEqual(spans[0].name, "flask.render_template")
resource = "tests.contrib.flask" if flask_version >= (2, 2, 0) else "test.html"
self.assertEqual(spans[0].resource, resource) # FIXME: should always be 'test.html'?
Expand Down Expand Up @@ -92,7 +92,7 @@ def test_render_template_string(self):
spans = self.get_spans()
self.assertEqual(len(spans), 3)

self.assertIsNone(spans[0].service)
self.assertEqual(spans[0].service, "tests.contrib.flask")
self.assertEqual(spans[0].name, "flask.render_template_string")
resource = "tests.contrib.flask" if flask_version >= (2, 2, 0) else "<memory>"
self.assertEqual(spans[0].resource, resource) # FIXME: should always be '<memory>'?
Expand Down
2 changes: 1 addition & 1 deletion tests/contrib/flask/test_views.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@

base_exception_name = "builtins.Exception"

EXPECTED_METADATA = {"component": "flask", "_dd.base_service": ""}
EXPECTED_METADATA = {"component": "flask", "_dd.base_service": "tests.contrib.flask"}


class FlaskViewTestCase(BaseFlaskTestCase):
Expand Down
Loading
Loading