forked from 47-studio-org/PostSharp.Samples
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathSamplePublisher.cs
105 lines (84 loc) · 2.88 KB
/
SamplePublisher.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
using Microsoft.ApplicationInsights;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading;
namespace PostSharp.Samples.Profiling
{
internal class MetricPublisher : IDisposable
{
readonly SampleCollector sampleCollector;
private readonly TelemetryClient telemetryClient;
private MetricData[] lastSamples = new MetricData[0];
Timer timer;
bool inProgress;
public MetricPublisher(SampleCollector collector, TelemetryClient telemetryClient)
{
this.telemetryClient = telemetryClient;
this.sampleCollector = collector;
}
public void Start(TimeSpan period)
{
this.timer?.Dispose();
this.timer = new Timer(this.OnTime, null, (int) period.TotalMilliseconds, (int) period.TotalMilliseconds);
}
private void OnTime(object state)
{
if (this.inProgress)
{
return;
}
this.inProgress = true;
try
{
this.PublishMetrics();
}
finally
{
this.inProgress = false;
}
}
public void PublishMetrics() => this.PublishMetrics(this.sampleCollector.MetricsMetadata, this.sampleCollector.GetMetrics());
private void PublishMetrics(IReadOnlyList<MetricMetadata> methods, MetricData[] metrics)
{
for (var i = 0; i < metrics.Length; i++)
{
MetricData lastSample;
if (i < this.lastSamples.Length)
{
lastSample = this.lastSamples[i];
}
else
{
lastSample = new MetricData { Timestamp = ProfilingServices.StartTimestamp };
}
metrics[i].GetDifference(lastSample, out var differenceSample);
this.PublishMetric(methods[i], differenceSample);
}
this.lastSamples = metrics;
}
private void PublishMetric(MetricMetadata method, in MetricData metric)
{
if (metric.ExecutionCount > 0)
{
var metrics = new Dictionary<string, double>
{
["ExceptionCount"] = metric.ExceptionCount,
["ExecutionCount"] = metric.ExecutionCount,
["CpuTime"] = metric.CpuTimeSpan.TotalMilliseconds,
["ThreadTime"] = metric.ThreadTimeSpan.TotalMilliseconds,
["ExclusiveCpuTime"] = metric.ExclusiveCpuTimeSpan.TotalMilliseconds,
["ExclusiveThreadTime"] = metric.ExclusiveThreadTimeSpan.TotalMilliseconds,
["AsyncTime"] = metric.AsyncTimeSpan.TotalMilliseconds,
["SampleTime"] = metric.SampleTimeSpan.TotalMilliseconds
};
Console.WriteLine(string.Format("TrackEvent: Name='{0,-70}', {1}", method.Name, string.Join(", ", metrics.Select(p => $"{p.Key}={p.Value,10}"))));
this.telemetryClient.TrackEvent(method.Name, metrics: metrics);
}
}
public void Dispose()
{
this.timer?.Dispose();
}
}
}