From 0ff2a8cb26b945605c2a28c5777edb1e44492a98 Mon Sep 17 00:00:00 2001 From: Johanna Szepanski <johanna.szepanski@softhouse.se> Date: Mon, 13 Jan 2025 09:29:03 +0100 Subject: [PATCH] improved UI for single and batch job details --- src/components/Job/JobDetails.tsx | 63 +---------- src/components/Job/JobDetailsBatchJob.jsx | 105 ++++++++++++++++++ src/components/Job/JobDetailsSection.tsx | 119 +++++++++++++++++++++ src/components/Job/JobDetailsSingleJob.tsx | 55 ++++++++++ 4 files changed, 282 insertions(+), 60 deletions(-) create mode 100644 src/components/Job/JobDetailsBatchJob.jsx create mode 100644 src/components/Job/JobDetailsSection.tsx create mode 100644 src/components/Job/JobDetailsSingleJob.tsx diff --git a/src/components/Job/JobDetails.tsx b/src/components/Job/JobDetails.tsx index d03c90c5..8513e2a6 100644 --- a/src/components/Job/JobDetails.tsx +++ b/src/components/Job/JobDetails.tsx @@ -1,20 +1,13 @@ import { useEffect, useState, useMemo } from "react"; -import { Typography, Stack, Box } from "@mui/material"; +import { Typography, Stack } from "@mui/material"; import { - KeyValueTable, SimpleAccordion, AlertBannerList, - ExternalLink, - InternalLink, - formatDateAndTime, - EmptyValue, - Duration, SingleStateStepper, STEPPER_STATES, AccessControl } from "@ess-ics/ce-ui-common"; -import { ActionTypeIconText } from "./JobIcons"; -import { JobDetailsTable } from "./JobDetailsTable"; +import { JobDetailsSection } from "./JobDetailsSection"; import { DeploymentJobOutput } from "../deployments/DeploymentJobOutput"; import { AWXJobDetails, Status } from "../../api/DataTypes"; import { JobDetails } from "../../store/deployApi"; @@ -30,47 +23,6 @@ const createAlert = (type: Status, message: string) => { }; }; -const getDeploymentDetails = (operation: JobDetails) => { - return { - user: ( - <InternalLink - to={`/user/${operation.createdBy}`} - label={`User profile, ${operation.createdBy}`} - > - {operation.createdBy} - </InternalLink> - ), - "AWX job link": operation?.jobUrl ? ( - <ExternalLink - href={operation.jobUrl} - label="AWX job" - > - {operation.jobUrl} - </ExternalLink> - ) : ( - <EmptyValue /> - ), - "created time": operation?.createdAt ? ( - formatDateAndTime(operation.createdAt) - ) : ( - <EmptyValue /> - ), - "AWX job start time": operation?.startTime ? ( - formatDateAndTime(operation.startTime) - ) : ( - <EmptyValue /> - ), - duration: operation ? ( - <Duration - createOrStartDate={new Date(operation.createdAt ?? new Date())} - finishedAt={new Date(operation.finishedAt ?? new Date())} - /> - ) : ( - <EmptyValue /> - ) - }; -}; - export const JobsDetails = ({ jobDetail: operation }: JobDetailsProps) => { const [alerts, setAlerts] = useState< { type?: Status; message?: string; link?: string }[] @@ -93,16 +45,7 @@ export const JobsDetails = ({ jobDetail: operation }: JobDetailsProps) => { return ( <Stack spacing={2}> <Stack>{<AlertBannerList alerts={alerts} />}</Stack> - <Box sx={{ paddingLeft: "46px" }}> - <ActionTypeIconText action={operation.action} /> - </Box> - <JobDetailsTable operation={operation} /> - <KeyValueTable - obj={getDeploymentDetails(operation)} - variant="overline" - sx={{ border: 0 }} - valueOptions={{ headerName: "" }} - /> + <JobDetailsSection job={operation} /> <AccessControl allowedRoles={["DeploymentToolAdmin", "DeploymentToolIntegrator"]} renderNoAccess={() => <></>} diff --git a/src/components/Job/JobDetailsBatchJob.jsx b/src/components/Job/JobDetailsBatchJob.jsx new file mode 100644 index 00000000..cad2712e --- /dev/null +++ b/src/components/Job/JobDetailsBatchJob.jsx @@ -0,0 +1,105 @@ +import { useMemo } from "react"; +import { + EllipsisText, + InternalLink, + Table, + SimpleAccordion +} from "@ess-ics/ce-ui-common"; +import { Typography, Stack } from "@mui/material"; +import { + getNoOfIOCs, + getNoOfHosts, + isDeploymentJob, + calculateHostText +} from "./JobUtils"; +import { JobRevisionChip } from "./JobRevisionChip"; + +const columns = [ + { + field: "ioc", + headerName: "IOC", + flex: 0.7 + }, + { + field: "host", + headerName: "Host", + flex: 1 + } +]; + +const createRow = (job, action) => { + return { + id: `${job.host.hostId}${job.iocId}`, + ioc: ( + <Stack + sx={{ padding: "8px 16px 8px 0" }} + gap={2} + > + <Stack + key={job.iocId} + flexDirection="row" + gap={1} + > + <InternalLink + to={`/iocs/${job.iocId}`} + label={`Ioc details, ${job.iocName}`} + > + <EllipsisText>{job.iocName}</EllipsisText> + </InternalLink> + {isDeploymentJob(action) && ( + <JobRevisionChip + gitReference={job.gitReference} + gitProjectId={job.gitProjectId} + /> + )} + </Stack> + </Stack> + ), + host: calculateHostText(job) + }; +}; + +export const JobDetailsBatchJob = ({ operation }) => { + const noOfIOCs = useMemo(() => { + return getNoOfIOCs(operation.jobs); + }, [operation]); + + const noOfHosts = useMemo(() => { + return getNoOfHosts(operation.jobs); + }, [operation]); + + return ( + <SimpleAccordion + summary={ + <Stack + flexDirection="row" + alignItems="end" + sx={{ width: "100%" }} + gap={1} + > + {" "} + <Typography + component="h2" + variant="h3" + > + Batch + </Typography> + <Typography + variant="body2" + sx={{ fontWeight: "600", marginRight: "10px" }} + > + {noOfIOCs} {noOfIOCs > 1 ? "IOCs" : "IOC"} + {", "} + {noOfHosts} {noOfHosts > 1 ? "Hosts" : "Host"} + </Typography> + </Stack> + } + > + <Table + columns={columns} + disableColumnSorting + rows={operation.jobs.map((job) => createRow(job, operation.action))} + /> + </SimpleAccordion> + ); +}; diff --git a/src/components/Job/JobDetailsSection.tsx b/src/components/Job/JobDetailsSection.tsx new file mode 100644 index 00000000..108220f9 --- /dev/null +++ b/src/components/Job/JobDetailsSection.tsx @@ -0,0 +1,119 @@ +import { useState } from "react"; +import { Paper, Box, Typography } from "@mui/material"; +import { + KeyValueTable, + ExternalLink, + InternalLink, + formatDateAndTime, + EmptyValue, + Duration, + SimpleAccordion +} from "@ess-ics/ce-ui-common"; +import { JobDetailsBatchJob } from "./JobDetailsBatchJob"; +import { JobDetailsSingleJob } from "./JobDetailsSingleJob"; +import { ActionTypeIconText } from "./JobIcons"; +import { isBatchJob } from "./JobUtils"; +import { JobDetails } from "../../store/deployApi"; + +interface JobDetailsSectionProps { + job: JobDetails; +} + +const getTimeDetails = (job: JobDetails) => { + return { + started: job?.createdAt ? formatDateAndTime(job.createdAt) : <EmptyValue />, + completed: job?.finishedAt ? ( + formatDateAndTime(job.finishedAt) + ) : ( + <EmptyValue /> + ), + duration: job?.finishedAt ? ( + <Duration + createOrStartDate={job.createdAt && new Date(job.createdAt)} + finishedAt={job.finishedAt && new Date(job.finishedAt)} + textOnly + /> + ) : ( + <EmptyValue /> + ) + }; +}; + +const getDeploymentDetails = (job: JobDetails) => { + return { + user: ( + <InternalLink + to={`/user/${job.createdBy}`} + label={`User profile, ${job.createdBy}`} + > + {job.createdBy} + </InternalLink> + ), + "AWX job link": job?.jobUrl ? ( + <ExternalLink + href={job.jobUrl} + label="AWX job" + > + {job.jobUrl} + </ExternalLink> + ) : ( + <EmptyValue /> + ) + }; +}; + +export const JobDetailsSection = ({ job }: JobDetailsSectionProps) => { + const [expanded, setExpanded] = useState(false); + return ( + <> + <Paper sx={{ padding: "12px 12px 0 12px" }}> + <Box sx={{ paddingLeft: "4px" }}> + <ActionTypeIconText action={job.action} /> + </Box> + {isBatchJob(job.action) ? ( + <Box sx={{ marginBottom: "16px" }}> + <JobDetailsBatchJob operation={job} /> + </Box> + ) : ( + <Box sx={{ marginBottom: "16px" }}> + <JobDetailsSingleJob job={job} /> + </Box> + )} + </Paper> + + <SimpleAccordion + expanded={expanded} + onChange={() => setExpanded((prev) => !prev)} + summary={ + <Typography + component="h2" + variant="h3" + sx={{ marginRight: "8px" }} + > + Job details + </Typography> + } + > + <KeyValueTable + obj={getDeploymentDetails(job)} + variant="overline" + sx={{ + border: 0, + "& .MuiBox-root": { + width: "100%" + } + }} + valueOptions={{ headerName: "" }} + /> + <Paper sx={{ marginTop: "8px", padding: "12px" }}> + <KeyValueTable + obj={getTimeDetails(job)} + variant="overline" + sx={{ border: 0 }} + valueOptions={{ headerName: "" }} + /> + </Paper> + </SimpleAccordion> + </> + ); +}; diff --git a/src/components/Job/JobDetailsSingleJob.tsx b/src/components/Job/JobDetailsSingleJob.tsx new file mode 100644 index 00000000..a6a2e77e --- /dev/null +++ b/src/components/Job/JobDetailsSingleJob.tsx @@ -0,0 +1,55 @@ +import { Stack } from "@mui/material"; +import { InternalLink, KeyValueTable } from "@ess-ics/ce-ui-common"; +import { calculateHostText } from "./JobUtils"; +import { JobGitRefLink } from "./JobGitRefLink"; +import { JobGitRefIcon } from "./JobGitRefIcon"; +import { JobDetails } from "../../store/deployApi"; + +interface JobDetailsSingleJobProps { + job: JobDetails; +} + +const getSingleOperationFields = (operation: JobDetails) => { + return { + ioc: ( + <InternalLink + to={`/iocs/${operation.iocName}`} + label={`IOC details, ${operation.iocName}`} + > + {operation.iocName} + </InternalLink> + ), + revision: + operation.gitReference && operation.gitProjectId ? ( + <Stack + flexDirection="row" + alignItems="center" + gap={0.5} + > + <JobGitRefIcon + gitReference={operation.gitReference} + gitProjectId={operation.gitProjectId} + /> + <JobGitRefLink + gitReference={operation.gitReference} + gitProjectId={operation.gitProjectId} + disableExternalLinkIcon + /> + </Stack> + ) : ( + "Unknown" + ), + host: calculateHostText(operation) + }; +}; + +export const JobDetailsSingleJob = ({ job }: JobDetailsSingleJobProps) => { + return ( + <KeyValueTable + obj={getSingleOperationFields(job)} + variant="overline" + sx={{ border: 0 }} + valueOptions={{ headerName: "" }} + /> + ); +}; -- GitLab