mirror of
				https://github.com/ClusterCockpit/cc-backend
				synced 2025-10-25 23:05:07 +02:00 
			
		
		
		
	add resource compare graph, add cursor sync, handle jobIds fitler
This commit is contained in:
		| @@ -15,6 +15,7 @@ | ||||
|  | ||||
| <script> | ||||
|   import { getContext } from "svelte"; | ||||
|   import uPlot from "uplot"; | ||||
|   import { | ||||
|     queryStore, | ||||
|     gql, | ||||
| @@ -40,11 +41,12 @@ | ||||
|   let filter = [...filterBuffer]; | ||||
|   let comparePlotData = {}; | ||||
|   let jobIds = []; | ||||
|   const sorting = { field: "startTime", type: "col", order: "DESC" }; | ||||
|  | ||||
|   /*uPlot*/ | ||||
|   let plotSync = uPlot.sync("compareJobsView"); | ||||
|  | ||||
|   /* GQL */ | ||||
|  | ||||
|   const client = getContextClient(); | ||||
|   const client = getContextClient();   | ||||
|   // Pull All Series For Metrics Statistics Only On Node Scope | ||||
|   const compareQuery = gql` | ||||
|   query ($filter: [JobFilter!]!, $metrics: [String!]!) { | ||||
| @@ -52,6 +54,9 @@ | ||||
|       jobId | ||||
|       startTime | ||||
|       duration | ||||
|       numNodes | ||||
|       numHWThreads | ||||
|       numAccelerators | ||||
|       stats { | ||||
|         name | ||||
|         data { | ||||
| @@ -111,11 +116,13 @@ | ||||
|  | ||||
|   function jobs2uplot(jobs, metrics) { | ||||
|     // Prep | ||||
|     // Resources Init | ||||
|     comparePlotData['resources'] = {unit:'', data: [[],[],[],[],[],[]]} // data: [X, XST, XRT, YNODES, YTHREADS, YACCS] | ||||
|     // Metric Init | ||||
|     for (let m of metrics) { | ||||
|       // Get Unit | ||||
|       const rawUnit = globalMetrics.find((gm) => gm.name == m)?.unit | ||||
|       const metricUnit = (rawUnit?.prefix ? rawUnit.prefix : "") + (rawUnit?.base ? rawUnit.base : "") | ||||
|       // Init | ||||
|       comparePlotData[m] = {unit: metricUnit, data: [[],[],[],[],[],[]]} // data: [X, XST, XRT, YMIN, YAVG, YMAX] | ||||
|     } | ||||
|  | ||||
| @@ -123,7 +130,16 @@ | ||||
|     if (jobs) { | ||||
|         let plotIndex = 0 | ||||
|         jobs.forEach((j) => { | ||||
|             // Collect JobIDs for X-Ticks | ||||
|             jobIds.push(j.jobId) | ||||
|             // Resources | ||||
|             comparePlotData['resources'].data[0].push(plotIndex) | ||||
|             comparePlotData['resources'].data[1].push(j.startTime) | ||||
|             comparePlotData['resources'].data[2].push(j.duration) | ||||
|             comparePlotData['resources'].data[3].push(j.numNodes) | ||||
|             comparePlotData['resources'].data[4].push(j?.numHWThreads?j.numHWThreads:0) | ||||
|             comparePlotData['resources'].data[5].push(j?.numAccelerators?j.numAccelerators:0) | ||||
|             // Metrics | ||||
|             for (let s of j.stats) { | ||||
|               comparePlotData[s.name].data[0].push(plotIndex) | ||||
|               comparePlotData[s.name].data[1].push(j.startTime) | ||||
| @@ -181,16 +197,34 @@ | ||||
|     </Col> | ||||
|   </Row> | ||||
| {:else} | ||||
|   <Row> | ||||
|     <Col> | ||||
|       <Comparogram | ||||
|         title={'Compare Resources'} | ||||
|         xlabel="JobIDs" | ||||
|         xticks={jobIds} | ||||
|         ylabel={'Resource Counts'} | ||||
|         data={comparePlotData['resources'].data} | ||||
|         {plotSync} | ||||
|         forResources | ||||
|       /> | ||||
|     </Col> | ||||
|   </Row> | ||||
|   {#each metrics as m} | ||||
|     <Comparogram | ||||
|       title={'Compare '+ m} | ||||
|       xlabel="JobIds" | ||||
|       xticks={jobIds} | ||||
|       ylabel={m} | ||||
|       metric={m} | ||||
|       yunit={comparePlotData[m].unit} | ||||
|       data={comparePlotData[m].data} | ||||
|     /> | ||||
|     <Row> | ||||
|       <Col> | ||||
|         <Comparogram | ||||
|           title={`Compare Metric '${m}'`} | ||||
|           xlabel="JobIDs" | ||||
|           xticks={jobIds} | ||||
|           ylabel={m} | ||||
|           metric={m} | ||||
|           yunit={comparePlotData[m].unit} | ||||
|           data={comparePlotData[m].data} | ||||
|           {plotSync} | ||||
|         /> | ||||
|       </Col> | ||||
|     </Row> | ||||
|   {/each} | ||||
|   <hr/><hr/> | ||||
|   {#each $compareData.data.jobsMetricStats as job, jindex (job.jobId)} | ||||
|   | ||||
| @@ -14,24 +14,24 @@ | ||||
|  | ||||
| <script> | ||||
|   import uPlot from "uplot"; | ||||
|   import { roundTwoDigits, formatTime } from "../units.js"; | ||||
|   import { roundTwoDigits, formatTime, formatNumber } from "../units.js"; | ||||
|   import { getContext, onMount, onDestroy } from "svelte"; | ||||
|   import { Card } from "@sveltestrap/sveltestrap"; | ||||
|  | ||||
|   export let metric; | ||||
|   export let metric = ""; | ||||
|   export let width = 0; | ||||
|   export let height = 300; | ||||
|   export let data; | ||||
|   export let xlabel; | ||||
|   export let xticks; | ||||
|   export let ylabel; | ||||
|   export let yunit; | ||||
|   export let title; | ||||
|   // export let cluster = ""; | ||||
|   // export let subCluster = ""; | ||||
|   export let data = null; | ||||
|   export let xlabel = ""; | ||||
|   export let xticks = []; | ||||
|   export let ylabel = ""; | ||||
|   export let yunit = ""; | ||||
|   export let title = ""; | ||||
|   export let forResources = false; | ||||
|   export let plotSync; | ||||
|  | ||||
|   // NOTE: Metric Thresholds non-required, Cluster Mixing Allowed | ||||
|  | ||||
|   const metricConfig = null // DEBUG FILLER | ||||
|   // const metricConfig = getContext("getMetricConfig")(cluster, subCluster, metric); // Args woher | ||||
|   const clusterCockpitConfig = getContext("cc-config"); | ||||
|   const lineWidth = clusterCockpitConfig.plot_general_lineWidth / window.devicePixelRatio; | ||||
|   const cbmode = clusterCockpitConfig?.plot_general_colorblindMode || false; | ||||
| @@ -80,9 +80,6 @@ | ||||
|       overEl.addEventListener("mouseleave", () => { | ||||
|         legendEl.style.display = "none"; | ||||
|       }); | ||||
|  | ||||
|       // let tooltip exit plot | ||||
|       // overEl.style.overflow = "visible"; | ||||
|     } | ||||
|  | ||||
|     function update(u) { | ||||
| @@ -100,19 +97,6 @@ | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   let maxY = null; | ||||
|   // TODO: Hilfreich! | ||||
|   // if (metricConfig !== null) { | ||||
|   //   maxY = data[3].reduce( // Data[3] is JobMaxs | ||||
|   //         (max, x) => Math.max(max, x), | ||||
|   //         metricConfig.normal, | ||||
|   //       ) || metricConfig.normal | ||||
|   //   if (maxY >= 10 * metricConfig.peak) { | ||||
|   //     // Hard y-range render limit if outliers in series data | ||||
|   //     maxY = 10 * metricConfig.peak; | ||||
|   //   } | ||||
|   // } | ||||
|  | ||||
|   const plotSeries = [ | ||||
|     { | ||||
|       label: "JobID", | ||||
| @@ -135,34 +119,62 @@ | ||||
|         return formatTime(ts); | ||||
|       }, | ||||
|     }, | ||||
|     { | ||||
|       label: "Min", | ||||
|       scale: "y", | ||||
|       width: lineWidth, | ||||
|       stroke: cbmode ? "rgb(0,255,0)" : "red", | ||||
|       value: (u, ts, sidx, didx) => { | ||||
|         return `${roundTwoDigits(ts)} ${yunit}`; | ||||
|   ] | ||||
|  | ||||
|   if (forResources) { | ||||
|     const resSeries = [ | ||||
|       { | ||||
|         label: "Nodes", | ||||
|         scale: "y", | ||||
|         width: lineWidth, | ||||
|         stroke: "black", | ||||
|       }, | ||||
|     }, | ||||
|     { | ||||
|       label: "Avg", | ||||
|       scale: "y", | ||||
|       width: lineWidth, | ||||
|       stroke: "black", | ||||
|       value: (u, ts, sidx, didx) => { | ||||
|         return `${roundTwoDigits(ts)} ${yunit}`; | ||||
|       { | ||||
|         label: "Threads", | ||||
|         scale: "y", | ||||
|         width: lineWidth, | ||||
|         stroke: "rgb(0,0,255)", | ||||
|       }, | ||||
|     }, | ||||
|     { | ||||
|       label: "Max", | ||||
|       scale: "y", | ||||
|       width: lineWidth, | ||||
|       stroke: cbmode ? "rgb(0,0,255)" : "green", | ||||
|       value: (u, ts, sidx, didx) => { | ||||
|         return `${roundTwoDigits(ts)} ${yunit}`; | ||||
|       { | ||||
|         label: "Accelerators", | ||||
|         scale: "y", | ||||
|         width: lineWidth, | ||||
|         stroke: cbmode ? "rgb(0,255,0)" : "red", | ||||
|       } | ||||
|     ]; | ||||
|     plotSeries.push(...resSeries) | ||||
|   } else { | ||||
|     const statsSeries = [ | ||||
|       { | ||||
|         label: "Min", | ||||
|         scale: "y", | ||||
|         width: lineWidth, | ||||
|         stroke: cbmode ? "rgb(0,255,0)" : "red", | ||||
|         value: (u, ts, sidx, didx) => { | ||||
|           return `${roundTwoDigits(ts)} ${yunit}`; | ||||
|         }, | ||||
|       }, | ||||
|     } | ||||
|   ]; | ||||
|       { | ||||
|         label: "Avg", | ||||
|         scale: "y", | ||||
|         width: lineWidth, | ||||
|         stroke: "black", | ||||
|         value: (u, ts, sidx, didx) => { | ||||
|           return `${roundTwoDigits(ts)} ${yunit}`; | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         label: "Max", | ||||
|         scale: "y", | ||||
|         width: lineWidth, | ||||
|         stroke: cbmode ? "rgb(0,0,255)" : "green", | ||||
|         value: (u, ts, sidx, didx) => { | ||||
|           return `${roundTwoDigits(ts)} ${yunit}`; | ||||
|         }, | ||||
|       } | ||||
|     ]; | ||||
|     plotSeries.push(...statsSeries) | ||||
|   }; | ||||
|  | ||||
|   const plotBands = [ | ||||
|     { series: [5, 4], fill: cbmode ? "rgba(0,0,255,0.1)" : "rgba(0,255,0,0.1)" }, | ||||
| @@ -198,19 +210,20 @@ | ||||
|         scale: "y", | ||||
|         grid: { show: true }, | ||||
|         labelFont: "sans-serif", | ||||
|         label: ylabel + (yunit ? ` (${yunit})` : '') | ||||
|         label: ylabel + (yunit ? ` (${yunit})` : ''), | ||||
|         values: (u, vals) => vals.map((v) => formatNumber(v)), | ||||
|       }, | ||||
|     ], | ||||
|     bands: plotBands, | ||||
|     padding: [5, 10, 0, 0], // 5, 10, -20, 0 | ||||
|     bands: forResources ? [] : plotBands, | ||||
|     padding: [5, 10, 0, 0], | ||||
|     hooks: { | ||||
|       draw: [ | ||||
|         (u) => { | ||||
|           // Draw plot type label: | ||||
|           let textl = "Metric Min/Avg/Max for Job Duration"; | ||||
|           let textl = forResources ? "Job Resources by Type" : "Metric Min/Avg/Max for Job Duration"; | ||||
|           let textr = "Earlier <- StartTime -> Later"; | ||||
|           u.ctx.save(); | ||||
|           u.ctx.textAlign = "start"; // 'end' | ||||
|           u.ctx.textAlign = "start"; | ||||
|           u.ctx.fillStyle = "black"; | ||||
|           u.ctx.fillText(textl, u.bbox.left + 10, u.bbox.top + 10); | ||||
|           u.ctx.textAlign = "end"; | ||||
| @@ -220,24 +233,8 @@ | ||||
|             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 (!metricConfig) { | ||||
|             u.ctx.restore(); | ||||
|             return; | ||||
|           } | ||||
|  | ||||
|           // TODO: Braucht MetricConf | ||||
|           let y = u.valToPos(metricConfig?.normal, "y", true); | ||||
|           u.ctx.save(); | ||||
|           u.ctx.lineWidth = lineWidth; | ||||
|           u.ctx.strokeStyle = "#000000"; // Black | ||||
|           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(); | ||||
|           return; | ||||
|         }, | ||||
|       ] | ||||
|     }, | ||||
| @@ -245,7 +242,7 @@ | ||||
|       x: { time: false }, | ||||
|       xst: { time: false }, | ||||
|       xrt: { time: false }, | ||||
|       y: maxY ? { min: 0, max: (maxY * 1.1) } : {auto: true}, // Add some space to upper render limit | ||||
|       y: {auto: true, distr: forResources ? 3 : 1}, | ||||
|     }, | ||||
|     legend: { | ||||
|       // Display legend | ||||
| @@ -254,6 +251,10 @@ | ||||
|     }, | ||||
|     cursor: {  | ||||
|       drag: { x: true, y: true }, | ||||
|       sync: {  | ||||
|         key: plotSync.key, | ||||
|         scales: ["x", null], | ||||
|       } | ||||
|     } | ||||
|   }; | ||||
|  | ||||
| @@ -267,6 +268,7 @@ | ||||
|       opts.width = ren_width; | ||||
|       opts.height = ren_height; | ||||
|       uplot = new uPlot(opts, data, plotWrapper); // Data is uplot formatted [[X][Ymin][Yavg][Ymax]] | ||||
|       plotSync.sub(uplot) | ||||
|     } else { | ||||
|       uplot.setSize({ width: ren_width, height: ren_height }); | ||||
|     } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user