opentelemetry-dotnet/test/OpenTelemetry.Shims.OpenTra.../TracerShimTests.cs

216 lines
8.5 KiB
C#

// <copyright file="TracerShimTests.cs" company="OpenTelemetry Authors">
// Copyright The OpenTelemetry Authors
//
// 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.
// </copyright>
using System.Collections;
using System.Diagnostics;
using Moq;
using OpenTelemetry.Context.Propagation;
using OpenTelemetry.Trace;
using OpenTracing;
using OpenTracing.Propagation;
using Xunit;
namespace OpenTelemetry.Shims.OpenTracing.Tests
{
public class TracerShimTests
{
private const string TracerName = "defaultactivitysource";
[Fact]
public void CtorArgumentValidation()
{
// null tracer and text format
Assert.Throws<ArgumentNullException>(() => new TracerShim(null, null));
// null tracer
Assert.Throws<ArgumentNullException>(() => new TracerShim(null, new TraceContextPropagator()));
// null context format
var tracerMock = new Mock<Tracer>();
Assert.Throws<ArgumentNullException>(() => new TracerShim(TracerProvider.Default.GetTracer("test"), null));
}
[Fact]
public void ScopeManager_NotNull()
{
var tracer = TracerProvider.Default.GetTracer(TracerName);
var shim = new TracerShim(tracer, new TraceContextPropagator());
// Internals of the ScopeManagerShim tested elsewhere
Assert.NotNull(shim.ScopeManager as ScopeManagerShim);
}
[Fact]
public void BuildSpan_NotNull()
{
var tracer = TracerProvider.Default.GetTracer(TracerName);
var shim = new TracerShim(tracer, new TraceContextPropagator());
// Internals of the SpanBuilderShim tested elsewhere
Assert.NotNull(shim.BuildSpan("foo") as SpanBuilderShim);
}
[Fact]
public void Inject_ArgumentValidation()
{
var tracer = TracerProvider.Default.GetTracer(TracerName);
var shim = new TracerShim(tracer, new TraceContextPropagator());
var spanContextShim = new SpanContextShim(new SpanContext(ActivityTraceId.CreateRandom(), ActivitySpanId.CreateRandom(), ActivityTraceFlags.None));
var mockFormat = new Mock<IFormat<ITextMap>>();
var mockCarrier = new Mock<ITextMap>();
Assert.Throws<ArgumentNullException>(() => shim.Inject(null, mockFormat.Object, mockCarrier.Object));
Assert.Throws<InvalidCastException>(() => shim.Inject(new Mock<ISpanContext>().Object, mockFormat.Object, mockCarrier.Object));
Assert.Throws<ArgumentNullException>(() => shim.Inject(spanContextShim, null, mockCarrier.Object));
Assert.Throws<ArgumentNullException>(() => shim.Inject(spanContextShim, mockFormat.Object, null));
}
[Fact]
public void Inject_UnknownFormatIgnored()
{
var tracer = TracerProvider.Default.GetTracer(TracerName);
var shim = new TracerShim(tracer, new TraceContextPropagator());
var spanContextShim = new SpanContextShim(new SpanContext(ActivityTraceId.CreateRandom(), ActivitySpanId.CreateRandom(), ActivityTraceFlags.Recorded));
// Only two specific types of ITextMap are supported, and neither is a Mock.
var mockCarrier = new Mock<ITextMap>();
shim.Inject(spanContextShim, new Mock<IFormat<ITextMap>>().Object, mockCarrier.Object);
// Verify that the carrier mock was never called.
mockCarrier.Verify(x => x.Set(It.IsAny<string>(), It.IsAny<string>()), Times.Never);
}
[Fact]
public void Extract_ArgumentValidation()
{
var tracer = TracerProvider.Default.GetTracer(TracerName);
var shim = new TracerShim(tracer, new TraceContextPropagator());
Assert.Throws<ArgumentNullException>(() => shim.Extract(null, new Mock<ITextMap>().Object));
Assert.Throws<ArgumentNullException>(() => shim.Extract(new Mock<IFormat<ITextMap>>().Object, null));
}
[Fact]
public void Extract_UnknownFormatIgnored()
{
var tracer = TracerProvider.Default.GetTracer(TracerName);
var shim = new TracerShim(tracer, new TraceContextPropagator());
var spanContextShim = new SpanContextShim(new SpanContext(ActivityTraceId.CreateRandom(), ActivitySpanId.CreateRandom(), ActivityTraceFlags.None));
// Only two specific types of ITextMap are supported, and neither is a Mock.
var mockCarrier = new Mock<ITextMap>();
var context = shim.Extract(new Mock<IFormat<ITextMap>>().Object, mockCarrier.Object);
// Verify that the carrier mock was never called.
mockCarrier.Verify(x => x.GetEnumerator(), Times.Never);
}
[Fact]
public void Extract_InvalidTraceParent()
{
var tracer = TracerProvider.Default.GetTracer(TracerName);
var shim = new TracerShim(tracer, new TraceContextPropagator());
var mockCarrier = new Mock<ITextMap>();
// The ProxyTracer uses OpenTelemetry.Context.Propagation.TraceContextPropagator, so we need to satisfy the traceparent key at the least
var carrierMap = new Dictionary<string, string>
{
// This is an invalid traceparent value
{ "traceparent", "unused" },
};
mockCarrier.Setup(x => x.GetEnumerator()).Returns(carrierMap.GetEnumerator());
var spanContextShim = shim.Extract(BuiltinFormats.TextMap, mockCarrier.Object) as SpanContextShim;
// Verify that the carrier was called
mockCarrier.Verify(x => x.GetEnumerator(), Times.Once);
Assert.Null(spanContextShim);
}
[Fact]
public void InjectExtract_TextMap_Ok()
{
var carrier = new TextMapCarrier();
var spanContextShim = new SpanContextShim(new SpanContext(ActivityTraceId.CreateRandom(), ActivitySpanId.CreateRandom(), ActivityTraceFlags.None));
var format = new TraceContextPropagator();
var tracer = TracerProvider.Default.GetTracer(TracerName);
var shim = new TracerShim(tracer, format);
// first inject
shim.Inject(spanContextShim, BuiltinFormats.TextMap, carrier);
// then extract
var extractedSpanContext = shim.Extract(BuiltinFormats.TextMap, carrier);
AssertOpenTracerSpanContextEqual(spanContextShim, extractedSpanContext);
}
private static void AssertOpenTracerSpanContextEqual(ISpanContext source, ISpanContext target)
{
Assert.Equal(source.TraceId, target.TraceId);
Assert.Equal(source.SpanId, target.SpanId);
// TODO BaggageItems are not implemented yet.
}
/// <summary>
/// Simple ITextMap implementation used for the inject/extract tests.
/// </summary>
/// <seealso cref="OpenTracing.Propagation.ITextMap" />
private class TextMapCarrier : ITextMap
{
private readonly Dictionary<string, string> map = new();
public IDictionary<string, string> Map => this.map;
public IEnumerator<KeyValuePair<string, string>> GetEnumerator() => this.map.GetEnumerator();
public void Set(string key, string value)
{
this.map[key] = value;
}
IEnumerator IEnumerable.GetEnumerator() => this.map.GetEnumerator();
}
/// <summary>
/// Simple IBinary implementation used for the inject/extract tests.
/// </summary>
/// <seealso cref="OpenTracing.Propagation.IBinary" />
private class BinaryCarrier : IBinary
{
private readonly MemoryStream carrierStream = new();
public MemoryStream Get() => this.carrierStream;
public void Set(MemoryStream stream)
{
this.carrierStream.SetLength(stream.Length);
this.carrierStream.Seek(0, SeekOrigin.Begin);
stream.CopyTo(this.carrierStream, (int)this.carrierStream.Length);
}
}
}
}