forked from zmc/pulpito-ng
-
Notifications
You must be signed in to change notification settings - Fork 9
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #51 from Devansh3712/kill-run
pages/Run/index.tsx: Add kill run button
- Loading branch information
Showing
7 changed files
with
235 additions
and
20 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
import Snackbar from "@mui/material/Snackbar"; | ||
import Alert from "@mui/material/Alert"; | ||
import { useState } from "react"; | ||
|
||
type AlertProps = { | ||
severity: "success" | "error", | ||
message: string, | ||
}; | ||
|
||
export default function AlertComponent(props: AlertProps) { | ||
const [isOpen, setIsOpen] = useState(true); | ||
const handleClose = ( | ||
event?: React.SyntheticEvent | Event, | ||
reason?: string | ||
) => { | ||
if (reason === "clickaway") { | ||
return; | ||
} | ||
setIsOpen(false); | ||
}; | ||
|
||
return ( | ||
<Snackbar autoHideDuration={3000} open={isOpen} onClose={handleClose}> | ||
<Alert onClose={handleClose} severity={props.severity} sx={{ width: "100%" }}> | ||
{props.message} | ||
</Alert> | ||
</Snackbar> | ||
); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,157 @@ | ||
import { useState } from "react"; | ||
import type { UseMutationResult, UseQueryResult } from "@tanstack/react-query"; | ||
import Button from "@mui/material/Button"; | ||
import Box from "@mui/material/Box"; | ||
import CircularProgress from "@mui/material/CircularProgress"; | ||
import DialogTitle from '@mui/material/DialogTitle'; | ||
import DialogContent from '@mui/material/DialogContent'; | ||
import Dialog from '@mui/material/Dialog'; | ||
import Paper from "@mui/material/Paper"; | ||
import Typography from "@mui/material/Typography"; | ||
import Tooltip from '@mui/material/Tooltip'; | ||
|
||
import CodeBlock from "../CodeBlock"; | ||
import type { Run as RunResponse } from "../../lib/paddles.d"; | ||
import { KillRunPayload } from "../../lib/teuthologyAPI.d"; | ||
import { useSession, useRunKill } from "../../lib/teuthologyAPI"; | ||
import Alert from "../Alert"; | ||
|
||
|
||
type KillButtonProps = { | ||
query: UseQueryResult<RunResponse>; | ||
}; | ||
|
||
type KillButtonDialogProps = { | ||
mutation: UseMutationResult; | ||
payload: KillRunPayload; | ||
open: boolean; | ||
handleClose: () => void; | ||
}; | ||
|
||
export default function KillButton({query: runQuery}: KillButtonProps) { | ||
const killMutation = useRunKill(); | ||
const [open, setOpen] = useState(false); | ||
const sessionQuery = useSession(); | ||
const data: RunResponse | undefined = runQuery.data; | ||
const run_owner = data?.jobs[0].owner || ""; | ||
const killPayload: KillRunPayload = { | ||
"--run": data?.name || "", | ||
"--owner": run_owner, | ||
"--machine-type": data?.machine_type || "", | ||
"--preserve-queue": true, | ||
} | ||
const loggedUser = sessionQuery.data?.session?.username; | ||
const isUserAdmin = sessionQuery.data?.session?.isUserAdmin; | ||
const owner = killPayload["--owner"].toLowerCase() | ||
const isOwner = (loggedUser?.toLowerCase() == owner) || (`scheduled_${loggedUser?.toLowerCase()}@teuthology` == owner) | ||
const isButtonDisabled = (!isOwner && !isUserAdmin) | ||
|
||
const getHelperMessage = () => { | ||
if (isButtonDisabled) { | ||
return `User (${loggedUser}) does not have admin privileges to kill runs owned by another user (${owner}). `; | ||
} else { | ||
if (!isOwner && isUserAdmin) return `Use admin privileges to kill run owned by '${owner}'. `; | ||
return "Terminate all jobs in this run"; | ||
} | ||
} | ||
|
||
const toggleDialog = () => { | ||
setOpen(!open); | ||
}; | ||
|
||
const refreshAndtoggle = () => { | ||
if (open && !killMutation.isIdle) { // on closing confirmation dialog after kill-run | ||
runQuery.refetch(); | ||
} | ||
toggleDialog(); | ||
killMutation.reset(); | ||
} | ||
|
||
if ((data?.status.includes("finished")) || !(sessionQuery.data?.session?.username)) { | ||
// run finished or user logged out | ||
return null; | ||
} | ||
|
||
|
||
return ( | ||
<div> | ||
<div style={{ display: "flex" }}> | ||
<Tooltip arrow title={getHelperMessage()}> | ||
<span> | ||
<Button | ||
variant="contained" | ||
color="error" | ||
size="large" | ||
onClick={refreshAndtoggle} | ||
disabled={isButtonDisabled} | ||
sx={{ marginBottom: "12px" }} | ||
> | ||
{(isOwner) ? "Kill Run" : "Kill Run As Admin"} | ||
</Button> | ||
</span> | ||
</Tooltip> | ||
<KillButtonDialog | ||
mutation={killMutation} | ||
payload={killPayload} | ||
open={open} | ||
handleClose={refreshAndtoggle} | ||
/> | ||
</div> | ||
{ (killMutation.isError) ? <Alert severity="error" message="Unable to kill run" /> : null } | ||
{ (killMutation.isSuccess) ? <Alert severity="success" message={`Run killed successfully! \n`} /> : null } | ||
</div> | ||
); | ||
}; | ||
|
||
function KillButtonDialog({mutation, open, handleClose, payload}: KillButtonDialogProps) { | ||
return ( | ||
<div> | ||
<Dialog onClose={handleClose} open={open} scroll="paper" fullWidth={true} maxWidth="md"> | ||
<DialogTitle variant="h6">KILL CONFIRMATION</DialogTitle> | ||
<DialogContent dividers> | ||
{ (mutation.isSuccess && mutation.data ) ? | ||
<div> | ||
<Typography variant="h6" display="block" color="green" gutterBottom> | ||
Successful! | ||
</Typography> | ||
<Paper> | ||
<CodeBlock value={mutation.data?.data?.logs || ""} language="python" /> | ||
</Paper> | ||
</div> : | ||
(mutation.isLoading) ? ( | ||
<div style={{display: "flex", alignItems: "center"}}> | ||
<CircularProgress size={20} color="inherit" style={{marginRight: "12px"}} /> | ||
<Typography variant="subtitle1" display="block"> | ||
Killing run... | ||
</Typography> | ||
</div> | ||
) : | ||
(mutation.isError) ? ( | ||
<div> | ||
<Typography variant="h6" display="block" color="red" gutterBottom> | ||
Failed! | ||
</Typography> | ||
<Paper> | ||
<CodeBlock value={(mutation.error?.response?.data?.detail) || ""} language="python" /> | ||
</Paper> | ||
</div> | ||
) : | ||
<div> | ||
<Typography variant="overline" display="block" gutterBottom> | ||
Are you sure you want to kill this run/job? | ||
</Typography> | ||
<Button | ||
variant="contained" | ||
color="error" | ||
size="large" | ||
onClick={() => mutation.mutate(payload)} | ||
> | ||
Yes, I'm sure | ||
</Button> | ||
</div> | ||
} | ||
</DialogContent> | ||
</Dialog> | ||
</div> | ||
) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,15 @@ | ||
|
||
export type Session = { | ||
session: { | ||
id: int, | ||
username: string, | ||
isUserAdmin: boolean, | ||
} | ||
} | ||
|
||
export type KillRunPayload = { | ||
"--run": string, | ||
"--owner": string, | ||
"--machine-type": string, | ||
"--preserve-queue": boolean, | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters