diff --git a/src/components/connection/connection-item.tsx b/src/components/connection/connection-item.tsx index e6379dfa654a15bb0e77e0454e2802497eb6fa30..3ec7583c7c96afbec0f92b3bbdf5017ff8d339d4 100644 --- a/src/components/connection/connection-item.tsx +++ b/src/components/connection/connection-item.tsx @@ -4,6 +4,7 @@ import { styled, ListItem, IconButton, ListItemText } from "@mui/material"; import { CloseRounded } from "@mui/icons-material"; import { ApiType } from "../../services/types"; import { deleteConnection } from "../../services/api"; +import parseTraffic from "../../utils/parse-traffic"; const Tag = styled("span")(({ theme }) => ({ display: "inline-block", @@ -23,32 +24,39 @@ interface Props { const ConnectionItem = (props: Props) => { const { value } = props; - const onDelete = useLockFn(async () => deleteConnection(value.id)); + const { id, metadata, chains, start, curUpload, curDownload } = value; + + const onDelete = useLockFn(async () => deleteConnection(id)); + const showTraffic = curUpload! > 1024 || curDownload! > 1024; return ( <ListItem dense secondaryAction={ - <IconButton edge="end" onClick={onDelete}> + <IconButton edge="end" color="inherit" onClick={onDelete}> <CloseRounded /> </IconButton> } > <ListItemText - primary={value.metadata.host || value.metadata.destinationIP} + primary={metadata.host || metadata.destinationIP} secondary={ <> <Tag sx={{ textTransform: "uppercase", color: "success" }}> - {value.metadata.network} + {metadata.network} </Tag> - <Tag>{value.metadata.type}</Tag> + <Tag>{metadata.type}</Tag> - {value.chains.length > 0 && ( - <Tag>{value.chains[value.chains.length - 1]}</Tag> - )} + {chains.length > 0 && <Tag>{chains[value.chains.length - 1]}</Tag>} - <Tag>{dayjs(value.start).fromNow()}</Tag> + <Tag>{dayjs(start).fromNow()}</Tag> + + {showTraffic && ( + <Tag> + {parseTraffic(curUpload!)} / {parseTraffic(curDownload!)} + </Tag> + )} </> } /> diff --git a/src/pages/connections.tsx b/src/pages/connections.tsx index f949de76d4a9f332849545a827e04be159761428..1cc6f7693b5396640b2a9a3fb594c5d63e973d02 100644 --- a/src/pages/connections.tsx +++ b/src/pages/connections.tsx @@ -1,6 +1,6 @@ -import { useEffect, useState } from "react"; +import { useEffect, useMemo, useState } from "react"; import { useLockFn } from "ahooks"; -import { Button, Paper } from "@mui/material"; +import { Box, Button, Paper, TextField } from "@mui/material"; import { Virtuoso } from "react-virtuoso"; import { useTranslation } from "react-i18next"; import { ApiType } from "../services/types"; @@ -8,12 +8,20 @@ import { closeAllConnections, getInfomation } from "../services/api"; import BasePage from "../components/base/base-page"; import ConnectionItem from "../components/connection/connection-item"; -const ConnectionsPage = () => { - const initConn = { uploadTotal: 0, downloadTotal: 0, connections: [] }; +const initConn = { uploadTotal: 0, downloadTotal: 0, connections: [] }; +const ConnectionsPage = () => { const { t } = useTranslation(); + + const [filterText, setFilterText] = useState(""); const [connData, setConnData] = useState<ApiType.Connections>(initConn); + const filterConn = useMemo(() => { + return connData.connections.filter((conn) => + (conn.metadata.host || conn.metadata.destinationIP)?.includes(filterText) + ); + }, [connData, filterText]); + useEffect(() => { let ws: WebSocket | null = null; @@ -23,32 +31,35 @@ const ConnectionsPage = () => { ws.addEventListener("message", (event) => { const data = JSON.parse(event.data) as ApiType.Connections; + + // 与å‰ä¸€æ¬¡connections的展示顺åºå°½é‡ä¿æŒä¸€è‡´ setConnData((old) => { const oldConn = old.connections; - const oldList = oldConn.map((each) => each.id); const maxLen = data.connections.length; const connections: typeof oldConn = []; - // 与å‰ä¸€æ¬¡è¿žæŽ¥çš„顺åºå°½é‡ä¿æŒä¸€è‡´ - data.connections - .filter((each) => { - const index = oldList.indexOf(each.id); - - if (index >= 0 && index < maxLen) { - connections[index] = each; - return false; - } - return true; - }) - .forEach((each) => { - for (let i = 0; i < maxLen; ++i) { - if (!connections[i]) { - connections[i] = each; - return; - } - } - }); + const rest = data.connections.filter((each) => { + const index = oldConn.findIndex((o) => o.id === each.id); + + if (index >= 0 && index < maxLen) { + const old = oldConn[index]; + each.curUpload = each.upload - old.upload; + each.curDownload = each.download - old.download; + + connections[index] = each; + return false; + } + return true; + }); + + for (let i = 0; i < maxLen; ++i) { + if (!connections[i] && rest.length > 0) { + connections[i] = rest.shift()!; + connections[i].curUpload = 0; + connections[i].curDownload = 0; + } + } return { ...data, connections }; }); @@ -76,11 +87,48 @@ const ConnectionsPage = () => { } > <Paper sx={{ boxShadow: 2, height: "100%" }}> - <Virtuoso - initialTopMostItemIndex={999} - data={connData.connections} - itemContent={(index, item) => <ConnectionItem value={item} />} - /> + <Box + sx={{ + pt: 1, + mb: 0.5, + mx: "12px", + height: "36px", + display: "flex", + alignItems: "center", + }} + > + {/* <Select + size="small" + autoComplete="off" + value={logState} + onChange={(e) => setLogState(e.target.value)} + sx={{ width: 120, mr: 1, '[role="button"]': { py: 0.65 } }} + > + <MenuItem value="all">ALL</MenuItem> + <MenuItem value="info">INFO</MenuItem> + <MenuItem value="warn">WARN</MenuItem> + </Select> */} + + <TextField + hiddenLabel + fullWidth + size="small" + autoComplete="off" + variant="outlined" + placeholder="Filter conditions" + value={filterText} + onChange={(e) => setFilterText(e.target.value)} + sx={{ input: { py: 0.65, px: 1.25 } }} + /> + </Box> + + <Box height="calc(100% - 50px)"> + <Virtuoso + initialTopMostItemIndex={999} + data={filterConn} + itemContent={(index, item) => <ConnectionItem value={item} />} + /> + </Box> </Paper> </BasePage> ); diff --git a/src/services/types.ts b/src/services/types.ts index 50b81fc3d21443e547fa765345937bdbca2acc43..4858679361875c066c9281e54f7e5dcb4d30de85 100644 --- a/src/services/types.ts +++ b/src/services/types.ts @@ -65,6 +65,8 @@ export namespace ApiType { chains: string[]; rule: string; rulePayload: string; + curUpload?: number; // calculate + curDownload?: number; // calculate } export interface Connections {