mirror of
https://github.com/ClusterCockpit/cc-backend
synced 2025-04-03 18:55:55 +02:00
565 lines
18 KiB
Svelte
565 lines
18 KiB
Svelte
<!--
|
|
@component Main plot component, based on uPlot; metricdata values by time
|
|
|
|
Only width/height should change reactively.
|
|
|
|
Properties:
|
|
- `metric String`: The metric name
|
|
- `scope String?`: Scope of the displayed data [Default: node]
|
|
- `height Number?`: The plot height [Default: 300]
|
|
- `timestep Number`: The timestep used for X-axis rendering
|
|
- `series [GraphQL.Series]`: The metric data object
|
|
- `useStatsSeries Bool?`: If this plot uses the statistics Min/Max/Median representation; automatically set to according bool [Default: null]
|
|
- `statisticsSeries [GraphQL.StatisticsSeries]?`: Min/Max/Median representation of metric data [Default: null]
|
|
- `cluster String`: Cluster name of the parent job / data
|
|
- `subCluster String`: Name of the subCluster of the parent job
|
|
- `isShared Bool?`: If this job used shared resources; will adapt threshold indicators accordingly [Default: false]
|
|
- `forNode Bool?`: If this plot is used for node data display; will render x-axis as negative time with $now as maximum [Default: false]
|
|
- `numhwthreads Number?`: Number of job HWThreads [Default: 0]
|
|
- `numaccs Number?`: Number of job Accelerators [Default: 0]
|
|
- `zoomState Object?`: The last zoom state to preserve on user zoom [Default: null]
|
|
-->
|
|
|
|
<script context="module">
|
|
function formatTime(t, forNode = false) {
|
|
if (t !== null) {
|
|
if (isNaN(t)) {
|
|
return t;
|
|
} else {
|
|
const tAbs = Math.abs(t);
|
|
const h = Math.floor(tAbs / 3600);
|
|
const m = Math.floor((tAbs % 3600) / 60);
|
|
// Re-Add "negativity" to time ticks only as string, so that if-cases work as intended
|
|
if (h == 0) return `${forNode && m != 0 ? "-" : ""}${m}m`;
|
|
else if (m == 0) return `${forNode ? "-" : ""}${h}h`;
|
|
else return `${forNode ? "-" : ""}${h}:${m}h`;
|
|
}
|
|
}
|
|
}
|
|
|
|
function timeIncrs(timestep, maxX, forNode) {
|
|
if (forNode === true) {
|
|
return [60, 120, 240, 300, 360, 480, 600, 900, 1800, 3600, 7200, 14400, 21600]; // forNode fixed increments
|
|
} else {
|
|
let incrs = [];
|
|
for (let t = timestep; t < maxX; t *= 10)
|
|
incrs.push(t, t * 2, t * 3, t * 5);
|
|
|
|
return incrs;
|
|
}
|
|
}
|
|
|
|
// removed arg "subcluster": input metricconfig and topology now directly derived from subcluster
|
|
function findJobAggregationThresholds(
|
|
subClusterTopology,
|
|
metricConfig,
|
|
scope,
|
|
isShared,
|
|
numhwthreads,
|
|
numaccs
|
|
) {
|
|
|
|
if (!subClusterTopology || !metricConfig || !scope) {
|
|
console.warn("Argument missing for findJobAggregationThresholds!");
|
|
return null;
|
|
}
|
|
|
|
// handle special *-stat scopes
|
|
if (scope.match(/(.*)-stat$/)) {
|
|
const statParts = scope.split('-');
|
|
scope = statParts[0]
|
|
}
|
|
|
|
if (
|
|
(scope == "node" && isShared == false) ||
|
|
metricConfig?.aggregation == "avg"
|
|
) {
|
|
return {
|
|
normal: metricConfig.normal,
|
|
caution: metricConfig.caution,
|
|
alert: metricConfig.alert,
|
|
peak: metricConfig.peak,
|
|
};
|
|
}
|
|
|
|
|
|
if (metricConfig?.aggregation == "sum") {
|
|
let divisor;
|
|
if (isShared == true) { // Shared
|
|
if (numaccs > 0) divisor = subClusterTopology.accelerators.length / numaccs;
|
|
else if (numhwthreads > 0) divisor = subClusterTopology.core.length / numhwthreads;
|
|
}
|
|
else if (scope == 'node') divisor = 1; // Use as configured for nodes
|
|
else if (scope == 'socket') divisor = subClusterTopology.socket.length;
|
|
else if (scope == "memoryDomain") divisor = subClusterTopology.memoryDomain.length;
|
|
else if (scope == "core") divisor = subClusterTopology.core.length;
|
|
else if (scope == "hwthread") divisor = subClusterTopology.core.length; // alt. name for core
|
|
else if (scope == "accelerator") divisor = subClusterTopology.accelerators.length;
|
|
else {
|
|
console.log('Unknown scope, return default aggregation thresholds ', scope)
|
|
divisor = 1;
|
|
}
|
|
|
|
return {
|
|
peak: metricConfig.peak / divisor,
|
|
normal: metricConfig.normal / divisor,
|
|
caution: metricConfig.caution / divisor,
|
|
alert: metricConfig.alert / divisor,
|
|
};
|
|
}
|
|
|
|
console.warn(
|
|
"Missing or unkown aggregation mode (sum/avg) for metric:",
|
|
metricConfig,
|
|
);
|
|
return null;
|
|
}
|
|
</script>
|
|
|
|
<script>
|
|
import uPlot from "uplot";
|
|
import { formatNumber } from "../units.js";
|
|
import { getContext, onMount, onDestroy, createEventDispatcher } from "svelte";
|
|
import { Card } from "@sveltestrap/sveltestrap";
|
|
|
|
export let metric;
|
|
export let scope = "node";
|
|
export let width = 0;
|
|
export let height = 300;
|
|
export let timestep;
|
|
export let series;
|
|
export let useStatsSeries = null;
|
|
export let statisticsSeries = null;
|
|
export let cluster = "";
|
|
export let subCluster;
|
|
export let isShared = false;
|
|
export let forNode = false;
|
|
export let numhwthreads = 0;
|
|
export let numaccs = 0;
|
|
export let zoomState = null;
|
|
export let thresholdState = null;
|
|
|
|
if (useStatsSeries == null) useStatsSeries = statisticsSeries != null;
|
|
if (useStatsSeries == false && series == null) useStatsSeries = true;
|
|
|
|
const usesMeanStatsSeries = (useStatsSeries && statisticsSeries.mean.length != 0)
|
|
const dispatch = createEventDispatcher();
|
|
const subClusterTopology = getContext("getHardwareTopology")(cluster, subCluster);
|
|
const metricConfig = getContext("getMetricConfig")(cluster, subCluster, metric);
|
|
const clusterCockpitConfig = getContext("cc-config");
|
|
const renderSleepTime = 200;
|
|
const normalLineColor = "#000000";
|
|
const lineWidth =
|
|
clusterCockpitConfig.plot_general_lineWidth / window.devicePixelRatio;
|
|
const lineColors = clusterCockpitConfig.plot_general_colorscheme;
|
|
const backgroundColors = {
|
|
normal: "rgba(255, 255, 255, 1.0)",
|
|
caution: "rgba(255, 128, 0, 0.3)",
|
|
alert: "rgba(255, 0, 0, 0.3)",
|
|
};
|
|
const thresholds = findJobAggregationThresholds(
|
|
subClusterTopology,
|
|
metricConfig,
|
|
scope,
|
|
isShared,
|
|
numhwthreads,
|
|
numaccs
|
|
);
|
|
|
|
const resampleConfig = getContext("resampling");
|
|
let resampleTrigger;
|
|
let resampleResolutions;
|
|
let resampleMinimum;
|
|
|
|
if (resampleConfig) {
|
|
resampleTrigger = Number(resampleConfig.trigger)
|
|
resampleResolutions = [...resampleConfig.resolutions];
|
|
resampleMinimum = Math.min(...resampleConfig.resolutions);
|
|
}
|
|
|
|
// UPLOT PLUGIN // converts the legend into a simple tooltip
|
|
function legendAsTooltipPlugin({
|
|
className,
|
|
style = { backgroundColor: "rgba(255, 249, 196, 0.92)", color: "black" },
|
|
} = {}) {
|
|
let legendEl;
|
|
const dataSize = series.length;
|
|
|
|
function init(u, opts) {
|
|
legendEl = u.root.querySelector(".u-legend");
|
|
|
|
legendEl.classList.remove("u-inline");
|
|
className && legendEl.classList.add(className);
|
|
|
|
uPlot.assign(legendEl.style, {
|
|
textAlign: "left",
|
|
pointerEvents: "none",
|
|
display: "none",
|
|
position: "absolute",
|
|
left: 0,
|
|
top: 0,
|
|
zIndex: 100,
|
|
boxShadow: "2px 2px 10px rgba(0,0,0,0.5)",
|
|
...style,
|
|
});
|
|
|
|
// conditional hide series color markers:
|
|
if (
|
|
useStatsSeries === true || // Min/Max/Median Self-Explanatory
|
|
dataSize === 1 || // Only one Y-Dataseries
|
|
dataSize > 6
|
|
) {
|
|
// More than 6 Y-Dataseries
|
|
const idents = legendEl.querySelectorAll(".u-marker");
|
|
for (let i = 0; i < idents.length; i++)
|
|
idents[i].style.display = "none";
|
|
}
|
|
|
|
const overEl = u.over;
|
|
overEl.style.overflow = "visible";
|
|
|
|
// move legend into plot bounds
|
|
overEl.appendChild(legendEl);
|
|
|
|
// show/hide tooltip on enter/exit
|
|
overEl.addEventListener("mouseenter", () => {
|
|
legendEl.style.display = null;
|
|
});
|
|
overEl.addEventListener("mouseleave", () => {
|
|
legendEl.style.display = "none";
|
|
});
|
|
|
|
// let tooltip exit plot
|
|
// overEl.style.overflow = "visible";
|
|
}
|
|
|
|
function update(u) {
|
|
const { left, top } = u.cursor;
|
|
const width = u?.over?.querySelector(".u-legend")?.offsetWidth ? u.over.querySelector(".u-legend").offsetWidth : 0;
|
|
legendEl.style.transform =
|
|
"translate(" + (left - width - 15) + "px, " + (top + 15) + "px)";
|
|
}
|
|
|
|
if (dataSize <= 12 || useStatsSeries === true) {
|
|
return {
|
|
hooks: {
|
|
init: init,
|
|
setCursor: update,
|
|
},
|
|
};
|
|
} else {
|
|
// Setting legend-opts show/live as object with false here will not work ...
|
|
return {};
|
|
}
|
|
}
|
|
|
|
// RETURN BG COLOR FROM THRESHOLD
|
|
function backgroundColor() {
|
|
if (
|
|
clusterCockpitConfig.plot_general_colorBackground == false ||
|
|
!thresholds ||
|
|
!(series && series.every((s) => s.statistics != null))
|
|
)
|
|
return backgroundColors.normal;
|
|
|
|
let cond =
|
|
thresholds.alert < thresholds.caution
|
|
? (a, b) => a <= b
|
|
: (a, b) => a >= b;
|
|
|
|
let avg =
|
|
series.reduce((sum, series) => sum + series.statistics.avg, 0) /
|
|
series.length;
|
|
|
|
if (Number.isNaN(avg)) return backgroundColors.normal;
|
|
|
|
if (cond(avg, thresholds.alert)) return backgroundColors.alert;
|
|
|
|
if (cond(avg, thresholds.caution)) return backgroundColors.caution;
|
|
|
|
return backgroundColors.normal;
|
|
}
|
|
|
|
// PREPARE UPLOT ...
|
|
function lineColor(i, n) {
|
|
if (n >= lineColors.length) return lineColors[i % lineColors.length];
|
|
else return lineColors[Math.floor((i / n) * lineColors.length)];
|
|
}
|
|
|
|
const longestSeries = useStatsSeries
|
|
? (usesMeanStatsSeries ? statisticsSeries.mean.length : statisticsSeries.median.length)
|
|
: series.reduce((n, series) => Math.max(n, series.data.length), 0);
|
|
const maxX = longestSeries * timestep;
|
|
let maxY = null;
|
|
|
|
if (thresholds !== null) {
|
|
maxY = useStatsSeries
|
|
? statisticsSeries.max.reduce(
|
|
(max, x) => Math.max(max, x),
|
|
thresholds.normal,
|
|
) || thresholds.normal
|
|
: series.reduce(
|
|
(max, series) => Math.max(max, series.statistics?.max),
|
|
thresholds.normal,
|
|
) || thresholds.normal;
|
|
|
|
if (maxY >= 10 * thresholds.peak) {
|
|
// Hard y-range render limit if outliers in series data
|
|
maxY = 10 * thresholds.peak;
|
|
}
|
|
}
|
|
|
|
const plotSeries = [
|
|
{
|
|
label: "Runtime",
|
|
value: (u, ts, sidx, didx) =>
|
|
didx == null ? null : formatTime(ts, forNode),
|
|
},
|
|
];
|
|
const plotData = [new Array(longestSeries)];
|
|
if (forNode === true) {
|
|
// Negative Timestamp Buildup
|
|
for (let i = 0; i <= longestSeries; i++) {
|
|
plotData[0][i] = (longestSeries - i) * timestep * -1;
|
|
}
|
|
} else {
|
|
// Positive Timestamp Buildup
|
|
for (
|
|
let j = 0;
|
|
j < longestSeries;
|
|
j++ // TODO: Cache/Reuse this array?
|
|
)
|
|
plotData[0][j] = j * timestep;
|
|
}
|
|
|
|
let plotBands = undefined;
|
|
if (useStatsSeries) {
|
|
plotData.push(statisticsSeries.min);
|
|
plotData.push(statisticsSeries.max);
|
|
if (usesMeanStatsSeries) {
|
|
plotData.push(statisticsSeries.mean);
|
|
} else {
|
|
plotData.push(statisticsSeries.median);
|
|
}
|
|
|
|
plotSeries.push({
|
|
label: "min",
|
|
scale: "y",
|
|
width: lineWidth,
|
|
stroke: "red",
|
|
});
|
|
plotSeries.push({
|
|
label: "max",
|
|
scale: "y",
|
|
width: lineWidth,
|
|
stroke: "green",
|
|
});
|
|
plotSeries.push({
|
|
label: usesMeanStatsSeries ? "mean" : "median",
|
|
scale: "y",
|
|
width: lineWidth,
|
|
stroke: "black",
|
|
});
|
|
|
|
plotBands = [
|
|
{ series: [2, 3], fill: "rgba(0,255,0,0.1)" },
|
|
{ series: [3, 1], fill: "rgba(255,0,0,0.1)" },
|
|
];
|
|
} else {
|
|
for (let i = 0; i < series.length; i++) {
|
|
plotData.push(series[i].data);
|
|
plotSeries.push({
|
|
label:
|
|
scope === "node"
|
|
? series[i].hostname
|
|
: scope + " #" + (i + 1),
|
|
scale: "y",
|
|
width: lineWidth,
|
|
stroke: lineColor(i, series.length),
|
|
});
|
|
}
|
|
}
|
|
|
|
const opts = {
|
|
width,
|
|
height,
|
|
plugins: [legendAsTooltipPlugin()],
|
|
series: plotSeries,
|
|
axes: [
|
|
{
|
|
scale: "x",
|
|
space: 35,
|
|
incrs: timeIncrs(timestep, maxX, forNode),
|
|
values: (_, vals) => vals.map((v) => formatTime(v, forNode)),
|
|
},
|
|
{
|
|
scale: "y",
|
|
grid: { show: true },
|
|
labelFont: "sans-serif",
|
|
values: (u, vals) => vals.map((v) => formatNumber(v)),
|
|
},
|
|
],
|
|
bands: plotBands,
|
|
padding: [5, 10, -20, 0],
|
|
hooks: {
|
|
init: [
|
|
(u) => {
|
|
/* IF Zoom Enabled */
|
|
if (resampleConfig) {
|
|
u.over.addEventListener("dblclick", (e) => {
|
|
// console.log('Dispatch Reset')
|
|
dispatch('zoom', {
|
|
lastZoomState: {
|
|
x: { time: false },
|
|
y: { auto: true }
|
|
}
|
|
});
|
|
});
|
|
};
|
|
},
|
|
],
|
|
draw: [
|
|
(u) => {
|
|
// Draw plot type label:
|
|
let textl = `${scope}${plotSeries.length > 2 ? "s" : ""}${
|
|
useStatsSeries
|
|
? (usesMeanStatsSeries ? ": min/mean/max" : ": min/median/max")
|
|
: metricConfig != null && scope != metricConfig.scope
|
|
? ` (${metricConfig.aggregation})`
|
|
: ""
|
|
}`;
|
|
let textr = `${isShared && scope != "core" && scope != "accelerator" ? "[Shared]" : ""}`;
|
|
u.ctx.save();
|
|
u.ctx.textAlign = "start"; // 'end'
|
|
u.ctx.fillStyle = "black";
|
|
u.ctx.fillText(textl, u.bbox.left + 10, u.bbox.top + 10);
|
|
u.ctx.textAlign = "end";
|
|
u.ctx.fillStyle = "black";
|
|
u.ctx.fillText(
|
|
textr,
|
|
u.bbox.left + u.bbox.width - 10,
|
|
u.bbox.top + 10,
|
|
);
|
|
// u.ctx.fillText(text, u.bbox.left + u.bbox.width - 10, u.bbox.top + u.bbox.height - 10) // Recipe for bottom right
|
|
|
|
if (!thresholds) {
|
|
u.ctx.restore();
|
|
return;
|
|
}
|
|
|
|
let y = u.valToPos(thresholds.normal, "y", true);
|
|
u.ctx.save();
|
|
u.ctx.lineWidth = lineWidth;
|
|
u.ctx.strokeStyle = normalLineColor;
|
|
u.ctx.setLineDash([5, 5]);
|
|
u.ctx.beginPath();
|
|
u.ctx.moveTo(u.bbox.left, y);
|
|
u.ctx.lineTo(u.bbox.left + u.bbox.width, y);
|
|
u.ctx.stroke();
|
|
u.ctx.restore();
|
|
},
|
|
],
|
|
setScale: [
|
|
(u, key) => { // If ZoomResample is Configured && Not System/Node View
|
|
if (resampleConfig && !forNode && key === 'x') {
|
|
const numX = (u.series[0].idxs[1] - u.series[0].idxs[0])
|
|
if (numX <= resampleTrigger && timestep !== resampleMinimum) {
|
|
/* Get closest zoom level; prevents multiple iterative zoom requests for big zoom-steps (e.g. 600 -> 300 -> 120 -> 60) */
|
|
// Which resolution to theoretically request to achieve 30 or more visible data points:
|
|
const target = (numX * timestep) / resampleTrigger
|
|
// Which configured resolution actually matches the closest to theoretical target:
|
|
const closest = resampleResolutions.reduce(function(prev, curr) {
|
|
return (Math.abs(curr - target) < Math.abs(prev - target) ? curr : prev);
|
|
});
|
|
// Prevents non-required dispatches
|
|
if (timestep !== closest) {
|
|
// console.log('Dispatch Zoom with Res from / to', timestep, closest)
|
|
dispatch('zoom', {
|
|
newRes: closest,
|
|
lastZoomState: u?.scales,
|
|
lastThreshold: thresholds?.normal
|
|
});
|
|
}
|
|
} else {
|
|
dispatch('zoom', {
|
|
lastZoomState: u?.scales,
|
|
lastThreshold: thresholds?.normal
|
|
});
|
|
};
|
|
};
|
|
},
|
|
]
|
|
},
|
|
scales: {
|
|
x: { time: false },
|
|
y: maxY ? { min: 0, max: (maxY * 1.1) } : {auto: true}, // Add some space to upper render limit
|
|
},
|
|
legend: {
|
|
// Display legend until max 12 Y-dataseries
|
|
show: series.length <= 12 || useStatsSeries === true ? true : false,
|
|
live: series.length <= 12 || useStatsSeries === true ? true : false,
|
|
},
|
|
cursor: { drag: { x: true, y: true } },
|
|
};
|
|
|
|
// RENDER HANDLING
|
|
let plotWrapper = null;
|
|
let uplot = null;
|
|
let timeoutId = null;
|
|
|
|
function render(ren_width, ren_height) {
|
|
if (!uplot) {
|
|
opts.width = ren_width;
|
|
opts.height = ren_height;
|
|
if (zoomState && metricConfig?.aggregation == "avg") {
|
|
opts.scales = {...zoomState}
|
|
} else if (zoomState && metricConfig?.aggregation == "sum") {
|
|
// Allow Zoom In === Ymin changed
|
|
if (zoomState.y.min !== 0) { // scope change?: only use zoomState if thresholds match
|
|
if ((thresholdState === thresholds?.normal)) { opts.scales = {...zoomState} };
|
|
} // else: reset scaling to default
|
|
}
|
|
uplot = new uPlot(opts, plotData, plotWrapper);
|
|
} else {
|
|
uplot.setSize({ width: ren_width, height: ren_height });
|
|
}
|
|
}
|
|
|
|
function onSizeChange(chg_width, chg_height) {
|
|
if (!uplot) return;
|
|
if (timeoutId != null) clearTimeout(timeoutId);
|
|
timeoutId = setTimeout(() => {
|
|
timeoutId = null;
|
|
render(chg_width, chg_height);
|
|
}, renderSleepTime);
|
|
}
|
|
|
|
onMount(() => {
|
|
if (plotWrapper) {
|
|
render(width, height);
|
|
}
|
|
});
|
|
|
|
onDestroy(() => {
|
|
if (timeoutId != null) clearTimeout(timeoutId);
|
|
if (uplot) uplot.destroy();
|
|
});
|
|
|
|
// This updates plot on all size changes if wrapper (== data) exists
|
|
$: if (plotWrapper) {
|
|
onSizeChange(width, height);
|
|
}
|
|
|
|
</script>
|
|
|
|
<!-- Define $width Wrapper and NoData Card -->
|
|
{#if series[0]?.data && series[0].data.length > 0}
|
|
<div bind:this={plotWrapper} bind:clientWidth={width}
|
|
style="background-color: {backgroundColor()};" class={forNode ? 'py-2 rounded' : 'rounded'}
|
|
/>
|
|
{:else}
|
|
<Card body color="warning" class="mx-4"
|
|
>Cannot render plot: No series data returned for <code>{metric}</code></Card
|
|
>
|
|
{/if}
|