Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Update to Latest Otel Packages #828

Merged
merged 9 commits into from
Mar 11, 2022
Merged
Show file tree
Hide file tree
Changes from 5 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
47 changes: 5 additions & 42 deletions src/Management/src/Abstractions/Diagnostics/DiagnosticsManager.cs
Original file line number Diff line number Diff line change
Expand Up @@ -17,51 +17,42 @@ public class DiagnosticsManager : IObserver<DiagnosticListener>, IDisposable, ID
internal IDisposable _listenersSubscription;
internal ILogger<DiagnosticsManager> _logger;
internal IList<IDiagnosticObserver> _observers;
internal IList<IPolledDiagnosticSource> _sources;
internal IList<IRuntimeDiagnosticSource> _sources;
internal IList<EventListener> _eventListeners;

internal Thread _workerThread;
internal bool _workerThreadShutdown = false;
internal int _started = 0;

private const int POLL_DELAY_MILLI = 15000;

private static readonly Lazy<DiagnosticsManager> AsSingleton = new (() => new DiagnosticsManager());

public static DiagnosticsManager Instance => AsSingleton.Value;

public DiagnosticsManager(
IEnumerable<IPolledDiagnosticSource> polledSources,
IEnumerable<IRuntimeDiagnosticSource> runtimeSources,
IEnumerable<IDiagnosticObserver> observers,
IEnumerable<EventListener> eventListeners,
ILogger<DiagnosticsManager> logger = null)
{
if (polledSources == null)
{
throw new ArgumentNullException(nameof(polledSources));
}

if (observers == null)
{
throw new ArgumentNullException(nameof(observers));
}

_logger = logger;
_observers = observers.ToList();
_sources = polledSources.ToList();
_sources = runtimeSources.ToList();
_eventListeners = eventListeners.ToList();
}

internal DiagnosticsManager(ILogger<DiagnosticsManager> logger = null)
{
_logger = logger;
_observers = new List<IDiagnosticObserver>();
_sources = new List<IPolledDiagnosticSource>();
_sources = new List<IRuntimeDiagnosticSource>();
}

public IList<IDiagnosticObserver> Observers => _observers;

public IList<IPolledDiagnosticSource> Sources => _sources;
public IList<IRuntimeDiagnosticSource> Sources => _sources;

public void OnCompleted()
{
Expand All @@ -86,13 +77,6 @@ public void Start()
if (Interlocked.CompareExchange(ref _started, 1, 0) == 0)
{
_listenersSubscription = DiagnosticListener.AllListeners.Subscribe(this);

_workerThread = new Thread(Poller)
{
IsBackground = true,
Name = "DiagnosticsPoller"
};
_workerThread.Start();
}
}

Expand Down Expand Up @@ -147,26 +131,5 @@ protected virtual void Dispose(bool disposing)
{
Dispose(false);
}

