|
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
using System.Collections.Concurrent;
using System.Diagnostics;
using System.Diagnostics.CodeAnalysis;
using System.Runtime.InteropServices;
using Aspire.Dashboard.Otlp.Storage;
using Google.Protobuf.Collections;
using OpenTelemetry.Proto.Common.V1;
using OpenTelemetry.Proto.Metrics.V1;
namespace Aspire.Dashboard.Otlp.Model;
[DebuggerDisplay("ApplicationName = {ApplicationName}, InstanceId = {InstanceId}")]
public class OtlpApplication
{
public const string SERVICE_NAME = "service.name";
public const string SERVICE_INSTANCE_ID = "service.instance.id";
public const string PROCESS_EXECUTABLE_NAME = "process.executable.name";
public string ApplicationName { get; }
public string InstanceId { get; }
public OtlpContext Context { get; }
public ApplicationKey ApplicationKey => new ApplicationKey(ApplicationName, InstanceId);
private readonly ReaderWriterLockSlim _metricsLock = new();
private readonly Dictionary<string, OtlpMeter> _meters = new();
private readonly Dictionary<OtlpInstrumentKey, OtlpInstrument> _instruments = new();
private readonly ConcurrentDictionary<KeyValuePair<string, string>[], OtlpApplicationView> _applicationViews = new(ApplicationViewKeyComparer.Instance);
public OtlpApplication(string name, string instanceId, OtlpContext context)
{
ApplicationName = name;
InstanceId = instanceId;
Context = context;
}
public void AddMetrics(AddContext context, RepeatedField<ScopeMetrics> scopeMetrics)
{
_metricsLock.EnterWriteLock();
try
{
// Temporary attributes array to use when adding metrics to the instruments.
KeyValuePair<string, string>[]? tempAttributes = null;
foreach (var sm in scopeMetrics)
{
foreach (var metric in sm.Metrics)
{
OtlpInstrument instrument;
try
{
if (string.IsNullOrEmpty(metric.Name))
{
throw new InvalidOperationException("Instrument name is required.");
}
var instrumentKey = new OtlpInstrumentKey(sm.Scope.Name, metric.Name);
ref var instrumentRef = ref CollectionsMarshal.GetValueRefOrAddDefault(_instruments, instrumentKey, out _);
// Adds to dictionary if not present.
instrumentRef ??= new OtlpInstrument
{
Summary = new OtlpInstrumentSummary
{
Name = metric.Name,
Description = metric.Description,
Unit = metric.Unit,
Type = MapMetricType(metric.DataCase),
Parent = GetMeter(sm.Scope)
},
Context = Context
};
instrument = instrumentRef;
}
catch (Exception ex)
{
// If we can't create the instrument then all data points for it are failures.
context.FailureCount += GetMetricDataPointCount(metric);
Context.Logger.LogInformation(ex, "Error adding metric instrument {MetricName}.", metric.Name);
continue;
}
AddMetrics(instrument, metric, context, ref tempAttributes);
}
}
}
finally
{
_metricsLock.ExitWriteLock();
}
}
private static int GetMetricDataPointCount(Metric metric)
{
return metric.DataCase switch
{
Metric.DataOneofCase.Gauge => metric.Gauge.DataPoints.Count,
Metric.DataOneofCase.Sum => metric.Sum.DataPoints.Count,
Metric.DataOneofCase.Histogram => metric.Histogram.DataPoints.Count,
Metric.DataOneofCase.Summary => metric.Summary.DataPoints.Count,
Metric.DataOneofCase.ExponentialHistogram => metric.ExponentialHistogram.DataPoints.Count,
_ => 0,
};
}
private void AddMetrics(OtlpInstrument instrument, Metric metric, AddContext context, ref KeyValuePair<string, string>[]? tempAttributes)
{
switch (metric.DataCase)
{
case Metric.DataOneofCase.Gauge:
foreach (var d in metric.Gauge.DataPoints)
{
try
{
instrument.FindScope(d.Attributes, ref tempAttributes).AddPointValue(d, Context);
}
catch (Exception ex)
{
context.FailureCount++;
Context.Logger.LogInformation(ex, "Error adding metric.");
}
}
break;
case Metric.DataOneofCase.Sum:
foreach (var d in metric.Sum.DataPoints)
{
try
{
instrument.FindScope(d.Attributes, ref tempAttributes).AddPointValue(d, Context);
}
catch (Exception ex)
{
context.FailureCount++;
Context.Logger.LogInformation(ex, "Error adding metric.");
}
}
break;
case Metric.DataOneofCase.Histogram:
foreach (var d in metric.Histogram.DataPoints)
{
try
{
instrument.FindScope(d.Attributes, ref tempAttributes).AddHistogramValue(d, Context);
}
catch (Exception ex)
{
context.FailureCount++;
Context.Logger.LogInformation(ex, "Error adding metric.");
}
}
break;
case Metric.DataOneofCase.Summary:
context.FailureCount += metric.Summary.DataPoints.Count;
Context.Logger.LogInformation("Error adding summary metrics. Summary is not supported.");
break;
case Metric.DataOneofCase.ExponentialHistogram:
context.FailureCount += metric.ExponentialHistogram.DataPoints.Count;
Context.Logger.LogInformation("Error adding exponential histogram metrics. Exponential histogram is not supported.");
break;
}
}
public void ClearMetrics()
{
_metricsLock.EnterWriteLock();
try
{
_instruments.Clear();
}
finally
{
_metricsLock.ExitWriteLock();
}
}
private static OtlpInstrumentType MapMetricType(Metric.DataOneofCase data)
{
return data switch
{
Metric.DataOneofCase.Gauge => OtlpInstrumentType.Gauge,
Metric.DataOneofCase.Sum => OtlpInstrumentType.Sum,
Metric.DataOneofCase.Histogram => OtlpInstrumentType.Histogram,
_ => OtlpInstrumentType.Unsupported
};
}
private OtlpMeter GetMeter(InstrumentationScope scope)
{
ref var meter = ref CollectionsMarshal.GetValueRefOrAddDefault(_meters, scope.Name, out _);
// Adds to dictionary if not present.
meter ??= new OtlpMeter(scope, Context);
return meter;
}
public OtlpInstrument? GetInstrument(string meterName, string instrumentName, DateTime? valuesStart, DateTime? valuesEnd)
{
_metricsLock.EnterReadLock();
try
{
if (!_instruments.TryGetValue(new OtlpInstrumentKey(meterName, instrumentName), out var instrument))
{
return null;
}
return OtlpInstrument.Clone(instrument, cloneData: true, valuesStart: valuesStart, valuesEnd: valuesEnd);
}
finally
{
_metricsLock.ExitReadLock();
}
}
public List<OtlpInstrumentSummary> GetInstrumentsSummary()
{
_metricsLock.EnterReadLock();
try
{
var instruments = new List<OtlpInstrumentSummary>(_instruments.Count);
foreach (var instrument in _instruments)
{
instruments.Add(instrument.Value.Summary);
}
return instruments;
}
finally
{
_metricsLock.ExitReadLock();
}
}
public static Dictionary<string, List<OtlpApplication>> GetReplicasByApplicationName(IEnumerable<OtlpApplication> allApplications)
{
return allApplications
.GroupBy(application => application.ApplicationName, StringComparers.ResourceName)
.ToDictionary(grouping => grouping.Key, grouping => grouping.ToList());
}
public static string GetResourceName(OtlpApplicationView app, List<OtlpApplication> allApplications) =>
GetResourceName(app.Application, allApplications);
public static string GetResourceName(OtlpApplication app, List<OtlpApplication> allApplications)
{
var count = 0;
foreach (var item in allApplications)
{
if (string.Equals(item.ApplicationName, app.ApplicationName, StringComparisons.ResourceName))
{
count++;
if (count >= 2)
{
var instanceId = app.InstanceId;
// Convert long GUID into a shorter, more human friendly format.
// Before: aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee
// After: aaaaaaaa
if (Guid.TryParse(instanceId, out var guid))
{
Span<char> chars = stackalloc char[32];
var result = guid.TryFormat(chars, charsWritten: out _, format: "N");
Debug.Assert(result, "Guid.TryFormat not successful.");
instanceId = chars.Slice(0, 8).ToString();
}
return $"{item.ApplicationName}-{instanceId}";
}
}
}
return app.ApplicationName;
}
internal List<OtlpApplicationView> GetViews() => _applicationViews.Values.ToList();
internal OtlpApplicationView GetView(RepeatedField<KeyValue> attributes)
{
// Inefficient to create this to possibly throw it away.
var view = new OtlpApplicationView(this, attributes);
if (_applicationViews.TryGetValue(view.Properties, out var applicationView))
{
return applicationView;
}
return _applicationViews.GetOrAdd(view.Properties, view);
}
/// <summary>
/// Application views are equal when all properties are equal.
/// </summary>
private sealed class ApplicationViewKeyComparer : IEqualityComparer<KeyValuePair<string, string>[]>
{
public static readonly ApplicationViewKeyComparer Instance = new();
public bool Equals(KeyValuePair<string, string>[]? x, KeyValuePair<string, string>[]? y)
{
if (x == y)
{
return true;
}
if (x == null || y == null)
{
return false;
}
if (x.Length != y.Length)
{
return false;
}
for (var i = 0; i < x.Length; i++)
{
if (!string.Equals(x[i].Key, y[i].Key, StringComparisons.OtlpAttribute))
{
return false;
}
if (!string.Equals(x[i].Value, y[i].Value, StringComparisons.OtlpAttribute))
{
return false;
}
}
return true;
}
public int GetHashCode([DisallowNull] KeyValuePair<string, string>[] obj)
{
var hashCode = new HashCode();
for (var i = 0; i < obj.Length; i++)
{
hashCode.Add(StringComparers.OtlpAttribute.GetHashCode(obj[i].Key));
hashCode.Add(StringComparers.OtlpAttribute.GetHashCode(obj[i].Value));
}
return hashCode.ToHashCode();
}
}
}
|