From 9254c170d644c853f7824e27e733e0930b4ee8f2 Mon Sep 17 00:00:00 2001 From: Owais Lone Date: Mon, 8 Mar 2021 23:04:01 +0530 Subject: [PATCH] Move `use_span()` method from Tracer to `opentelemetry.trace.use_span()` (#1668) Fixes: #1630 --- .github/workflows/test.yml | 2 +- CHANGELOG.md | 5 ++ docs/examples/basic_context/async_context.py | 2 +- .../src/opentelemetry/trace/__init__.py | 85 ++++++++++++------- opentelemetry-api/tests/trace/test_globals.py | 71 ++++++++++++++++ opentelemetry-api/tests/trace/test_tracer.py | 5 -- .../src/opentelemetry/sdk/trace/__init__.py | 50 ++--------- opentelemetry-sdk/tests/trace/test_trace.py | 24 ++---- .../shim/opentracing_shim/__init__.py | 11 ++- 9 files changed, 151 insertions(+), 104 deletions(-) diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index 73baecb7851..12f49c14914 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -10,7 +10,7 @@ env: # Otherwise, set variable to the commit of your branch on # opentelemetry-python-contrib which is compatible with these Core repo # changes. - CONTRIB_REPO_SHA: 16ae58b341b960df52c1cf4541695164821b3638 + CONTRIB_REPO_SHA: 8783e0ff97ad123006ff1ff2c2cf3f52161a406f jobs: build: diff --git a/CHANGELOG.md b/CHANGELOG.md index 47db7726653..cd126d8a52c 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -9,6 +9,11 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ### Added - Add `max_attr_value_length` support to Jaeger exporter ([#1633])(https://github.com/open-telemetry/opentelemetry-python/pull/1633) +- Moved `use_span` from Tracer to `opentelemetry.trace.use_span`. + ([#1668](https://github.com/open-telemetry/opentelemetry-python/pull/1668)) +- `opentelemetry.trace.use_span()` will now overwrite previously set status on span in case an + exception is raised inside the context manager and `set_status_on_exception` is set to `True`. + ([#1668](https://github.com/open-telemetry/opentelemetry-python/pull/1668)) ### Changed - Rename `IdsGenerator` to `IdGenerator` diff --git a/docs/examples/basic_context/async_context.py b/docs/examples/basic_context/async_context.py index 41e049e2a6b..d80ccb31e01 100644 --- a/docs/examples/basic_context/async_context.py +++ b/docs/examples/basic_context/async_context.py @@ -24,7 +24,7 @@ async def async_span(span): - with tracer.use_span(span): + with trace.use_span(span): ctx = baggage.set_baggage("foo", "bar") return ctx diff --git a/opentelemetry-api/src/opentelemetry/trace/__init__.py b/opentelemetry-api/src/opentelemetry/trace/__init__.py index 54b5abd87b8..6e32fb352d2 100644 --- a/opentelemetry-api/src/opentelemetry/trace/__init__.py +++ b/opentelemetry-api/src/opentelemetry/trace/__init__.py @@ -80,9 +80,11 @@ from logging import getLogger from typing import Iterator, Optional, Sequence, cast +from opentelemetry import context as context_api from opentelemetry.context.context import Context from opentelemetry.environment_variables import OTEL_PYTHON_TRACER_PROVIDER from opentelemetry.trace.propagation import ( + SPAN_KEY, get_current_span, set_span_in_context, ) @@ -101,7 +103,7 @@ format_span_id, format_trace_id, ) -from opentelemetry.trace.status import Status +from opentelemetry.trace.status import Status, StatusCode from opentelemetry.util import types from opentelemetry.util._providers import _load_provider @@ -324,7 +326,7 @@ def start_as_current_span( is equivalent to:: span = tracer.start_span(name) - with tracer.use_span(span, end_on_exit=True): + with opentelemetry.trace.use_span(span, end_on_exit=True): do_work() Args: @@ -350,28 +352,6 @@ def start_as_current_span( The newly-created span. """ - @contextmanager # type: ignore - @abstractmethod - def use_span( - self, span: "Span", end_on_exit: bool = False, - ) -> Iterator[None]: - """Context manager for setting the passed span as the - current span in the context, as well as resetting the - context back upon exiting the context manager. - - Set the given span as the current span in this tracer's context. - - On exiting the context manager set the span that was previously active - as the current span (this is usually but not necessarily the parent of - the given span). If ``end_on_exit`` is ``True``, then the span is also - ended when exiting the context manager. - - Args: - span: The span to start and make current. - end_on_exit: Whether to end the span automatically when leaving the - context manager. - """ - class DefaultTracer(Tracer): """The default Tracer, used when no Tracer implementation is available. @@ -409,13 +389,6 @@ def start_as_current_span( # pylint: disable=unused-argument,no-self-use yield INVALID_SPAN - @contextmanager # type: ignore - def use_span( - self, span: "Span", end_on_exit: bool = False, - ) -> Iterator[None]: - # pylint: disable=unused-argument,no-self-use - yield - _TRACER_PROVIDER = None @@ -466,6 +439,55 @@ def get_tracer_provider() -> TracerProvider: return _TRACER_PROVIDER +@contextmanager # type: ignore +def use_span( + span: Span, + end_on_exit: bool = False, + record_exception: bool = True, + set_status_on_exception: bool = True, +) -> Iterator[Span]: + """Takes a non-active span and activates it in the current context. + + Args: + span: The span that should be activated in the current context. + end_on_exit: Whether to end the span automatically when leaving the + context manager scope. + record_exception: Whether to record any exceptions raised within the + context as error event on the span. + set_status_on_exception: Only relevant if the returned span is used + in a with/context manager. Defines wether the span status will + be automatically set to ERROR when an uncaught exception is + raised in the span with block. The span status won't be set by + this mechanism if it was previously set manually. + """ + try: + token = context_api.attach(context_api.set_value(SPAN_KEY, span)) + try: + yield span + finally: + context_api.detach(token) + + except Exception as exc: # pylint: disable=broad-except + if isinstance(span, Span) and span.is_recording(): + # Record the exception as an event + if record_exception: + span.record_exception(exc) + + # Set status in case exception was raised + if set_status_on_exception: + span.set_status( + Status( + status_code=StatusCode.ERROR, + description="{}: {}".format(type(exc).__name__, exc), + ) + ) + raise + + finally: + if end_on_exit: + span.end() + + __all__ = [ "DEFAULT_TRACE_OPTIONS", "DEFAULT_TRACE_STATE", @@ -492,5 +514,6 @@ def get_tracer_provider() -> TracerProvider: "get_tracer_provider", "set_tracer_provider", "set_span_in_context", + "use_span", "Status", ] diff --git a/opentelemetry-api/tests/trace/test_globals.py b/opentelemetry-api/tests/trace/test_globals.py index 55b040f2bc2..a6cfb9eb2ce 100644 --- a/opentelemetry-api/tests/trace/test_globals.py +++ b/opentelemetry-api/tests/trace/test_globals.py @@ -2,6 +2,27 @@ from unittest.mock import patch from opentelemetry import context, trace +from opentelemetry.trace.status import Status, StatusCode + + +class TestSpan(trace.NonRecordingSpan): + has_ended = False + recorded_exception = None + recorded_status = Status(status_code=StatusCode.UNSET) + + def set_status(self, status): + self.recorded_status = status + + def end(self, end_time=None): + self.has_ended = True + + def is_recording(self): + return not self.has_ended + + def record_exception( + self, exception, attributes=None, timestamp=None, escaped=False + ): + self.recorded_exception = exception class TestGlobals(unittest.TestCase): @@ -38,3 +59,53 @@ def test_get_current_span(self): finally: context.detach(token) self.assertEqual(trace.get_current_span(), trace.INVALID_SPAN) + + +class TestUseTracer(unittest.TestCase): + def test_use_span(self): + self.assertEqual(trace.get_current_span(), trace.INVALID_SPAN) + span = trace.NonRecordingSpan(trace.INVALID_SPAN_CONTEXT) + with trace.use_span(span): + self.assertIs(trace.get_current_span(), span) + self.assertEqual(trace.get_current_span(), trace.INVALID_SPAN) + + def test_use_span_end_on_exit(self): + + test_span = TestSpan(trace.INVALID_SPAN_CONTEXT) + + with trace.use_span(test_span): + pass + self.assertFalse(test_span.has_ended) + + with trace.use_span(test_span, end_on_exit=True): + pass + self.assertTrue(test_span.has_ended) + + def test_use_span_exception(self): + class TestUseSpanException(Exception): + pass + + test_span = TestSpan(trace.INVALID_SPAN_CONTEXT) + exception = TestUseSpanException("test exception") + with self.assertRaises(TestUseSpanException): + with trace.use_span(test_span): + raise exception + + self.assertEqual(test_span.recorded_exception, exception) + + def test_use_span_set_status(self): + class TestUseSpanException(Exception): + pass + + test_span = TestSpan(trace.INVALID_SPAN_CONTEXT) + with self.assertRaises(TestUseSpanException): + with trace.use_span(test_span): + raise TestUseSpanException("test error") + + self.assertEqual( + test_span.recorded_status.status_code, StatusCode.ERROR + ) + self.assertEqual( + test_span.recorded_status.description, + "TestUseSpanException: test error", + ) diff --git a/opentelemetry-api/tests/trace/test_tracer.py b/opentelemetry-api/tests/trace/test_tracer.py index 0ae3a0505b0..d0c451e1f24 100644 --- a/opentelemetry-api/tests/trace/test_tracer.py +++ b/opentelemetry-api/tests/trace/test_tracer.py @@ -29,11 +29,6 @@ def test_start_as_current_span(self): with self.tracer.start_as_current_span("") as span: self.assertIsInstance(span, trace.Span) - def test_use_span(self): - span = trace.NonRecordingSpan(trace.INVALID_SPAN_CONTEXT) - with self.tracer.use_span(span): - pass - def test_get_current_span(self): with self.tracer.start_as_current_span("test") as span: trace.get_current_span().set_attribute("test", "test") diff --git a/opentelemetry-sdk/src/opentelemetry/sdk/trace/__init__.py b/opentelemetry-sdk/src/opentelemetry/sdk/trace/__init__.py index f7b3ecd5668..6fc2fcddd36 100644 --- a/opentelemetry-sdk/src/opentelemetry/sdk/trace/__init__.py +++ b/opentelemetry-sdk/src/opentelemetry/sdk/trace/__init__.py @@ -776,10 +776,7 @@ def __exit__( self.record_exception(exception=exc_val, escaped=True) # Records status if span is used as context manager # i.e. with tracer.start_span() as span: - if ( - self._status.status_code is StatusCode.UNSET - and self._set_status_on_exception - ): + if self._set_status_on_exception: self.set_status( Status( status_code=StatusCode.ERROR, @@ -869,7 +866,12 @@ def start_as_current_span( record_exception=record_exception, set_status_on_exception=set_status_on_exception, ) - with self.use_span(span, end_on_exit=end_on_exit) as span_context: + with trace_api.use_span( + span, + end_on_exit=end_on_exit, + record_exception=record_exception, + set_status_on_exception=set_status_on_exception, + ) as span_context: yield span_context def start_span( # pylint: disable=too-many-locals @@ -950,44 +952,6 @@ def start_span( # pylint: disable=too-many-locals span = trace_api.NonRecordingSpan(context=span_context) return span - @contextmanager - def use_span( - self, span: trace_api.Span, end_on_exit: bool = False, - ) -> Iterator[trace_api.Span]: - try: - token = context_api.attach(context_api.set_value(SPAN_KEY, span)) - try: - yield span - finally: - context_api.detach(token) - - except Exception as exc: # pylint: disable=broad-except - # Record the exception as an event - if isinstance(span, Span) and span.is_recording(): - # pylint:disable=protected-access - if span._record_exception: - span.record_exception(exc) - - # Records status if use_span is used - # i.e. with tracer.start_as_current_span() as span: - if ( - span._status.status_code is StatusCode.UNSET - and span._set_status_on_exception - ): - span.set_status( - Status( - status_code=StatusCode.ERROR, - description="{}: {}".format( - type(exc).__name__, exc - ), - ) - ) - raise - - finally: - if end_on_exit: - span.end() - class TracerProvider(trace_api.TracerProvider): """See `opentelemetry.trace.TracerProvider`.""" diff --git a/opentelemetry-sdk/tests/trace/test_trace.py b/opentelemetry-sdk/tests/trace/test_trace.py index e2acad09b7b..2bcc1bdd19a 100644 --- a/opentelemetry-sdk/tests/trace/test_trace.py +++ b/opentelemetry-sdk/tests/trace/test_trace.py @@ -122,18 +122,6 @@ def run_general_code(shutdown_on_exit, explicit_shutdown): out = run_general_code(False, False) self.assertTrue(out.startswith(b"0")) - def test_use_span_exception(self): - class TestUseSpanException(Exception): - pass - - default_span = trace_api.NonRecordingSpan( - trace_api.INVALID_SPAN_CONTEXT - ) - tracer = new_tracer() - with self.assertRaises(TestUseSpanException): - with tracer.use_span(default_span): - raise TestUseSpanException() - def test_tracer_provider_accepts_concurrent_multi_span_processor(self): span_processor = trace.ConcurrentMultiSpanProcessor(2) tracer_provider = trace.TracerProvider( @@ -307,7 +295,7 @@ def test_start_span_implicit(self): self.assertIsNone(root.end_time) self.assertEqual(root.kind, trace_api.SpanKind.INTERNAL) - with tracer.use_span(root, True): + with trace_api.use_span(root, True): self.assertIs(trace_api.get_current_span(), root) with tracer.start_span( @@ -364,7 +352,7 @@ def test_start_span_explicit(self): self.assertIsNone(root.end_time) # Test with the implicit root span - with tracer.use_span(root, True): + with trace_api.use_span(root, True): self.assertIs(trace_api.get_current_span(), root) with tracer.start_span("stepchild", other_parent_context) as child: @@ -947,7 +935,7 @@ def error_status_test(context): .start_as_current_span("root") ) - def test_override_error_status(self): + def test_last_status_wins(self): def error_status_test(context): with self.assertRaises(AssertionError): with context as root: @@ -956,8 +944,10 @@ def error_status_test(context): ) raise AssertionError("unknown") - self.assertIs(root.status.status_code, StatusCode.OK) - self.assertEqual(root.status.description, "OK") + self.assertIs(root.status.status_code, StatusCode.ERROR) + self.assertEqual( + root.status.description, "AssertionError: unknown" + ) error_status_test( trace.TracerProvider().get_tracer(__name__).start_span("root") diff --git a/shim/opentelemetry-opentracing-shim/src/opentelemetry/shim/opentracing_shim/__init__.py b/shim/opentelemetry-opentracing-shim/src/opentelemetry/shim/opentracing_shim/__init__.py index c1dffbc1f9b..b7a365302f9 100644 --- a/shim/opentelemetry-opentracing-shim/src/opentelemetry/shim/opentracing_shim/__init__.py +++ b/shim/opentelemetry-opentracing-shim/src/opentelemetry/shim/opentracing_shim/__init__.py @@ -112,6 +112,7 @@ TracerProvider, get_current_span, set_span_in_context, + use_span, ) from opentelemetry.util.types import Attributes @@ -322,7 +323,7 @@ class ScopeShim(Scope): It is necessary to have both ways for constructing `ScopeShim` objects because in some cases we need to create the object from an OpenTelemetry `opentelemetry.trace.Span` context manager (as returned by - :meth:`opentelemetry.trace.Tracer.use_span`), in which case our only way of + :meth:`opentelemetry.trace.use_span`), in which case our only way of retrieving a `opentelemetry.trace.Span` object is by calling the ``__enter__()`` method on the context manager, which makes the span active in the OpenTelemetry tracer; whereas in other cases we need to accept a @@ -365,7 +366,7 @@ def from_context_manager(cls, manager: "ScopeManagerShim", span_cm): Example usage:: span = otel_tracer.start_span("TestSpan") - span_cm = otel_tracer.use_span(span) + span_cm = opentelemetry.trace.use_span(span) scope_shim = ScopeShim.from_context_manager( scope_manager_shim, span_cm=span_cm, @@ -375,7 +376,7 @@ def from_context_manager(cls, manager: "ScopeManagerShim", span_cm): manager: The :class:`ScopeManagerShim` that created this :class:`ScopeShim`. span_cm: A context manager as returned by - :meth:`opentelemetry.trace.Tracer.use_span`. + :meth:`opentelemetry.trace.use_span`. """ otel_span = span_cm.__enter__() @@ -452,9 +453,7 @@ def activate(self, span: SpanShim, finish_on_close: bool) -> "ScopeShim": A :class:`ScopeShim` representing the activated span. """ - span_cm = self._tracer.unwrap().use_span( - span.unwrap(), end_on_exit=finish_on_close - ) + span_cm = use_span(span.unwrap(), end_on_exit=finish_on_close) return ScopeShim.from_context_manager(self, span_cm=span_cm) @property