2022-06-22 11:20:57 +02:00
|
|
|
<!--
|
|
|
|
@component
|
|
|
|
|
|
|
|
Properties:
|
|
|
|
- job: GraphQL.Job (constant/key)
|
|
|
|
- metrics: [String] (can change)
|
|
|
|
- plotWidth: Number
|
|
|
|
- plotHeight: Number
|
|
|
|
-->
|
|
|
|
|
|
|
|
<script>
|
2023-05-05 10:07:12 +02:00
|
|
|
import { queryStore, gql, getContextClient } from "@urql/svelte";
|
|
|
|
import { getContext } from "svelte";
|
|
|
|
import { Card, Spinner } from "sveltestrap";
|
|
|
|
import MetricPlot from "../plots/MetricPlot.svelte";
|
|
|
|
import JobInfo from "./JobInfo.svelte";
|
|
|
|
import { maxScope } from "../utils.js";
|
2022-06-22 11:20:57 +02:00
|
|
|
|
2023-05-05 10:07:12 +02:00
|
|
|
export let job;
|
|
|
|
export let metrics;
|
|
|
|
export let plotWidth;
|
|
|
|
export let plotHeight = 275;
|
2022-06-22 11:20:57 +02:00
|
|
|
|
2023-05-05 10:07:12 +02:00
|
|
|
let { id } = job;
|
|
|
|
let scopes = [job.numNodes == 1 ? "core" : "node"];
|
2022-06-22 11:20:57 +02:00
|
|
|
|
2023-05-05 10:07:12 +02:00
|
|
|
const cluster = getContext("clusters").find((c) => c.name == job.cluster);
|
2023-05-09 11:58:57 +02:00
|
|
|
const metricConfig = getContext("metrics"); // Get all MetricConfs which include subCluster-specific settings for this job
|
|
|
|
const client = getContextClient();
|
|
|
|
const query = gql`
|
|
|
|
query ($id: ID!, $metrics: [String!]!, $scopes: [MetricScope!]!) {
|
|
|
|
jobMetrics(id: $id, metrics: $metrics, scopes: $scopes) {
|
|
|
|
name
|
|
|
|
scope
|
|
|
|
metric {
|
|
|
|
unit {
|
|
|
|
prefix
|
|
|
|
base
|
|
|
|
}
|
|
|
|
timestep
|
|
|
|
statisticsSeries {
|
|
|
|
min
|
|
|
|
mean
|
|
|
|
max
|
|
|
|
}
|
|
|
|
series {
|
|
|
|
hostname
|
|
|
|
id
|
|
|
|
data
|
|
|
|
statistics {
|
2023-05-05 10:07:12 +02:00
|
|
|
min
|
2023-05-09 11:58:57 +02:00
|
|
|
avg
|
2023-05-05 10:07:12 +02:00
|
|
|
max
|
|
|
|
}
|
|
|
|
}
|
2022-06-22 11:20:57 +02:00
|
|
|
}
|
|
|
|
}
|
2023-05-09 11:58:57 +02:00
|
|
|
}
|
|
|
|
`;
|
|
|
|
|
|
|
|
$: metricsQuery = queryStore({
|
2023-05-12 11:19:37 +02:00
|
|
|
client: client,
|
|
|
|
query: query,
|
|
|
|
variables: { id, metrics, scopes }
|
2023-05-05 10:07:12 +02:00
|
|
|
});
|
2022-06-22 11:20:57 +02:00
|
|
|
|
2023-05-09 11:58:57 +02:00
|
|
|
function refresh() {
|
|
|
|
queryStore({
|
2023-05-12 11:19:37 +02:00
|
|
|
client: client,
|
|
|
|
query: query,
|
|
|
|
variables: { id, metrics, scopes }
|
2023-05-09 11:58:57 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-05-05 10:07:12 +02:00
|
|
|
const selectScope = (jobMetrics) =>
|
|
|
|
jobMetrics.reduce(
|
|
|
|
(a, b) =>
|
|
|
|
maxScope([a.scope, b.scope]) == a.scope
|
|
|
|
? job.numNodes > 1
|
|
|
|
? a
|
|
|
|
: b
|
|
|
|
: job.numNodes > 1
|
|
|
|
? b
|
|
|
|
: a,
|
|
|
|
jobMetrics[0]
|
|
|
|
);
|
2022-06-22 11:20:57 +02:00
|
|
|
|
2023-05-05 10:07:12 +02:00
|
|
|
const sortAndSelectScope = (jobMetrics) =>
|
|
|
|
metrics
|
|
|
|
.map(function (name) {
|
|
|
|
// Get MetricConf for this selected/requested metric
|
|
|
|
let thisConfig = metricConfig(cluster, name);
|
2023-05-08 17:43:34 +02:00
|
|
|
let thisSCIndex = -1
|
|
|
|
if (thisConfig) {
|
|
|
|
thisSCIndex = thisConfig.subClusters.findIndex(
|
|
|
|
(sc) => sc.name == job.subCluster
|
|
|
|
);
|
|
|
|
};
|
2023-05-05 10:07:12 +02:00
|
|
|
// Check if Subcluster has MetricConf: If not found (index == -1), no further remove flag check required
|
|
|
|
if (thisSCIndex >= 0) {
|
|
|
|
// SubCluster Config present: Check if remove flag is set
|
|
|
|
if (thisConfig.subClusters[thisSCIndex].remove == true) {
|
|
|
|
// Return null data and informational flag
|
|
|
|
return { removed: true, data: null };
|
|
|
|
} else {
|
|
|
|
// load and return metric, if data available
|
|
|
|
let thisMetric = jobMetrics.filter(
|
|
|
|
(jobMetric) => jobMetric.name == name
|
|
|
|
); // Returns Array
|
|
|
|
if (thisMetric.length > 0) {
|
|
|
|
return { removed: false, data: thisMetric };
|
|
|
|
} else {
|
|
|
|
return { removed: false, data: null };
|
|
|
|
}
|
|
|
|
}
|
2023-03-30 15:21:35 +02:00
|
|
|
} else {
|
2023-05-05 10:07:12 +02:00
|
|
|
// No specific subCluster config: 'remove' flag not set, deemed false -> load and return metric, if data available
|
|
|
|
let thisMetric = jobMetrics.filter(
|
|
|
|
(jobMetric) => jobMetric.name == name
|
|
|
|
); // Returns Array
|
2023-03-30 15:21:35 +02:00
|
|
|
if (thisMetric.length > 0) {
|
2023-05-05 10:07:12 +02:00
|
|
|
return { removed: false, data: thisMetric };
|
2023-03-30 15:21:35 +02:00
|
|
|
} else {
|
2023-05-05 10:07:12 +02:00
|
|
|
return { removed: false, data: null };
|
2023-03-30 15:21:35 +02:00
|
|
|
}
|
|
|
|
}
|
2023-05-05 10:07:12 +02:00
|
|
|
})
|
|
|
|
.map(function (jobMetrics) {
|
|
|
|
if (jobMetrics.data != null && jobMetrics.data.length > 0) {
|
|
|
|
return {
|
|
|
|
removed: jobMetrics.removed,
|
|
|
|
data: selectScope(jobMetrics.data),
|
|
|
|
};
|
2023-03-30 15:21:35 +02:00
|
|
|
} else {
|
2023-05-05 10:07:12 +02:00
|
|
|
return jobMetrics;
|
2023-03-30 15:21:35 +02:00
|
|
|
}
|
2023-05-05 10:07:12 +02:00
|
|
|
});
|
2022-06-22 11:20:57 +02:00
|
|
|
|
2023-05-09 11:58:57 +02:00
|
|
|
if (job.monitoringStatus) refresh();
|
2022-06-22 11:20:57 +02:00
|
|
|
</script>
|
|
|
|
|
|
|
|
<tr>
|
|
|
|
<td>
|
2023-05-05 10:07:12 +02:00
|
|
|
<JobInfo {job} />
|
2022-06-22 11:20:57 +02:00
|
|
|
</td>
|
|
|
|
{#if job.monitoringStatus == 0 || job.monitoringStatus == 2}
|
2023-05-05 10:07:12 +02:00
|
|
|
<td colspan={metrics.length}>
|
2022-06-22 11:20:57 +02:00
|
|
|
<Card body color="warning">Not monitored or archiving failed</Card>
|
|
|
|
</td>
|
|
|
|
{:else if $metricsQuery.fetching}
|
2023-05-05 10:07:12 +02:00
|
|
|
<td colspan={metrics.length} style="text-align: center;">
|
2022-06-22 11:20:57 +02:00
|
|
|
<Spinner secondary />
|
|
|
|
</td>
|
|
|
|
{:else if $metricsQuery.error}
|
2023-05-05 10:07:12 +02:00
|
|
|
<td colspan={metrics.length}>
|
2022-06-22 11:20:57 +02:00
|
|
|
<Card body color="danger" class="mb-3">
|
|
|
|
{$metricsQuery.error.message.length > 500
|
2023-05-05 10:07:12 +02:00
|
|
|
? $metricsQuery.error.message.substring(0, 499) + "..."
|
2022-06-22 11:20:57 +02:00
|
|
|
: $metricsQuery.error.message}
|
|
|
|
</Card>
|
|
|
|
</td>
|
|
|
|
{:else}
|
|
|
|
{#each sortAndSelectScope($metricsQuery.data.jobMetrics) as metric, i (metric || i)}
|
|
|
|
<td>
|
2023-05-05 10:07:12 +02:00
|
|
|
<!-- Subluster Metricconfig remove keyword for jobtables (joblist main, user joblist, project joblist) to be used here as toplevel case-->
|
|
|
|
{#if metric.removed == false && metric.data != null}
|
|
|
|
<MetricPlot
|
|
|
|
width={plotWidth}
|
|
|
|
height={plotHeight}
|
|
|
|
timestep={metric.data.metric.timestep}
|
|
|
|
scope={metric.data.scope}
|
|
|
|
series={metric.data.metric.series}
|
|
|
|
statisticsSeries={metric.data.metric.statisticsSeries}
|
|
|
|
metric={metric.data.name}
|
|
|
|
{cluster}
|
|
|
|
subCluster={job.subCluster}
|
|
|
|
/>
|
|
|
|
{:else if metric.removed == true && metric.data == null}
|
|
|
|
<Card body color="info"
|
|
|
|
>Metric disabled for subcluster '{job.subCluster}'</Card
|
|
|
|
>
|
|
|
|
{:else}
|
|
|
|
<Card body color="warning">Missing Data</Card>
|
|
|
|
{/if}
|
2022-06-22 11:20:57 +02:00
|
|
|
</td>
|
|
|
|
{/each}
|
|
|
|
{/if}
|
|
|
|
</tr>
|