2023-09-13 11:25:08 +02:00
|
|
|
|
using Core;
|
2024-06-06 09:54:50 +02:00
|
|
|
|
using IdentityModel;
|
2023-11-12 10:07:23 +01:00
|
|
|
|
using KubernetesWorkflow.Types;
|
2023-09-26 14:32:28 +02:00
|
|
|
|
using Logging;
|
2023-09-13 11:25:08 +02:00
|
|
|
|
using System.Globalization;
|
|
|
|
|
|
|
|
|
|
namespace MetricsPlugin
|
|
|
|
|
{
|
|
|
|
|
public class MetricsQuery
|
|
|
|
|
{
|
2024-03-26 09:10:06 +01:00
|
|
|
|
private readonly IEndpoint endpoint;
|
2023-09-26 14:32:28 +02:00
|
|
|
|
private readonly ILog log;
|
2023-09-13 11:25:08 +02:00
|
|
|
|
|
2023-09-14 15:30:09 +02:00
|
|
|
|
public MetricsQuery(IPluginTools tools, RunningContainer runningContainer)
|
2023-09-13 11:25:08 +02:00
|
|
|
|
{
|
2023-09-14 15:30:09 +02:00
|
|
|
|
RunningContainer = runningContainer;
|
2023-09-26 14:32:28 +02:00
|
|
|
|
log = tools.GetLog();
|
2024-09-23 10:52:12 +02:00
|
|
|
|
var address = RunningContainer.GetAddress(PrometheusContainerRecipe.PortTag);
|
2024-03-26 09:10:06 +01:00
|
|
|
|
endpoint = tools
|
2024-07-25 10:10:11 +02:00
|
|
|
|
.CreateHttp(address.ToString())
|
|
|
|
|
.CreateEndpoint(address, "/api/v1/");
|
2023-09-13 11:25:08 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-14 15:30:09 +02:00
|
|
|
|
public RunningContainer RunningContainer { get; }
|
2023-09-13 11:25:08 +02:00
|
|
|
|
|
|
|
|
|
public Metrics? GetMostRecent(string metricName, IMetricsScrapeTarget target)
|
|
|
|
|
{
|
|
|
|
|
var response = GetLastOverTime(metricName, GetInstanceStringForNode(target));
|
|
|
|
|
if (response == null) return null;
|
|
|
|
|
|
2023-09-26 14:32:28 +02:00
|
|
|
|
var result = new Metrics
|
2023-09-13 11:25:08 +02:00
|
|
|
|
{
|
|
|
|
|
Sets = response.data.result.Select(r =>
|
|
|
|
|
{
|
|
|
|
|
return new MetricsSet
|
|
|
|
|
{
|
|
|
|
|
Instance = r.metric.instance,
|
|
|
|
|
Values = MapSingleValue(r.value)
|
|
|
|
|
};
|
|
|
|
|
}).ToArray()
|
|
|
|
|
};
|
2023-09-26 14:32:28 +02:00
|
|
|
|
|
|
|
|
|
Log(target, metricName, result);
|
|
|
|
|
return result;
|
2023-09-13 11:25:08 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Metrics? GetMetrics(string metricName)
|
|
|
|
|
{
|
|
|
|
|
var response = GetAll(metricName);
|
|
|
|
|
if (response == null) return null;
|
2023-09-26 14:32:28 +02:00
|
|
|
|
var result = MapResponseToMetrics(response);
|
|
|
|
|
Log(metricName, result);
|
|
|
|
|
return result;
|
2023-09-13 11:25:08 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Metrics? GetAllMetricsForNode(IMetricsScrapeTarget target)
|
|
|
|
|
{
|
2024-03-26 09:10:06 +01:00
|
|
|
|
var response = endpoint.HttpGetJson<PrometheusQueryResponse>($"query?query={GetInstanceStringForNode(target)}{GetQueryTimeRange()}");
|
2023-09-13 11:25:08 +02:00
|
|
|
|
if (response.status != "success") return null;
|
2023-09-26 14:32:28 +02:00
|
|
|
|
var result = MapResponseToMetrics(response);
|
|
|
|
|
Log(target, result);
|
|
|
|
|
return result;
|
2023-09-13 11:25:08 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private PrometheusQueryResponse? GetLastOverTime(string metricName, string instanceString)
|
|
|
|
|
{
|
2024-03-26 09:10:06 +01:00
|
|
|
|
var response = endpoint.HttpGetJson<PrometheusQueryResponse>($"query?query=last_over_time({metricName}{instanceString}{GetQueryTimeRange()})");
|
2023-09-13 11:25:08 +02:00
|
|
|
|
if (response.status != "success") return null;
|
|
|
|
|
return response;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private PrometheusQueryResponse? GetAll(string metricName)
|
|
|
|
|
{
|
2024-03-26 09:10:06 +01:00
|
|
|
|
var response = endpoint.HttpGetJson<PrometheusQueryResponse>($"query?query={metricName}{GetQueryTimeRange()}");
|
2023-09-13 11:25:08 +02:00
|
|
|
|
if (response.status != "success") return null;
|
|
|
|
|
return response;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private Metrics MapResponseToMetrics(PrometheusQueryResponse response)
|
|
|
|
|
{
|
|
|
|
|
return new Metrics
|
|
|
|
|
{
|
|
|
|
|
Sets = response.data.result.Select(r =>
|
|
|
|
|
{
|
|
|
|
|
return new MetricsSet
|
|
|
|
|
{
|
|
|
|
|
Name = r.metric.__name__,
|
|
|
|
|
Instance = r.metric.instance,
|
|
|
|
|
Values = MapMultipleValues(r.values)
|
|
|
|
|
};
|
|
|
|
|
}).ToArray()
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private MetricsSetValue[] MapSingleValue(object[] value)
|
|
|
|
|
{
|
|
|
|
|
if (value != null && value.Length > 0)
|
|
|
|
|
{
|
|
|
|
|
return new[]
|
|
|
|
|
{
|
|
|
|
|
MapValue(value)
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
return Array.Empty<MetricsSetValue>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private MetricsSetValue[] MapMultipleValues(object[][] values)
|
|
|
|
|
{
|
|
|
|
|
if (values != null && values.Length > 0)
|
|
|
|
|
{
|
|
|
|
|
return values.Select(v => MapValue(v)).ToArray();
|
|
|
|
|
}
|
|
|
|
|
return Array.Empty<MetricsSetValue>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private MetricsSetValue MapValue(object[] value)
|
|
|
|
|
{
|
|
|
|
|
if (value.Length != 2) throw new InvalidOperationException("Expected value to be [double, string].");
|
|
|
|
|
|
|
|
|
|
return new MetricsSetValue
|
|
|
|
|
{
|
|
|
|
|
Timestamp = ToTimestamp(value[0]),
|
|
|
|
|
Value = ToValue(value[1])
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private string GetInstanceNameForNode(IMetricsScrapeTarget target)
|
|
|
|
|
{
|
2023-11-07 12:02:17 +01:00
|
|
|
|
return ScrapeTargetHelper.FormatTarget(log, target);
|
2023-09-13 11:25:08 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private string GetInstanceStringForNode(IMetricsScrapeTarget target)
|
|
|
|
|
{
|
|
|
|
|
return "{instance=\"" + GetInstanceNameForNode(target) + "\"}";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private string GetQueryTimeRange()
|
|
|
|
|
{
|
|
|
|
|
return "[12h]";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private double ToValue(object v)
|
|
|
|
|
{
|
|
|
|
|
return Convert.ToDouble(v, CultureInfo.InvariantCulture);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private DateTime ToTimestamp(object v)
|
|
|
|
|
{
|
|
|
|
|
var unixSeconds = ToValue(v);
|
|
|
|
|
return new DateTime(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc).AddSeconds(unixSeconds);
|
|
|
|
|
}
|
2023-09-26 14:32:28 +02:00
|
|
|
|
|
|
|
|
|
private void Log(IMetricsScrapeTarget target, string metricName, Metrics result)
|
|
|
|
|
{
|
2023-11-06 15:27:23 +01:00
|
|
|
|
Log($"{target.Container.Name} '{metricName}' = {result}");
|
2023-09-26 14:32:28 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void Log(string metricName, Metrics result)
|
|
|
|
|
{
|
|
|
|
|
Log($"'{metricName}' = {result}");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void Log(IMetricsScrapeTarget target, Metrics result)
|
|
|
|
|
{
|
2023-11-06 15:27:23 +01:00
|
|
|
|
Log($"{target.Container.Name} => {result}");
|
2023-09-26 14:32:28 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void Log(string msg)
|
|
|
|
|
{
|
|
|
|
|
log.Log(msg);
|
|
|
|
|
}
|
2023-09-13 11:25:08 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class Metrics
|
|
|
|
|
{
|
|
|
|
|
public MetricsSet[] Sets { get; set; } = Array.Empty<MetricsSet>();
|
2023-09-26 14:32:28 +02:00
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
|
|
|
|
return "[" + string.Join(',', Sets.Select(s => s.ToString())) + "]";
|
|
|
|
|
}
|
2024-06-06 09:54:50 +02:00
|
|
|
|
|
|
|
|
|
public string AsCsv()
|
|
|
|
|
{
|
|
|
|
|
var allTimestamps = Sets.SelectMany(s => s.Values.Select(v => v.Timestamp)).Distinct().OrderDescending().ToArray();
|
|
|
|
|
|
|
|
|
|
var lines = new List<string>();
|
|
|
|
|
MakeLine(lines, e =>
|
|
|
|
|
{
|
|
|
|
|
e.Add("Metrics");
|
|
|
|
|
foreach (var ts in allTimestamps) e.Add(ts.ToEpochTime().ToString());
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
foreach (var set in Sets)
|
|
|
|
|
{
|
|
|
|
|
MakeLine(lines, e =>
|
|
|
|
|
{
|
|
|
|
|
e.Add(set.Name);
|
|
|
|
|
foreach (var ts in allTimestamps)
|
|
|
|
|
{
|
|
|
|
|
var value = set.Values.SingleOrDefault(v => v.Timestamp == ts);
|
|
|
|
|
if (value == null) e.Add(" ");
|
|
|
|
|
else e.Add(value.Value.ToString());
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return string.Join(Environment.NewLine, lines.ToArray());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void MakeLine(List<string> lines, Action<List<string>> values)
|
|
|
|
|
{
|
|
|
|
|
var list = new List<string>();
|
|
|
|
|
values(list);
|
|
|
|
|
lines.Add(string.Join(",", list));
|
|
|
|
|
}
|
2023-09-13 11:25:08 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class MetricsSet
|
|
|
|
|
{
|
|
|
|
|
public string Name { get; set; } = string.Empty;
|
|
|
|
|
public string Instance { get; set; } = string.Empty;
|
|
|
|
|
public MetricsSetValue[] Values { get; set; } = Array.Empty<MetricsSetValue>();
|
2023-09-26 14:32:28 +02:00
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
|
|
|
|
return $"{Name} ({Instance}) : {{{string.Join(",", Values.Select(v => v.ToString()))}}}";
|
|
|
|
|
}
|
2023-09-13 11:25:08 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class MetricsSetValue
|
|
|
|
|
{
|
|
|
|
|
public DateTime Timestamp { get; set; }
|
|
|
|
|
public double Value { get; set; }
|
2023-09-26 14:32:28 +02:00
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
|
|
|
|
return $"<{Timestamp.ToString("o")}={Value}>";
|
|
|
|
|
}
|
2023-09-13 11:25:08 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class PrometheusQueryResponse
|
|
|
|
|
{
|
|
|
|
|
public string status { get; set; } = string.Empty;
|
|
|
|
|
public PrometheusQueryResponseData data { get; set; } = new();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class PrometheusQueryResponseData
|
|
|
|
|
{
|
|
|
|
|
public string resultType { get; set; } = string.Empty;
|
|
|
|
|
public PrometheusQueryResponseDataResultEntry[] result { get; set; } = Array.Empty<PrometheusQueryResponseDataResultEntry>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class PrometheusQueryResponseDataResultEntry
|
|
|
|
|
{
|
|
|
|
|
public ResultEntryMetric metric { get; set; } = new();
|
|
|
|
|
public object[] value { get; set; } = Array.Empty<object>();
|
|
|
|
|
public object[][] values { get; set; } = Array.Empty<object[]>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class ResultEntryMetric
|
|
|
|
|
{
|
|
|
|
|
public string __name__ { get; set; } = string.Empty;
|
|
|
|
|
public string instance { get; set; } = string.Empty;
|
|
|
|
|
public string job { get; set; } = string.Empty;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class PrometheusAllNamesResponse
|
|
|
|
|
{
|
|
|
|
|
public string status { get; set; } = string.Empty;
|
|
|
|
|
public string[] data { get; set; } = Array.Empty<string>();
|
|
|
|
|
}
|
|
|
|
|
}
|