Create servers view

This commit is contained in:
Ken Hibino 2020-12-29 06:40:11 -08:00
parent db0749892b
commit 1f3897c570
5 changed files with 205 additions and 1 deletions

View File

@ -20,6 +20,7 @@ import MenuIcon from "@material-ui/icons/Menu";
import BarChartIcon from "@material-ui/icons/BarChart";
import LayersIcon from "@material-ui/icons/Layers";
import SettingsIcon from "@material-ui/icons/Settings";
import ScheduleIcon from "@material-ui/icons/Schedule";
import FeedbackIcon from "@material-ui/icons/Feedback";
import CloseIcon from "@material-ui/icons/Close";
import { AppState } from "./store";
@ -30,6 +31,7 @@ import SchedulersView from "./views/SchedulersView";
import DashboardView from "./views/DashboardView";
import TasksView from "./views/TasksView";
import SettingsView from "./views/SettingsView";
import ServersView from "./views/ServersView";
const drawerWidth = 220;
@ -213,10 +215,15 @@ function App(props: ConnectedProps<typeof connector>) {
primary="Queues"
icon={<BarChartIcon />}
/>
<ListItemLink
to={paths.SERVERS}
primary="Servers"
icon={<LayersIcon />}
/>
<ListItemLink
to={paths.SCHEDULERS}
primary="Schedulers"
icon={<LayersIcon />}
icon={<ScheduleIcon />}
/>
</div>
</List>
@ -250,6 +257,9 @@ function App(props: ConnectedProps<typeof connector>) {
<Route exact path={paths.SCHEDULERS}>
<SchedulersView />
</Route>
<Route exact path={paths.SERVERS}>
<ServersView />
</Route>
<Route exact path={paths.SETTINGS}>
<SettingsView />
</Route>

View File

@ -128,6 +128,10 @@ export interface DeadTask extends BaseTask {
error_message: string;
}
export interface ServerInfo {
// TODO: fill this out
}
export interface SchedulerEntry {
id: string;
spec: string;

View File

@ -0,0 +1,147 @@
import React, { useState } from "react";
import clsx from "clsx";
import { makeStyles } from "@material-ui/core/styles";
import IconButton from "@material-ui/core/IconButton";
import Table from "@material-ui/core/Table";
import TableBody from "@material-ui/core/TableBody";
import TableCell from "@material-ui/core/TableCell";
import TableContainer from "@material-ui/core/TableContainer";
import TableHead from "@material-ui/core/TableHead";
import TableRow from "@material-ui/core/TableRow";
import TableSortLabel from "@material-ui/core/TableSortLabel";
import Alert from "@material-ui/lab/Alert";
import AlertTitle from "@material-ui/lab/AlertTitle";
import { ServerInfo } from "../api";
import { SortDirection, SortableTableColumn } from "../types/table";
const useStyles = makeStyles((theme) => ({
table: {
minWidth: 650,
},
fixedCell: {
position: "sticky",
zIndex: 1,
left: 0,
background: theme.palette.common.white,
},
}));
enum SortBy {
Host,
PID,
Status,
ActiveWorkers,
Queues,
Started,
}
const colConfigs: SortableTableColumn<SortBy>[] = [
{
label: "Host",
key: "host",
sortBy: SortBy.Host,
align: "left",
},
{
label: "PID",
key: "pid",
sortBy: SortBy.PID,
align: "left",
},
{
label: "Status",
key: "status",
sortBy: SortBy.Status,
align: "left",
},
{
label: "Active Workers",
key: "workers",
sortBy: SortBy.ActiveWorkers,
align: "left",
},
{
label: "Queues",
key: "queues",
sortBy: SortBy.Queues,
align: "left",
},
{
label: "Started",
key: "started",
sortBy: SortBy.Started,
align: "left",
},
];
// sortServers takes a array of server-infos and return a sorted array.
// It returns a new array and leave the original array untouched.
function sortServerInfos(
entries: ServerInfo[],
cmpFn: (first: ServerInfo, second: ServerInfo) => number
): ServerInfo[] {
let copy = [...entries];
copy.sort(cmpFn);
return copy;
}
interface Props {
servers: ServerInfo[];
}
export default function ServersTable(props: Props) {
const classes = useStyles();
const [sortBy, setSortBy] = useState<SortBy>(SortBy.Host);
const [sortDir, setSortDir] = useState<SortDirection>(SortDirection.Asc);
const createSortClickHandler = (sortKey: SortBy) => (e: React.MouseEvent) => {
if (sortKey === sortBy) {
// Toggle sort direction.
const nextSortDir =
sortDir === SortDirection.Asc ? SortDirection.Desc : SortDirection.Asc;
setSortDir(nextSortDir);
} else {
// Change the sort key.
setSortBy(sortKey);
}
};
const cmpFunc = (s1: ServerInfo, s2: ServerInfo): number => {
return 0; // TODO: implement this
};
if (props.servers.length === 0) {
return (
<Alert severity="info">
<AlertTitle>Info</AlertTitle>
No servers found at this time.
</Alert>
);
}
return (
<TableContainer>
<Table className={classes.table} aria-label="server info table">
<TableHead>
<TableRow>
{colConfigs.map((cfg, i) => (
<TableCell
key={cfg.key}
align={cfg.align}
className={clsx(i === 0 && classes.fixedCell)}
>
<TableSortLabel
active={cfg.sortBy === sortBy}
direction={sortDir}
onClick={createSortClickHandler(cfg.sortBy)}
>
{cfg.label}
</TableSortLabel>
</TableCell>
))}
</TableRow>
</TableHead>
<TableBody></TableBody>
</Table>
</TableContainer>
);
}

View File

@ -1,6 +1,7 @@
export const paths = {
HOME: "/",
SETTINGS: "/settings",
SERVERS: "/servers",
SCHEDULERS: "/schedulers",
QUEUE_DETAILS: "/queues/:qname",
};

View File

@ -0,0 +1,42 @@
import React from "react";
import Container from "@material-ui/core/Container";
import { makeStyles } from "@material-ui/core/styles";
import Grid from "@material-ui/core/Grid";
import Paper from "@material-ui/core/Paper";
import Typography from "@material-ui/core/Typography";
import ServersTable from "../components/ServersTable";
const useStyles = makeStyles((theme) => ({
container: {
paddingTop: theme.spacing(4),
paddingBottom: theme.spacing(4),
},
paper: {
padding: theme.spacing(2),
display: "flex",
overflow: "auto",
flexDirection: "column",
},
heading: {
paddingLeft: theme.spacing(2),
marginBottom: theme.spacing(1),
},
}));
export default function ServersView() {
const classes = useStyles();
return (
<Container maxWidth="lg" className={classes.container}>
<Grid container spacing={3}>
<Grid item xs={12}>
<Paper className={classes.paper} variant="outlined">
<Typography variant="h6" className={classes.heading}>
Servers
</Typography>
<ServersTable servers={[]} />
</Paper>
</Grid>
</Grid>
</Container>
);
}