private void Poller(object obj)
{
while (!_workerThreadShutdown)
{
try
{
foreach (var source in _sources)
{
source.Poll();
}

Thread.Sleep(POLL_DELAY_MILLI);
}
catch (Exception e)
{
_logger?.LogError(e, "Diagnostic source poller exception, terminating");
return;
}
}
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -4,8 +4,7 @@

namespace Steeltoe.Common.Diagnostics
{
public interface IPolledDiagnosticSource
public interface IRuntimeDiagnosticSource
{
void Poll();
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -6,5 +6,6 @@ namespace Steeltoe.Management.Endpoint.Metrics
{
public interface IMetricsEndpointOptions : IEndpointOptions
{
long ScrapeResponseCacheDurationMilliseconds { get; }
}
}
186 changes: 16 additions & 170 deletions src/Management/src/EndpointBase/Metrics/MetricsEndpoint.cs
Original file line number Diff line number Diff line change
Expand Up @@ -3,25 +3,19 @@
// See the LICENSE file in the project root for more information.

using Microsoft.Extensions.Logging;
using Steeltoe.Management.OpenTelemetry.Metrics.Export;
using Steeltoe.Management.OpenTelemetry.Metrics.Exporter;
using Steeltoe.Management.OpenTelemetry.Metrics.Processor;
using Steeltoe.Management.OpenTelemetry.Exporters;
using Steeltoe.Management.OpenTelemetry.Metrics;
using System;
using System.Collections.Generic;
using System.Linq;

namespace Steeltoe.Management.Endpoint.Metrics
{
#pragma warning disable CS0618 // Type or member is obsolete
public class MetricsEndpoint : AbstractEndpoint<IMetricsResponse, MetricsRequest>, IMetricsEndpoint
{
private readonly SteeltoeExporter _exporter;
private readonly ILogger<MetricsEndpoint> _logger;

private List<ProcessedMetric<long>> LongMetrics { get; set; }

private List<ProcessedMetric<double>> DoubleMetrics { get; set; }

public MetricsEndpoint(IMetricsEndpointOptions options, SteeltoeExporter exporter, ILogger<MetricsEndpoint> logger = null)
: base(options)
{
Expand Down Expand Up @@ -53,7 +47,7 @@ public override IMetricsResponse Invoke(MetricsRequest request)
return null;
}

protected internal List<MetricSample> GetMetricSamplesByTags(MetricDictionary<List<MetricSample>> measurements, string metricName, IEnumerable<KeyValuePair<string, string>> tags)
protected internal List<MetricSample> GetMetricSamplesByTags(MetricsCollection<List<MetricSample>> measurements, string metricName, IEnumerable<KeyValuePair<string, string>> tags)
{
IEnumerable<MetricSample> filtered = measurements[metricName];
var sampleList = new List<MetricSample>();
Expand All @@ -63,6 +57,7 @@ protected internal List<MetricSample> GetMetricSamplesByTags(MetricDictionary<Li
}

static MetricSample SumAggregator(MetricSample current, MetricSample next) => new (current.Statistic, current.Value + next.Value, current.Tags);
static MetricSample MaxAggregator(MetricSample current, MetricSample next) => new (current.Statistic, current.Value > next.Value ? current.Value : next.Value, current.Tags);

var valueSamples = filtered.Where(sample => sample.Statistic == MetricStatistic.VALUE);
if (valueSamples.Any())
Expand All @@ -89,177 +84,28 @@ protected internal List<MetricSample> GetMetricSamplesByTags(MetricDictionary<Li
sampleList.Add(countSamples.Aggregate(SumAggregator));
}

return sampleList;
}

protected internal MetricsResponse GetMetric(MetricsRequest request, List<MetricSample> measurements, List<MetricTag> availTags)
{
return new MetricsResponse(request.MetricName, measurements, availTags);
}

protected internal void GetMetricsCollection(out MetricDictionary<List<MetricSample>> measurements, out MetricDictionary<List<MetricTag>> availTags)
{
measurements = new MetricDictionary<List<MetricSample>>();
availTags = new MetricDictionary<List<MetricTag>>();

var doubleMetrics = _exporter.GetAndClearDoubleMetrics();
if (doubleMetrics == null || doubleMetrics.Count <= 0)
{
doubleMetrics = DoubleMetrics;
}
else
{
DoubleMetrics = doubleMetrics;
}

if (doubleMetrics != null)
{
for (var i = 0; i < doubleMetrics.Count; i++)
{
var metric = doubleMetrics[i];
var labels = metric.Labels;

switch (metric.AggregationType)
{
case AggregationType.DoubleSum:
{
var doubleSum = metric.Data as SumData<double>;

var doubleValue = doubleSum.Sum;

measurements[metric.MetricName].Add(new MetricSample(MetricStatistic.COUNT, doubleValue, labels));

AddLabelsToTags(availTags, metric.MetricName, labels);

break;
}

case AggregationType.Summary:
{
var doubleSummary = metric.Data as SummaryData<double>;

var value = doubleSummary.Count > 0 ? doubleSummary.Sum / doubleSummary.Count : 0;
measurements[metric.MetricName].Add(new MetricSample(MetricStatistic.VALUE, value, labels));

// If labels contain time, Total time
if (labels.Any(l => l.Key.Equals("TimeUnit", StringComparison.OrdinalIgnoreCase)))
{
measurements[metric.MetricName].Add(new MetricSample(MetricStatistic.TOTAL_TIME, doubleSummary.Sum, labels));
}
else
{
measurements[metric.MetricName].Add(new MetricSample(MetricStatistic.TOTAL, doubleSummary.Sum, labels));
}

AddLabelsToTags(availTags, metric.MetricName, labels);

break;
}

default:
_logger.LogDebug($"Handle Agg Type {metric.AggregationType} in doubleMetrics");
break;
}
}
}

var longMetrics = _exporter.GetAndClearLongMetrics();
if (longMetrics == null || longMetrics.Count <= 0)
{
longMetrics = LongMetrics;
}
else
var maxSamples = filtered.Where(sample => sample.Statistic == MetricStatistic.MAX);
if (maxSamples.Any())
{
LongMetrics = longMetrics;
var sample = maxSamples.Aggregate(MaxAggregator);
sampleList.Add(new MetricSample(MetricStatistic.MAX, sample.Value, sample.Tags));
}

if (longMetrics != null)
{
foreach (var metric in longMetrics)
{
var labels = metric.Labels;
switch (metric.AggregationType)
{
case AggregationType.LongSum:
{
var longSum = metric.Data as SumData<long>;
var longValue = longSum.Sum;

measurements[metric.MetricName].Add(new MetricSample(MetricStatistic.COUNT, longValue, labels));
AddLabelsToTags(availTags, metric.MetricName, labels);

break;
}

case AggregationType.Summary:
{
var longSummary = metric.Data as SummaryData<long>;

var value = longSummary.Count > 0 ? longSummary.Sum / longSummary.Count : 0;
measurements[metric.MetricName].Add(new MetricSample(MetricStatistic.VALUE, value, labels));

// If labels contain time, Total time
if (labels.Any(l => l.Key.Equals("TimeUnit", StringComparison.OrdinalIgnoreCase)))
{
measurements[metric.MetricName].Add(new MetricSample(MetricStatistic.TOTAL_TIME, longSummary.Sum, labels));
}
else
{
measurements[metric.MetricName].Add(new MetricSample(MetricStatistic.TOTAL, longSummary.Sum, labels));
}

AddLabelsToTags(availTags, metric.MetricName, labels);

break;
}

default:
_logger.LogDebug($"Handle Agg Type {metric.AggregationType} in longMetrics");
break;
}
}
}
return sampleList;
}

private void AddLabelsToTags(MetricDictionary<List<MetricTag>> availTags, string name, IEnumerable<KeyValuePair<string, string>> labels)
protected internal MetricsResponse GetMetric(MetricsRequest request, List<MetricSample> metricSamples, List<MetricTag> availTags)
{
foreach (var label in labels)
{
var currentTags = availTags[name];
var existingTag = currentTags.FirstOrDefault(tag => tag.Tag.Equals(label.Key, StringComparison.OrdinalIgnoreCase));

if (existingTag != null)
{
existingTag.Values.Add(label.Value);
}
else
{
currentTags.Add(new MetricTag(label.Key, new HashSet<string>(new List<string> { label.Value })));
}
}
return new MetricsResponse(request.MetricName, metricSamples, availTags);
}
#pragma warning restore CS0618 // Type or member is obsolete

protected internal class MetricDictionary<T>
: Dictionary<string, T>
where T : new()
protected internal void GetMetricsCollection(out MetricsCollection<List<MetricSample>> metricSamples, out MetricsCollection<List<MetricTag>> availTags)
{
public MetricDictionary()
{
}
var collectionResponse = (SteeltoeCollectionResponse)_exporter.CollectionManager.EnterCollect().Result;
metricSamples = collectionResponse.MetricSamples;
availTags = collectionResponse.AvailableTags;

public new T this[string key]
{
get
{
if (!ContainsKey(key))
{
base[key] = new T();
}

return base[key];
}
}
// TODO: update the response header with actual updatetime
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -45,5 +45,7 @@ public MetricsEndpointOptions(IConfiguration config)
public string IngressIgnorePattern { get; set; }

public string EgressIgnorePattern { get; set; }

public long ScrapeResponseCacheDurationMilliseconds { get; set; }
}
}
1 change: 1 addition & 0 deletions src/Management/src/EndpointBase/Metrics/MetricsResponse.cs
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@
// The .NET Foundation licenses this file to you under the Apache 2.0 License.
// See the LICENSE file in the project root for more information.

using Steeltoe.Management.OpenTelemetry.Metrics;
using System.Collections.Generic;
using System.Text.Json.Serialization;

Expand Down
Loading