From 3048b8757a75ed9dc2bac8f0c43fbb7266817bfd Mon Sep 17 00:00:00 2001 From: Marissa Date: Thu, 9 Nov 2023 17:38:58 -0500 Subject: [PATCH 1/4] Add `PassThroughPropagator` Add `PassThroughPropagator` that works with any context implementation. --- build.sbt | 2 +- .../propagation/PassThroughPropagator.scala | 80 +++++++++++++++++++ .../PassThroughPropagatorProps.scala | 65 +++++++++++++++ .../PassThroughPropagatorSuite.scala | 39 +++++++++ 4 files changed, 185 insertions(+), 1 deletion(-) create mode 100644 core/common/src/main/scala/org/typelevel/otel4s/context/propagation/PassThroughPropagator.scala create mode 100644 core/common/src/test/scala/org/typelevel/otel4s/context/propagation/PassThroughPropagatorProps.scala create mode 100644 core/common/src/test/scala/org/typelevel/otel4s/context/propagation/PassThroughPropagatorSuite.scala diff --git a/build.sbt b/build.sbt index 92e280e34..812744bd8 100644 --- a/build.sbt +++ b/build.sbt @@ -99,7 +99,7 @@ lazy val `core-common` = crossProject(JVMPlatform, JSPlatform, NativePlatform) name := "otel4s-core-common", libraryDependencies ++= Seq( "org.typelevel" %%% "cats-core" % CatsVersion, - "org.typelevel" %%% "cats-effect-kernel" % CatsEffectVersion, + "org.typelevel" %%% "cats-effect" % CatsEffectVersion, "org.typelevel" %%% "cats-mtl" % CatsMtlVersion, "org.typelevel" %%% "vault" % VaultVersion % Test, "org.typelevel" %%% "cats-laws" % CatsVersion % Test, diff --git a/core/common/src/main/scala/org/typelevel/otel4s/context/propagation/PassThroughPropagator.scala b/core/common/src/main/scala/org/typelevel/otel4s/context/propagation/PassThroughPropagator.scala new file mode 100644 index 000000000..517c11684 --- /dev/null +++ b/core/common/src/main/scala/org/typelevel/otel4s/context/propagation/PassThroughPropagator.scala @@ -0,0 +1,80 @@ +/* + * 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.context +package propagation + +import cats.effect.SyncIO +import org.typelevel.otel4s.context.syntax._ + +/** A [[TextMapPropagator]] that extracts a specified collection of fields and + * stores them in a context, and extracts them from a context later for + * injection. It does not interact with telemetry. + */ +final class PassThroughPropagator[Ctx, K[X] <: Key[X]] private ( + val fields: Iterable[String], + entriesKey: K[List[(String, String)]] +)(implicit c: Contextual.Keyed[Ctx, K]) + extends TextMapPropagator[Ctx] { + + def extract[A](ctx: Ctx, carrier: A)(implicit + getter: TextMapGetter[A] + ): Ctx = { + val list = fields.view + .flatMap(k => getter.get(carrier, k).map(k -> _)) + .toList + if (list.isEmpty) ctx else ctx.updated(entriesKey, list) + } + + def inject[A](ctx: Ctx, carrier: A)(implicit updater: TextMapUpdater[A]): A = + ctx + .getOrElse(entriesKey, Nil) + .foldLeft(carrier) { case (c, k -> v) => updater.updated(c, k, v) } + + override def toString: String = + s"PassThroughPropagator{fields=${fields.mkString("[", ", ", "]")}}" +} + +object PassThroughPropagator { + private def forDistinctFields[Ctx, K[X] <: Key[X]](fields: Seq[String])( + implicit + c: Contextual.Keyed[Ctx, K], + kp: Key.Provider[SyncIO, K] + ): TextMapPropagator[Ctx] = + if (fields.isEmpty) TextMapPropagator.noop + else { + new PassThroughPropagator( + fields, + kp.uniqueKey[List[(String, String)]]( + "otel4s-PassThroughPropagator-entries" + ).unsafeRunSync() + ) + } + + /** Creates a `PassThroughPropagator` that propagates the given fields. */ + def apply[Ctx, K[X] <: Key[X]](fields: String*)(implicit + c: Contextual.Keyed[Ctx, K], + kp: Key.Provider[SyncIO, K] + ): TextMapPropagator[Ctx] = + forDistinctFields(fields.distinct) + + /** Creates a `PassThroughPropagator` that propagates the given fields. */ + def apply[Ctx, K[X] <: Key[X]](fields: Iterable[String])(implicit + c: Contextual.Keyed[Ctx, K], + kp: Key.Provider[SyncIO, K] + ): TextMapPropagator[Ctx] = + forDistinctFields(fields.iterator.distinct.toSeq) +} diff --git a/core/common/src/test/scala/org/typelevel/otel4s/context/propagation/PassThroughPropagatorProps.scala b/core/common/src/test/scala/org/typelevel/otel4s/context/propagation/PassThroughPropagatorProps.scala new file mode 100644 index 000000000..21efc6606 --- /dev/null +++ b/core/common/src/test/scala/org/typelevel/otel4s/context/propagation/PassThroughPropagatorProps.scala @@ -0,0 +1,65 @@ +/* + * 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.context.propagation + +import munit.ScalaCheckSuite +import org.scalacheck.Prop.forAll +import org.typelevel.otel4s.context.vault.VaultContext + +class PassThroughPropagatorProps extends ScalaCheckSuite { + private[this] def vaultPropagator( + fields: Iterable[String] + ): TextMapPropagator[VaultContext] = + PassThroughPropagator(fields) + + property("retrieves all stored values matching fields and no others") { + forAll { + ( + entries: Map[String, String], + extraFields: Set[String], + unpropagated: Map[String, String] + ) => + val propagator = vaultPropagator(entries.keys.view ++ extraFields) + val toSkip = unpropagated.view + .filterKeys(s => !entries.contains(s) && !extraFields.contains(s)) + .toMap + val extracted = propagator.extract(VaultContext.root, entries ++ toSkip) + val injected = propagator.inject(extracted, Map.empty[String, String]) + assertEquals(injected, entries) + for (key -> _ <- toSkip) assert(!injected.contains(key)) // redundant + } + } + + property("retrieves no values when none were stored") { + forAll { (entries: Map[String, String], extraFields: Seq[String]) => + val propagator = vaultPropagator(entries.keys.view ++ extraFields) + val injected = + propagator.inject(VaultContext.root, Map.empty[String, String]) + assert(injected.isEmpty) + } + } + + property("retrieves no values when created with no fields") { + forAll { (entries: Map[String, String]) => + val propagator = vaultPropagator(Nil) + val extracted = propagator.extract(VaultContext.root, entries) + val injected = propagator.inject(extracted, Map.empty[String, String]) + assertEquals(extracted, VaultContext.root) + assert(injected.isEmpty) + } + } +} diff --git a/core/common/src/test/scala/org/typelevel/otel4s/context/propagation/PassThroughPropagatorSuite.scala b/core/common/src/test/scala/org/typelevel/otel4s/context/propagation/PassThroughPropagatorSuite.scala new file mode 100644 index 000000000..7d4771a1b --- /dev/null +++ b/core/common/src/test/scala/org/typelevel/otel4s/context/propagation/PassThroughPropagatorSuite.scala @@ -0,0 +1,39 @@ +/* + * 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.context.propagation + +import munit.FunSuite +import org.typelevel.otel4s.context.vault.VaultContext + +class PassThroughPropagatorSuite extends FunSuite { + private val propagator: TextMapPropagator[VaultContext] = + PassThroughPropagator("foo", "bar") + + test("propagates only fields given to constructor") { + val entries = Map("foo" -> "0", "bar" -> "1", "baz" -> "2", "qux" -> "3") + val extracted = propagator.extract(VaultContext.root, entries) + val injected = propagator.inject(extracted, Map.empty[String, String]) + assertEquals(injected, Map("foo" -> "0", "bar" -> "1")) + } + + test("propagates nothing when no matching fields") { + val entries = Map("baz" -> "2", "qux" -> "3") + val extracted = propagator.extract(VaultContext.root, entries) + val injected = propagator.inject(extracted, Map.empty[String, String]) + assert(injected.isEmpty) + } +} From e15a5b93dc43a2136ca59fdff85d3a987af9aafb Mon Sep 17 00:00:00 2001 From: Marissa Date: Mon, 13 Nov 2023 12:51:14 -0500 Subject: [PATCH 2/4] Improve stability of `TextMapPropagator.of` Change `TextMapPropagator` to always return the same instance when passed a `Seq` with exactly one element, even when that element is a `TextMapPropagator.Noop` instance. --- .../otel4s/context/propagation/TextMapPropagator.scala | 7 +++++-- .../context/propagation/TextMapPropagatorSuite.scala | 10 +++++++++- 2 files changed, 14 insertions(+), 3 deletions(-) diff --git a/core/common/src/main/scala/org/typelevel/otel4s/context/propagation/TextMapPropagator.scala b/core/common/src/main/scala/org/typelevel/otel4s/context/propagation/TextMapPropagator.scala index 7952de835..ce04ab081 100644 --- a/core/common/src/main/scala/org/typelevel/otel4s/context/propagation/TextMapPropagator.scala +++ b/core/common/src/main/scala/org/typelevel/otel4s/context/propagation/TextMapPropagator.scala @@ -97,8 +97,11 @@ object TextMapPropagator { */ def of[Ctx]( propagators: TextMapPropagator[Ctx]* - ): TextMapPropagator[Ctx] = - propagators.combineAll + ): TextMapPropagator[Ctx] = { + // reference stability for noop + if (propagators.lengthIs == 1) propagators.head + else propagators.combineAll + } /** Creates a no-op implementation of the [[TextMapPropagator]]. * diff --git a/core/common/src/test/scala/org/typelevel/otel4s/context/propagation/TextMapPropagatorSuite.scala b/core/common/src/test/scala/org/typelevel/otel4s/context/propagation/TextMapPropagatorSuite.scala index 7bdce9367..826a80d7e 100644 --- a/core/common/src/test/scala/org/typelevel/otel4s/context/propagation/TextMapPropagatorSuite.scala +++ b/core/common/src/test/scala/org/typelevel/otel4s/context/propagation/TextMapPropagatorSuite.scala @@ -39,10 +39,18 @@ class TextMapPropagatorSuite extends FunSuite { val propagator = new TestPropagator[String](fields, "TestPropagator") val composite = TextMapPropagator.of(propagator) - assertEquals(composite.fields, fields) + assert(composite eq propagator) assertEquals(composite.toString, "TestPropagator") } + test("of (single input, noop) - use same reference") { + val propagator = TextMapPropagator.noop[String] + val composite = TextMapPropagator.of(propagator) + + assert(composite eq propagator) + assertEquals(composite.toString, "TextMapPropagator.Noop") + } + test("of (multiple) - create a multi instance") { val fieldsA = List("a", "b") val fieldsB = List("c", "d") From ea5b5a640efd690ce4dba36e2d405cc480182d99 Mon Sep 17 00:00:00 2001 From: Marissa Date: Wed, 1 Nov 2023 15:26:42 -0400 Subject: [PATCH 3/4] Redesign and improve `TextMap` converters Redesign converters between Java and Scala `TextMap{Getter,Propagator}`s to follow the pattern from the Scala standard library, supporting converting in either direction between the types. Remove unused conversion for `TextMapSetter`. --- NOTICE | 15 + .../org/typelevel/otel4s/java/OtelJava.scala | 5 +- .../typelevel/otel4s/java/Conversions.scala | 26 -- .../otel4s/java/TextMapPropagatorImpl.scala | 55 ---- .../propagation/PropagatorConverters.scala | 65 +++++ .../convert/AsJavaConverters.scala | 77 +++++ .../convert/AsJavaExtensions.scala | 74 +++++ .../convert/AsScalaConverters.scala | 77 +++++ .../convert/AsScalaExtensions.scala | 74 +++++ .../convert/JavaPropagatorWrappers.scala | 105 +++++++ .../convert/PropagatorConverters.scala | 55 ++++ .../PropagatorConvertersProps.scala | 270 ++++++++++++++++++ .../otel4s/java/trace/TracerSuite.scala | 54 ++-- licenses/LICENSE_scala | 201 +++++++++++++ 14 files changed, 1050 insertions(+), 103 deletions(-) create mode 100644 NOTICE delete mode 100644 java/common/src/main/scala/org/typelevel/otel4s/java/TextMapPropagatorImpl.scala create mode 100644 java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/PropagatorConverters.scala create mode 100644 java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/AsJavaConverters.scala create mode 100644 java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/AsJavaExtensions.scala create mode 100644 java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/AsScalaConverters.scala create mode 100644 java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/AsScalaExtensions.scala create mode 100644 java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/JavaPropagatorWrappers.scala create mode 100644 java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/PropagatorConverters.scala create mode 100644 java/common/src/test/scala/org/typelevel/otel4s/java/context/propagation/PropagatorConvertersProps.scala create mode 100644 licenses/LICENSE_scala diff --git a/NOTICE b/NOTICE new file mode 100644 index 000000000..0fb7bbb7c --- /dev/null +++ b/NOTICE @@ -0,0 +1,15 @@ +otel4s +Copyright 2022-2023 Typelevel +Licensed under Apache License 2.0 (see LICENSE) + +This software contains portions of code derived from scala +https://github.com/scala/scala +Scala +Copyright (c) 2002-2023 EPFL +Copyright (c) 2011-2023 Lightbend, Inc. + +Scala includes software developed at +LAMP/EPFL (https://lamp.epfl.ch/) and +Lightbend, Inc. (https://www.lightbend.com/). + +Licensed under the Apache License, Version 2.0 (see licenses/LICENSE_scala). diff --git a/java/all/src/main/scala/org/typelevel/otel4s/java/OtelJava.scala b/java/all/src/main/scala/org/typelevel/otel4s/java/OtelJava.scala index baa63ec82..e69719831 100644 --- a/java/all/src/main/scala/org/typelevel/otel4s/java/OtelJava.scala +++ b/java/all/src/main/scala/org/typelevel/otel4s/java/OtelJava.scala @@ -30,6 +30,7 @@ import org.typelevel.otel4s.context.propagation.ContextPropagators import org.typelevel.otel4s.java.Conversions.asyncFromCompletableResultCode import org.typelevel.otel4s.java.context.Context import org.typelevel.otel4s.java.context.LocalContext +import org.typelevel.otel4s.java.context.propagation.PropagatorConverters._ import org.typelevel.otel4s.java.instances._ import org.typelevel.otel4s.java.metrics.Metrics import org.typelevel.otel4s.java.trace.Traces @@ -66,9 +67,7 @@ object OtelJava { def local[F[_]: Async: LocalContext]( jOtel: JOpenTelemetry ): OtelJava[F] = { - val contextPropagators = ContextPropagators.of( - new TextMapPropagatorImpl(jOtel.getPropagators.getTextMapPropagator) - ) + val contextPropagators = jOtel.getPropagators.asScala val metrics = Metrics.forAsync(jOtel) val traces = Traces.local(jOtel, contextPropagators) diff --git a/java/common/src/main/scala/org/typelevel/otel4s/java/Conversions.scala b/java/common/src/main/scala/org/typelevel/otel4s/java/Conversions.scala index 0d9c12e79..0dac35d73 100644 --- a/java/common/src/main/scala/org/typelevel/otel4s/java/Conversions.scala +++ b/java/common/src/main/scala/org/typelevel/otel4s/java/Conversions.scala @@ -17,17 +17,10 @@ package org.typelevel.otel4s package java -import _root_.java.lang.{Iterable => JIterable} import cats.effect.kernel.Async import cats.syntax.either._ import io.opentelemetry.api.common.{Attributes => JAttributes} -import io.opentelemetry.context.propagation.{TextMapGetter => JTextMapGetter} -import io.opentelemetry.context.propagation.{TextMapSetter => JTextMapSetter} import io.opentelemetry.sdk.common.CompletableResultCode -import org.typelevel.otel4s.context.propagation.TextMapGetter -import org.typelevel.otel4s.context.propagation.TextMapSetter - -import scala.jdk.CollectionConverters._ private[java] object Conversions { @@ -83,23 +76,4 @@ private[java] object Conversions { } ) ) - - implicit def fromTextMapGetter[A](implicit - getter: TextMapGetter[A] - ): JTextMapGetter[A] = - new JTextMapGetter[A] { - def get(carrier: A, key: String) = - getter.get(carrier, key).orNull - - def keys(carrier: A): JIterable[String] = - getter.keys(carrier).asJava - } - - implicit def fromTextMapSetter[A](implicit - setter: TextMapSetter[A] - ): JTextMapSetter[A] = - new JTextMapSetter[A] { - def set(carrier: A, key: String, value: String) = - setter.unsafeSet(carrier, key, value) - } } diff --git a/java/common/src/main/scala/org/typelevel/otel4s/java/TextMapPropagatorImpl.scala b/java/common/src/main/scala/org/typelevel/otel4s/java/TextMapPropagatorImpl.scala deleted file mode 100644 index 280923461..000000000 --- a/java/common/src/main/scala/org/typelevel/otel4s/java/TextMapPropagatorImpl.scala +++ /dev/null @@ -1,55 +0,0 @@ -/* - * 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 -package java - -import io.opentelemetry.context.propagation.{ - TextMapPropagator => JTextMapPropagator -} -import org.typelevel.otel4s.context.propagation.TextMapGetter -import org.typelevel.otel4s.context.propagation.TextMapPropagator -import org.typelevel.otel4s.context.propagation.TextMapUpdater -import org.typelevel.otel4s.java.Conversions._ -import org.typelevel.otel4s.java.context.Context -import org.typelevel.scalaccompat.annotation.threadUnsafe3 - -import scala.jdk.CollectionConverters._ - -private[java] class TextMapPropagatorImpl( - jPropagator: JTextMapPropagator -) extends TextMapPropagator[Context] { - @threadUnsafe3 - lazy val fields: Iterable[String] = - jPropagator.fields().asScala - - def extract[A: TextMapGetter](ctx: Context, carrier: A): Context = - ctx.map(jPropagator.extract(_, carrier, fromTextMapGetter)) - - def inject[A](ctx: Context, carrier: A)(implicit - injector: TextMapUpdater[A] - ): A = { - var injectedCarrier = carrier - jPropagator.inject[Null]( - ctx.underlying, - null, // explicitly allowed per opentelemetry-java, so our setter can be a lambda! - (_, key, value) => { - injectedCarrier = injector.updated(injectedCarrier, key, value) - } - ) - injectedCarrier - } -} diff --git a/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/PropagatorConverters.scala b/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/PropagatorConverters.scala new file mode 100644 index 000000000..6d7939482 --- /dev/null +++ b/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/PropagatorConverters.scala @@ -0,0 +1,65 @@ +/* + * 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. + */ + +/* + * The scaladocs in this file are adapted from those in the file + * scala/jdk/CollectionConverters.scala + * in the Scala standard library. + */ + +package org.typelevel.otel4s.java.context.propagation + +/** This object provides extension methods that convert between Scala and Java + * `TextMapGetter`s, `TextMapPropagator`s, and `ContextPropagators` using + * `asScala` and `asJava` extension methods. + * + * For the rare instances where they are needed, explicit conversion methods + * are defined in + * [[org.typelevel.otel4s.java.context.propagation.convert.PropagatorConverters]]. + * + * {{{ + * import io.opentelemetry.api.trace.propagation.W3CTraceContextPropagator + * import org.typelevel.otel4s.context.propagation.TextMapPropagator + * import org.typelevel.otel4s.java.context.Context + * import org.typelevel.otel4s.java.context.propagation.TextMapOperatorConverters._ + * + * val propagator: TextMapPropagator[Context] = W3CTraceContextPropagator.getInstance().asScala + * }}} + * + * The conversions return wrappers for the TextMap operators, and converting + * from a source type to a target type and back again will return the original + * source object. For example: + * + * {{{ + * import io.opentelemetry.api.trace.propagation.W3CTraceContextPropagator + * import org.typelevel.otel4s.context.propagation.TextMapPropagator + * import org.typelevel.otel4s.java.context.Context + * import org.typelevel.otel4s.java.context.propagation.TextMapOperatorConverters._ + * + * val source: W3CTraceContextPropagator = W3CTraceContextPropagator.getInstance() + * val target: TextMapPropagator[Context] = source.asScala + * val other: io.opentelemetry.context.propagation.TextMapPropagator = target.asJava + * assert(source eq other) + * }}} + * + * Currently, `ContextPropagators` for both Java and Scala are simple wrappers + * around a `TextMapPropagator` instance of the corresponding type. + * Consequently, conversions between `ContextPropagators` convert the + * `TextMapPropagator` and do not use a custom wrapper. + */ +object PropagatorConverters + extends convert.AsJavaExtensions + with convert.AsScalaExtensions diff --git a/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/AsJavaConverters.scala b/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/AsJavaConverters.scala new file mode 100644 index 000000000..1afebb327 --- /dev/null +++ b/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/AsJavaConverters.scala @@ -0,0 +1,77 @@ +/* + * 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. + */ + +/* + * The scaladocs in this file are adapted from those in the file + * scala/collection/convert/AsJavaConverters.scala + * in the Scala standard library. + */ + +package org.typelevel.otel4s.java.context +package propagation.convert + +import io.opentelemetry.context.propagation.{ + ContextPropagators => JContextPropagators +} +import io.opentelemetry.context.propagation.{TextMapGetter => JTextMapGetter} +import io.opentelemetry.context.propagation.{ + TextMapPropagator => JTextMapPropagator +} +import org.typelevel.otel4s.context.propagation.ContextPropagators +import org.typelevel.otel4s.context.propagation.TextMapGetter +import org.typelevel.otel4s.context.propagation.TextMapPropagator + +import scala.{unchecked => uc} + +/** Defines explicit conversion methods from Scala to Java for `TextMapGetter`s, + * `TextMapPropagator`s, and `ContextPropagators`. These methods are available + * through + * [[org.typelevel.otel4s.java.context.propagation.convert.PropagatorConverters]]. + */ +trait AsJavaConverters { + import JavaPropagatorWrappers._ + + /** Converts a Scala `TextMapGetter` to a Java `TextMapGetter`. + * + * The returned Java `TextMapGetter` is backed by the provided Scala + * `TextMapGetter` unless the Scala `TextMapGetter` was previously obtained + * from an implicit or explicit call of `asScala`, in which case the original + * Java `TextMapGetter` will be returned. + */ + def asJava[A](getter: TextMapGetter[A]): JTextMapGetter[A] = getter match { + case null => null + case wrapper: JTextMapGetterWrapper[A @uc] => wrapper.underlying + case _ => new TextMapGetterWrapper(getter) + } + + /** Converts a Scala `TextMapPropagator` to a Java `TextMapPropagator`. + * + * The returned Java `TextMapPropagator` is backed by the provided Scala + * `TextMapPropagator`unless the Scala `TextMapPropagator` was previously + * obtained from an implicit or explicit call of `asScala`, in which case the + * original Java `TextMapPropagator` will be returned. + */ + def asJava(propagator: TextMapPropagator[Context]): JTextMapPropagator = + propagator match { + case null => null + case wrapper: JTextMapPropagatorWrapper => wrapper.underlying + case _ => new TextMapPropagatorWrapper(propagator) + } + + /** Converts a Scala `ContextPropagators` to a Java `ContextPropagators`. */ + def asJava(propagators: ContextPropagators[Context]): JContextPropagators = + JContextPropagators.create(asJava(propagators.textMapPropagator)) +} diff --git a/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/AsJavaExtensions.scala b/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/AsJavaExtensions.scala new file mode 100644 index 000000000..f650c7d2a --- /dev/null +++ b/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/AsJavaExtensions.scala @@ -0,0 +1,74 @@ +/* + * 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. + */ + +/* + * The scaladocs in this file are adapted from those in the file + * scala/collection/convert/AsJavaExtensions.scala + * in the Scala standard library. + */ + +package org.typelevel.otel4s.java.context +package propagation +package convert + +import io.opentelemetry.context.propagation.{ + ContextPropagators => JContextPropagators +} +import io.opentelemetry.context.propagation.{TextMapGetter => JTextMapGetter} +import io.opentelemetry.context.propagation.{ + TextMapPropagator => JTextMapPropagator +} +import org.typelevel.otel4s.context.propagation.ContextPropagators +import org.typelevel.otel4s.context.propagation.TextMapGetter +import org.typelevel.otel4s.context.propagation.TextMapPropagator + +/** Defines `asJava` extension methods for `TextMapGetter`s and + * `TextMapPropagator`s, and `ContextPropagators`, available through + * [[org.typelevel.otel4s.java.context.propagation.PropagatorConverters]]. + */ +trait AsJavaExtensions { + import convert.{PropagatorConverters => conv} + + implicit class TextMapGetterHasAsJava[A](getter: TextMapGetter[A]) { + + /** Converts a Scala `TextMapGetter` to a Java `TextMapGetter`. + * + * @see + * [[org.typelevel.otel4s.java.context.propagation.convert.AsJavaConverters.asJava[A](getter* `convert.PropagatorConverters.asJava`]]. + */ + def asJava: JTextMapGetter[A] = conv.asJava(getter) + } + + implicit class TextMapPropagatorHasAsJava(prop: TextMapPropagator[Context]) { + + /** Converts a Scala `TextMapPropagator` to a Java `TextMapPropagator`. + * + * @see + * [[org.typelevel.otel4s.java.context.propagation.convert.AsJavaConverters.asJava(propagator:* `convert.PropagatorConverters.asJava`]]. + */ + def asJava: JTextMapPropagator = conv.asJava(prop) + } + + implicit class ContextPropagatorsHasAsJava(cp: ContextPropagators[Context]) { + + /** Converts a Scala `ContextPropagators` to a Java `ContextPropagators`. + * + * @see + * [[org.typelevel.otel4s.java.context.propagation.convert.AsJavaConverters.asJava(propagators* `convert.PropagatorConverters.asJava`]]. + */ + def asJava: JContextPropagators = conv.asJava(cp) + } +} diff --git a/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/AsScalaConverters.scala b/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/AsScalaConverters.scala new file mode 100644 index 000000000..813f83479 --- /dev/null +++ b/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/AsScalaConverters.scala @@ -0,0 +1,77 @@ +/* + * 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. + */ + +/* + * The scaladocs in this file are adapted from those in the file + * scala/collection/convert/AsScalaConverters.scala + * in the Scala standard library. + */ + +package org.typelevel.otel4s.java.context +package propagation.convert + +import io.opentelemetry.context.propagation.{ + ContextPropagators => JContextPropagators +} +import io.opentelemetry.context.propagation.{TextMapGetter => JTextMapGetter} +import io.opentelemetry.context.propagation.{ + TextMapPropagator => JTextMapPropagator +} +import org.typelevel.otel4s.context.propagation.ContextPropagators +import org.typelevel.otel4s.context.propagation.TextMapGetter +import org.typelevel.otel4s.context.propagation.TextMapPropagator + +import scala.{unchecked => uc} + +/** Defines explicit conversion methods from Java to Scala for `TextMapGetter`s, + * `TextMapPropagator`s, and `ContextPropagators`. These methods are available + * through + * [[org.typelevel.otel4s.java.context.propagation.convert.PropagatorConverters]]. + */ +trait AsScalaConverters { + import JavaPropagatorWrappers._ + + /** Converts a Java `TextMapGetter` to a Scala `TextMapGetter`. + * + * The returned Scala `TextMapGetter` is backed by the provided Java + * `TextMapGetter` unless the Java `TextMapGetter` was previously obtained + * from an implicit or explicit call of `asJava`, in which case the original + * Scala `TextMapGetter` will be returned. + */ + def asScala[A](getter: JTextMapGetter[A]): TextMapGetter[A] = getter match { + case null => null + case wrapper: TextMapGetterWrapper[A @uc] => wrapper.underlying + case _ => new JTextMapGetterWrapper(getter) + } + + /** Converts a Java `TextMapPropagator` to a Scala `TextMapPropagator`. + * + * The returned Scala `TextMapPropagator` is backed by the provided Java + * `TextMapPropagator` unless the Java `TextMapPropagator` was previously + * obtained from an implicit or explicit call of `asJava`, in which case the + * original Scala `TextMapPropagator` will be returned. + */ + def asScala(propagator: JTextMapPropagator): TextMapPropagator[Context] = + propagator match { + case null => null + case wrapper: TextMapPropagatorWrapper => wrapper.underlying + case _ => new JTextMapPropagatorWrapper(propagator) + } + + /** Converts a Java `ContextPropagators` to a Scala `ContextPropagators`. */ + def asScala(propagators: JContextPropagators): ContextPropagators[Context] = + ContextPropagators.of(asScala(propagators.getTextMapPropagator)) +} diff --git a/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/AsScalaExtensions.scala b/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/AsScalaExtensions.scala new file mode 100644 index 000000000..64f37d78e --- /dev/null +++ b/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/AsScalaExtensions.scala @@ -0,0 +1,74 @@ +/* + * 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. + */ + +/* + * The scaladocs in this file are adapted from those in the file + * scala/collection/convert/AsScalaExtensions.scala + * in the Scala standard library. + */ + +package org.typelevel.otel4s.java.context +package propagation +package convert + +import io.opentelemetry.context.propagation.{ + ContextPropagators => JContextPropagators +} +import io.opentelemetry.context.propagation.{TextMapGetter => JTextMapGetter} +import io.opentelemetry.context.propagation.{ + TextMapPropagator => JTextMapPropagator +} +import org.typelevel.otel4s.context.propagation.ContextPropagators +import org.typelevel.otel4s.context.propagation.TextMapGetter +import org.typelevel.otel4s.context.propagation.TextMapPropagator + +/** Defines `asScala` extension methods for `TextMapGetter`s, + * `TextMapPropagator`s, and `ContextPropagators`, available through + * [[org.typelevel.otel4s.java.context.propagation.PropagatorConverters]]. + */ +trait AsScalaExtensions { + import convert.{PropagatorConverters => conv} + + implicit class TextMapGetterHasAsScala[A](getter: JTextMapGetter[A]) { + + /** Converts a Java `TextMapGetter` to a Scala `TextMapGetter`. + * + * @see + * [[org.typelevel.otel4s.java.context.propagation.convert.AsScalaConverters.asScala[A](getter* `convert.PropagatorConverters.asScala`]]. + */ + def asScala: TextMapGetter[A] = conv.asScala(getter) + } + + implicit class TextMapPropagatorHasAsScala(propagator: JTextMapPropagator) { + + /** Converts a Java `TextMapPropagator` to a Scala `TextMapPropagator`. + * + * @see + * [[org.typelevel.otel4s.java.context.propagation.convert.AsScalaConverters.asScala(propagator:* `convert.PropagatorConverters.asScala`]]. + */ + def asScala: TextMapPropagator[Context] = conv.asScala(propagator) + } + + implicit class ContextPropagatorsHasAsScala(cp: JContextPropagators) { + + /** Converts a Java `TextMapPropagator` to a Scala `TextMapPropagator`. + * + * @see + * [[org.typelevel.otel4s.java.context.propagation.convert.AsScalaConverters.asScala(propagators* `convert.PropagatorConverters.asScala`]]. + */ + def asScala: ContextPropagators[Context] = conv.asScala(cp) + } +} diff --git a/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/JavaPropagatorWrappers.scala b/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/JavaPropagatorWrappers.scala new file mode 100644 index 000000000..730552221 --- /dev/null +++ b/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/JavaPropagatorWrappers.scala @@ -0,0 +1,105 @@ +/* + * 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.java.context +package propagation.convert + +import io.opentelemetry.context.{Context => JContext} +import io.opentelemetry.context.propagation.{TextMapGetter => JTextMapGetter} +import io.opentelemetry.context.propagation.{ + TextMapPropagator => JTextMapPropagator +} +import io.opentelemetry.context.propagation.{TextMapSetter => JTextMapSetter} +import org.typelevel.otel4s.context.propagation.TextMapGetter +import org.typelevel.otel4s.context.propagation.TextMapPropagator +import org.typelevel.otel4s.context.propagation.TextMapUpdater + +import java.{lang => jl} +import java.{util => ju} +import scala.jdk.CollectionConverters._ + +private[convert] object JavaPropagatorWrappers { + class TextMapGetterWrapper[C](val underlying: TextMapGetter[C]) + extends JTextMapGetter[C] { + def keys(carrier: C): jl.Iterable[String] = + underlying.keys(carrier).asJava + def get(carrier: C /* may be `null` */, key: String): String = + Option(carrier) + .flatMap(underlying.get(_, key)) + .orNull + } + + class JTextMapGetterWrapper[C](val underlying: JTextMapGetter[C]) + extends TextMapGetter[C] { + def get(carrier: C, key: String): Option[String] = + Option(underlying.get(carrier, key) /* may return `null` */ ) + def keys(carrier: C): Iterable[String] = + underlying.keys(carrier).asScala + } + + class TextMapPropagatorWrapper(val underlying: TextMapPropagator[Context]) + extends JTextMapPropagator { + def fields(): ju.Collection[String] = + underlying.fields.asJavaCollection + def inject[C]( + context: JContext, + carrier: C, // may be `null` + setter: JTextMapSetter[C] + ): Unit = { + // sadly, this must essentially inject twice + val immutableRes = + underlying.inject(Context.wrap(context), List.empty[(String, String)]) + for ((key, value) <- immutableRes) setter.set(carrier, key, value) + } + def extract[C]( + context: JContext, + carrier: C, // may be `null` + getter: JTextMapGetter[C] + ): JContext = { + implicit val tmg: TextMapGetter[C] = + PropagatorConverters.asScala(getter) // don't double-wrap + Option(carrier) + .fold(context)(underlying.extract(Context.wrap(context), _).underlying) + } + } + + class JTextMapPropagatorWrapper(val underlying: JTextMapPropagator) + extends TextMapPropagator[Context] { + def fields: Iterable[String] = + underlying.fields().asScala + def extract[A](ctx: Context, carrier: A)(implicit + getter: TextMapGetter[A] + ): Context = + ctx.map( + underlying.extract( + _, + carrier, + PropagatorConverters.asJava(getter) // don't double-wrap + ) + ) + def inject[A](ctx: Context, carrier: A)(implicit + updater: TextMapUpdater[A] + ): A = { + var res = carrier + underlying.inject[Null]( + ctx.underlying, + null, // explicitly allowed per opentelemetry-java, so our setter can be a lambda! + (_, key, value) => res = updater.updated(res, key, value) + ) + res + } + } +} diff --git a/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/PropagatorConverters.scala b/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/PropagatorConverters.scala new file mode 100644 index 000000000..25542c380 --- /dev/null +++ b/java/common/src/main/scala/org/typelevel/otel4s/java/context/propagation/convert/PropagatorConverters.scala @@ -0,0 +1,55 @@ +/* + * 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. + */ + +/* + * The scaladocs in this file are adapted from those in the file + * scala/jdk/javaapi/CollectionConverters.scala + * in the Scala standard library. + */ + +package org.typelevel.otel4s.java.context.propagation.convert + +/** This object contains methods that convert between Scala and Java + * `TextMapGetter`s, `TextMapPropagator`s, and `ContextPropagators` using + * explicit `asScala` and `asJava` methods. + * + * In general, it is preferable to use the extension methods defined in + * [[org.typelevel.otel4s.java.context.propagation.PropagatorConverters]]. + * + * {{{ + * // Java Code + * import io.opentelemetry.api.trace.propagation.W3CTraceContextPropagator; + * import org.typelevel.otel4s.context.propagation.TextMapPropagator; + * import org.typelevel.otel4s.java.context.Context; + * import org.typelevel.otel4s.java.context.propagation.convert.TextMapOperatorConverters; + * + * public class Example { + * public void foo(W3CTraceContextPropagator jp) { + * TextMapPropagator sp = TextMapOperatorConverters.asScala(jp); + * } + * } + * }}} + * + * The conversions return wrappers for the TextMap operators, and converting + * from a source type to a target type and back again will return the original + * source object. + * + * Currently, `ContextPropagators` for both Java and Scala are simple wrappers + * around a `TextMapPropagator` instance of the corresponding type. + * Consequently, conversions between `ContextPropagators` convert the + * `TextMapPropagator` and do not use a custom wrapper. + */ +object PropagatorConverters extends AsJavaConverters with AsScalaConverters diff --git a/java/common/src/test/scala/org/typelevel/otel4s/java/context/propagation/PropagatorConvertersProps.scala b/java/common/src/test/scala/org/typelevel/otel4s/java/context/propagation/PropagatorConvertersProps.scala new file mode 100644 index 000000000..912793e34 --- /dev/null +++ b/java/common/src/test/scala/org/typelevel/otel4s/java/context/propagation/PropagatorConvertersProps.scala @@ -0,0 +1,270 @@ +/* + * 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.java.context.propagation + +import io.opentelemetry.context.propagation.{ + ContextPropagators => JContextPropagators +} +import io.opentelemetry.context.propagation.{TextMapGetter => JTextMapGetter} +import io.opentelemetry.context.propagation.{ + TextMapPropagator => JTextMapPropagator +} +import io.opentelemetry.extension.incubator.propagation.{ + PassThroughPropagator => JPassThroughPropagator +} +import munit.ScalaCheckSuite +import org.scalacheck.Arbitrary +import org.scalacheck.Gen +import org.scalacheck.Prop.forAll +import org.typelevel.otel4s.context.propagation.ContextPropagators +import org.typelevel.otel4s.context.propagation.PassThroughPropagator +import org.typelevel.otel4s.context.propagation.TextMapGetter +import org.typelevel.otel4s.context.propagation.TextMapPropagator +import org.typelevel.otel4s.java.context.Context +import org.typelevel.otel4s.java.context.propagation.PropagatorConverters._ + +import java.{lang => jl} +import scala.jdk.CollectionConverters._ + +class PropagatorConvertersProps extends ScalaCheckSuite { + private type Carrier = Map[String, String] + + private def mkJTMG: JTextMapGetter[Carrier] = + new JTextMapGetter[Carrier] { + def keys(carrier: Carrier): jl.Iterable[String] = + carrier.keys.asJava + + def get(carrier: Carrier, key: String): String = + carrier.get(key).orNull + } + private implicit def arbitraryJTMG: Arbitrary[JTextMapGetter[Carrier]] = + Arbitrary(Gen.delay(Gen.const(mkJTMG))) + private implicit def arbitrarySTMG: Arbitrary[TextMapGetter[Carrier]] = + Arbitrary(Gen.delay(Gen.const(implicitly))) + + private implicit def arbitraryJTMP: Arbitrary[JTextMapPropagator] = + Arbitrary { + for (fields <- Gen.listOf(Gen.alphaNumStr)) + yield JPassThroughPropagator.create(fields.asJava) + } + private implicit def arbitrarySTMP: Arbitrary[TextMapPropagator[Context]] = + Arbitrary { + for (fields <- Gen.listOf(Gen.alphaNumStr)) + yield PassThroughPropagator[Context, Context.Key](fields) + } + + private implicit def arbitraryJCP: Arbitrary[JContextPropagators] = + Arbitrary { + for (jtmp <- arbitraryJTMP.arbitrary) + yield JContextPropagators.create(jtmp) + } + private implicit def arbitrarySCP: Arbitrary[ContextPropagators[Context]] = + Arbitrary { + for (stmp <- arbitrarySTMP.arbitrary) + yield ContextPropagators.of(stmp) + } + + property("Java TextMapGetter round trip reference identity") { + forAll { (tmg: JTextMapGetter[Carrier]) => + assert(tmg.asScala.asJava eq tmg) + } + } + property("Scala TextMapGetter round trip reference identity") { + forAll { (tmg: TextMapGetter[Carrier]) => + assert(tmg.asJava.asScala eq tmg) + } + } + + property("Java TextMapPropagator round trip reference identity") { + forAll { (tmp: JTextMapPropagator) => + assert(tmp.asScala.asJava eq tmp) + } + } + property("Scala TextMapPropagator round trip reference identity") { + forAll { (tmp: TextMapPropagator[Context]) => + assert(tmp.asJava.asScala eq tmp) + } + } + + property("Java ContextPropagators round trip sort of identity") { + forAll { (cp: JContextPropagators) => + assert(cp.asScala.asJava.getTextMapPropagator eq cp.getTextMapPropagator) + } + } + property("Scala ContextPropagators round trip sort of identity") { + forAll { (cp: ContextPropagators[Context]) => + assert(cp.asJava.asScala.textMapPropagator eq cp.textMapPropagator) + } + } + + property("Java TextMapGetter and wrapper behavior equivalence") { + forAll { + ( + tmg: JTextMapGetter[Carrier], + carrier: Carrier, + otherKeys: Seq[String] + ) => + val wrapper = tmg.asScala + assertEquals( + wrapper.keys(carrier).toSeq, + tmg.keys(carrier).asScala.toSeq + ) + for (key <- carrier.keys ++ otherKeys) { + assertEquals(wrapper.get(carrier, key), Option(tmg.get(carrier, key))) + } + } + } + property("Scala TextMapGetter and wrapper behavior equivalence") { + forAll { + ( + tmg: TextMapGetter[Carrier], + carrier: Carrier, + otherKeys: Seq[String] + ) => + val wrapper = tmg.asJava + assertEquals( + wrapper.keys(carrier).asScala.toSeq, + tmg.keys(carrier).toSeq + ) + for (key <- carrier.keys ++ otherKeys) { + assertEquals(Option(wrapper.get(carrier, key)), tmg.get(carrier, key)) + } + } + } + + property("Java TextMapPropagator and wrapper behavior equivalence") { + forAll { + (entries: Carrier, extraFields: Set[String], unpropagated: Carrier) => + val tmp = + JPassThroughPropagator.create((entries.keySet ++ extraFields).asJava) + val wrapper = tmp.asScala + assertEquals(wrapper.fields.toSeq, tmp.fields().asScala.toSeq) + val toSkip = unpropagated.view + .filterKeys(k => !entries.contains(k) && !extraFields.contains(k)) + .toMap + val extracted = + tmp.extract(Context.root.underlying, entries ++ toSkip, mkJTMG) + val wrapperExtracted = + wrapper.extract(Context.root, entries ++ toSkip) + var injected = Map.empty[String, String] + tmp.inject[Carrier]( + extracted, + null, + (_, k, v) => injected = injected.updated(k, v) + ) + val wrapperInjected = + wrapper.inject(wrapperExtracted, Map.empty[String, String]) + assertEquals(wrapperInjected, injected) + } + } + property("Scala TextMapPropagator and wrapper behavior equivalence") { + forAll { + (entries: Carrier, extraFields: Set[String], unpropagated: Carrier) => + val tmp: TextMapPropagator[Context] = + PassThroughPropagator(entries.keySet ++ extraFields) + val wrapper = tmp.asJava + assertEquals(wrapper.fields().asScala.toSeq, tmp.fields.toSeq) + val toSkip = unpropagated.view + .filterKeys(k => !entries.contains(k) && !extraFields.contains(k)) + .toMap + val extracted = + tmp.extract(Context.root, entries ++ toSkip) + val wrapperExtracted = + wrapper.extract(Context.root.underlying, entries ++ toSkip, mkJTMG) + val injected = tmp.inject(extracted, Map.empty[String, String]) + var wrapperInjected = Map.empty[String, String] + wrapper.inject[Carrier]( + wrapperExtracted, + null, + (_, k, v) => wrapperInjected = wrapperInjected.updated(k, v) + ) + assertEquals(wrapperInjected, injected) + } + } + + property("Java ContextPropagators and wrapper behavior equivalence") { + forAll { + (entries: Carrier, extraFields: Set[String], unpropagated: Carrier) => + val cp = JContextPropagators.create( + JPassThroughPropagator.create((entries.keySet ++ extraFields).asJava) + ) + val wrapper = cp.asScala + assertEquals( + wrapper.textMapPropagator.fields.toSeq, + cp.getTextMapPropagator.fields().asScala.toSeq + ) + val toSkip = unpropagated.view + .filterKeys(k => !entries.contains(k) && !extraFields.contains(k)) + .toMap + val extracted = + cp.getTextMapPropagator.extract( + Context.root.underlying, + entries ++ toSkip, + mkJTMG + ) + val wrapperExtracted = + wrapper.textMapPropagator.extract(Context.root, entries ++ toSkip) + var injected = Map.empty[String, String] + cp.getTextMapPropagator.inject[Carrier]( + extracted, + null, + (_, k, v) => injected = injected.updated(k, v) + ) + val wrapperInjected = + wrapper.textMapPropagator.inject( + wrapperExtracted, + Map.empty[String, String] + ) + assertEquals(wrapperInjected, injected) + } + } + property("Scala ContextPropagators and wrapper behavior equivalence") { + forAll { + (entries: Carrier, extraFields: Set[String], unpropagated: Carrier) => + val cp = ContextPropagators.of( + PassThroughPropagator[Context, Context.Key]( + entries.keySet ++ extraFields + ) + ) + val wrapper = cp.asJava + assertEquals( + wrapper.getTextMapPropagator.fields().asScala.toSeq, + cp.textMapPropagator.fields.toSeq + ) + val toSkip = unpropagated.view + .filterKeys(k => !entries.contains(k) && !extraFields.contains(k)) + .toMap + val extracted = + cp.textMapPropagator.extract(Context.root, entries ++ toSkip) + val wrapperExtracted = + wrapper.getTextMapPropagator.extract( + Context.root.underlying, + entries ++ toSkip, + mkJTMG + ) + val injected = + cp.textMapPropagator.inject(extracted, Map.empty[String, String]) + var wrapperInjected = Map.empty[String, String] + wrapper.getTextMapPropagator.inject[Carrier]( + wrapperExtracted, + null, + (_, k, v) => wrapperInjected = wrapperInjected.updated(k, v) + ) + assertEquals(wrapperInjected, injected) + } + } +} 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 647c85630..ded5b0423 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 @@ -48,9 +48,12 @@ import io.opentelemetry.sdk.trace.`export`.SimpleSpanProcessor import io.opentelemetry.sdk.trace.internal.data.ExceptionEventData import munit.CatsEffectSuite import org.typelevel.otel4s.Attribute +import org.typelevel.otel4s.context.propagation.{ + PassThroughPropagator => SPassThroughPropagator +} import org.typelevel.otel4s.context.propagation.ContextPropagators -import org.typelevel.otel4s.java.TextMapPropagatorImpl import org.typelevel.otel4s.java.context.Context +import org.typelevel.otel4s.java.context.propagation.PropagatorConverters._ import org.typelevel.otel4s.java.instances._ import org.typelevel.otel4s.trace.Span import org.typelevel.otel4s.trace.Tracer @@ -948,22 +951,37 @@ class TracerSuite extends CatsEffectSuite { // typelevel/otel4s#277 test("retain all of a provided context through propagation") { - TestControl.executeEmbed { - for { - sdk <- makeSdk(additionalPropagators = - Seq(PassThroughPropagator.create("foo", "bar")) - ) - tracer <- sdk.provider.get("tracer") - _ <- tracer.joinOrRoot(Map("foo" -> "1", "baz" -> "2")) { - for { - carrier <- tracer.propagate(Map.empty[String, String]) - } yield { - assertEquals(carrier.size, 1) - assertEquals(carrier.get("foo"), Some("1")) - } + for { + sdk <- makeSdk(additionalPropagators = + Seq(PassThroughPropagator.create("foo", "bar")) + ) + tracer <- sdk.provider.get("tracer") + _ <- tracer.joinOrRoot(Map("foo" -> "1", "baz" -> "2")) { + for { + carrier <- tracer.propagate(Map.empty[String, String]) + } yield { + assertEquals(carrier.size, 1) + assertEquals(carrier.get("foo"), Some("1")) } - } yield () - } + } + } yield () + } + + test("retain context when using Java and Scala propagators together") { + for { + sdk <- makeSdk(additionalPropagators = + Seq(SPassThroughPropagator[Context, Context.Key]("foo", "bar").asJava) + ) + tracer <- sdk.provider.get("tracer") + _ <- tracer.joinOrRoot(Map("foo" -> "1", "baz" -> "2")) { + for { + carrier <- tracer.propagate(Map.empty[String, String]) + } yield { + assertEquals(carrier.size, 1) + assertEquals(carrier.get("foo"), Some("1")) + } + } + } yield () } test("nested SpanOps#surround for Tracer[IO]") { @@ -1203,9 +1221,7 @@ class TracerSuite extends CatsEffectSuite { W3CTraceContextPropagator.getInstance() +: additionalPropagators val propagators = ContextPropagators.of( - new TextMapPropagatorImpl( - JTextMapPropagator.composite(textMapPropagators.asJava) - ) + JTextMapPropagator.composite(textMapPropagators.asJava).asScala ) val provider = TracerProviderImpl.local[IO](tracerProvider, propagators) diff --git a/licenses/LICENSE_scala b/licenses/LICENSE_scala new file mode 100644 index 000000000..261eeb9e9 --- /dev/null +++ b/licenses/LICENSE_scala @@ -0,0 +1,201 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright [yyyy] [name of copyright owner] + + 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. From c1565ad6e54cf3025263e20b30d11777ca71e5c5 Mon Sep 17 00:00:00 2001 From: Marissa Date: Wed, 1 Nov 2023 15:35:51 -0400 Subject: [PATCH 4/4] Delete `TextMapSetter` --- .../context/propagation/TextMapGetter.scala | 2 - .../context/propagation/TextMapSetter.scala | 68 ------------------- .../context/propagation/TextMapUpdater.scala | 2 - .../propagation/TextMapSetterProps.scala | 58 ---------------- .../propagation/TextMapSetterSuite.scala | 60 ---------------- 5 files changed, 190 deletions(-) delete mode 100644 core/common/src/main/scala/org/typelevel/otel4s/context/propagation/TextMapSetter.scala delete mode 100644 core/common/src/test/scala/org/typelevel/otel4s/context/propagation/TextMapSetterProps.scala delete mode 100644 core/common/src/test/scala/org/typelevel/otel4s/context/propagation/TextMapSetterSuite.scala diff --git a/core/common/src/main/scala/org/typelevel/otel4s/context/propagation/TextMapGetter.scala b/core/common/src/main/scala/org/typelevel/otel4s/context/propagation/TextMapGetter.scala index 25e8347e4..38805837a 100644 --- a/core/common/src/main/scala/org/typelevel/otel4s/context/propagation/TextMapGetter.scala +++ b/core/common/src/main/scala/org/typelevel/otel4s/context/propagation/TextMapGetter.scala @@ -34,8 +34,6 @@ import scala.collection.generic.IsSeq * behavior of such implicit instances will be made to match the specification. * * @see - * See [[TextMapSetter]] to set a value to a mutable carrier - * @see * See [[TextMapUpdater]] to update values of an immutable carrier * * @tparam A diff --git a/core/common/src/main/scala/org/typelevel/otel4s/context/propagation/TextMapSetter.scala b/core/common/src/main/scala/org/typelevel/otel4s/context/propagation/TextMapSetter.scala deleted file mode 100644 index c79d9c216..000000000 --- a/core/common/src/main/scala/org/typelevel/otel4s/context/propagation/TextMapSetter.scala +++ /dev/null @@ -1,68 +0,0 @@ -/* - * 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.context.propagation - -import scala.collection.mutable -import scala.collection.mutable.Buffer - -/** Offers a way to store a string value associated with a given key. - * - * A trait that defines a method to set a value in a key-value store. - * - * Implicit instances of `TextMapSetter` are provided for - * [[scala.collection.mutable.Map]] and [[scala.collection.mutable.Buffer]] - * types. The behavior of `TextMapSetter[Buffer[(String, String)]]` when - * duplicate keys are present is unspecified, and may change at any time. In - * particular, if the behavior of `Buffer` types with duplicate keys is ever - * specified by open telemetry, the behavior of such implicit instances will be - * made to match the specification. - * - * @see - * See [[TextMapGetter]] to get a value from the carrier - * @see - * See [[TextMapUpdater]] to update values of an immutable carrier - * - * @tparam A - * the type of the carrier - */ -trait TextMapSetter[A] { - - /** Sets the `value` associated with the given `key` in the `carrier`. - * - * '''Important:''' the carrier must to be '''mutable'''. - * - * @param carrier - * the carrier to store the key-value pair at - * - * @param key - * the key to associate the value with - * - * @param value - * the value to set - */ - def unsafeSet(carrier: A, key: String, value: String): Unit -} - -object TextMapSetter { - def apply[A](implicit setter: TextMapSetter[A]): TextMapSetter[A] = setter - - implicit def forMap[C <: mutable.Map[String, String]]: TextMapSetter[C] = - (carrier: C, key: String, value: String) => carrier.update(key, value) - - implicit def forBuffer[C <: Buffer[(String, String)]]: TextMapSetter[C] = - (carrier: C, key: String, value: String) => carrier.append(key -> value) -} diff --git a/core/common/src/main/scala/org/typelevel/otel4s/context/propagation/TextMapUpdater.scala b/core/common/src/main/scala/org/typelevel/otel4s/context/propagation/TextMapUpdater.scala index 5bc179da5..ca3c70399 100644 --- a/core/common/src/main/scala/org/typelevel/otel4s/context/propagation/TextMapUpdater.scala +++ b/core/common/src/main/scala/org/typelevel/otel4s/context/propagation/TextMapUpdater.scala @@ -35,8 +35,6 @@ import scala.collection.immutable.SortedMapOps * * @see * See [[TextMapGetter]] to get a value from the carrier - * @see - * See [[TextMapSetter]] to set values to a mutable carrier * * @tparam A * the type of the carrier diff --git a/core/common/src/test/scala/org/typelevel/otel4s/context/propagation/TextMapSetterProps.scala b/core/common/src/test/scala/org/typelevel/otel4s/context/propagation/TextMapSetterProps.scala deleted file mode 100644 index df2cacf08..000000000 --- a/core/common/src/test/scala/org/typelevel/otel4s/context/propagation/TextMapSetterProps.scala +++ /dev/null @@ -1,58 +0,0 @@ -/* - * 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.context.propagation - -import lgbt.princess.platform.Platform -import munit.ScalaCheckSuite -import org.scalacheck.Arbitrary -import org.scalacheck.Prop._ - -import scala.collection.generic.IsIterable -import scala.collection.mutable -import scala.reflect.ClassTag - -class TextMapSetterProps extends ScalaCheckSuite { - def mkProp[A: Arbitrary](implicit - tag: ClassTag[A], - setter: TextMapSetter[A], - conv: IsIterable[A] { type A = (String, String) } - ): Unit = - property(s"TextMapSetter for ${tag.runtimeClass.getName}") { - forAll { (coll: A, key: String, value: String) => - setter.unsafeSet(coll, key, value) - assert(conv(coll).exists(_ == (key -> value))) - } - } - - mkProp[mutable.Map[String, String]] - mkProp[mutable.HashMap[String, String]] - mkProp[mutable.CollisionProofHashMap[String, String]] - mkProp[mutable.SortedMap[String, String]] - mkProp[mutable.TreeMap[String, String]] - mkProp[mutable.SeqMap[String, String]] - mkProp[mutable.LinkedHashMap[String, String]] - if (Platform.isJVM) mkProp[collection.concurrent.TrieMap[String, String]] - - mkProp[mutable.Buffer[(String, String)]] - mkProp[mutable.ListBuffer[(String, String)]] - mkProp[mutable.IndexedBuffer[(String, String)]] - mkProp[mutable.ArrayBuffer[(String, String)]] - mkProp[mutable.Queue[(String, String)]] - mkProp[mutable.Stack[(String, String)]] - mkProp[mutable.ArrayDeque[(String, String)]] - mkProp[mutable.UnrolledBuffer[(String, String)]] -} diff --git a/core/common/src/test/scala/org/typelevel/otel4s/context/propagation/TextMapSetterSuite.scala b/core/common/src/test/scala/org/typelevel/otel4s/context/propagation/TextMapSetterSuite.scala deleted file mode 100644 index 349679361..000000000 --- a/core/common/src/test/scala/org/typelevel/otel4s/context/propagation/TextMapSetterSuite.scala +++ /dev/null @@ -1,60 +0,0 @@ -/* - * 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.context.propagation - -import munit.FunSuite - -import scala.collection.mutable -import scala.collection.mutable.ArrayBuffer -import scala.collection.mutable.Buffer -import scala.collection.mutable.ListBuffer - -class TextMapSetterSuite extends FunSuite { - // `TextMapSetter[C]` is not implicitly summoned by this method - // so that it tests that instances are available in a non-generic - // context. - def check[C](tms: TextMapSetter[C])(carrier: C)(expected: C): Unit = { - tms.unsafeSet(carrier, "1", "one") - tms.unsafeSet(carrier, "2", "two") - tms.unsafeSet(carrier, "3", "three") - assertEquals(carrier, expected) - } - - test("TextMapSetter[mutable.Map[String, String]") { - check(TextMapSetter[mutable.HashMap[String, String]])( - mutable.HashMap.empty - )(mutable.HashMap("1" -> "one", "2" -> "two", "3" -> "three")) - check(TextMapSetter[mutable.TreeMap[String, String]])( - mutable.TreeMap.empty - )(mutable.TreeMap("1" -> "one", "2" -> "two", "3" -> "three")) - check(TextMapSetter[mutable.Map[String, String]])( - mutable.Map.empty - )(mutable.Map("1" -> "one", "2" -> "two", "3" -> "three")) - } - - test("TextMapSetter[Buffer[(String, String)]") { - check(TextMapSetter[ListBuffer[(String, String)]])( - ListBuffer.empty - )(ListBuffer("1" -> "one", "2" -> "two", "3" -> "three")) - check(TextMapSetter[ArrayBuffer[(String, String)]])( - ArrayBuffer.empty - )(ArrayBuffer("1" -> "one", "2" -> "two", "3" -> "three")) - check(TextMapSetter[Buffer[(String, String)]])(Buffer.empty)( - Buffer("1" -> "one", "2" -> "two", "3" -> "three") - ) - } -}