mirror of
				https://github.com/ClusterCockpit/cc-backend
				synced 2025-10-26 14:25:06 +01:00 
			
		
		
		
	Merge branch 'sample_resolution_select' into dev
This commit is contained in:
		| @@ -48,6 +48,7 @@ | ||||
|       href: `/monitoring/user/${username}`, | ||||
|       icon: "bar-chart-line-fill", | ||||
|       perCluster: false, | ||||
|       listOptions: false, | ||||
|       menu: "none", | ||||
|     }, | ||||
|     { | ||||
| @@ -56,6 +57,7 @@ | ||||
|       href: `/monitoring/jobs/`, | ||||
|       icon: "card-list", | ||||
|       perCluster: false, | ||||
|       listOptions: false, | ||||
|       menu: "none", | ||||
|     }, | ||||
|     { | ||||
| @@ -63,7 +65,8 @@ | ||||
|       requiredRole: roles.manager, | ||||
|       href: "/monitoring/users/", | ||||
|       icon: "people-fill", | ||||
|       perCluster: false, | ||||
|       perCluster: true, | ||||
|       listOptions: true, | ||||
|       menu: "Groups", | ||||
|     }, | ||||
|     { | ||||
| @@ -71,7 +74,8 @@ | ||||
|       requiredRole: roles.support, | ||||
|       href: "/monitoring/projects/", | ||||
|       icon: "folder", | ||||
|       perCluster: false, | ||||
|       perCluster: true, | ||||
|       listOptions: true, | ||||
|       menu: "Groups", | ||||
|     }, | ||||
|     { | ||||
| @@ -80,6 +84,7 @@ | ||||
|       href: "/monitoring/tags/", | ||||
|       icon: "tags", | ||||
|       perCluster: false, | ||||
|       listOptions: false, | ||||
|       menu: "Groups", | ||||
|     }, | ||||
|     { | ||||
| @@ -88,6 +93,7 @@ | ||||
|       href: "/monitoring/analysis/", | ||||
|       icon: "graph-up", | ||||
|       perCluster: true, | ||||
|       listOptions: false, | ||||
|       menu: "Stats", | ||||
|     }, | ||||
|     { | ||||
| @@ -96,6 +102,7 @@ | ||||
|       href: "/monitoring/systems/", | ||||
|       icon: "cpu", | ||||
|       perCluster: true, | ||||
|       listOptions: false, | ||||
|       menu: "Groups", | ||||
|     }, | ||||
|     { | ||||
| @@ -104,6 +111,7 @@ | ||||
|       href: "/monitoring/status/", | ||||
|       icon: "cpu", | ||||
|       perCluster: true, | ||||
|       listOptions: false, | ||||
|       menu: "Stats", | ||||
|     }, | ||||
|   ]; | ||||
|   | ||||
| @@ -56,7 +56,8 @@ | ||||
|     selectedScopes = []; | ||||
|  | ||||
|   let plots = {}, | ||||
|     roofWidth | ||||
|     roofWidth, | ||||
|     statsTable | ||||
|  | ||||
|   let missingMetrics = [], | ||||
|     missingHosts = [], | ||||
| @@ -119,15 +120,6 @@ | ||||
|     variables: { dbid, selectedMetrics, selectedScopes }, | ||||
|   }); | ||||
|  | ||||
|   function loadAllScopes() { | ||||
|     selectedScopes = [...selectedScopes, "socket", "core"] | ||||
|     jobMetrics = queryStore({ | ||||
|       client: client, | ||||
|       query: query, | ||||
|       variables: { dbid, selectedMetrics, selectedScopes}, | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   // Handle Job Query on Init -> is not executed anymore | ||||
|   getContext("on-init")(() => { | ||||
|     let job = $initq.data.job; | ||||
| @@ -352,7 +344,7 @@ | ||||
|             {#if item.data} | ||||
|               <Metric | ||||
|                 bind:this={plots[item.metric]} | ||||
|                 on:load-all={loadAllScopes} | ||||
|                 on:more-loaded={({ detail }) => statsTable.moreLoaded(detail)} | ||||
|                 job={$initq.data.job} | ||||
|                 metricName={item.metric} | ||||
|                 metricUnit={$initq.data.globalMetrics.find((gm) => gm.name == item.metric)?.unit} | ||||
| @@ -418,6 +410,7 @@ | ||||
|             {#if $jobMetrics?.data?.jobMetrics} | ||||
|               {#key $jobMetrics.data.jobMetrics} | ||||
|                 <StatsTable | ||||
|                   bind:this={statsTable} | ||||
|                   job={$initq.data.job} | ||||
|                   jobMetrics={$jobMetrics.data.jobMetrics} | ||||
|                 /> | ||||
|   | ||||
| @@ -90,11 +90,10 @@ | ||||
|     }, | ||||
|   }); | ||||
|  | ||||
|   let itemsPerPage = ccconfig.plot_list_jobsPerPage; | ||||
|   let page = 1; | ||||
|   let paging = { itemsPerPage, page }; | ||||
|   let sorting = { field: "startTime", type: "col", order: "DESC" }; | ||||
|   $: filter = [ | ||||
|  | ||||
|   const paging = { itemsPerPage: 50, page: 1 }; | ||||
|   const sorting = { field: "startTime", type: "col", order: "DESC" }; | ||||
|   const filter = [ | ||||
|     { cluster: { eq: cluster } }, | ||||
|     { node: { contains: hostname } }, | ||||
|     { state: ["running"] }, | ||||
| @@ -207,7 +206,6 @@ | ||||
|             cluster={clusters.find((c) => c.name == cluster)} | ||||
|             subCluster={$nodeMetricsData.data.nodeMetrics[0].subCluster} | ||||
|             series={item.metric.series} | ||||
|             resources={[{ hostname: hostname }]} | ||||
|             forNode={true} | ||||
|           /> | ||||
|         {:else if item.disabled === true && item.metric} | ||||
|   | ||||
| @@ -206,7 +206,6 @@ | ||||
|             metric={item.data.name} | ||||
|             cluster={clusters.find((c) => c.name == cluster)} | ||||
|             subCluster={item.subCluster} | ||||
|             resources={[{ hostname: item.host }]} | ||||
|             forNode={true} | ||||
|           /> | ||||
|         {:else if item.disabled === true && item.data} | ||||
|   | ||||
| @@ -9,6 +9,7 @@ new Config({ | ||||
|         username: username | ||||
|     }, | ||||
|     context: new Map([ | ||||
|             ['cc-config', clusterCockpitConfig] | ||||
|             ['cc-config', clusterCockpitConfig], | ||||
|             ['resampling', resampleConfig] | ||||
|     ]) | ||||
| }) | ||||
|   | ||||
| @@ -51,7 +51,5 @@ | ||||
|   <Col> | ||||
|     <EditProject on:reload={getUserList} /> | ||||
|   </Col> | ||||
|   <Col> | ||||
|     <Options /> | ||||
|   </Col> | ||||
|   <Options /> | ||||
| </Row> | ||||
|   | ||||
| @@ -3,11 +3,13 @@ | ||||
|  --> | ||||
|  | ||||
| <script> | ||||
|   import { onMount } from "svelte"; | ||||
|   import { Card, CardBody, CardTitle } from "@sveltestrap/sveltestrap"; | ||||
|   import { getContext, onMount } from "svelte"; | ||||
|   import { Col, Card, CardBody, CardTitle } from "@sveltestrap/sveltestrap"; | ||||
|  | ||||
|   let scrambled; | ||||
|  | ||||
|   const resampleConfig = getContext("resampling"); | ||||
|  | ||||
|   onMount(() => { | ||||
|     scrambled = window.localStorage.getItem("cc-scramble-names") != null; | ||||
|   }); | ||||
| @@ -23,16 +25,30 @@ | ||||
|   } | ||||
| </script> | ||||
|  | ||||
| <Card class="h-100"> | ||||
|   <CardBody> | ||||
|     <CardTitle class="mb-3">Scramble Names / Presentation Mode</CardTitle> | ||||
|     <input | ||||
|       type="checkbox" | ||||
|       id="scramble-names-checkbox" | ||||
|       style="margin-right: 1em;" | ||||
|       on:click={handleScramble} | ||||
|       bind:checked={scrambled} | ||||
|     /> | ||||
|     Active? | ||||
|   </CardBody> | ||||
| </Card> | ||||
| <Col> | ||||
|   <Card class="h-100"> | ||||
|     <CardBody> | ||||
|       <CardTitle class="mb-3">Scramble Names / Presentation Mode</CardTitle> | ||||
|       <input | ||||
|         type="checkbox" | ||||
|         id="scramble-names-checkbox" | ||||
|         style="margin-right: 1em;" | ||||
|         on:click={handleScramble} | ||||
|         bind:checked={scrambled} | ||||
|       /> | ||||
|       Active? | ||||
|     </CardBody> | ||||
|   </Card> | ||||
| </Col> | ||||
|  | ||||
| {#if resampleConfig} | ||||
|   <Col>  | ||||
|     <Card class="h-100"> | ||||
|       <CardBody> | ||||
|         <CardTitle class="mb-3">Metric Plot Resampling</CardTitle> | ||||
|         <p>Triggered at {resampleConfig.trigger} datapoints.</p> | ||||
|         <p>Configured resolutions: {resampleConfig.resolutions}</p> | ||||
|       </CardBody> | ||||
|     </Card> | ||||
|   </Col> | ||||
| {/if} | ||||
|   | ||||
| @@ -26,18 +26,23 @@ | ||||
|   export let showFootprint; | ||||
|   export let triggerMetricRefresh = false; | ||||
|  | ||||
|   const resampleConfig = getContext("resampling") || null; | ||||
|   const resampleDefault = resampleConfig ? Math.max(...resampleConfig.resolutions) : 0; | ||||
|    | ||||
|   let { id } = job; | ||||
|   let scopes = job.numNodes == 1 | ||||
|     ? job.numAcc >= 1 | ||||
|       ? ["core", "accelerator"] | ||||
|       : ["core"] | ||||
|     : ["node"]; | ||||
|   let selectedResolution = resampleDefault; | ||||
|   let zoomStates = {}; | ||||
|  | ||||
|   const cluster = getContext("clusters").find((c) => c.name == job.cluster); | ||||
|   const client = getContextClient(); | ||||
|   const query = gql` | ||||
|     query ($id: ID!, $metrics: [String!]!, $scopes: [MetricScope!]!) { | ||||
|       jobMetrics(id: $id, metrics: $metrics, scopes: $scopes) { | ||||
|     query ($id: ID!, $metrics: [String!]!, $scopes: [MetricScope!]!, $selectedResolution: Int) { | ||||
|       jobMetrics(id: $id, metrics: $metrics, scopes: $scopes, resolution: $selectedResolution) { | ||||
|         name | ||||
|         scope | ||||
|         metric { | ||||
| @@ -66,17 +71,30 @@ | ||||
|     } | ||||
|   `; | ||||
|  | ||||
|   function handleZoom(detail, metric) { | ||||
|     if ( // States have to differ, causes deathloop if just set | ||||
|         (zoomStates[metric]?.x?.min !== detail?.lastZoomState?.x?.min) && | ||||
|         (zoomStates[metric]?.y?.max !== detail?.lastZoomState?.y?.max) | ||||
|     ) { | ||||
|         zoomStates[metric] = {...detail.lastZoomState} | ||||
|     } | ||||
|  | ||||
|     if (detail?.newRes) { // Triggers GQL | ||||
|         selectedResolution = detail.newRes | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   $: metricsQuery = queryStore({ | ||||
|     client: client, | ||||
|     query: query, | ||||
|     variables: { id, metrics, scopes }, | ||||
|     variables: { id, metrics, scopes, selectedResolution }, | ||||
|   }); | ||||
|    | ||||
|   function refreshMetrics() { | ||||
|     metricsQuery = queryStore({ | ||||
|       client: client, | ||||
|       query: query, | ||||
|       variables: { id, metrics, scopes }, | ||||
|       variables: { id, metrics, scopes, selectedResolution }, | ||||
|       // requestPolicy: 'network-only' // use default cache-first for refresh | ||||
|     }); | ||||
|   } | ||||
| @@ -159,6 +177,7 @@ | ||||
|         <!-- Subluster Metricconfig remove keyword for jobtables (joblist main, user joblist, project joblist) to be used here as toplevel case--> | ||||
|         {#if metric.disabled == false && metric.data} | ||||
|           <MetricPlot | ||||
|             on:zoom={({detail}) => { handleZoom(detail, metric.data.name) }} | ||||
|             width={plotWidth} | ||||
|             height={plotHeight} | ||||
|             timestep={metric.data.metric.timestep} | ||||
| @@ -169,9 +188,9 @@ | ||||
|             {cluster} | ||||
|             subCluster={job.subCluster} | ||||
|             isShared={job.exclusive != 1} | ||||
|             resources={job.resources} | ||||
|             numhwthreads={job.numHWThreads} | ||||
|             numaccs={job.numAcc} | ||||
|             zoomState={zoomStates[metric.data.name] || null} | ||||
|           /> | ||||
|         {:else if metric.disabled == true && metric.data} | ||||
|           <Card body color="info" | ||||
|   | ||||
| @@ -6,7 +6,6 @@ | ||||
|     Properties: | ||||
|     - `metric String`: The metric name | ||||
|     - `scope String?`: Scope of the displayed data [Default: node] | ||||
|     - `resources [GraphQL.Resource]`: List of resources used for parent job | ||||
|     - `width Number`: The plot width | ||||
|     - `height Number`: The plot height | ||||
|     - `timestep Number`: The timestep used for X-axis rendering | ||||
| @@ -16,9 +15,10 @@ | ||||
|     - `cluster GraphQL.Cluster`: Cluster Object of the parent job | ||||
|     - `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] | ||||
|     - `forNode Bool?`: If this plot is used for node data display; will ren[data, err := metricdata.LoadNodeData(cluster, metrics, nodes, scopes, from, to, ctx)](https://github.com/ClusterCockpit/cc-backend/blob/9fe7cdca9215220a19930779a60c8afc910276a3/internal/graph/schema.resolvers.go#L391-L392)der 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"> | ||||
| @@ -40,7 +40,7 @@ | ||||
|  | ||||
|   function timeIncrs(timestep, maxX, forNode) { | ||||
|     if (forNode === true) { | ||||
|       return [60, 300, 900, 1800, 3600, 7200, 14400, 21600]; // forNode fixed increments | ||||
|       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) | ||||
| @@ -113,12 +113,11 @@ | ||||
| <script> | ||||
|   import uPlot from "uplot"; | ||||
|   import { formatNumber } from "../units.js"; | ||||
|   import { getContext, onMount, onDestroy } from "svelte"; | ||||
|   import { getContext, onMount, onDestroy, createEventDispatcher } from "svelte"; | ||||
|   import { Card } from "@sveltestrap/sveltestrap"; | ||||
|  | ||||
|   export let metric; | ||||
|   export let scope = "node"; | ||||
|   export let resources = []; | ||||
|   export let width; | ||||
|   export let height; | ||||
|   export let timestep; | ||||
| @@ -131,11 +130,13 @@ | ||||
|   export let forNode = false; | ||||
|   export let numhwthreads = 0; | ||||
|   export let numaccs = 0; | ||||
|   export let zoomState = null; | ||||
|  | ||||
|   if (useStatsSeries == null) useStatsSeries = statisticsSeries != null; | ||||
|  | ||||
|   if (useStatsSeries == false && series == null) useStatsSeries = true; | ||||
|  | ||||
|   const dispatch = createEventDispatcher(); | ||||
|   const subClusterTopology = getContext("getHardwareTopology")(cluster, subCluster); | ||||
|   const metricConfig = getContext("getMetricConfig")(cluster, subCluster, metric); | ||||
|   const clusterCockpitConfig = getContext("cc-config"); | ||||
| @@ -158,6 +159,17 @@ | ||||
|     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); | ||||
|   } | ||||
|  | ||||
|   // converts the legend into a simple tooltip | ||||
|   function legendAsTooltipPlugin({ | ||||
|     className, | ||||
| @@ -296,7 +308,6 @@ | ||||
|     }, | ||||
|   ]; | ||||
|   const plotData = [new Array(longestSeries)]; | ||||
|  | ||||
|   if (forNode === true) { | ||||
|     // Negative Timestamp Buildup | ||||
|     for (let i = 0; i <= longestSeries; i++) { | ||||
| @@ -317,15 +328,15 @@ | ||||
|     plotData.push(statisticsSeries.min); | ||||
|     plotData.push(statisticsSeries.max); | ||||
|     plotData.push(statisticsSeries.median); | ||||
|     // plotData.push(statisticsSeries.mean); | ||||
|  | ||||
|     if (forNode === true) { | ||||
|       // timestamp 0 with null value for reversed time axis | ||||
|       if (plotData[1].length != 0) plotData[1].push(null); | ||||
|       if (plotData[2].length != 0) plotData[2].push(null); | ||||
|       if (plotData[3].length != 0) plotData[3].push(null); | ||||
|       // if (plotData[4].length != 0) plotData[4].push(null); | ||||
|     } | ||||
|     /* deprecated: sparse data handled by uplot */ | ||||
|     // if (forNode === true) { | ||||
|     //   if (plotData[1][-1] != null && plotData[2][-1] != null && plotData[3][-1] != null) { | ||||
|     //     if (plotData[1].length != 0) plotData[1].push(null); | ||||
|     //     if (plotData[2].length != 0) plotData[2].push(null); | ||||
|     //     if (plotData[3].length != 0) plotData[3].push(null); | ||||
|     //   } | ||||
|     // } | ||||
|  | ||||
|     plotSeries.push({ | ||||
|       label: "min", | ||||
| @@ -345,12 +356,6 @@ | ||||
|       width: lineWidth, | ||||
|       stroke: "black", | ||||
|     }); | ||||
|     // plotSeries.push({ | ||||
|     //   label: "mean", | ||||
|     //   scale: "y", | ||||
|     //   width: lineWidth, | ||||
|     //   stroke: "blue", | ||||
|     // }); | ||||
|  | ||||
|     plotBands = [ | ||||
|       { series: [2, 3], fill: "rgba(0,255,0,0.1)" }, | ||||
| @@ -359,13 +364,18 @@ | ||||
|   } else { | ||||
|     for (let i = 0; i < series.length; i++) { | ||||
|       plotData.push(series[i].data); | ||||
|       if (forNode === true && plotData[1].length != 0) plotData[1].push(null); // timestamp 0 with null value for reversed time axis | ||||
|       /* deprecated: sparse data handled by uplot */ | ||||
|       // if (forNode === true && plotData[1].length != 0) { | ||||
|       //   if (plotData[1][-1] != null) { | ||||
|       //     plotData[1].push(null); | ||||
|       //   }; | ||||
|       // }; | ||||
|  | ||||
|       plotSeries.push({ | ||||
|         label: | ||||
|           scope === "node" | ||||
|             ? resources[i].hostname | ||||
|             : // scope === 'accelerator' ? resources[0].accelerators[i] : | ||||
|               scope + " #" + (i + 1), | ||||
|             ? series[i].hostname | ||||
|             : scope + " #" + (i + 1), | ||||
|         scale: "y", | ||||
|         width: lineWidth, | ||||
|         stroke: lineColor(i, series.length), | ||||
| @@ -395,6 +405,22 @@ | ||||
|     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: | ||||
| @@ -436,6 +462,34 @@ | ||||
|           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 | ||||
|                 }); | ||||
|               } | ||||
|             } else { | ||||
|               dispatch('zoom', { | ||||
|                 lastZoomState: u?.scales | ||||
|               }); | ||||
|             }; | ||||
|           }; | ||||
|         }, | ||||
|       ] | ||||
|     }, | ||||
|     scales: { | ||||
|       x: { time: false }, | ||||
| @@ -466,6 +520,9 @@ | ||||
|     if (!uplot) { | ||||
|       opts.width = width; | ||||
|       opts.height = height; | ||||
|       if (zoomState) { | ||||
|         opts.scales = {...zoomState} | ||||
|       } | ||||
|       uplot = new uPlot(opts, plotData, plotWrapper); | ||||
|     } else { | ||||
|       uplot.setSize({ width, height }); | ||||
| @@ -474,7 +531,6 @@ | ||||
|  | ||||
|   function onSizeChange() { | ||||
|     if (!uplot) return; | ||||
|  | ||||
|     if (timeoutId != null) clearTimeout(timeoutId); | ||||
|  | ||||
|     timeoutId = setTimeout(() => { | ||||
|   | ||||
| @@ -69,6 +69,7 @@ | ||||
|  | ||||
| <InputGroup class="inline-from"> | ||||
|   <InputGroupText><Icon name="clock-history" /></InputGroupText> | ||||
|   <InputGroupText>Range</InputGroupText> | ||||
|   <select | ||||
|     class="form-select" | ||||
|     bind:value={timeRange} | ||||
|   | ||||
| @@ -21,7 +21,41 @@ | ||||
| </script> | ||||
|  | ||||
| {#each links as item} | ||||
|   {#if !item.perCluster} | ||||
|   {#if item.listOptions} | ||||
|     <Dropdown nav inNavbar> | ||||
|       <DropdownToggle nav caret> | ||||
|         <Icon name={item.icon} /> | ||||
|         {item.title} | ||||
|       </DropdownToggle> | ||||
|       <DropdownMenu class="dropdown-menu-lg-end"> | ||||
|         <DropdownItem | ||||
|           href={item.href} | ||||
|         > | ||||
|           All Clusters | ||||
|         </DropdownItem> | ||||
|         <DropdownItem divider /> | ||||
|         {#each clusters as cluster} | ||||
|           <Dropdown nav direction="right"> | ||||
|             <DropdownToggle nav caret class="dropdown-item py-1 px-2"> | ||||
|               {cluster.name} | ||||
|             </DropdownToggle> | ||||
|             <DropdownMenu> | ||||
|               <DropdownItem class="py-1 px-2" | ||||
|                 href={item.href + '?cluster=' + cluster.name} | ||||
|               > | ||||
|                 All Jobs | ||||
|               </DropdownItem> | ||||
|               <DropdownItem class="py-1 px-2" | ||||
|                 href={item.href + '?cluster=' + cluster.name + '&state=running'} | ||||
|               > | ||||
|                 Running Jobs | ||||
|               </DropdownItem> | ||||
|             </DropdownMenu> | ||||
|           </Dropdown> | ||||
|         {/each} | ||||
|       </DropdownMenu> | ||||
|     </Dropdown> | ||||
|   {:else if !item.perCluster} | ||||
|     <NavLink href={item.href} active={window.location.pathname == item.href} | ||||
|       ><Icon name={item.icon} /> {item.title}</NavLink | ||||
|     > | ||||
|   | ||||
| @@ -10,6 +10,7 @@ new Job({ | ||||
|         roles: roles | ||||
|     }, | ||||
|     context: new Map([ | ||||
|             ['cc-config', clusterCockpitConfig] | ||||
|             ['cc-config', clusterCockpitConfig], | ||||
|             ['resampling', resampleConfig] | ||||
|     ]) | ||||
| }) | ||||
|   | ||||
| @@ -13,14 +13,24 @@ | ||||
|  --> | ||||
|  | ||||
| <script> | ||||
|   import { createEventDispatcher } from "svelte"; | ||||
|   import {  | ||||
|     getContext, | ||||
|     createEventDispatcher  | ||||
|   } from "svelte"; | ||||
|   import {  | ||||
|     queryStore, | ||||
|     gql, | ||||
|     getContextClient  | ||||
|   } from "@urql/svelte"; | ||||
|   import { | ||||
|     InputGroup, | ||||
|     InputGroupText, | ||||
|     Spinner, | ||||
|     Card, | ||||
|   } from "@sveltestrap/sveltestrap"; | ||||
|   import { minScope } from "../generic/utils"; | ||||
|   import {  | ||||
|     minScope, | ||||
|   } from "../generic/utils.js"; | ||||
|   import Timeseries from "../generic/plots/MetricPlot.svelte"; | ||||
|  | ||||
|   export let job; | ||||
| @@ -32,32 +42,132 @@ | ||||
|   export let rawData; | ||||
|   export let isShared = false; | ||||
|  | ||||
|   const dispatch = createEventDispatcher(); | ||||
|   const unit = (metricUnit?.prefix ? metricUnit.prefix : "") + (metricUnit?.base ? metricUnit.base : "") | ||||
|      | ||||
|   let selectedHost = null, | ||||
|     plot, | ||||
|     fetching = false, | ||||
|     error = null; | ||||
|   const resampleConfig = getContext("resampling") || null; | ||||
|   const resampleDefault = resampleConfig ? Math.max(...resampleConfig.resolutions) : 0; | ||||
|  | ||||
|   let selectedHost = null; | ||||
|   let error = null; | ||||
|   let selectedScope = minScope(scopes); | ||||
|   let selectedResolution = null; | ||||
|   let pendingResolution = resampleDefault; | ||||
|   let selectedScopeIndex = scopes.findIndex((s) => s == minScope(scopes)); | ||||
|   let patternMatches = false; | ||||
|   let nodeOnly = false; // If, after load-all, still only node scope returned | ||||
|   let statsSeries = rawData.map((data) => data?.statisticsSeries ? data.statisticsSeries : null); | ||||
|   let zoomState = null; | ||||
|   let pendingZoomState = null; | ||||
|  | ||||
|   let statsPattern = /(.*)-stat$/ | ||||
|   let statsSeries = rawData.map((data) => data?.statisticsSeries ? data.statisticsSeries : null) | ||||
|   let selectedScopeIndex | ||||
|  | ||||
|   $: availableScopes = scopes; | ||||
|   $: patternMatches = statsPattern.exec(selectedScope) | ||||
|   $: if (!patternMatches) { | ||||
|       selectedScopeIndex = scopes.findIndex((s) => s == selectedScope); | ||||
|     } else { | ||||
|       selectedScopeIndex = scopes.findIndex((s) => s == patternMatches[1]); | ||||
|   const dispatch = createEventDispatcher(); | ||||
|   const statsPattern = /(.*)-stat$/; | ||||
|   const unit = (metricUnit?.prefix ? metricUnit.prefix : "") + (metricUnit?.base ? metricUnit.base : ""); | ||||
|   const client = getContextClient(); | ||||
|   const subQuery = gql` | ||||
|     query ($dbid: ID!, $selectedMetrics: [String!]!, $selectedScopes: [MetricScope!]!, $selectedResolution: Int) { | ||||
|       singleUpdate: jobMetrics(id: $dbid, metrics: $selectedMetrics, scopes: $selectedScopes, resolution: $selectedResolution) { | ||||
|         name | ||||
|         scope | ||||
|         metric { | ||||
|           unit { | ||||
|             prefix | ||||
|             base | ||||
|           } | ||||
|           timestep | ||||
|           statisticsSeries { | ||||
|             min | ||||
|             median | ||||
|             max | ||||
|           } | ||||
|           series { | ||||
|             hostname | ||||
|             id | ||||
|             data | ||||
|             statistics { | ||||
|               min | ||||
|               avg | ||||
|               max | ||||
|             } | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   `; | ||||
|  | ||||
|   function handleZoom(detail) { | ||||
|       if ( // States have to differ, causes deathloop if just set | ||||
|           (pendingZoomState?.x?.min !== detail?.lastZoomState?.x?.min) && | ||||
|           (pendingZoomState?.y?.max !== detail?.lastZoomState?.y?.max) | ||||
|       ) { | ||||
|           pendingZoomState = {...detail.lastZoomState} | ||||
|       } | ||||
|  | ||||
|       if (detail?.newRes) { // Triggers GQL | ||||
|           pendingResolution = detail.newRes | ||||
|       } | ||||
|   } | ||||
|  | ||||
|   let metricData; | ||||
|   let selectedScopes = [...scopes] | ||||
|   const dbid = job.id; | ||||
|   const selectedMetrics = [metricName] | ||||
|  | ||||
|   $: if (selectedScope || pendingResolution) { | ||||
|     if (!selectedResolution) { | ||||
|       // Skips reactive data load on init | ||||
|       selectedResolution = Number(pendingResolution) | ||||
|  | ||||
|     } else { | ||||
|       if (selectedScope == "load-all") { | ||||
|         selectedScopes = [...scopes, "socket", "core", "accelerator"] | ||||
|       } | ||||
|  | ||||
|       if (pendingResolution) { | ||||
|         selectedResolution = Number(pendingResolution) | ||||
|       } | ||||
|  | ||||
|       metricData = queryStore({ | ||||
|         client: client, | ||||
|         query: subQuery, | ||||
|         variables: { dbid, selectedMetrics, selectedScopes, selectedResolution }, | ||||
|         // Never user network-only: causes reactive load-loop! | ||||
|       }); | ||||
|  | ||||
|       if ($metricData && !$metricData.fetching) { | ||||
|         rawData = $metricData.data.singleUpdate.map((x) => x.metric) | ||||
|         scopes  = $metricData.data.singleUpdate.map((x) => x.scope) | ||||
|         statsSeries    = rawData.map((data) => data?.statisticsSeries ? data.statisticsSeries : null) | ||||
|  | ||||
|         // Keep Zoomlevel if ResChange By Zoom | ||||
|         if (pendingZoomState) { | ||||
|           zoomState = {...pendingZoomState} | ||||
|         } | ||||
|  | ||||
|         // Set selected scope to min of returned scopes | ||||
|         if (selectedScope == "load-all") { | ||||
|           selectedScope = minScope(scopes) | ||||
|           nodeOnly = (selectedScope == "node") // "node" still only scope after load-all | ||||
|         } | ||||
|  | ||||
|         const statsTableData = $metricData.data.singleUpdate.filter((x) => x.scope !== "node") | ||||
|         if (statsTableData.length > 0) { | ||||
|           dispatch("more-loaded", statsTableData); | ||||
|         } | ||||
|  | ||||
|         patternMatches = statsPattern.exec(selectedScope) | ||||
|  | ||||
|         if (!patternMatches) { | ||||
|           selectedScopeIndex = scopes.findIndex((s) => s == selectedScope); | ||||
|         } else { | ||||
|           selectedScopeIndex = scopes.findIndex((s) => s == patternMatches[1]); | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   $: data = rawData[selectedScopeIndex]; | ||||
|   $: series = data?.series.filter( | ||||
|    | ||||
|   $: series = data?.series?.filter( | ||||
|     (series) => selectedHost == null || series.hostname == selectedHost, | ||||
|   ); | ||||
|  | ||||
|   $: if (selectedScope == "load-all") dispatch("load-all"); | ||||
| </script> | ||||
|  | ||||
| <InputGroup> | ||||
| @@ -65,13 +175,13 @@ | ||||
|     {metricName} ({unit}) | ||||
|   </InputGroupText> | ||||
|   <select class="form-select" bind:value={selectedScope}> | ||||
|     {#each availableScopes as scope, index} | ||||
|     {#each scopes as scope, index} | ||||
|       <option value={scope}>{scope}</option> | ||||
|       {#if statsSeries[index]} | ||||
|         <option value={scope + '-stat'}>stats series ({scope})</option> | ||||
|       {/if} | ||||
|     {/each} | ||||
|     {#if availableScopes.length == 1 && nativeScope != "node"} | ||||
|     {#if scopes.length == 1 && nativeScope != "node" && !nodeOnly} | ||||
|       <option value={"load-all"}>Load all...</option> | ||||
|     {/if} | ||||
|   </select> | ||||
| @@ -85,13 +195,13 @@ | ||||
|   {/if} | ||||
| </InputGroup> | ||||
| {#key series} | ||||
|   {#if fetching == true} | ||||
|   {#if $metricData?.fetching == true} | ||||
|     <Spinner /> | ||||
|   {:else if error != null} | ||||
|     <Card body color="danger">{error.message}</Card> | ||||
|   {:else if series != null && !patternMatches} | ||||
|     <Timeseries | ||||
|       bind:this={plot} | ||||
|       on:zoom={({detail}) => { handleZoom(detail) }} | ||||
|       {width} | ||||
|       height={300} | ||||
|       cluster={job.cluster} | ||||
| @@ -101,11 +211,11 @@ | ||||
|       metric={metricName} | ||||
|       {series} | ||||
|       {isShared} | ||||
|       resources={job.resources} | ||||
|       {zoomState} | ||||
|     /> | ||||
|   {:else if statsSeries[selectedScopeIndex] != null && patternMatches} | ||||
|     <Timeseries | ||||
|       bind:this={plot} | ||||
|       on:zoom={({detail}) => { handleZoom(detail) }} | ||||
|       {width} | ||||
|       height={300} | ||||
|       cluster={job.cluster} | ||||
| @@ -115,7 +225,7 @@ | ||||
|       metric={metricName} | ||||
|       {series} | ||||
|       {isShared} | ||||
|       resources={job.resources} | ||||
|       {zoomState} | ||||
|       statisticsSeries={statsSeries[selectedScopeIndex]} | ||||
|       useStatsSeries={!!statsSeries[selectedScopeIndex]} | ||||
|     /> | ||||
|   | ||||
| @@ -4,6 +4,9 @@ | ||||
|     Properties: | ||||
|     - `job Object`: The job object | ||||
|     - `jobMetrics [Object]`: The jobs metricdata | ||||
|  | ||||
|     Exported: | ||||
|     - `moreLoaded`: Adds additional scopes requested from Metric.svelte in Job-View | ||||
|  --> | ||||
|  | ||||
| <script> | ||||
| @@ -23,8 +26,8 @@ | ||||
|   export let job; | ||||
|   export let jobMetrics; | ||||
|  | ||||
|   const allMetrics = [...new Set(jobMetrics.map((m) => m.name))].sort(), | ||||
|     scopesForMetric = (metric) => | ||||
|   const allMetrics = [...new Set(jobMetrics.map((m) => m.name))].sort() | ||||
|   const scopesForMetric = (metric) => | ||||
|       jobMetrics.filter((jm) => jm.name == metric).map((jm) => jm.scope); | ||||
|  | ||||
|   let hosts = job.resources.map((r) => r.hostname).sort(), | ||||
| @@ -83,6 +86,14 @@ | ||||
|       return s.dir != "up" ? s1[stat] - s2[stat] : s2[stat] - s1[stat]; | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   export function moreLoaded(moreJobMetrics) { | ||||
|     moreJobMetrics.forEach(function (newMetric) { | ||||
|       if (!jobMetrics.some((m) => m.scope == newMetric.scope)) { | ||||
|         jobMetrics = [...jobMetrics, newMetric] | ||||
|       } | ||||
|     }); | ||||
|   }; | ||||
| </script> | ||||
|  | ||||
| <Table class="mb-0"> | ||||
|   | ||||
| @@ -9,6 +9,7 @@ new Jobs({ | ||||
|         roles: roles | ||||
|     }, | ||||
|     context: new Map([ | ||||
|             ['cc-config', clusterCockpitConfig] | ||||
|             ['cc-config', clusterCockpitConfig], | ||||
|             ['resampling', resampleConfig] | ||||
|     ]) | ||||
| }) | ||||
|   | ||||
| @@ -8,6 +8,7 @@ new User({ | ||||
|         user: userInfos | ||||
|     }, | ||||
|     context: new Map([ | ||||
|             ['cc-config', clusterCockpitConfig] | ||||
|             ['cc-config', clusterCockpitConfig], | ||||
|             ['resampling', resampleConfig] | ||||
|     ]) | ||||
| }) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user