Skip to content

Commit

Permalink
Merge pull request #349 from NthPortal/current-span/PR
Browse files Browse the repository at this point in the history
Add `Tracer#currentSpanOrNoop`
  • Loading branch information
iRevive authored Nov 7, 2023
2 parents dfbed85 + 46733bb commit b5ba469
Show file tree
Hide file tree
Showing 6 changed files with 85 additions and 5 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -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
Expand All @@ -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]].
*
Expand Down Expand Up @@ -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
Expand All @@ -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] =
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -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)
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -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 {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -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,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -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
Expand All @@ -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)

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -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) =
Expand Down

0 comments on commit b5ba469

Please sign in to comment.