All Downloads are FREE. Search and download functionalities are using the official Maven repository.

sttp.client3.opentelemetry.zio.OpenTelemetryTracingZioBackend.scala Maven / Gradle / Ivy

package sttp.client3.opentelemetry.zio

import io.opentelemetry.api.trace.{SpanKind, StatusCode}
import io.opentelemetry.api.trace.propagation.W3CTraceContextPropagator
import io.opentelemetry.context.propagation.{TextMapPropagator, TextMapSetter}
import sttp.capabilities.Effect
import sttp.client3._
import zio._
import zio.telemetry.opentelemetry.TracingSyntax.OpenTelemetryZioOps
import zio.telemetry.opentelemetry._

import scala.collection.mutable

private class OpenTelemetryTracingZioBackend[+P](
    delegate: SttpBackend[Task, P],
    tracer: OpenTelemetryZioTracer,
    tracing: Tracing
) extends DelegateSttpBackend[Task, P](delegate) {
  def send[T, R >: P with Effect[Task]](request: Request[T, R]): Task[Response[T]] = {
    val carrier: mutable.Map[String, String] = mutable.Map().empty
    val propagator: TextMapPropagator = W3CTraceContextPropagator.getInstance()
    val setter: TextMapSetter[mutable.Map[String, String]] = (carrier, key, value) => carrier.update(key, value)

    (for {
      _ <- Tracing.inject(propagator, carrier, setter)
      _ <- tracer.before(request)
      resp <- delegate.send(request.headers(carrier.toMap))
      _ <- tracer.after(resp)
    } yield resp)
      .span(tracer.spanName(request), SpanKind.CLIENT, { case _ => StatusCode.ERROR })
      .provideLayer(ZLayer.succeed(tracing))
  }
}

object OpenTelemetryTracingZioBackend {
  def apply[P](
      other: SttpBackend[Task, P],
      tracing: Tracing,
      tracer: OpenTelemetryZioTracer = OpenTelemetryZioTracer.Default
  ): SttpBackend[Task, P] =
    new OpenTelemetryTracingZioBackend[P](other, tracer, tracing)

}

trait OpenTelemetryZioTracer {
  def spanName[T](request: Request[T, Nothing]): String
  def before[T](request: Request[T, Nothing]): RIO[Tracing, Unit]
  def after[T](response: Response[T]): RIO[Tracing, Unit]
}

object OpenTelemetryZioTracer {
  val Default: OpenTelemetryZioTracer = new OpenTelemetryZioTracer {
    override def spanName[T](request: Request[T, Nothing]): String = s"HTTP ${request.method.method}"
    override def before[T](request: Request[T, Nothing]): RIO[Tracing, Unit] =
      Tracing.setAttribute("http.method", request.method.method) *>
        Tracing.setAttribute("http.url", request.uri.toString()) *>
        ZIO.unit
    override def after[T](response: Response[T]): RIO[Tracing, Unit] =
      Tracing.setAttribute("http.status_code", response.code.code.toLong) *>
        ZIO.unit
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy