From 46733bb7f3aeac592c77aa4c0912593b7df6d3cf Mon Sep 17 00:00:00 2001 From: Marissa Date: Fri, 27 Oct 2023 19:52:34 -0400 Subject: [PATCH] Add `Tracer#currentSpanOrNoop` --- .../org/typelevel/otel4s/trace/Tracer.scala | 13 ++++- .../typelevel/otel4s/trace/TracerSuite.scala | 5 ++ .../otel4s/java/trace/SpanBackendImpl.scala | 2 + .../otel4s/java/trace/SpanRunner.scala | 5 +- .../otel4s/java/trace/TracerImpl.scala | 10 ++++ .../otel4s/java/trace/TracerSuite.scala | 55 +++++++++++++++++++ 6 files changed, 85 insertions(+), 5 deletions(-) diff --git a/core/trace/src/main/scala/org/typelevel/otel4s/trace/Tracer.scala b/core/trace/src/main/scala/org/typelevel/otel4s/trace/Tracer.scala index 0dd3aa0c2..2966a3928 100644 --- a/core/trace/src/main/scala/org/typelevel/otel4s/trace/Tracer.scala +++ b/core/trace/src/main/scala/org/typelevel/otel4s/trace/Tracer.scala @@ -19,6 +19,7 @@ package trace import cats.Applicative import cats.effect.kernel.MonadCancelThrow +import cats.syntax.functor._ import cats.~> import org.typelevel.otel4s.context.propagation.TextMapGetter import org.typelevel.otel4s.context.propagation.TextMapUpdater @@ -45,10 +46,16 @@ trait Tracer[F[_]] extends TracerMacro[F] { */ def meta: Tracer.Meta[F] - /** Returns the context of a span when it is available in the scope. + /** Returns the context of the current span when a span that is not no-op + * exists in the local scope. */ def currentSpanContext: F[Option[SpanContext]] + /** Returns the current span if one exists in the local scope, or a no-op span + * otherwise. + */ + def currentSpanOrNoop: F[Span[F]] + /** Creates a new [[SpanBuilder]]. The builder can be used to make a fully * customized [[Span]]. * @@ -255,6 +262,8 @@ object Tracer { private val builder = SpanBuilder.noop(noopBackend) val meta: Meta[F] = Meta.disabled val currentSpanContext: F[Option[SpanContext]] = Applicative[F].pure(None) + val currentSpanOrNoop: F[Span[F]] = + Applicative[F].pure(Span.fromBackend(noopBackend)) def rootScope[A](fa: F[A]): F[A] = fa def noopScope[A](fa: F[A]): F[A] = fa def childScope[A](parent: SpanContext)(fa: F[A]): F[A] = fa @@ -272,6 +281,8 @@ object Tracer { def meta: Meta[G] = tracer.meta.mapK[G] def currentSpanContext: G[Option[SpanContext]] = kt.liftK(tracer.currentSpanContext) + def currentSpanOrNoop: G[Span[G]] = + kt.liftK(tracer.currentSpanOrNoop.map(_.mapK[G])) def spanBuilder(name: String): SpanBuilder[G] = tracer.spanBuilder(name).mapK[G] def childScope[A](parent: SpanContext)(ga: G[A]): G[A] = diff --git a/core/trace/src/test/scala/org/typelevel/otel4s/trace/TracerSuite.scala b/core/trace/src/test/scala/org/typelevel/otel4s/trace/TracerSuite.scala index 746399ffa..546682ced 100644 --- a/core/trace/src/test/scala/org/typelevel/otel4s/trace/TracerSuite.scala +++ b/core/trace/src/test/scala/org/typelevel/otel4s/trace/TracerSuite.scala @@ -78,4 +78,9 @@ class TracerSuite extends CatsEffectSuite { } yield assert(!allocated) } + test("`currentSpanOrNoop` is not valid when instrument is noop") { + val tracer = Tracer.noop[IO] + for (span <- tracer.currentSpanOrNoop) + yield assert(!span.context.isValid) + } } diff --git a/java/trace/src/main/scala/org/typelevel/otel4s/java/trace/SpanBackendImpl.scala b/java/trace/src/main/scala/org/typelevel/otel4s/java/trace/SpanBackendImpl.scala index 2c9f63446..608308f94 100644 --- a/java/trace/src/main/scala/org/typelevel/otel4s/java/trace/SpanBackendImpl.scala +++ b/java/trace/src/main/scala/org/typelevel/otel4s/java/trace/SpanBackendImpl.scala @@ -103,6 +103,8 @@ private[java] class SpanBackendImpl[F[_]: Sync]( } private[java] object SpanBackendImpl { + def fromJSpan[F[_]: Sync](jSpan: JSpan): SpanBackendImpl[F] = + new SpanBackendImpl(jSpan, WrappedSpanContext(jSpan.getSpanContext)) private def toJStatus(status: Status): JStatusCode = status match { diff --git a/java/trace/src/main/scala/org/typelevel/otel4s/java/trace/SpanRunner.scala b/java/trace/src/main/scala/org/typelevel/otel4s/java/trace/SpanRunner.scala index e6643c8ed..43a854bbb 100644 --- a/java/trace/src/main/scala/org/typelevel/otel4s/java/trace/SpanRunner.scala +++ b/java/trace/src/main/scala/org/typelevel/otel4s/java/trace/SpanRunner.scala @@ -83,10 +83,7 @@ private[java] object SpanRunner { if (hasStartTimestamp) Sync[F].pure(b) else Sync[F].realTime.map(t => b.setStartTimestamp(t.length, t.unit)) jSpan <- Sync[F].delay(builder.startSpan()) - } yield new SpanBackendImpl( - jSpan, - WrappedSpanContext(jSpan.getSpanContext) - ) + } yield SpanBackendImpl.fromJSpan(jSpan) private def startManaged[F[_]: Sync]( builder: JSpanBuilder, diff --git a/java/trace/src/main/scala/org/typelevel/otel4s/java/trace/TracerImpl.scala b/java/trace/src/main/scala/org/typelevel/otel4s/java/trace/TracerImpl.scala index c50f092bb..0ec9be296 100644 --- a/java/trace/src/main/scala/org/typelevel/otel4s/java/trace/TracerImpl.scala +++ b/java/trace/src/main/scala/org/typelevel/otel4s/java/trace/TracerImpl.scala @@ -24,6 +24,7 @@ import org.typelevel.otel4s.context.propagation.TextMapGetter import org.typelevel.otel4s.context.propagation.TextMapUpdater import org.typelevel.otel4s.java.context.Context import org.typelevel.otel4s.java.context.LocalContext +import org.typelevel.otel4s.trace.Span import org.typelevel.otel4s.trace.SpanBuilder import org.typelevel.otel4s.trace.SpanContext import org.typelevel.otel4s.trace.Tracer @@ -47,6 +48,15 @@ private[java] class TracerImpl[F[_]: Sync]( .map(jSpan => new WrappedSpanContext(jSpan.getSpanContext)) } + def currentSpanOrNoop: F[Span[F]] = + L.reader { ctx => + Span.fromBackend( + SpanBackendImpl.fromJSpan( + JSpan.fromContext(ctx.underlying) + ) + ) + } + def spanBuilder(name: String): SpanBuilder[F] = new SpanBuilderImpl[F](jTracer, name, runner) diff --git a/java/trace/src/test/scala/org/typelevel/otel4s/java/trace/TracerSuite.scala b/java/trace/src/test/scala/org/typelevel/otel4s/java/trace/TracerSuite.scala index d2c4e2410..647c85630 100644 --- a/java/trace/src/test/scala/org/typelevel/otel4s/java/trace/TracerSuite.scala +++ b/java/trace/src/test/scala/org/typelevel/otel4s/java/trace/TracerSuite.scala @@ -349,6 +349,61 @@ class TracerSuite extends CatsEffectSuite { } } + test("`currentSpanOrNoop` outside of a span (root scope)") { + for { + sdk <- makeSdk() + tracer <- sdk.provider.get("tracer") + currentSpan <- tracer.currentSpanOrNoop + _ <- currentSpan.addAttribute(Attribute("string-attribute", "value")) + spans <- sdk.finishedSpans + } yield { + assert(!currentSpan.context.isValid) + assertEquals(spans.length, 0) + } + } + + test("`currentSpanOrNoop` in noop scope") { + for { + sdk <- makeSdk() + tracer <- sdk.provider.get("tracer") + _ <- tracer.noopScope { + for { + currentSpan <- tracer.currentSpanOrNoop + _ <- currentSpan.addAttribute(Attribute("string-attribute", "value")) + } yield assert(!currentSpan.context.isValid) + } + spans <- sdk.finishedSpans + } yield assertEquals(spans.length, 0) + } + + test("`currentSpanOrNoop` inside a span") { + def expected(now: FiniteDuration): List[SpanNode] = + List(SpanNode("span", now, now, Nil)) + + TestControl.executeEmbed { + for { + now <- IO.monotonic.delayBy(1.second) // otherwise returns 0 + sdk <- makeSdk() + tracer <- sdk.provider.get("tracer") + _ <- tracer.span("span").surround { + for { + currentSpan <- tracer.currentSpanOrNoop + _ <- currentSpan + .addAttribute(Attribute("string-attribute", "value")) + } yield assert(currentSpan.context.isValid) + } + spans <- sdk.finishedSpans + tree <- IO.pure(SpanNode.fromSpans(spans)) + // _ <- IO.println(tree.map(SpanNode.render).mkString("\n")) + } yield { + assertEquals(tree, expected(now)) + val key = JAttributeKey.stringKey("string-attribute") + val attr = spans.map(data => Option(data.getAttributes.get(key))) + assertEquals(attr.flatten, List("value")) + } + } + } + test("create a new scope with a custom parent") { def expected(now: FiniteDuration) =