2024-07-31 23:13:36 +02:00
|
|
|
// Copyright 2024, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
|
|
|
import { useHeight } from "@/app/hook/useHeight";
|
|
|
|
import { useWidth } from "@/app/hook/useWidth";
|
2024-09-05 09:21:08 +02:00
|
|
|
import { getConnStatusAtom, globalStore, waveEventSubscribe, WOS } from "@/store/global";
|
2024-07-31 23:13:36 +02:00
|
|
|
import { WshServer } from "@/store/wshserver";
|
2024-08-30 20:33:04 +02:00
|
|
|
import * as util from "@/util/util";
|
2024-07-31 23:13:36 +02:00
|
|
|
import * as Plot from "@observablehq/plot";
|
|
|
|
import dayjs from "dayjs";
|
|
|
|
import * as htl from "htl";
|
|
|
|
import * as jotai from "jotai";
|
|
|
|
import * as React from "react";
|
|
|
|
|
|
|
|
import "./cpuplot.less";
|
|
|
|
|
2024-08-30 20:33:04 +02:00
|
|
|
const DefaultNumPoints = 120;
|
|
|
|
|
|
|
|
type DataItem = {
|
|
|
|
ts: number;
|
|
|
|
[k: string]: number;
|
|
|
|
};
|
|
|
|
|
|
|
|
const SysInfoMetricNames = {
|
|
|
|
cpu: "CPU %",
|
|
|
|
"mem:total": "Memory Total",
|
|
|
|
"mem:used": "Memory Used",
|
|
|
|
"mem:free": "Memory Free",
|
|
|
|
"mem:available": "Memory Available",
|
2024-07-31 23:13:36 +02:00
|
|
|
};
|
2024-08-30 20:33:04 +02:00
|
|
|
for (let i = 0; i < 32; i++) {
|
|
|
|
SysInfoMetricNames[`cpu:${i}`] = `CPU[${i}] %`;
|
|
|
|
}
|
|
|
|
|
|
|
|
function convertWaveEventToDataItem(event: WaveEvent): DataItem {
|
|
|
|
const eventData: TimeSeriesData = event.data;
|
|
|
|
if (eventData == null || eventData.ts == null || eventData.values == null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
const dataItem = { ts: eventData.ts };
|
|
|
|
for (const key in eventData.values) {
|
|
|
|
dataItem[key] = eventData.values[key];
|
|
|
|
}
|
|
|
|
return dataItem;
|
|
|
|
}
|
2024-07-31 23:13:36 +02:00
|
|
|
|
|
|
|
class CpuPlotViewModel {
|
2024-08-23 01:25:53 +02:00
|
|
|
viewType: string;
|
2024-07-31 23:13:36 +02:00
|
|
|
blockAtom: jotai.Atom<Block>;
|
|
|
|
termMode: jotai.Atom<string>;
|
|
|
|
htmlElemFocusRef: React.RefObject<HTMLInputElement>;
|
|
|
|
blockId: string;
|
|
|
|
viewIcon: jotai.Atom<string>;
|
|
|
|
viewText: jotai.Atom<string>;
|
|
|
|
viewName: jotai.Atom<string>;
|
2024-08-30 20:33:04 +02:00
|
|
|
dataAtom: jotai.PrimitiveAtom<Array<DataItem>>;
|
|
|
|
addDataAtom: jotai.WritableAtom<unknown, [DataItem[]], void>;
|
2024-08-01 09:57:06 +02:00
|
|
|
incrementCount: jotai.WritableAtom<unknown, [], Promise<void>>;
|
2024-08-30 20:33:04 +02:00
|
|
|
loadingAtom: jotai.PrimitiveAtom<boolean>;
|
|
|
|
numPoints: jotai.Atom<number>;
|
|
|
|
metrics: jotai.Atom<string[]>;
|
|
|
|
connection: jotai.Atom<string>;
|
|
|
|
manageConnection: jotai.Atom<boolean>;
|
2024-09-05 09:21:08 +02:00
|
|
|
connStatus: jotai.Atom<ConnStatus>;
|
2024-07-31 23:13:36 +02:00
|
|
|
|
|
|
|
constructor(blockId: string) {
|
2024-08-23 01:25:53 +02:00
|
|
|
this.viewType = "cpuplot";
|
2024-07-31 23:13:36 +02:00
|
|
|
this.blockId = blockId;
|
2024-08-01 09:57:06 +02:00
|
|
|
this.blockAtom = WOS.getWaveObjectAtom<Block>(`block:${blockId}`);
|
2024-08-30 20:33:04 +02:00
|
|
|
this.addDataAtom = jotai.atom(null, (get, set, points) => {
|
|
|
|
const targetLen = get(this.numPoints) + 1;
|
|
|
|
let data = get(this.dataAtom);
|
|
|
|
try {
|
|
|
|
if (data.length > targetLen) {
|
|
|
|
data = data.slice(data.length - targetLen);
|
|
|
|
}
|
|
|
|
if (data.length < targetLen) {
|
|
|
|
const defaultData = this.getDefaultData();
|
|
|
|
data = [...defaultData.slice(defaultData.length - targetLen + data.length), ...data];
|
|
|
|
}
|
|
|
|
const newData = [...data.slice(points.length), ...points];
|
|
|
|
set(this.dataAtom, newData);
|
|
|
|
} catch (e) {
|
|
|
|
console.log("Error adding data to cpuplot", e);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
this.manageConnection = jotai.atom(true);
|
|
|
|
this.loadingAtom = jotai.atom(true);
|
|
|
|
this.numPoints = jotai.atom((get) => {
|
|
|
|
const blockData = get(this.blockAtom);
|
|
|
|
const metaNumPoints = blockData?.meta?.["graph:numpoints"];
|
|
|
|
if (metaNumPoints == null || metaNumPoints <= 0) {
|
|
|
|
return DefaultNumPoints;
|
|
|
|
}
|
|
|
|
return metaNumPoints;
|
|
|
|
});
|
|
|
|
this.metrics = jotai.atom((get) => {
|
|
|
|
const blockData = get(this.blockAtom);
|
|
|
|
const metrics = blockData?.meta?.["graph:metrics"];
|
|
|
|
if (metrics == null || !Array.isArray(metrics)) {
|
|
|
|
return ["cpu"];
|
|
|
|
}
|
|
|
|
return metrics;
|
2024-07-31 23:13:36 +02:00
|
|
|
});
|
|
|
|
this.viewIcon = jotai.atom((get) => {
|
|
|
|
return "chart-line"; // should not be hardcoded
|
|
|
|
});
|
|
|
|
this.viewName = jotai.atom((get) => {
|
|
|
|
return "CPU %"; // should not be hardcoded
|
|
|
|
});
|
2024-08-01 09:57:06 +02:00
|
|
|
this.incrementCount = jotai.atom(null, async (get, set) => {
|
|
|
|
const meta = get(this.blockAtom).meta;
|
|
|
|
const count = meta.count ?? 0;
|
|
|
|
await WshServer.SetMetaCommand({ oref: WOS.makeORef("block", this.blockId), meta: { count: count + 1 } });
|
|
|
|
});
|
2024-08-30 20:33:04 +02:00
|
|
|
this.connection = jotai.atom((get) => {
|
|
|
|
const blockData = get(this.blockAtom);
|
|
|
|
const connValue = blockData?.meta?.connection;
|
|
|
|
if (util.isBlank(connValue)) {
|
|
|
|
return "local";
|
|
|
|
}
|
|
|
|
return connValue;
|
|
|
|
});
|
|
|
|
this.dataAtom = jotai.atom(this.getDefaultData());
|
|
|
|
this.loadInitialData();
|
2024-09-05 09:21:08 +02:00
|
|
|
this.connStatus = jotai.atom((get) => {
|
|
|
|
const blockData = get(this.blockAtom);
|
|
|
|
const connName = blockData?.meta?.connection;
|
|
|
|
const connAtom = getConnStatusAtom(connName);
|
|
|
|
return get(connAtom);
|
|
|
|
});
|
2024-08-30 20:33:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async loadInitialData() {
|
|
|
|
globalStore.set(this.loadingAtom, true);
|
|
|
|
try {
|
|
|
|
const numPoints = globalStore.get(this.numPoints);
|
|
|
|
const connName = globalStore.get(this.connection);
|
|
|
|
const initialData = await WshServer.EventReadHistoryCommand({
|
|
|
|
event: "sysinfo",
|
|
|
|
scope: connName,
|
|
|
|
maxitems: numPoints,
|
|
|
|
});
|
|
|
|
if (initialData == null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const initialDataItems: DataItem[] = initialData.map(convertWaveEventToDataItem);
|
|
|
|
globalStore.set(this.addDataAtom, initialDataItems);
|
|
|
|
} catch (e) {
|
|
|
|
console.log("Error loading initial data for cpuplot", e);
|
|
|
|
} finally {
|
|
|
|
globalStore.set(this.loadingAtom, false);
|
|
|
|
}
|
2024-07-31 23:13:36 +02:00
|
|
|
}
|
|
|
|
|
2024-08-30 20:33:04 +02:00
|
|
|
getDefaultData(): Array<DataItem> {
|
2024-07-31 23:13:36 +02:00
|
|
|
// set it back one to avoid backwards line being possible
|
2024-08-30 20:33:04 +02:00
|
|
|
const numPoints = globalStore.get(this.numPoints);
|
|
|
|
const currentTime = Date.now() - 1000;
|
|
|
|
const points: DataItem[] = [];
|
|
|
|
for (let i = numPoints; i > -1; i--) {
|
|
|
|
points.push({ ts: currentTime - i * 1000 });
|
2024-07-31 23:13:36 +02:00
|
|
|
}
|
|
|
|
return points;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function makeCpuPlotViewModel(blockId: string): CpuPlotViewModel {
|
|
|
|
const cpuPlotViewModel = new CpuPlotViewModel(blockId);
|
|
|
|
return cpuPlotViewModel;
|
|
|
|
}
|
|
|
|
|
2024-08-30 20:33:04 +02:00
|
|
|
const plotColors = ["#58C142", "#FFC107", "#FF5722", "#2196F3", "#9C27B0", "#00BCD4", "#FFEB3B", "#795548"];
|
|
|
|
|
2024-09-05 09:21:08 +02:00
|
|
|
type CpuPlotViewProps = {
|
|
|
|
blockId: string;
|
|
|
|
model: CpuPlotViewModel;
|
|
|
|
};
|
|
|
|
|
|
|
|
function CpuPlotView({ model, blockId }: CpuPlotViewProps) {
|
2024-08-30 20:33:04 +02:00
|
|
|
const connName = jotai.useAtomValue(model.connection);
|
|
|
|
const lastConnName = React.useRef(connName);
|
2024-09-05 09:21:08 +02:00
|
|
|
const connStatus = jotai.useAtomValue(model.connStatus);
|
|
|
|
const addPlotData = jotai.useSetAtom(model.addDataAtom);
|
|
|
|
const loading = jotai.useAtomValue(model.loadingAtom);
|
2024-07-31 23:13:36 +02:00
|
|
|
|
|
|
|
React.useEffect(() => {
|
2024-09-05 09:21:08 +02:00
|
|
|
if (connStatus?.status != "connected") {
|
|
|
|
return;
|
|
|
|
}
|
2024-08-30 20:33:04 +02:00
|
|
|
if (lastConnName.current !== connName) {
|
2024-09-05 09:21:08 +02:00
|
|
|
lastConnName.current = connName;
|
2024-08-30 20:33:04 +02:00
|
|
|
model.loadInitialData();
|
|
|
|
}
|
|
|
|
const unsubFn = waveEventSubscribe("sysinfo", connName, (event: WaveEvent) => {
|
|
|
|
const loading = globalStore.get(model.loadingAtom);
|
|
|
|
if (loading) {
|
|
|
|
return;
|
2024-07-31 23:13:36 +02:00
|
|
|
}
|
2024-08-30 20:33:04 +02:00
|
|
|
const dataItem = convertWaveEventToDataItem(event);
|
|
|
|
addPlotData([dataItem]);
|
|
|
|
});
|
|
|
|
return () => {
|
|
|
|
unsubFn();
|
2024-07-31 23:13:36 +02:00
|
|
|
};
|
2024-08-30 20:33:04 +02:00
|
|
|
}, [connName]);
|
2024-09-05 09:21:08 +02:00
|
|
|
React.useEffect(() => {}, [connName]);
|
|
|
|
if (connStatus?.status != "connected") {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
if (loading) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return <CpuPlotViewInner key={connStatus?.connection ?? "local"} blockId={blockId} model={model} />;
|
|
|
|
}
|
|
|
|
|
|
|
|
const CpuPlotViewInner = React.memo(({ model }: CpuPlotViewProps) => {
|
|
|
|
const containerRef = React.useRef<HTMLInputElement>();
|
|
|
|
const plotData = jotai.useAtomValue(model.dataAtom);
|
|
|
|
const parentHeight = useHeight(containerRef);
|
|
|
|
const parentWidth = useWidth(containerRef);
|
|
|
|
const yvals = jotai.useAtomValue(model.metrics);
|
2024-07-31 23:13:36 +02:00
|
|
|
|
|
|
|
React.useEffect(() => {
|
2024-08-30 20:33:04 +02:00
|
|
|
const marks: Plot.Markish[] = [];
|
|
|
|
marks.push(
|
|
|
|
() => htl.svg`<defs>
|
2024-07-31 23:13:36 +02:00
|
|
|
<linearGradient id="gradient" gradientTransform="rotate(90)">
|
|
|
|
<stop offset="0%" stop-color="#58C142" stop-opacity="0.7" />
|
|
|
|
<stop offset="100%" stop-color="#58C142" stop-opacity="0" />
|
|
|
|
</linearGradient>
|
2024-08-30 20:33:04 +02:00
|
|
|
</defs>`
|
|
|
|
);
|
|
|
|
if (yvals.length == 0) {
|
|
|
|
// nothing
|
|
|
|
} else if (yvals.length == 1) {
|
|
|
|
marks.push(
|
2024-07-31 23:13:36 +02:00
|
|
|
Plot.lineY(plotData, {
|
2024-08-30 20:33:04 +02:00
|
|
|
stroke: plotColors[0],
|
2024-07-31 23:13:36 +02:00
|
|
|
strokeWidth: 2,
|
2024-08-30 20:33:04 +02:00
|
|
|
x: "ts",
|
|
|
|
y: yvals[0],
|
|
|
|
})
|
|
|
|
);
|
|
|
|
marks.push(
|
2024-07-31 23:13:36 +02:00
|
|
|
Plot.areaY(plotData, {
|
|
|
|
fill: "url(#gradient)",
|
2024-08-30 20:33:04 +02:00
|
|
|
x: "ts",
|
|
|
|
y: yvals[0],
|
|
|
|
})
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
let idx = 0;
|
|
|
|
for (const yval of yvals) {
|
|
|
|
marks.push(
|
|
|
|
Plot.lineY(plotData, {
|
|
|
|
stroke: plotColors[idx % plotColors.length],
|
|
|
|
strokeWidth: 1,
|
|
|
|
x: "ts",
|
|
|
|
y: yval,
|
|
|
|
})
|
|
|
|
);
|
|
|
|
idx++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const plot = Plot.plot({
|
|
|
|
x: { grid: true, label: "time", tickFormat: (d) => `${dayjs.unix(d / 1000).format("HH:mm:ss")}` },
|
|
|
|
y: { label: "%", domain: [0, 100] },
|
|
|
|
width: parentWidth,
|
|
|
|
height: parentHeight,
|
|
|
|
marks: marks,
|
2024-07-31 23:13:36 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
if (plot !== undefined) {
|
|
|
|
containerRef.current.append(plot);
|
|
|
|
}
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
if (plot !== undefined) {
|
|
|
|
plot.remove();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}, [plotData, parentHeight, parentWidth]);
|
|
|
|
|
|
|
|
return <div className="plot-view" ref={containerRef} />;
|
2024-09-05 09:21:08 +02:00
|
|
|
});
|
2024-07-31 23:13:36 +02:00
|
|
|
|
|
|
|
export { CpuPlotView, CpuPlotViewModel, makeCpuPlotViewModel };
|