212 lines
9.1 KiB
C#
212 lines
9.1 KiB
C#
// <copyright file="JaegerUdpBatcherTests.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;
|
|
using System.Collections.Generic;
|
|
using System.Diagnostics;
|
|
using System.Linq;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
using OpenTelemetry.Exporter.Jaeger.Implementation;
|
|
using OpenTelemetry.Resources;
|
|
using OpenTelemetry.Trace;
|
|
using OpenTelemetry.Trace.Export;
|
|
using Xunit;
|
|
|
|
namespace OpenTelemetry.Exporter.Jaeger.Tests.Implementation
|
|
{
|
|
public class JaegerUdpBatcherTests
|
|
{
|
|
public const string TestPayloadBase64 = "goEBCWVtaXRCYXRjaBwcGAx0ZXN0IHByb2Nlc3MZHBgQdGVzdF9wcm9jZXNzX3RhZxUAGAp0ZXN0X3ZhbHVlAAAZHBab5cuG2OehhdwBFuPakI2n2cCVLhaAjfWp6NHt6dQBFrK5moSni5GXGBgETmFtZRkcFQAWm+XLhtjnoYXcARbj2pCNp9nAlS4W/Y6j+bqS9fbuAQAVAhaAgLPexpa/BRaAnJw5GYwYCXN0cmluZ0tleRUAGAV2YWx1ZQAYB2xvbmdLZXkVBkYCABgIbG9uZ0tleTIVBkYCABgJZG91YmxlS2V5FQInAAAAAAAA8D8AGApkb3VibGVLZXkyFQInAAAAAAAA8D8AGAdib29sS2V5FQQxABgJc3Bhbi5raW5kFQAYBmNsaWVudAAYDm90LnN0YXR1c19jb2RlFQAYAk9rABksFoCAs97Glr8FGSwYA2tleRUAGAV2YWx1ZQAYB21lc3NhZ2UVABgGRXZlbnQxAAAWgICz3saWvwUZLBgDa2V5FQAYBXZhbHVlABgHbWVzc2FnZRUAGAZFdmVudDIAAAAAAA==";
|
|
|
|
internal static Process TestProcess { get; } = new Process("test process", new Dictionary<string, object> { { "test_process_tag", "test_value" } });
|
|
|
|
[Fact]
|
|
public async Task JaegerUdpBatcherTests_BuildBatchesToTransmit_DefaultBatch()
|
|
{
|
|
// Arrange
|
|
var options = new JaegerExporterOptions { ServiceName = "TestService", MaxFlushInterval = TimeSpan.FromHours(1) };
|
|
|
|
var jaegerUdpBatcher = new JaegerUdpBatcher(options);
|
|
|
|
// Act
|
|
await jaegerUdpBatcher.AppendAsync(CreateTestJaegerSpan(), CancellationToken.None).ConfigureAwait(false);
|
|
await jaegerUdpBatcher.AppendAsync(CreateTestJaegerSpan(), CancellationToken.None).ConfigureAwait(false);
|
|
await jaegerUdpBatcher.AppendAsync(CreateTestJaegerSpan(), CancellationToken.None).ConfigureAwait(false);
|
|
|
|
var batches = jaegerUdpBatcher.CurrentBatches.Values;
|
|
|
|
// Assert
|
|
Assert.Single(batches);
|
|
Assert.Equal("TestService", batches.First().Process.ServiceName);
|
|
Assert.Equal(3, batches.First().SpanMessages.Count());
|
|
}
|
|
|
|
[Fact]
|
|
public async Task JaegerUdpBatcherTests_BuildBatchesToTransmit_MultipleBatches()
|
|
{
|
|
// Arrange
|
|
var options = new JaegerExporterOptions { ServiceName = "TestService", MaxFlushInterval = TimeSpan.FromHours(1) };
|
|
|
|
var jaegerUdpBatcher = new JaegerUdpBatcher(options);
|
|
|
|
// Act
|
|
await jaegerUdpBatcher.AppendAsync(CreateTestJaegerSpan(), CancellationToken.None).ConfigureAwait(false);
|
|
await jaegerUdpBatcher.AppendAsync(
|
|
CreateTestJaegerSpan(
|
|
additionalAttributes: new Dictionary<string, object>
|
|
{
|
|
["peer.service"] = "MySQL",
|
|
}),
|
|
CancellationToken.None).ConfigureAwait(false);
|
|
await jaegerUdpBatcher.AppendAsync(CreateTestJaegerSpan(), CancellationToken.None).ConfigureAwait(false);
|
|
|
|
var batches = jaegerUdpBatcher.CurrentBatches.Values;
|
|
|
|
// Assert
|
|
Assert.Equal(2, batches.Count());
|
|
|
|
var primaryBatch = batches.Where(b => b.Process.ServiceName == "TestService");
|
|
Assert.Single(primaryBatch);
|
|
Assert.Equal(2, primaryBatch.First().SpanMessages.Count());
|
|
|
|
var mySQLBatch = batches.Where(b => b.Process.ServiceName == "MySQL");
|
|
Assert.Single(mySQLBatch);
|
|
Assert.Single(mySQLBatch.First().SpanMessages);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task JaegerUdpBatcherTests_BuildBatchesToTransmit_FlushedBatch()
|
|
{
|
|
// Arrange
|
|
var options = new JaegerExporterOptions { ServiceName = "TestService", MaxFlushInterval = TimeSpan.FromHours(1), MaxPacketSize = 750 };
|
|
|
|
var jaegerUdpBatcher = new JaegerUdpBatcher(options);
|
|
|
|
// Act
|
|
await jaegerUdpBatcher.AppendAsync(CreateTestJaegerSpan(), CancellationToken.None).ConfigureAwait(false);
|
|
await jaegerUdpBatcher.AppendAsync(CreateTestJaegerSpan(), CancellationToken.None).ConfigureAwait(false);
|
|
var flushCount = await jaegerUdpBatcher.AppendAsync(CreateTestJaegerSpan(), CancellationToken.None).ConfigureAwait(false);
|
|
|
|
var batches = jaegerUdpBatcher.CurrentBatches.Values;
|
|
|
|
// Assert
|
|
Assert.Equal(2, flushCount);
|
|
Assert.Single(batches);
|
|
Assert.Equal("TestService", batches.First().Process.ServiceName);
|
|
Assert.Single(batches.First().SpanMessages);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task JaegerUdpBatcher_IntegrationTest()
|
|
{
|
|
var validJaegerThriftPayload = Convert.FromBase64String(TestPayloadBase64);
|
|
|
|
var memoryTransport = new InMemoryTransport();
|
|
|
|
using var jaegerUdpBatcher = new JaegerUdpBatcher(
|
|
new JaegerExporterOptions { MaxFlushInterval = TimeSpan.FromHours(1) },
|
|
memoryTransport);
|
|
jaegerUdpBatcher.Process = TestProcess;
|
|
|
|
await jaegerUdpBatcher.AppendAsync(CreateTestPayloadJaegerSpan(), CancellationToken.None).ConfigureAwait(false);
|
|
|
|
await jaegerUdpBatcher.FlushAsync(CancellationToken.None).ConfigureAwait(false);
|
|
|
|
Assert.Equal(Convert.ToBase64String(validJaegerThriftPayload), Convert.ToBase64String(memoryTransport.ToArray()));
|
|
|
|
memoryTransport.Reset();
|
|
|
|
await jaegerUdpBatcher.AppendAsync(CreateTestPayloadJaegerSpan(), CancellationToken.None).ConfigureAwait(false);
|
|
|
|
await jaegerUdpBatcher.FlushAsync(CancellationToken.None).ConfigureAwait(false);
|
|
|
|
// SeqNo is the second byte.
|
|
validJaegerThriftPayload[2]++;
|
|
|
|
Assert.Equal(Convert.ToBase64String(validJaegerThriftPayload), Convert.ToBase64String(memoryTransport.ToArray()));
|
|
}
|
|
|
|
internal static JaegerSpan CreateTestPayloadJaegerSpan()
|
|
{
|
|
var startTimestamp = new DateTimeOffset(2019, 1, 1, 0, 0, 0, TimeSpan.Zero);
|
|
var endTimestamp = startTimestamp.AddSeconds(60);
|
|
var eventTimestamp = new DateTimeOffset(2019, 1, 1, 0, 0, 0, TimeSpan.Zero);
|
|
|
|
var traceId = ActivityTraceId.CreateFromString("e8ea7e9ac72de94e91fabc613f9686b2".AsSpan());
|
|
var spanId = ActivitySpanId.CreateFromString("6a69db47429ea340".AsSpan());
|
|
var parentSpanId = ActivitySpanId.CreateFromBytes(new byte[] { 12, 23, 34, 45, 56, 67, 78, 89 });
|
|
var attributes = new Dictionary<string, object>
|
|
{
|
|
{ "stringKey", "value" },
|
|
{ "longKey", 1L },
|
|
{ "longKey2", 1 },
|
|
{ "doubleKey", 1D },
|
|
{ "doubleKey2", 1F },
|
|
{ "boolKey", true },
|
|
};
|
|
var events = new List<Event>
|
|
{
|
|
new Event(
|
|
"Event1",
|
|
eventTimestamp,
|
|
new Dictionary<string, object>
|
|
{
|
|
{ "key", "value" },
|
|
}),
|
|
new Event(
|
|
"Event2",
|
|
eventTimestamp,
|
|
new Dictionary<string, object>
|
|
{
|
|
{ "key", "value" },
|
|
}),
|
|
};
|
|
|
|
var linkedSpanId = ActivitySpanId.CreateFromString("888915b6286b9c41".AsSpan());
|
|
|
|
var link = new Link(new SpanContext(
|
|
traceId,
|
|
linkedSpanId,
|
|
ActivityTraceFlags.Recorded));
|
|
|
|
return new SpanData(
|
|
"Name",
|
|
new SpanContext(traceId, spanId, ActivityTraceFlags.Recorded),
|
|
parentSpanId,
|
|
SpanKind.Client,
|
|
startTimestamp,
|
|
attributes,
|
|
events,
|
|
new[] { link, },
|
|
null,
|
|
Status.Ok,
|
|
endTimestamp).ToJaegerSpan();
|
|
}
|
|
|
|
internal static JaegerSpan CreateTestJaegerSpan(
|
|
bool setAttributes = true,
|
|
Dictionary<string, object> additionalAttributes = null,
|
|
bool addEvents = true,
|
|
bool addLinks = true,
|
|
Resource resource = null,
|
|
SpanKind kind = SpanKind.Client)
|
|
{
|
|
return JaegerSpanConverterTest.CreateTestSpan(
|
|
setAttributes, additionalAttributes, addEvents, addLinks, resource, kind).ToJaegerSpan();
|
|
}
|
|
}
|
|
}
|