mirror of
				https://github.com/hibiken/asynqmon.git
				synced 2025-10-26 16:26:12 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			177 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			177 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import Checkbox from "@material-ui/core/Checkbox";
 | |
| import IconButton from "@material-ui/core/IconButton";
 | |
| import TableCell from "@material-ui/core/TableCell";
 | |
| import TableRow from "@material-ui/core/TableRow";
 | |
| import Tooltip from "@material-ui/core/Tooltip";
 | |
| import DeleteIcon from "@material-ui/icons/Delete";
 | |
| import FileCopyOutlinedIcon from "@material-ui/icons/FileCopyOutlined";
 | |
| import MoreHorizIcon from "@material-ui/icons/MoreHoriz";
 | |
| import React from "react";
 | |
| import { connect, ConnectedProps } from "react-redux";
 | |
| import { useHistory } from "react-router-dom";
 | |
| import { taskRowsPerPageChange } from "../actions/settingsActions";
 | |
| import {
 | |
|   batchDeleteCompletedTasksAsync,
 | |
|   deleteAllCompletedTasksAsync,
 | |
|   deleteCompletedTaskAsync,
 | |
|   listCompletedTasksAsync,
 | |
| } from "../actions/tasksActions";
 | |
| import { taskDetailsPath } from "../paths";
 | |
| import { AppState } from "../store";
 | |
| import { TableColumn } from "../types/table";
 | |
| import {
 | |
|   durationFromSeconds,
 | |
|   prettifyPayload,
 | |
|   stringifyDuration,
 | |
|   timeAgo,
 | |
|   uuidPrefix,
 | |
| } from "../utils";
 | |
| import SyntaxHighlighter from "./SyntaxHighlighter";
 | |
| import TasksTable, { RowProps, useRowStyles } from "./TasksTable";
 | |
| 
 | |
| 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 = {
 | |
|   listTasks: listCompletedTasksAsync,
 | |
|   deleteTask: deleteCompletedTaskAsync,
 | |
|   deleteAllTasks: deleteAllCompletedTasksAsync,
 | |
|   batchDeleteTasks: 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.
 | |
| }
 | |
| 
 | |
| 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" },
 | |
| ];
 | |
| 
 | |
| 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))}
 | |
|     >
 | |
|       {!window.READ_ONLY && (
 | |
|         <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" className={classes.idCell}>
 | |
|         <div className={classes.IdGroup}>
 | |
|           {uuidPrefix(task.id)}
 | |
|           <Tooltip title="Copy full ID to clipboard">
 | |
|             <IconButton
 | |
|               onClick={(e) => {
 | |
|                 e.stopPropagation();
 | |
|                 navigator.clipboard.writeText(task.id);
 | |
|               }}
 | |
|               size="small"
 | |
|               className={classes.copyButton}
 | |
|             >
 | |
|               <FileCopyOutlinedIcon fontSize="small" />
 | |
|             </IconButton>
 | |
|           </Tooltip>
 | |
|         </div>
 | |
|       </TableCell>
 | |
|       <TableCell>{task.type}</TableCell>
 | |
|       <TableCell>
 | |
|         <SyntaxHighlighter
 | |
|           language="json"
 | |
|           customStyle={{ margin: 0, maxWidth: 400 }}
 | |
|         >
 | |
|           {prettifyPayload(task.payload)}
 | |
|         </SyntaxHighlighter>
 | |
|       </TableCell>
 | |
|       <TableCell>{timeAgo(task.completed_at)}</TableCell>
 | |
|       <TableCell>
 | |
|         <SyntaxHighlighter
 | |
|           language="json"
 | |
|           customStyle={{ margin: 0, maxWidth: 400 }}
 | |
|         >
 | |
|           {prettifyPayload(task.result)}
 | |
|         </SyntaxHighlighter>
 | |
|       </TableCell>
 | |
|       <TableCell>
 | |
|         {task.ttl_seconds > 0
 | |
|           ? `${stringifyDuration(durationFromSeconds(task.ttl_seconds))} left`
 | |
|           : `expired`}
 | |
|       </TableCell>
 | |
|       {!window.READ_ONLY && (
 | |
|         <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>
 | |
|   );
 | |
| }
 | |
| 
 | |
| function CompletedTasksTable(props: Props & ReduxProps) {
 | |
|   return (
 | |
|     <TasksTable
 | |
|       taskState="completed"
 | |
|       columns={columns}
 | |
|       renderRow={(rowProps: RowProps) => <Row {...rowProps} />}
 | |
|       {...props}
 | |
|     />
 | |
|   );
 | |
| }
 | |
| export default connector(CompletedTasksTable);
 |