Add groups support #146
@ -1,11 +1,16 @@
|
|||||||
import { APIClient } from "./ApiClient";
|
import { APIClient } from "./ApiClient";
|
||||||
import { VMGroup } from "./ServerApi";
|
import { VMGroup } from "./ServerApi";
|
||||||
import { VMState } from "./VMApi";
|
import { VMInfo, VMState } from "./VMApi";
|
||||||
|
|
||||||
export interface GroupVMState {
|
export interface GroupVMState {
|
||||||
[key: string]: VMState;
|
[key: string]: VMState;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface TreatmentResult {
|
||||||
|
ok: number;
|
||||||
|
failed: number;
|
||||||
|
}
|
||||||
|
|
||||||
export class GroupApi {
|
export class GroupApi {
|
||||||
/**
|
/**
|
||||||
* Get the state of the VMs of a group
|
* Get the state of the VMs of a group
|
||||||
@ -15,4 +20,16 @@ export class GroupApi {
|
|||||||
await APIClient.exec({ method: "GET", uri: `/group/${g.id}/vm/state` })
|
await APIClient.exec({ method: "GET", uri: `/group/${g.id}/vm/state` })
|
||||||
).data;
|
).data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Request to start the VM of a group
|
||||||
|
*/
|
||||||
|
static async StartVM(g: VMGroup, vm?: VMInfo): Promise<TreatmentResult> {
|
||||||
|
return (
|
||||||
|
await APIClient.exec({
|
||||||
|
method: "GET",
|
||||||
|
uri: `/group/${g.id}/vm/start` + (vm ? `?vm_id=${vm.uuid}` : ""),
|
||||||
|
})
|
||||||
|
).data;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,12 @@
|
|||||||
import { Button, Toolbar, Tooltip } from "@fluentui/react-components";
|
import { Button, Spinner, Toolbar, Tooltip } from "@fluentui/react-components";
|
||||||
import { PlayRegular } from "@fluentui/react-icons";
|
import { PlayRegular } from "@fluentui/react-icons";
|
||||||
import { VMGroup } from "../api/ServerApi";
|
import { VMGroup } from "../api/ServerApi";
|
||||||
import { VMInfo, VMState } from "../api/VMApi";
|
import { VMInfo, VMState } from "../api/VMApi";
|
||||||
|
import { GroupApi, TreatmentResult } from "../api/GroupApi";
|
||||||
|
import React from "react";
|
||||||
|
import { useToast } from "../hooks/providers/ToastProvider";
|
||||||
|
import { useConfirm } from "../hooks/providers/ConfirmDialogProvider";
|
||||||
|
import { useAlert } from "../hooks/providers/AlertDialogProvider";
|
||||||
|
|
||||||
export function GroupVMAction(p: {
|
export function GroupVMAction(p: {
|
||||||
group: VMGroup;
|
group: VMGroup;
|
||||||
@ -19,7 +24,7 @@ export function GroupVMAction(p: {
|
|||||||
allowedStates={["Shutdown", "Shutoff", "Crashed"]}
|
allowedStates={["Shutdown", "Shutoff", "Crashed"]}
|
||||||
currState={p.state}
|
currState={p.state}
|
||||||
needConfirm={false}
|
needConfirm={false}
|
||||||
action={async () => {}}
|
action={GroupApi.StartVM}
|
||||||
/>
|
/>
|
||||||
</Toolbar>
|
</Toolbar>
|
||||||
);
|
);
|
||||||
@ -28,7 +33,7 @@ export function GroupVMAction(p: {
|
|||||||
function GroupVMButton(p: {
|
function GroupVMButton(p: {
|
||||||
enabled: boolean;
|
enabled: boolean;
|
||||||
icon: React.ReactElement;
|
icon: React.ReactElement;
|
||||||
action: (group: VMGroup, vm?: VMGroup) => Promise<void>;
|
action: (group: VMGroup, vm?: VMInfo) => Promise<TreatmentResult>;
|
||||||
tooltip: string;
|
tooltip: string;
|
||||||
currState?: VMState;
|
currState?: VMState;
|
||||||
allowedStates: VMState[];
|
allowedStates: VMState[];
|
||||||
@ -36,24 +41,62 @@ function GroupVMButton(p: {
|
|||||||
vm?: VMInfo;
|
vm?: VMInfo;
|
||||||
needConfirm: boolean;
|
needConfirm: boolean;
|
||||||
}): React.ReactElement {
|
}): React.ReactElement {
|
||||||
const process = () => {};
|
const toast = useToast();
|
||||||
|
const confirm = useConfirm();
|
||||||
|
const alert = useAlert();
|
||||||
|
|
||||||
|
const [running, setRunning] = React.useState(false);
|
||||||
|
|
||||||
|
const target = p.vm
|
||||||
|
? `the VM ${p.vm.name}`
|
||||||
|
: `all the VM of the group ${p.group.id}`;
|
||||||
|
|
||||||
const allowed =
|
const allowed =
|
||||||
!p.vm || (p.currState && p.allowedStates.includes(p.currState));
|
!p.vm || (p.currState && p.allowedStates.includes(p.currState));
|
||||||
|
|
||||||
|
const perform = async () => {
|
||||||
|
if (running || !allowed) return;
|
||||||
|
try {
|
||||||
|
if (
|
||||||
|
(!p.vm || p.needConfirm) &&
|
||||||
|
!(await confirm(
|
||||||
|
`Do you want to perform ${p.tooltip} action on ${target}?`,
|
||||||
|
`Confirmation`,
|
||||||
|
p.tooltip
|
||||||
|
))
|
||||||
|
) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setRunning(true);
|
||||||
|
|
||||||
|
const result = await p.action(p.group, p.vm);
|
||||||
|
|
||||||
|
toast(
|
||||||
|
p.tooltip,
|
||||||
|
`${p.tooltip} action on ${target}: ${result.ok} OK / ${result.failed} Failed`,
|
||||||
|
"success"
|
||||||
|
);
|
||||||
|
} catch (e) {
|
||||||
|
console.error("Failed to perform group action!", e);
|
||||||
|
|
||||||
|
alert(`Failed to perform ${p.tooltip} action on ${target}: ${e}`);
|
||||||
|
} finally {
|
||||||
|
setRunning(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
if (!p.enabled) return <></>;
|
if (!p.enabled) return <></>;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Tooltip
|
<Tooltip
|
||||||
content={`${p.tooltip} ${
|
content={`${p.tooltip} ${target}`}
|
||||||
p.vm ? `the VM ${p.vm.name}` : `all the VM of the group ${p.group.id}`
|
|
||||||
}`}
|
|
||||||
relationship="description"
|
relationship="description"
|
||||||
withArrow
|
withArrow
|
||||||
>
|
>
|
||||||
<Button
|
<Button
|
||||||
icon={p.icon}
|
icon={running ? <Spinner size="tiny" /> : p.icon}
|
||||||
onClick={process}
|
onClick={allowed ? perform : undefined}
|
||||||
disabled={!allowed}
|
disabled={!allowed}
|
||||||
appearance="subtle"
|
appearance="subtle"
|
||||||
/>
|
/>
|
||||||
|
Loading…
Reference in New Issue
Block a user