mirror of
https://github.com/ClusterCockpit/cc-backend
synced 2025-01-24 10:29:06 +01:00
346 lines
7.7 KiB
Go
346 lines
7.7 KiB
Go
// Copyright (C) NHR@FAU, University Erlangen-Nuremberg.
|
|
// All rights reserved.
|
|
// Use of this source code is governed by a MIT-style
|
|
// license that can be found in the LICENSE file.
|
|
package schema
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"math"
|
|
"sort"
|
|
"unsafe"
|
|
|
|
"github.com/ClusterCockpit/cc-backend/internal/util"
|
|
)
|
|
|
|
type JobData map[string]map[MetricScope]*JobMetric
|
|
|
|
type JobMetric struct {
|
|
Unit Unit `json:"unit"`
|
|
Timestep int `json:"timestep"`
|
|
Series []Series `json:"series"`
|
|
StatisticsSeries *StatsSeries `json:"statisticsSeries,omitempty"`
|
|
}
|
|
|
|
type Series struct {
|
|
Hostname string `json:"hostname"`
|
|
Id *string `json:"id,omitempty"`
|
|
Statistics MetricStatistics `json:"statistics"`
|
|
Data []Float `json:"data"`
|
|
}
|
|
|
|
type MetricStatistics struct {
|
|
Avg float64 `json:"avg"`
|
|
Min float64 `json:"min"`
|
|
Max float64 `json:"max"`
|
|
}
|
|
|
|
type StatsSeries struct {
|
|
Mean []Float `json:"mean"`
|
|
Median []Float `json:"median"`
|
|
Min []Float `json:"min"`
|
|
Max []Float `json:"max"`
|
|
Percentiles map[int][]Float `json:"percentiles,omitempty"`
|
|
}
|
|
|
|
type MetricScope string
|
|
|
|
const (
|
|
MetricScopeInvalid MetricScope = "invalid_scope"
|
|
|
|
MetricScopeNode MetricScope = "node"
|
|
MetricScopeSocket MetricScope = "socket"
|
|
MetricScopeMemoryDomain MetricScope = "memoryDomain"
|
|
MetricScopeCore MetricScope = "core"
|
|
MetricScopeHWThread MetricScope = "hwthread"
|
|
|
|
MetricScopeAccelerator MetricScope = "accelerator"
|
|
)
|
|
|
|
var metricScopeGranularity map[MetricScope]int = map[MetricScope]int{
|
|
MetricScopeNode: 10,
|
|
MetricScopeSocket: 5,
|
|
MetricScopeMemoryDomain: 4,
|
|
MetricScopeCore: 3,
|
|
MetricScopeHWThread: 2,
|
|
/* Special-Case Accelerator
|
|
* -> No conversion possible if native scope is HWTHREAD
|
|
* -> Therefore needs to be less than HWTREAD, else max() would return unhandled case
|
|
* -> If nativeScope is accelerator, accelerator metrics return correctly
|
|
*/
|
|
MetricScopeAccelerator: 1,
|
|
|
|
MetricScopeInvalid: -1,
|
|
}
|
|
|
|
func (e *MetricScope) LT(other MetricScope) bool {
|
|
a := metricScopeGranularity[*e]
|
|
b := metricScopeGranularity[other]
|
|
return a < b
|
|
}
|
|
|
|
func (e *MetricScope) LTE(other MetricScope) bool {
|
|
a := metricScopeGranularity[*e]
|
|
b := metricScopeGranularity[other]
|
|
return a <= b
|
|
}
|
|
|
|
func (e *MetricScope) Max(other MetricScope) MetricScope {
|
|
a := metricScopeGranularity[*e]
|
|
b := metricScopeGranularity[other]
|
|
if a > b {
|
|
return *e
|
|
}
|
|
return other
|
|
}
|
|
|
|
func (e *MetricScope) UnmarshalGQL(v interface{}) error {
|
|
str, ok := v.(string)
|
|
if !ok {
|
|
return fmt.Errorf("SCHEMA/METRICS > enums must be strings")
|
|
}
|
|
|
|
*e = MetricScope(str)
|
|
if !e.Valid() {
|
|
return fmt.Errorf("SCHEMA/METRICS > %s is not a valid MetricScope", str)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (e MetricScope) MarshalGQL(w io.Writer) {
|
|
fmt.Fprintf(w, "\"%s\"", e)
|
|
}
|
|
|
|
func (e MetricScope) Valid() bool {
|
|
gran, ok := metricScopeGranularity[e]
|
|
return ok && gran > 0
|
|
}
|
|
|
|
func (jd *JobData) Size() int {
|
|
n := 128
|
|
for _, scopes := range *jd {
|
|
for _, metric := range scopes {
|
|
if metric.StatisticsSeries != nil {
|
|
n += len(metric.StatisticsSeries.Max)
|
|
n += len(metric.StatisticsSeries.Median)
|
|
n += len(metric.StatisticsSeries.Min)
|
|
}
|
|
|
|
for _, series := range metric.Series {
|
|
n += len(series.Data)
|
|
}
|
|
}
|
|
}
|
|
return n * int(unsafe.Sizeof(Float(0)))
|
|
}
|
|
|
|
const smooth bool = false
|
|
|
|
func (jm *JobMetric) AddStatisticsSeries() {
|
|
if jm.StatisticsSeries != nil || len(jm.Series) < 4 {
|
|
return
|
|
}
|
|
|
|
n, m := 0, len(jm.Series[0].Data)
|
|
for _, series := range jm.Series {
|
|
if len(series.Data) > n {
|
|
n = len(series.Data)
|
|
}
|
|
if len(series.Data) < m {
|
|
m = len(series.Data)
|
|
}
|
|
}
|
|
|
|
// mean := make([]Float, n)
|
|
min, median, max := make([]Float, n), make([]Float, n), make([]Float, n)
|
|
i := 0
|
|
for ; i < m; i++ {
|
|
seriesCount := len(jm.Series)
|
|
// ssum := 0.0
|
|
smin, smed, smax := math.MaxFloat32, make([]float64, seriesCount), -math.MaxFloat32
|
|
notnan := 0
|
|
for j := 0; j < seriesCount; j++ {
|
|
x := float64(jm.Series[j].Data[i])
|
|
if math.IsNaN(x) {
|
|
continue
|
|
}
|
|
|
|
notnan += 1
|
|
// ssum += x
|
|
smed[j] = x
|
|
smin = math.Min(smin, x)
|
|
smax = math.Max(smax, x)
|
|
}
|
|
|
|
if notnan < 3 {
|
|
min[i] = NaN
|
|
// mean[i] = NaN
|
|
median[i] = NaN
|
|
max[i] = NaN
|
|
} else {
|
|
min[i] = Float(smin)
|
|
// mean[i] = Float(ssum / float64(notnan))
|
|
max[i] = Float(smax)
|
|
|
|
medianRaw, err := util.Median(smed)
|
|
if err != nil {
|
|
median[i] = NaN
|
|
} else {
|
|
median[i] = Float(medianRaw)
|
|
}
|
|
}
|
|
}
|
|
|
|
for ; i < n; i++ {
|
|
min[i] = NaN
|
|
// mean[i] = NaN
|
|
median[i] = NaN
|
|
max[i] = NaN
|
|
}
|
|
|
|
if smooth {
|
|
for i := 2; i < len(median)-2; i++ {
|
|
if min[i].IsNaN() {
|
|
continue
|
|
}
|
|
|
|
min[i] = (min[i-2] + min[i-1] + min[i] + min[i+1] + min[i+2]) / 5
|
|
max[i] = (max[i-2] + max[i-1] + max[i] + max[i+1] + max[i+2]) / 5
|
|
// mean[i] = (mean[i-2] + mean[i-1] + mean[i] + mean[i+1] + mean[i+2]) / 5
|
|
// Reduce Median further
|
|
smoothRaw := []float64{float64(median[i-2]), float64(median[i-1]), float64(median[i]), float64(median[i+1]), float64(median[i+2])}
|
|
smoothMedian, err := util.Median(smoothRaw)
|
|
if err != nil {
|
|
median[i] = NaN
|
|
} else {
|
|
median[i] = Float(smoothMedian)
|
|
}
|
|
}
|
|
}
|
|
|
|
jm.StatisticsSeries = &StatsSeries{Median: median, Min: min, Max: max} // Mean: mean
|
|
}
|
|
|
|
func (jd *JobData) AddNodeScope(metric string) bool {
|
|
scopes, ok := (*jd)[metric]
|
|
if !ok {
|
|
return false
|
|
}
|
|
|
|
var maxScope MetricScope = MetricScopeInvalid
|
|
for scope := range scopes {
|
|
maxScope = maxScope.Max(scope)
|
|
}
|
|
|
|
if maxScope == MetricScopeInvalid || maxScope == MetricScopeNode {
|
|
return false
|
|
}
|
|
|
|
jm := scopes[maxScope]
|
|
hosts := make(map[string][]Series, 32)
|
|
for _, series := range jm.Series {
|
|
hosts[series.Hostname] = append(hosts[series.Hostname], series)
|
|
}
|
|
|
|
nodeJm := &JobMetric{
|
|
Unit: jm.Unit,
|
|
Timestep: jm.Timestep,
|
|
Series: make([]Series, 0, len(hosts)),
|
|
}
|
|
for hostname, series := range hosts {
|
|
min, sum, max := math.MaxFloat32, 0.0, -math.MaxFloat32
|
|
for _, series := range series {
|
|
sum += series.Statistics.Avg
|
|
min = math.Min(min, series.Statistics.Min)
|
|
max = math.Max(max, series.Statistics.Max)
|
|
}
|
|
|
|
n, m := 0, len(jm.Series[0].Data)
|
|
for _, series := range jm.Series {
|
|
if len(series.Data) > n {
|
|
n = len(series.Data)
|
|
}
|
|
if len(series.Data) < m {
|
|
m = len(series.Data)
|
|
}
|
|
}
|
|
|
|
i, data := 0, make([]Float, len(series[0].Data))
|
|
for ; i < m; i++ {
|
|
x := Float(0.0)
|
|
for _, series := range jm.Series {
|
|
x += series.Data[i]
|
|
}
|
|
data[i] = x
|
|
}
|
|
|
|
for ; i < n; i++ {
|
|
data[i] = NaN
|
|
}
|
|
|
|
nodeJm.Series = append(nodeJm.Series, Series{
|
|
Hostname: hostname,
|
|
Statistics: MetricStatistics{Min: min, Avg: sum / float64(len(series)), Max: max},
|
|
Data: data,
|
|
})
|
|
}
|
|
|
|
scopes[MetricScopeNode] = nodeJm
|
|
return true
|
|
}
|
|
|
|
func (jm *JobMetric) AddPercentiles(ps []int) bool {
|
|
if jm.StatisticsSeries == nil {
|
|
jm.AddStatisticsSeries()
|
|
}
|
|
|
|
if len(jm.Series) < 3 {
|
|
return false
|
|
}
|
|
|
|
if jm.StatisticsSeries.Percentiles == nil {
|
|
jm.StatisticsSeries.Percentiles = make(map[int][]Float, len(ps))
|
|
}
|
|
|
|
n := 0
|
|
for _, series := range jm.Series {
|
|
if len(series.Data) > n {
|
|
n = len(series.Data)
|
|
}
|
|
}
|
|
|
|
data := make([][]float64, n)
|
|
for i := 0; i < n; i++ {
|
|
vals := make([]float64, 0, len(jm.Series))
|
|
for _, series := range jm.Series {
|
|
if i < len(series.Data) {
|
|
vals = append(vals, float64(series.Data[i]))
|
|
}
|
|
}
|
|
|
|
sort.Float64s(vals)
|
|
data[i] = vals
|
|
}
|
|
|
|
for _, p := range ps {
|
|
if p < 1 || p > 99 {
|
|
panic("SCHEMA/METRICS > invalid percentile")
|
|
}
|
|
|
|
if _, ok := jm.StatisticsSeries.Percentiles[p]; ok {
|
|
continue
|
|
}
|
|
|
|
percentiles := make([]Float, n)
|
|
for i := 0; i < n; i++ {
|
|
sorted := data[i]
|
|
percentiles[i] = Float(sorted[(len(sorted)*p)/100])
|
|
}
|
|
|
|
jm.StatisticsSeries.Percentiles[p] = percentiles
|
|
}
|
|
|
|
return true
|
|
}
|