diff --git a/core/common/src/main/scala/org/typelevel/otel4s/KindTransformer.scala b/core/common/src/main/scala/org/typelevel/otel4s/KindTransformer.scala new file mode 100644 index 000000000..4f811f5c6 --- /dev/null +++ b/core/common/src/main/scala/org/typelevel/otel4s/KindTransformer.scala @@ -0,0 +1,123 @@ +/* + * Copyright 2022 Typelevel + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.typelevel.otel4s + +import cats.Applicative +import cats.Functor +import cats.Monad +import cats.data.EitherT +import cats.data.IorT +import cats.data.Kleisli +import cats.data.Nested +import cats.data.OptionT +import cats.data.StateT +import cats.effect.MonadCancelThrow +import cats.effect.Resource +import cats.implicits.toFunctorOps +import cats.syntax.all._ +import cats.~> + +/** A utility for transforming the higher-kinded type `F` to another + * higher-kinded type `G`. + */ +@annotation.implicitNotFound("No transformer defined from ${F} to ${G}") +trait KindTransformer[F[_], G[_]] { + + /** A higher-kinded function that lifts the kind `F` into a `G`. + * + * @note + * This method is usually best implemented by a `liftK` method on `G`'s + * companion object. + */ + val liftK: F ~> G + + /** Modify the context of `G[A]` using the natural transformation `f`. + * + * This method is "limited" in the sense that while most `mapK` methods can + * modify the context using arbitrary transformations, this method can only + * modify the context using natural transformations. + * + * @note + * This method is usually best implemented by a `mapK` method on `G`. + */ + def limitedMapK[A](ga: G[A])(f: F ~> F): G[A] + + /** Lifts a natural transformation from `F` to `F` into a natural + * transformation from `G` to `G`. + */ + final def liftFunctionK(f: F ~> F): G ~> G = + new (G ~> G) { + def apply[A](ga: G[A]): G[A] = limitedMapK(ga)(f) + } +} + +object KindTransformer { + implicit def optionT[F[_]: Functor]: KindTransformer[F, OptionT[F, *]] = + new KindTransformer[F, OptionT[F, *]] { + val liftK: F ~> OptionT[F, *] = OptionT.liftK + def limitedMapK[A](ga: OptionT[F, A])(f: F ~> F): OptionT[F, A] = + ga.mapK(f) + } + + implicit def eitherT[F[_]: Functor, L]: KindTransformer[F, EitherT[F, L, *]] = + new KindTransformer[F, EitherT[F, L, *]] { + val liftK: F ~> EitherT[F, L, *] = EitherT.liftK + def limitedMapK[R](ga: EitherT[F, L, R])(f: F ~> F): EitherT[F, L, R] = + ga.mapK(f) + } + + implicit def iorT[F[_]: Functor, L]: KindTransformer[F, IorT[F, L, *]] = + new KindTransformer[F, IorT[F, L, *]] { + val liftK: F ~> IorT[F, L, *] = IorT.liftK + def limitedMapK[R](ga: IorT[F, L, R])(f: F ~> F): IorT[F, L, R] = + ga.mapK(f) + } + + implicit def kleisli[F[_], A]: KindTransformer[F, Kleisli[F, A, *]] = + new KindTransformer[F, Kleisli[F, A, *]] { + val liftK: F ~> Kleisli[F, A, *] = Kleisli.liftK + def limitedMapK[B](ga: Kleisli[F, A, B])(f: F ~> F): Kleisli[F, A, B] = + ga.mapK(f) + } + + implicit def stateT[F[_]: Monad, S]: KindTransformer[F, StateT[F, S, *]] = + new KindTransformer[F, StateT[F, S, *]] { + val liftK: F ~> StateT[F, S, *] = StateT.liftK + def limitedMapK[A](ga: StateT[F, S, A])(f: F ~> F): StateT[F, S, A] = + ga.mapK(f) + } + + implicit def resource[F[_]: MonadCancelThrow] + : KindTransformer[F, Resource[F, *]] = + new KindTransformer[F, Resource[F, *]] { + val liftK: F ~> Resource[F, *] = Resource.liftK + def limitedMapK[A](ga: Resource[F, A])(f: F ~> F): Resource[F, A] = + ga.mapK(f) + } + + implicit def nested[F[_]: Functor, G[_]: Applicative] + : KindTransformer[F, Nested[F, G, *]] = + new KindTransformer[F, Nested[F, G, *]] { + val liftK: F ~> Nested[F, G, *] = + new (F ~> Nested[F, G, *]) { + def apply[A](fa: F[A]): Nested[F, G, A] = + fa.map(_.pure[G]).nested + } + def limitedMapK[A](ga: Nested[F, G, A])(f: F ~> F): Nested[F, G, A] = + ga.mapK(f) + } +} diff --git a/core/common/src/main/scala/org/typelevel/otel4s/meta/InstrumentMeta.scala b/core/common/src/main/scala/org/typelevel/otel4s/meta/InstrumentMeta.scala index 946d13076..f4b78b31e 100644 --- a/core/common/src/main/scala/org/typelevel/otel4s/meta/InstrumentMeta.scala +++ b/core/common/src/main/scala/org/typelevel/otel4s/meta/InstrumentMeta.scala @@ -17,6 +17,7 @@ package org.typelevel.otel4s.meta import cats.Applicative +import cats.~> trait InstrumentMeta[F[_]] { @@ -28,6 +29,9 @@ trait InstrumentMeta[F[_]] { */ def unit: F[Unit] + /** Modify the context `F` using the transformation `f`. */ + def mapK[G[_]](f: F ~> G): InstrumentMeta[G] = + new InstrumentMeta.MappedK(this)(f) } object InstrumentMeta { @@ -44,4 +48,9 @@ object InstrumentMeta { val unit: F[Unit] = Applicative[F].unit } + private class MappedK[F[_], G[_]](meta: InstrumentMeta[F])(f: F ~> G) + extends InstrumentMeta[G] { + def isEnabled: Boolean = meta.isEnabled + def unit: G[Unit] = f(meta.unit) + } } diff --git a/core/trace/src/main/scala/org/typelevel/otel4s/trace/Span.scala b/core/trace/src/main/scala/org/typelevel/otel4s/trace/Span.scala index 2a1258526..66cc1709e 100644 --- a/core/trace/src/main/scala/org/typelevel/otel4s/trace/Span.scala +++ b/core/trace/src/main/scala/org/typelevel/otel4s/trace/Span.scala @@ -18,6 +18,7 @@ package org.typelevel.otel4s package trace import cats.Applicative +import cats.~> import org.typelevel.otel4s.meta.InstrumentMeta import scala.concurrent.duration.FiniteDuration @@ -94,6 +95,15 @@ trait Span[F[_]] extends SpanMacro[F] { */ final def end(timestamp: FiniteDuration): F[Unit] = backend.end(timestamp) + + /** Modify the context `F` using the transformation `f`. */ + def mapK[G[_]](f: F ~> G): Span[G] = Span.fromBackend(backend.mapK(f)) + + /** Modify the context `F` using an implicit [[KindTransformer]] from `F` to + * `G`. + */ + final def mapK[G[_]](implicit kt: KindTransformer[F, G]): Span[G] = + mapK(kt.liftK) } object Span { @@ -121,6 +131,15 @@ object Span { private[otel4s] def end: F[Unit] private[otel4s] def end(timestamp: FiniteDuration): F[Unit] + + /** Modify the context `F` using the transformation `f`. */ + def mapK[G[_]](f: F ~> G): Backend[G] = new Backend.MappedK(this)(f) + + /** Modify the context `F` using an implicit [[KindTransformer]] from `F` to + * `G`. + */ + final def mapK[G[_]](implicit kt: KindTransformer[F, G]): Backend[G] = + mapK(kt.liftK) } object Backend { @@ -151,6 +170,36 @@ object Span { private[otel4s] def end: F[Unit] = unit private[otel4s] def end(timestamp: FiniteDuration): F[Unit] = unit } + + /** Implementation for [[Backend.mapK]]. */ + private class MappedK[F[_], G[_]](backend: Backend[F])(f: F ~> G) + extends Backend[G] { + def meta: InstrumentMeta[G] = + backend.meta.mapK(f) + def context: SpanContext = backend.context + def addAttributes(attributes: Attribute[_]*): G[Unit] = + f(backend.addAttributes(attributes: _*)) + def addEvent(name: String, attributes: Attribute[_]*): G[Unit] = + f(backend.addEvent(name, attributes: _*)) + def addEvent( + name: String, + timestamp: FiniteDuration, + attributes: Attribute[_]* + ): G[Unit] = + f(backend.addEvent(name, timestamp, attributes: _*)) + def recordException( + exception: Throwable, + attributes: Attribute[_]* + ): G[Unit] = + f(backend.recordException(exception, attributes: _*)) + def setStatus(status: Status): G[Unit] = + f(backend.setStatus(status)) + def setStatus(status: Status, description: String): G[Unit] = + f(backend.setStatus(status, description)) + private[otel4s] def end: G[Unit] = f(backend.end) + private[otel4s] def end(timestamp: FiniteDuration): G[Unit] = + f(backend.end(timestamp)) + } } private[otel4s] def fromBackend[F[_]](back: Backend[F]): Span[F] = diff --git a/core/trace/src/main/scala/org/typelevel/otel4s/trace/SpanBuilder.scala b/core/trace/src/main/scala/org/typelevel/otel4s/trace/SpanBuilder.scala index c93c669cf..4e559137d 100644 --- a/core/trace/src/main/scala/org/typelevel/otel4s/trace/SpanBuilder.scala +++ b/core/trace/src/main/scala/org/typelevel/otel4s/trace/SpanBuilder.scala @@ -19,6 +19,7 @@ package trace import cats.Applicative import cats.arrow.FunctionK +import cats.effect.MonadCancelThrow import cats.effect.Resource import scala.concurrent.duration.FiniteDuration @@ -112,6 +113,15 @@ trait SpanBuilder[F[_]] { def withParent(parent: SpanContext): SpanBuilder[F] def build: SpanOps[F] + + /** Modify the context `F` using an implicit [[KindTransformer]] from `F` to + * `G`. + */ + def mapK[G[_]: MonadCancelThrow](implicit + F: MonadCancelThrow[F], + kt: KindTransformer[F, G] + ): SpanBuilder[G] = + new SpanBuilder.MappedK(this) } object SpanBuilder { @@ -152,4 +162,31 @@ object SpanBuilder { } } + /** Implementation for [[SpanBuilder.mapK]]. */ + private class MappedK[F[_]: MonadCancelThrow, G[_]: MonadCancelThrow]( + builder: SpanBuilder[F] + )(implicit kt: KindTransformer[F, G]) + extends SpanBuilder[G] { + def addAttribute[A](attribute: Attribute[A]): SpanBuilder[G] = + new MappedK(builder.addAttribute(attribute)) + def addAttributes(attributes: Attribute[_]*): SpanBuilder[G] = + new MappedK(builder.addAttributes(attributes: _*)) + def addLink( + spanContext: SpanContext, + attributes: Attribute[_]* + ): SpanBuilder[G] = + new MappedK(builder.addLink(spanContext, attributes: _*)) + def withFinalizationStrategy( + strategy: SpanFinalizer.Strategy + ): SpanBuilder[G] = + new MappedK(builder.withFinalizationStrategy(strategy)) + def withSpanKind(spanKind: SpanKind): SpanBuilder[G] = + new MappedK(builder.withSpanKind(spanKind)) + def withStartTimestamp(timestamp: FiniteDuration): SpanBuilder[G] = + new MappedK(builder.withStartTimestamp(timestamp)) + def root: SpanBuilder[G] = new MappedK(builder.root) + def withParent(parent: SpanContext): SpanBuilder[G] = + new MappedK(builder.withParent(parent)) + def build: SpanOps[G] = builder.build.mapK[G] + } } diff --git a/core/trace/src/main/scala/org/typelevel/otel4s/trace/SpanOps.scala b/core/trace/src/main/scala/org/typelevel/otel4s/trace/SpanOps.scala index 712f7097a..1d14a453b 100644 --- a/core/trace/src/main/scala/org/typelevel/otel4s/trace/SpanOps.scala +++ b/core/trace/src/main/scala/org/typelevel/otel4s/trace/SpanOps.scala @@ -14,9 +14,12 @@ * limitations under the License. */ -package org.typelevel.otel4s.trace +package org.typelevel.otel4s +package trace +import cats.effect.MonadCancelThrow import cats.effect.Resource +import cats.syntax.functor._ import cats.~> trait SpanOps[F[_]] { @@ -138,6 +141,15 @@ trait SpanOps[F[_]] { * See [[use]] for more details regarding lifecycle strategy */ final def surround[A](fa: F[A]): F[A] = use(_ => fa) + + /** Modify the context `F` using an implicit [[KindTransformer]] from `F` to + * `G`. + */ + def mapK[G[_]: MonadCancelThrow](implicit + F: MonadCancelThrow[F], + kt: KindTransformer[F, G] + ): SpanOps[G] = + new SpanOps.MappedK(this) } object SpanOps { @@ -156,6 +168,12 @@ object SpanOps { * [[cats.arrow.FunctionK FunctionK]] will not be traced. */ def trace: F ~> F + + /** Modify the context `F` using an implicit [[KindTransformer]] from `F` to + * `G`. + */ + def mapK[G[_]](implicit kt: KindTransformer[F, G]): Res[G] = + Res(span.mapK[G], kt.liftFunctionK(trace)) } object Res { @@ -168,4 +186,21 @@ object SpanOps { def apply[F[_]](span: Span[F], trace: F ~> F): Res[F] = Impl(span, trace) } + + /** Implementation for [[SpanOps.mapK]]. */ + private class MappedK[F[_]: MonadCancelThrow, G[_]: MonadCancelThrow]( + ops: SpanOps[F] + )(implicit kt: KindTransformer[F, G]) + extends SpanOps[G] { + def startUnmanaged: G[Span[G]] = + kt.liftK(ops.startUnmanaged).map(_.mapK[G]) + + def resource: Resource[G, Res[G]] = + ops.resource.mapK(kt.liftK).map(res => res.mapK[G]) + + def use[A](f: Span[G] => G[A]): G[A] = + resource.use { res => res.trace(f(res.span)) } + + def use_ : G[Unit] = kt.liftK(ops.use_) + } } 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 2d9029a62..ec093e705 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 @@ -18,6 +18,8 @@ package org.typelevel.otel4s package trace import cats.Applicative +import cats.effect.MonadCancelThrow +import cats.~> import org.typelevel.otel4s.meta.InstrumentMeta @annotation.implicitNotFound(""" @@ -184,6 +186,14 @@ trait Tracer[F[_]] extends TracerMacro[F] { */ def propagate[C: TextMapUpdater](carrier: C): F[C] + /** Modify the context `F` using an implicit [[KindTransformer]] from `F` to + * `G`. + */ + def mapK[G[_]: MonadCancelThrow](implicit + F: MonadCancelThrow[F], + kt: KindTransformer[F, G] + ): Tracer[G] = + new Tracer.MappedK(this) } object Tracer { @@ -192,6 +202,15 @@ object Tracer { trait Meta[F[_]] extends InstrumentMeta[F] { def noopSpanBuilder: SpanBuilder[F] + + /** Modify the context `F` using an implicit [[KindTransformer]] from `F` to + * `G`. + */ + def mapK[G[_]: MonadCancelThrow](implicit + F: MonadCancelThrow[F], + kt: KindTransformer[F, G] + ): Meta[G] = + new Meta.MappedK(this) } object Meta { @@ -208,6 +227,17 @@ object Tracer { val noopSpanBuilder: SpanBuilder[F] = SpanBuilder.noop(noopBackend) } + + /** Implementation for [[Meta.mapK]]. */ + private class MappedK[F[_]: MonadCancelThrow, G[_]: MonadCancelThrow]( + meta: Meta[F] + )(implicit kt: KindTransformer[F, G]) + extends Meta[G] { + def noopSpanBuilder: SpanBuilder[G] = + meta.noopSpanBuilder.mapK[G] + def isEnabled: Boolean = meta.isEnabled + def unit: G[Unit] = kt.liftK(meta.unit) + } } def noop[F[_]: Applicative]: Tracer[F] = @@ -225,6 +255,36 @@ object Tracer { Applicative[F].pure(carrier) } + /** Implementation for [[Tracer.mapK]]. */ + private class MappedK[F[_]: MonadCancelThrow, G[_]: MonadCancelThrow]( + tracer: Tracer[F] + )(implicit kt: KindTransformer[F, G]) + extends Tracer[G] { + def meta: Meta[G] = tracer.meta.mapK[G] + def currentSpanContext: G[Option[SpanContext]] = + kt.liftK(tracer.currentSpanContext) + def spanBuilder(name: String): SpanBuilder[G] = + tracer.spanBuilder(name).mapK[G] + def childScope[A](parent: SpanContext)(ga: G[A]): G[A] = + kt.limitedMapK(ga)(new (F ~> F) { + def apply[B](fb: F[B]): F[B] = tracer.childScope(parent)(fb) + }) + def joinOrRoot[A, C: TextMapGetter](carrier: C)(ga: G[A]): G[A] = + kt.limitedMapK(ga)(new (F ~> F) { + def apply[B](fb: F[B]): F[B] = tracer.joinOrRoot(carrier)(fb) + }) + def rootScope[A](ga: G[A]): G[A] = + kt.limitedMapK(ga)(new (F ~> F) { + def apply[B](fb: F[B]): F[B] = tracer.rootScope(fb) + }) + def noopScope[A](ga: G[A]): G[A] = + kt.limitedMapK(ga)(new (F ~> F) { + def apply[B](fb: F[B]): F[B] = tracer.noopScope(fb) + }) + def propagate[C: TextMapUpdater](carrier: C): G[C] = + kt.liftK(tracer.propagate(carrier)) + } + object Implicits { implicit def noop[F[_]: Applicative]: Tracer[F] = Tracer.noop } 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 e981a96ab..91fbcfab9 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 @@ -16,13 +16,17 @@ package org.typelevel.otel4s.java.trace +import cats.data.EitherT +import cats.data.IorT +import cats.data.Kleisli +import cats.data.OptionT +import cats.data.StateT import cats.effect.IO import cats.effect.IOLocal +import cats.effect.MonadCancelThrow import cats.effect.Resource -import cats.effect.kernel.MonadCancelThrow import cats.effect.testkit.TestControl -import cats.syntax.apply._ -import cats.syntax.functor._ +import cats.syntax.all._ import cats.~> import io.opentelemetry.api.common.{AttributeKey => JAttributeKey} import io.opentelemetry.api.common.Attributes @@ -61,6 +65,7 @@ import scala.jdk.CollectionConverters._ import scala.util.control.NoStackTrace class TracerSuite extends CatsEffectSuite { + import TracerSuite.NestedSurround test("propagate instrumentation info") { val expected = InstrumentationScopeInfo @@ -891,6 +896,222 @@ class TracerSuite extends CatsEffectSuite { } } + test("nested SpanOps#surround for Tracer[IO]") { + TestControl.executeEmbed { + for { + now <- IO.monotonic.delayBy(1.second) // otherwise returns 0 + sdk <- makeSdk() + tracer <- sdk.provider.get("tracer") + _ <- tracer + .span("outer") + .surround { + for { + _ <- IO.sleep(NestedSurround.preBodyDuration) + _ <- tracer + .span("body-1") + .surround(IO.sleep(NestedSurround.body1Duration)) + _ <- tracer + .span("body-2") + .surround(IO.sleep(NestedSurround.body2Duration)) + } yield () + } + spans <- sdk.finishedSpans + tree <- IO.pure(SpanNode.fromSpans(spans)) + } yield assertEquals(tree, List(NestedSurround.expected(now))) + } + } + + test("nested SpanOps#surround for Tracer[OptionT[IO, *]]") { + TestControl.executeEmbed { + for { + now <- IO.monotonic.delayBy(1.second) // otherwise returns 0 + sdk <- makeSdk() + tracerIO <- sdk.provider.get("tracer") + tracer = tracerIO.mapK[OptionT[IO, *]] + _ <- tracer + .span("outer") + .surround { + for { + _ <- OptionT.liftF(IO.sleep(NestedSurround.preBodyDuration)) + _ <- tracer + .span("body-1") + .surround(OptionT.liftF(IO.sleep(NestedSurround.body1Duration))) + _ <- tracer + .span("body-2") + .surround(OptionT.liftF(IO.sleep(NestedSurround.body2Duration))) + } yield () + } + .value + spans <- sdk.finishedSpans + tree <- IO.pure(SpanNode.fromSpans(spans)) + } yield assertEquals(tree, List(NestedSurround.expected(now))) + } + } + + test("nested SpanOps#surround for Tracer[EitherT[IO, String, *]]") { + TestControl.executeEmbed { + for { + now <- IO.monotonic.delayBy(1.second) // otherwise returns 0 + sdk <- makeSdk() + tracerIO <- sdk.provider.get("tracer") + tracer = tracerIO.mapK[EitherT[IO, String, *]] + _ <- tracer + .span("outer") + .surround { + for { + _ <- EitherT.liftF(IO.sleep(NestedSurround.preBodyDuration)) + _ <- tracer + .span("body-1") + .surround(EitherT.liftF(IO.sleep(NestedSurround.body1Duration))) + _ <- tracer + .span("body-2") + .surround(EitherT.liftF(IO.sleep(NestedSurround.body2Duration))) + } yield () + } + .value + spans <- sdk.finishedSpans + tree <- IO.pure(SpanNode.fromSpans(spans)) + } yield assertEquals(tree, List(NestedSurround.expected(now))) + } + } + + test("nested SpanOps#surround for Tracer[IorT[IO, String, *]]") { + TestControl.executeEmbed { + for { + now <- IO.monotonic.delayBy(1.second) // otherwise returns 0 + sdk <- makeSdk() + tracerIO <- sdk.provider.get("tracer") + tracer = tracerIO.mapK[IorT[IO, String, *]] + _ <- tracer + .span("outer") + .surround { + for { + _ <- IorT.liftF(IO.sleep(NestedSurround.preBodyDuration)) + _ <- tracer + .span("body-1") + .surround(IorT.liftF(IO.sleep(NestedSurround.body1Duration))) + _ <- tracer + .span("body-2") + .surround(IorT.liftF(IO.sleep(NestedSurround.body2Duration))) + } yield () + } + .value + spans <- sdk.finishedSpans + tree <- IO.pure(SpanNode.fromSpans(spans)) + } yield assertEquals(tree, List(NestedSurround.expected(now))) + } + } + + test("nested SpanOps#surround for Tracer[Kleisli[IO, String, *]]") { + TestControl.executeEmbed { + for { + now <- IO.monotonic.delayBy(1.second) // otherwise returns 0 + sdk <- makeSdk() + tracerIO <- sdk.provider.get("tracer") + tracer = tracerIO.mapK[Kleisli[IO, String, *]] + _ <- tracer + .span("outer") + .surround { + for { + _ <- Kleisli.liftF(IO.sleep(NestedSurround.preBodyDuration)) + _ <- tracer + .span("body-1") + .surround(Kleisli.liftF(IO.sleep(NestedSurround.body1Duration))) + _ <- tracer + .span("body-2") + .surround(Kleisli.liftF(IO.sleep(NestedSurround.body2Duration))) + } yield () + } + .run("unused") + spans <- sdk.finishedSpans + tree <- IO.pure(SpanNode.fromSpans(spans)) + } yield assertEquals(tree, List(NestedSurround.expected(now))) + } + } + + test("nested SpanOps#surround for Tracer[StateT[IO, Int, *]]") { + TestControl.executeEmbed { + for { + now <- IO.monotonic.delayBy(1.second) // otherwise returns 0 + sdk <- makeSdk() + tracerIO <- sdk.provider.get("tracer") + tracer = tracerIO.mapK[StateT[IO, Int, *]] + _ <- tracer + .span("outer") + .surround { + for { + _ <- StateT.liftF(IO.sleep(NestedSurround.preBodyDuration)) + _ <- tracer + .span("body-1") + .surround(StateT.liftF(IO.sleep(NestedSurround.body1Duration))) + _ <- tracer + .span("body-2") + .surround(StateT.liftF(IO.sleep(NestedSurround.body2Duration))) + } yield () + } + .run(0) + spans <- sdk.finishedSpans + tree <- IO.pure(SpanNode.fromSpans(spans)) + } yield assertEquals(tree, List(NestedSurround.expected(now))) + } + } + + test("nested SpanOps#surround for Tracer[Resource[IO, *]]") { + TestControl.executeEmbed { + for { + now <- IO.monotonic.delayBy(1.second) // otherwise returns 0 + sdk <- makeSdk() + tracerIO <- sdk.provider.get("tracer") + tracer = tracerIO.mapK[Resource[IO, *]] + _ <- tracer + .span("outer") + .surround { + for { + _ <- Resource.eval(IO.sleep(NestedSurround.preBodyDuration)) + _ <- tracer + .span("body-1") + .surround(Resource.eval(IO.sleep(NestedSurround.body1Duration))) + _ <- tracer + .span("body-2") + .surround(Resource.eval(IO.sleep(NestedSurround.body2Duration))) + } yield () + } + .use_ + spans <- sdk.finishedSpans + tree <- IO.pure(SpanNode.fromSpans(spans)) + } yield assertEquals(tree, List(NestedSurround.expected(now))) + } + } + + /* + test("nested SpanOps#surround for Tracer[Nested[IO, List, *]]") { + TestControl.executeEmbed { + for { + now <- IO.monotonic.delayBy(1.second) // otherwise returns 0 + sdk <- makeSdk() + tracerIO <- sdk.provider.get("tracer") + tracer = tracerIO.mapK[Nested[IO, List, *]] + _ <- tracer + .span("outer") + .surround { + for { + _ <- IO.sleep(NestedSurround.preBodyDuration).map(List(_)).nested + _ <- tracer + .span("body-1") + .surround(IO.sleep(NestedSurround.body1Duration).map(List(_)).nested) + _ <- tracer + .span("body-2") + .surround(IO.sleep(NestedSurround.body2Duration).map(List(_)).nested) + } yield () + } + .value + spans <- sdk.finishedSpans + tree <- IO.pure(SpanNode.fromSpans(spans)) + } yield assertEquals(tree, List(NestedSurround.expected(now))) + } + } + */ + private def assertIdsNotEqual(s1: Span[IO], s2: Span[IO]): Unit = { assertNotEquals(s1.context.traceIdHex, s2.context.traceIdHex) assertNotEquals(s1.context.spanIdHex, s2.context.spanIdHex) @@ -941,4 +1162,35 @@ object TracerSuite { } + private object NestedSurround { + val preBodyDuration: FiniteDuration = 25.millis + val body1Duration: FiniteDuration = 100.millis + val body2Duration: FiniteDuration = 50.millis + + def expected(start: FiniteDuration): SpanNode = { + val body1Start = start + preBodyDuration + val body1End = body1Start + body1Duration + val end = body1End + body2Duration + + SpanNode( + "outer", + start, + end, + List( + SpanNode( + "body-1", + body1Start, + body1End, + Nil + ), + SpanNode( + "body-2", + body1End, + end, + Nil + ) + ) + ) + } + } }