using System.Globalization; namespace CodexDistTestCore.Metrics { public class MetricsDownloader { private readonly TestLog log; private readonly Dictionary activePrometheuses; public MetricsDownloader(TestLog log, Dictionary activePrometheuses) { this.log = log; this.activePrometheuses = activePrometheuses; } public void DownloadAllMetrics() { foreach (var pair in activePrometheuses) { DownloadAllMetrics(pair.Key, pair.Value); } } private void DownloadAllMetrics(MetricsQuery query, OnlineCodexNode[] nodes) { foreach (var node in nodes) { DownloadAllMetricsForNode(query, node); } } private void DownloadAllMetricsForNode(MetricsQuery query, OnlineCodexNode node) { var metrics = query.GetAllMetricsForNode(node); if (metrics == null || metrics.Sets.Length == 0 || metrics.Sets.All(s => s.Values.Length == 0)) return; var headers = new[] { "timestamp" }.Concat(metrics.Sets.Select(s => s.Name)).ToArray(); var map = CreateValueMap(metrics); WriteToFile(node.GetName(), headers, map); } private void WriteToFile(string nodeName, string[] headers, Dictionary> map) { var file = log.CreateSubfile("csv"); log.Log($"Downloading metrics for {nodeName} to file {file.FilenameWithoutPath}"); file.WriteRaw(string.Join(",", headers)); foreach (var pair in map) { file.WriteRaw(string.Join(",", new[] { FormatTimestamp(pair.Key) }.Concat(pair.Value))); } } private Dictionary> CreateValueMap(Metrics metrics) { var map = CreateForAllTimestamps(metrics); foreach (var metric in metrics.Sets) { AddToMap(map, metric); } return map; } private Dictionary> CreateForAllTimestamps(Metrics metrics) { var result = new Dictionary>(); var timestamps = metrics.Sets.SelectMany(s => s.Values).Select(v => v.Timestamp).Distinct().ToArray(); foreach (var timestamp in timestamps) result.Add(timestamp, new List()); return result; } private void AddToMap(Dictionary> map, MetricsSet metric) { foreach (var key in map.Keys) { map[key].Add(GetValueAtTimestamp(key, metric)); } } private string GetValueAtTimestamp(DateTime key, MetricsSet metric) { var value = metric.Values.SingleOrDefault(v => v.Timestamp == key); if (value == null) return ""; return value.Value.ToString(CultureInfo.InvariantCulture); } private string FormatTimestamp(DateTime key) { var origin = new DateTime(1970, 1, 1, 0, 0, 0, 0, DateTimeKind.Utc); var diff = key - origin; return Math.Floor(diff.TotalSeconds).ToString(CultureInfo.InvariantCulture); } } }