mirror of
				https://github.com/hibiken/asynqmon.git
				synced 2025-10-26 16:26:12 +08:00 
			
		
		
		
	Add completed state
This commit is contained in:
		| @@ -37,7 +37,7 @@ import { taskRowsPerPageChange } from "../actions/settingsActions"; | ||||
| import TableActions from "./TableActions"; | ||||
| import { timeAgo, uuidPrefix } from "../utils"; | ||||
| import { usePolling } from "../hooks"; | ||||
| import { ArchivedTaskExtended } from "../reducers/tasksReducer"; | ||||
| import { TaskInfoExtended } from "../reducers/tasksReducer"; | ||||
| import { TableColumn } from "../types/table"; | ||||
| import { taskDetailsPath } from "../paths"; | ||||
|  | ||||
| @@ -311,7 +311,7 @@ const useRowStyles = makeStyles((theme) => ({ | ||||
| })); | ||||
|  | ||||
| interface RowProps { | ||||
|   task: ArchivedTaskExtended; | ||||
|   task: TaskInfoExtended; | ||||
|   isSelected: boolean; | ||||
|   onSelectChange: (checked: boolean) => void; | ||||
|   onRunClick: () => void; | ||||
|   | ||||
							
								
								
									
										376
									
								
								ui/src/components/CompletedTasksTable.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										376
									
								
								ui/src/components/CompletedTasksTable.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,376 @@ | ||||
| import React, { useCallback, useState } from "react"; | ||||
| import { useHistory } from "react-router-dom"; | ||||
| import { connect, ConnectedProps } from "react-redux"; | ||||
| import { makeStyles } from "@material-ui/core/styles"; | ||||
| import Table from "@material-ui/core/Table"; | ||||
| import TableBody from "@material-ui/core/TableBody"; | ||||
| import TableCell from "@material-ui/core/TableCell"; | ||||
| import Checkbox from "@material-ui/core/Checkbox"; | ||||
| import TableContainer from "@material-ui/core/TableContainer"; | ||||
| import TableHead from "@material-ui/core/TableHead"; | ||||
| import TableRow from "@material-ui/core/TableRow"; | ||||
| import Tooltip from "@material-ui/core/Tooltip"; | ||||
| import Paper from "@material-ui/core/Paper"; | ||||
| import IconButton from "@material-ui/core/IconButton"; | ||||
| import DeleteIcon from "@material-ui/icons/Delete"; | ||||
| import MoreHorizIcon from "@material-ui/icons/MoreHoriz"; | ||||
| import TableFooter from "@material-ui/core/TableFooter"; | ||||
| import TablePagination from "@material-ui/core/TablePagination"; | ||||
| import Alert from "@material-ui/lab/Alert"; | ||||
| import AlertTitle from "@material-ui/lab/AlertTitle"; | ||||
| import SyntaxHighlighter from "./SyntaxHighlighter"; | ||||
| import { AppState } from "../store"; | ||||
| import { | ||||
|   listCompletedTasksAsync, | ||||
|   deleteAllCompletedTasksAsync, | ||||
|   deleteCompletedTaskAsync, | ||||
|   batchDeleteCompletedTasksAsync, | ||||
| } from "../actions/tasksActions"; | ||||
| import TablePaginationActions, { | ||||
|   rowsPerPageOptions, | ||||
| } from "./TablePaginationActions"; | ||||
| import { taskRowsPerPageChange } from "../actions/settingsActions"; | ||||
| import TableActions from "./TableActions"; | ||||
| import { | ||||
|   durationFromSeconds, | ||||
|   stringifyDuration, | ||||
|   timeAgo, | ||||
|   uuidPrefix, | ||||
| } from "../utils"; | ||||
| import { usePolling } from "../hooks"; | ||||
| import { TaskInfoExtended } from "../reducers/tasksReducer"; | ||||
| import { TableColumn } from "../types/table"; | ||||
| import { taskDetailsPath } from "../paths"; | ||||
|  | ||||
| const useStyles = makeStyles((theme) => ({ | ||||
|   table: { | ||||
|     minWidth: 650, | ||||
|   }, | ||||
|   stickyHeaderCell: { | ||||
|     background: theme.palette.background.paper, | ||||
|   }, | ||||
|   alert: { | ||||
|     borderTopLeftRadius: 0, | ||||
|     borderTopRightRadius: 0, | ||||
|   }, | ||||
|   pagination: { | ||||
|     border: "none", | ||||
|   }, | ||||
| })); | ||||
|  | ||||
| function mapStateToProps(state: AppState) { | ||||
|   return { | ||||
|     loading: state.tasks.completedTasks.loading, | ||||
|     error: state.tasks.completedTasks.error, | ||||
|     tasks: state.tasks.completedTasks.data, | ||||
|     batchActionPending: state.tasks.completedTasks.batchActionPending, | ||||
|     allActionPending: state.tasks.completedTasks.allActionPending, | ||||
|     pollInterval: state.settings.pollInterval, | ||||
|     pageSize: state.settings.taskRowsPerPage, | ||||
|   }; | ||||
| } | ||||
|  | ||||
| const mapDispatchToProps = { | ||||
|   listCompletedTasksAsync, | ||||
|   deleteCompletedTaskAsync, | ||||
|   deleteAllCompletedTasksAsync, | ||||
|   batchDeleteCompletedTasksAsync, | ||||
|   taskRowsPerPageChange, | ||||
| }; | ||||
|  | ||||
| const connector = connect(mapStateToProps, mapDispatchToProps); | ||||
|  | ||||
| type ReduxProps = ConnectedProps<typeof connector>; | ||||
|  | ||||
| interface Props { | ||||
|   queue: string; // name of the queue. | ||||
|   totalTaskCount: number; // totoal number of completed tasks. | ||||
| } | ||||
|  | ||||
| function CompletedTasksTable(props: Props & ReduxProps) { | ||||
|   const { pollInterval, listCompletedTasksAsync, queue, pageSize } = props; | ||||
|   const classes = useStyles(); | ||||
|   const [page, setPage] = useState(0); | ||||
|   const [selectedIds, setSelectedIds] = useState<string[]>([]); | ||||
|   const [activeTaskId, setActiveTaskId] = useState<string>(""); | ||||
|  | ||||
|   const handlePageChange = ( | ||||
|     event: React.MouseEvent<HTMLButtonElement> | null, | ||||
|     newPage: number | ||||
|   ) => { | ||||
|     setPage(newPage); | ||||
|   }; | ||||
|  | ||||
|   const handleRowsPerPageChange = ( | ||||
|     event: React.ChangeEvent<HTMLInputElement | HTMLTextAreaElement> | ||||
|   ) => { | ||||
|     props.taskRowsPerPageChange(parseInt(event.target.value, 10)); | ||||
|     setPage(0); | ||||
|   }; | ||||
|  | ||||
|   const handleSelectAllClick = (event: React.ChangeEvent<HTMLInputElement>) => { | ||||
|     if (event.target.checked) { | ||||
|       const newSelected = props.tasks.map((t) => t.id); | ||||
|       setSelectedIds(newSelected); | ||||
|     } else { | ||||
|       setSelectedIds([]); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const handleDeleteAllClick = () => { | ||||
|     props.deleteAllCompletedTasksAsync(queue); | ||||
|   }; | ||||
|  | ||||
|   const handleBatchDeleteClick = () => { | ||||
|     props | ||||
|       .batchDeleteCompletedTasksAsync(queue, selectedIds) | ||||
|       .then(() => setSelectedIds([])); | ||||
|   }; | ||||
|  | ||||
|   const fetchData = useCallback(() => { | ||||
|     const pageOpts = { page: page + 1, size: pageSize }; | ||||
|     listCompletedTasksAsync(queue, pageOpts); | ||||
|   }, [page, pageSize, queue, listCompletedTasksAsync]); | ||||
|  | ||||
|   usePolling(fetchData, pollInterval); | ||||
|  | ||||
|   if (props.error.length > 0) { | ||||
|     return ( | ||||
|       <Alert severity="error" className={classes.alert}> | ||||
|         <AlertTitle>Error</AlertTitle> | ||||
|         {props.error} | ||||
|       </Alert> | ||||
|     ); | ||||
|   } | ||||
|   if (props.tasks.length === 0) { | ||||
|     return ( | ||||
|       <Alert severity="info" className={classes.alert}> | ||||
|         <AlertTitle>Info</AlertTitle> | ||||
|         No completed tasks at this time. | ||||
|       </Alert> | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   const columns: TableColumn[] = [ | ||||
|     { key: "id", label: "ID", align: "left" }, | ||||
|     { key: "type", label: "Type", align: "left" }, | ||||
|     { key: "payload", label: "Payload", align: "left" }, | ||||
|     { key: "completed_at", label: "Completed", align: "left" }, | ||||
|     { key: "result", label: "Result", align: "left" }, | ||||
|     { key: "ttl", label: "TTL", align: "left" }, | ||||
|     { key: "actions", label: "Actions", align: "center" }, | ||||
|   ]; | ||||
|  | ||||
|   const rowCount = props.tasks.length; | ||||
|   const numSelected = selectedIds.length; | ||||
|   return ( | ||||
|     <div> | ||||
|       <TableActions | ||||
|         showIconButtons={numSelected > 0} | ||||
|         iconButtonActions={[ | ||||
|           { | ||||
|             tooltip: "Delete", | ||||
|             icon: <DeleteIcon />, | ||||
|             onClick: handleBatchDeleteClick, | ||||
|             disabled: props.batchActionPending, | ||||
|           }, | ||||
|         ]} | ||||
|         menuItemActions={[ | ||||
|           { | ||||
|             label: "Delete All", | ||||
|             onClick: handleDeleteAllClick, | ||||
|             disabled: props.allActionPending, | ||||
|           }, | ||||
|         ]} | ||||
|       /> | ||||
|       <TableContainer component={Paper}> | ||||
|         <Table | ||||
|           stickyHeader={true} | ||||
|           className={classes.table} | ||||
|           aria-label="archived tasks table" | ||||
|           size="small" | ||||
|         > | ||||
|           <TableHead> | ||||
|             <TableRow> | ||||
|               <TableCell | ||||
|                 padding="checkbox" | ||||
|                 classes={{ stickyHeader: classes.stickyHeaderCell }} | ||||
|               > | ||||
|                 <IconButton> | ||||
|                   <Checkbox | ||||
|                     indeterminate={numSelected > 0 && numSelected < rowCount} | ||||
|                     checked={rowCount > 0 && numSelected === rowCount} | ||||
|                     onChange={handleSelectAllClick} | ||||
|                     inputProps={{ | ||||
|                       "aria-label": "select all tasks shown in the table", | ||||
|                     }} | ||||
|                   /> | ||||
|                 </IconButton> | ||||
|               </TableCell> | ||||
|               {columns.map((col) => ( | ||||
|                 <TableCell | ||||
|                   key={col.key} | ||||
|                   align={col.align} | ||||
|                   classes={{ stickyHeader: classes.stickyHeaderCell }} | ||||
|                 > | ||||
|                   {col.label} | ||||
|                 </TableCell> | ||||
|               ))} | ||||
|             </TableRow> | ||||
|           </TableHead> | ||||
|           <TableBody> | ||||
|             {props.tasks.map((task) => ( | ||||
|               <Row | ||||
|                 key={task.id} | ||||
|                 task={task} | ||||
|                 isSelected={selectedIds.includes(task.id)} | ||||
|                 onSelectChange={(checked: boolean) => { | ||||
|                   if (checked) { | ||||
|                     setSelectedIds(selectedIds.concat(task.id)); | ||||
|                   } else { | ||||
|                     setSelectedIds(selectedIds.filter((id) => id !== task.id)); | ||||
|                   } | ||||
|                 }} | ||||
|                 onDeleteClick={() => { | ||||
|                   props.deleteCompletedTaskAsync(queue, task.id); | ||||
|                 }} | ||||
|                 allActionPending={props.allActionPending} | ||||
|                 onActionCellEnter={() => setActiveTaskId(task.id)} | ||||
|                 onActionCellLeave={() => setActiveTaskId("")} | ||||
|                 showActions={activeTaskId === task.id} | ||||
|               /> | ||||
|             ))} | ||||
|           </TableBody> | ||||
|           <TableFooter> | ||||
|             <TableRow> | ||||
|               <TablePagination | ||||
|                 rowsPerPageOptions={rowsPerPageOptions} | ||||
|                 colSpan={columns.length + 1} | ||||
|                 count={props.totalTaskCount} | ||||
|                 rowsPerPage={pageSize} | ||||
|                 page={page} | ||||
|                 SelectProps={{ | ||||
|                   inputProps: { "aria-label": "rows per page" }, | ||||
|                   native: true, | ||||
|                 }} | ||||
|                 onPageChange={handlePageChange} | ||||
|                 onRowsPerPageChange={handleRowsPerPageChange} | ||||
|                 ActionsComponent={TablePaginationActions} | ||||
|                 className={classes.pagination} | ||||
|               /> | ||||
|             </TableRow> | ||||
|           </TableFooter> | ||||
|         </Table> | ||||
|       </TableContainer> | ||||
|     </div> | ||||
|   ); | ||||
| } | ||||
|  | ||||
| const useRowStyles = makeStyles((theme) => ({ | ||||
|   root: { | ||||
|     cursor: "pointer", | ||||
|     "&:hover": { | ||||
|       boxShadow: theme.shadows[2], | ||||
|     }, | ||||
|     "&:hover .MuiTableCell-root": { | ||||
|       borderBottomColor: theme.palette.background.paper, | ||||
|     }, | ||||
|   }, | ||||
|   actionCell: { | ||||
|     width: "96px", | ||||
|   }, | ||||
|   actionButton: { | ||||
|     marginLeft: 3, | ||||
|     marginRight: 3, | ||||
|   }, | ||||
| })); | ||||
|  | ||||
| interface RowProps { | ||||
|   task: TaskInfoExtended; | ||||
|   isSelected: boolean; | ||||
|   onSelectChange: (checked: boolean) => void; | ||||
|   onDeleteClick: () => void; | ||||
|   allActionPending: boolean; | ||||
|   showActions: boolean; | ||||
|   onActionCellEnter: () => void; | ||||
|   onActionCellLeave: () => void; | ||||
| } | ||||
|  | ||||
| function Row(props: RowProps) { | ||||
|   const { task } = props; | ||||
|   const classes = useRowStyles(); | ||||
|   const history = useHistory(); | ||||
|   return ( | ||||
|     <TableRow | ||||
|       key={task.id} | ||||
|       className={classes.root} | ||||
|       selected={props.isSelected} | ||||
|       onClick={() => history.push(taskDetailsPath(task.queue, task.id))} | ||||
|     > | ||||
|       <TableCell padding="checkbox" onClick={(e) => e.stopPropagation()}> | ||||
|         <IconButton> | ||||
|           <Checkbox | ||||
|             onChange={(event: React.ChangeEvent<HTMLInputElement>) => | ||||
|               props.onSelectChange(event.target.checked) | ||||
|             } | ||||
|             checked={props.isSelected} | ||||
|           /> | ||||
|         </IconButton> | ||||
|       </TableCell> | ||||
|       <TableCell component="th" scope="row"> | ||||
|         {uuidPrefix(task.id)} | ||||
|       </TableCell> | ||||
|       <TableCell>{task.type}</TableCell> | ||||
|       <TableCell> | ||||
|         <SyntaxHighlighter | ||||
|           language="json" | ||||
|           customStyle={{ margin: 0, maxWidth: 400 }} | ||||
|         > | ||||
|           {task.payload} | ||||
|         </SyntaxHighlighter> | ||||
|       </TableCell> | ||||
|       <TableCell>{timeAgo(task.completed_at)}</TableCell> | ||||
|       <TableCell> | ||||
|         <SyntaxHighlighter | ||||
|           language="json" | ||||
|           customStyle={{ margin: 0, maxWidth: 400 }} | ||||
|         > | ||||
|           {task.result} | ||||
|         </SyntaxHighlighter> | ||||
|       </TableCell> | ||||
|       <TableCell> | ||||
|         {task.ttl_seconds > 0 | ||||
|           ? `${stringifyDuration(durationFromSeconds(task.ttl_seconds))} left` | ||||
|           : `expired`} | ||||
|       </TableCell> | ||||
|       <TableCell | ||||
|         align="center" | ||||
|         className={classes.actionCell} | ||||
|         onMouseEnter={props.onActionCellEnter} | ||||
|         onMouseLeave={props.onActionCellLeave} | ||||
|         onClick={(e) => e.stopPropagation()} | ||||
|       > | ||||
|         {props.showActions ? ( | ||||
|           <React.Fragment> | ||||
|             <Tooltip title="Delete"> | ||||
|               <IconButton | ||||
|                 className={classes.actionButton} | ||||
|                 onClick={props.onDeleteClick} | ||||
|                 disabled={task.requestPending || props.allActionPending} | ||||
|                 size="small" | ||||
|               > | ||||
|                 <DeleteIcon fontSize="small" /> | ||||
|               </IconButton> | ||||
|             </Tooltip> | ||||
|           </React.Fragment> | ||||
|         ) : ( | ||||
|           <IconButton size="small" onClick={props.onActionCellEnter}> | ||||
|             <MoreHorizIcon fontSize="small" /> | ||||
|           </IconButton> | ||||
|         )} | ||||
|       </TableCell> | ||||
|     </TableRow> | ||||
|   ); | ||||
| } | ||||
|  | ||||
| export default connector(CompletedTasksTable); | ||||
| @@ -38,7 +38,7 @@ import { AppState } from "../store"; | ||||
| import { usePolling } from "../hooks"; | ||||
| import { uuidPrefix } from "../utils"; | ||||
| import { TableColumn } from "../types/table"; | ||||
| import { PendingTaskExtended } from "../reducers/tasksReducer"; | ||||
| import { TaskInfoExtended } from "../reducers/tasksReducer"; | ||||
| import { taskDetailsPath } from "../paths"; | ||||
|  | ||||
| const useStyles = makeStyles((theme) => ({ | ||||
| @@ -313,7 +313,7 @@ const useRowStyles = makeStyles((theme) => ({ | ||||
| })); | ||||
|  | ||||
| interface RowProps { | ||||
|   task: PendingTaskExtended; | ||||
|   task: TaskInfoExtended; | ||||
|   isSelected: boolean; | ||||
|   onSelectChange: (checked: boolean) => void; | ||||
|   onDeleteClick: () => void; | ||||
|   | ||||
| @@ -23,6 +23,7 @@ interface TaskBreakdown { | ||||
|   scheduled: number; // number of scheduled tasks in the queue. | ||||
|   retry: number; // number of retry tasks in the queue. | ||||
|   archived: number; // number of archived tasks in the queue. | ||||
|   completed: number; // number of completed tasks in the queue. | ||||
| } | ||||
|  | ||||
| function QueueSizeChart(props: Props) { | ||||
| @@ -55,6 +56,7 @@ function QueueSizeChart(props: Props) { | ||||
|         <Bar dataKey="scheduled" stackId="a" fill="#fdd663" /> | ||||
|         <Bar dataKey="retry" stackId="a" fill="#f666a9" /> | ||||
|         <Bar dataKey="archived" stackId="a" fill="#ac4776" /> | ||||
|         <Bar dataKey="completed" stackId="a" fill="#4bb543" /> | ||||
|       </BarChart> | ||||
|     </ResponsiveContainer> | ||||
|   ); | ||||
|   | ||||
| @@ -41,7 +41,7 @@ import { taskRowsPerPageChange } from "../actions/settingsActions"; | ||||
| import TableActions from "./TableActions"; | ||||
| import { durationBefore, uuidPrefix } from "../utils"; | ||||
| import { usePolling } from "../hooks"; | ||||
| import { RetryTaskExtended } from "../reducers/tasksReducer"; | ||||
| import { TaskInfoExtended } from "../reducers/tasksReducer"; | ||||
| import { TableColumn } from "../types/table"; | ||||
| import { taskDetailsPath } from "../paths"; | ||||
|  | ||||
| @@ -344,7 +344,7 @@ const useRowStyles = makeStyles((theme) => ({ | ||||
| })); | ||||
|  | ||||
| interface RowProps { | ||||
|   task: RetryTaskExtended; | ||||
|   task: TaskInfoExtended; | ||||
|   isSelected: boolean; | ||||
|   onSelectChange: (checked: boolean) => void; | ||||
|   onDeleteClick: () => void; | ||||
|   | ||||
| @@ -41,7 +41,7 @@ import TablePaginationActions, { | ||||
| import TableActions from "./TableActions"; | ||||
| import { durationBefore, uuidPrefix } from "../utils"; | ||||
| import { usePolling } from "../hooks"; | ||||
| import { ScheduledTaskExtended } from "../reducers/tasksReducer"; | ||||
| import { TaskInfoExtended } from "../reducers/tasksReducer"; | ||||
| import { TableColumn } from "../types/table"; | ||||
| import { taskDetailsPath } from "../paths"; | ||||
|  | ||||
| @@ -341,7 +341,7 @@ const useRowStyles = makeStyles((theme) => ({ | ||||
| })); | ||||
|  | ||||
| interface RowProps { | ||||
|   task: ScheduledTaskExtended; | ||||
|   task: TaskInfoExtended; | ||||
|   isSelected: boolean; | ||||
|   onSelectChange: (checked: boolean) => void; | ||||
|   onRunClick: () => void; | ||||
|   | ||||
| @@ -11,6 +11,7 @@ import PendingTasksTable from "./PendingTasksTable"; | ||||
| import ScheduledTasksTable from "./ScheduledTasksTable"; | ||||
| import RetryTasksTable from "./RetryTasksTable"; | ||||
| import ArchivedTasksTable from "./ArchivedTasksTable"; | ||||
| import CompletedTasksTable from "./CompletedTasksTable"; | ||||
| import { useHistory } from "react-router-dom"; | ||||
| import { queueDetailsPath, taskDetailsPath } from "../paths"; | ||||
| import { QueueInfo } from "../reducers/queuesReducer"; | ||||
| @@ -56,6 +57,7 @@ function mapStatetoProps(state: AppState, ownProps: Props) { | ||||
|         scheduled: 0, | ||||
|         retry: 0, | ||||
|         archived: 0, | ||||
|         completed: 0, | ||||
|         processed: 0, | ||||
|         failed: 0, | ||||
|         timestamp: "n/a", | ||||
| @@ -104,7 +106,8 @@ const useStyles = makeStyles((theme) => ({ | ||||
|     marginLeft: "2px", | ||||
|   }, | ||||
|   searchbar: { | ||||
|     marginLeft: theme.spacing(4), | ||||
|     paddingLeft: theme.spacing(2), | ||||
|     paddingRight: theme.spacing(2), | ||||
|   }, | ||||
|   search: { | ||||
|     position: "relative", | ||||
| @@ -147,6 +150,7 @@ function TasksTable(props: Props & ReduxProps) { | ||||
|     { key: "scheduled", label: "Scheduled", count: currentStats.scheduled }, | ||||
|     { key: "retry", label: "Retry", count: currentStats.retry }, | ||||
|     { key: "archived", label: "Archived", count: currentStats.archived }, | ||||
|     { key: "completed", label: "Completed", count: currentStats.completed }, | ||||
|   ]; | ||||
|  | ||||
|   const [searchQuery, setSearchQuery] = useState<string>(""); | ||||
| @@ -229,6 +233,12 @@ function TasksTable(props: Props & ReduxProps) { | ||||
|           totalTaskCount={currentStats.archived} | ||||
|         /> | ||||
|       </TabPanel> | ||||
|       <TabPanel value="completed" selected={props.selected}> | ||||
|         <CompletedTasksTable | ||||
|           queue={props.queue} | ||||
|           totalTaskCount={currentStats.completed} | ||||
|         /> | ||||
|       </TabPanel> | ||||
|     </Paper> | ||||
|   ); | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user