Skip to content

Commit 95e1919

Browse files
committed
feat: new console-based dashboard
`codeflare dashboard` command ```shell codeflare dashboard all|cpu|memory|status ``` this currently assumes that the aggregator is already running against the job no gpu metrics, yet This also adds some helpers: ``` codeflare dump logs|cpu|gpu|memory|status ```
1 parent f89e4e2 commit 95e1919

29 files changed

+2816
-1410
lines changed

package-lock.json

+1,106-1,390
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

+1
Original file line numberDiff line numberDiff line change
@@ -156,6 +156,7 @@
156156
"@kui-shell/plugin-bash-like": "13.1.3-dev-20230329-123301",
157157
"@kui-shell/plugin-client-common": "13.1.3-dev-20230329-123301",
158158
"@kui-shell/plugin-codeflare": "file:./plugins/plugin-codeflare",
159+
"@kui-shell/plugin-codeflare-dashboard": "file:./plugins/plugin-codeflare-dashboard",
159160
"@kui-shell/plugin-core-support": "13.1.3-dev-20230329-123301",
160161
"@kui-shell/plugin-electron-components": "13.1.3-dev-20230329-123301",
161162
"@kui-shell/plugin-kubectl": "13.1.3-dev-20230329-123301",
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
dist
2+
mdist
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
build
2+
src
3+
tests
4+
dist/test
5+
tsconfig*
6+
/*.tgz
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
{
2+
"name": "@kui-shell/plugin-codeflare-dashboard",
3+
"version": "0.0.1",
4+
"description": "ML dashboarding",
5+
"types": "./mdist/index.d.ts",
6+
"module": "./mdist/index.js",
7+
"main": "./mdist/index.js",
8+
"license": "Apache-2.0",
9+
"homepage": "https://github.com/IBM/kui#readme",
10+
"bugs": {
11+
"url": "https://github.com/IBM/kui/issues/new"
12+
},
13+
"repository": {
14+
"type": "git",
15+
"url": "git+https://github.com/IBM/kui.git"
16+
},
17+
"keywords": [
18+
"kui",
19+
"plugin"
20+
],
21+
"private": true,
22+
"publishConfig": {
23+
"access": "public"
24+
},
25+
"devDependencies": {
26+
"@types/split2": "^3.2.1",
27+
"react-devtools-core": "^4.27.2"
28+
},
29+
"dependencies": {
30+
"@logdna/tail-file": "^3.0.1",
31+
"chokidar": "^3.5.3",
32+
"ink": "^3.2.0",
33+
"madwizard": "^7.0.3",
34+
"pretty-ms": "^7.0.1",
35+
"strip-ansi": "6.0.0"
36+
}
37+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,133 @@
1+
/*
2+
* Copyright 2023 The Kubernetes Authors
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* http://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
17+
import React from "react"
18+
import { Box, Text, TextProps } from "ink"
19+
import type { Arguments } from "@kui-shell/core"
20+
21+
type WorkerState = "Pending" | "Scheduled" | "Installing" | "Running" | "Failed" | "Success"
22+
23+
type Props = unknown
24+
type State = {
25+
workers: WorkerState[]
26+
}
27+
28+
class Dashboard extends React.PureComponent<Props, State> {
29+
public constructor(props: Props) {
30+
super(props)
31+
32+
const styles = Object.keys(this.styleOf) as WorkerState[]
33+
const randoState = () => styles[Math.round(Math.random() * styles.length) % styles.length]
34+
this.state = {
35+
workers: Array(50).fill(1).map(randoState),
36+
}
37+
38+
setInterval(() => {
39+
this.setState((curState) => {
40+
const idx = Math.round(Math.random() * this.state.workers.length) % this.state.workers.length
41+
const styles = Object.keys(this.styleOf)
42+
const newState = styles[Math.round(Math.random() * styles.length) % styles.length]
43+
return {
44+
workers: [...curState.workers.slice(0, idx), newState as WorkerState, ...curState.workers.slice(idx + 1)],
45+
}
46+
})
47+
}, 1000)
48+
}
49+
50+
// private readonly sizes = ["▁▁", "▃▃", "▅▅", "▆▆", "██", "■■"]
51+
// private readonly sizes = "■".repeat(5)
52+
private readonly sizes = Array(5).fill("▇▇")
53+
// private readonly sizes = "█".repeat(4)
54+
55+
private readonly styleOf: Record<WorkerState, TextProps> = {
56+
Pending: { color: "gray", children: this.sizes[3] },
57+
Scheduled: { color: "white", dimColor: true, children: this.sizes[3] },
58+
// Pulling: { color: "yellow", dimColor: true, children: this.sizes[3] },
59+
Installing: { color: "yellow", children: this.sizes[3] },
60+
Running: { color: "cyan", children: this.sizes[4] },
61+
Failed: { color: "red", children: this.sizes[4] },
62+
Success: { color: "blue", children: this.sizes[4] },
63+
}
64+
65+
private matrix(): WorkerState[][] {
66+
const N = Math.ceil(Math.sqrt(this.state.workers.length))
67+
const matrix = Array(N)
68+
for (let i = 0; i < N; i++) {
69+
matrix[i] = Array(N)
70+
for (let j = 0; j < N; j++) {
71+
matrix[i][j] = this.state.workers[i * N + j]
72+
}
73+
}
74+
return matrix
75+
}
76+
77+
private histo(): number[] {
78+
const keys = Object.keys(this.styleOf)
79+
const indexer = keys.reduce((M, worker, idx) => {
80+
M[worker] = idx
81+
return M
82+
}, {} as Record<string, number>)
83+
84+
return this.state.workers.reduce((H, worker) => {
85+
H[indexer[worker]]++
86+
return H
87+
}, Array(keys.length).fill(0))
88+
}
89+
90+
public render() {
91+
const M = this.matrix()
92+
const H = this.histo()
93+
94+
return (
95+
<Box flexDirection="column" margin={1}>
96+
<Box>
97+
{Object.keys(this.styleOf).map((_, idx) => (
98+
<Box width="20%" borderStyle="singleDouble" marginRight={1} key={_}>
99+
<Box flexDirection="column">
100+
<Box>
101+
<Text color={this.styleOf[_ as WorkerState].color} bold inverse>
102+
{_}
103+
</Text>
104+
</Box>
105+
<Box>
106+
<Text>{H[idx]}</Text>
107+
</Box>
108+
</Box>
109+
</Box>
110+
))}
111+
</Box>
112+
113+
<Box marginTop={1} width={M.reduce((N, c) => (N += c.length), 0)} flexDirection="column">
114+
{M.map((row, ridx) => (
115+
<Box key={ridx}>
116+
{row.map((worker, cidx) => (
117+
<Text key={cidx} {...this.styleOf[worker]} />
118+
))}
119+
</Box>
120+
))}
121+
</Box>
122+
</Box>
123+
)
124+
}
125+
}
126+
127+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
128+
export default async function dashboard(args: Arguments) {
129+
const { render } = await import("ink")
130+
await render(<Dashboard />)
131+
await new Promise(() => {}) // eslint-disable-line @typescript-eslint/no-empty-function
132+
return true
133+
}

0 commit comments

Comments
 (0)