anvil/striker-ui/components/StrikerInitForm.tsx

241 lines
8.1 KiB
TypeScript

import { Grid } from '@mui/material';
import { FC, useCallback, useMemo, useRef, useState } from 'react';
import api from '../lib/api';
import ConfirmDialog from './ConfirmDialog';
import ContainedButton from './ContainedButton';
import FlexBox from './FlexBox';
import GeneralInitForm, {
GeneralInitFormForwardedRefContent,
GeneralInitFormValues,
} from './GeneralInitForm';
import handleAPIError from '../lib/handleAPIError';
import MessageBox, { Message } from './MessageBox';
import NetworkInitForm, {
NetworkInitFormForwardedRefContent,
NetworkInitFormValues,
} from './NetworkInitForm';
import { Panel, PanelHeader } from './Panels';
import Spinner from './Spinner';
import { BodyText, HeaderText, InlineMonoText, MonoText } from './Text';
const StrikerInitForm: FC = () => {
const [submitMessage, setSubmitMessage] = useState<Message | undefined>();
const [requestBody, setRequestBody] = useState<
(GeneralInitFormValues & NetworkInitFormValues) | undefined
>();
const [isOpenConfirm, setIsOpenConfirm] = useState<boolean>(false);
const [isDisableSubmit, setIsDisableSubmit] = useState<boolean>(true);
const [isGeneralInitFormValid, setIsGeneralInitFormValid] =
useState<boolean>(false);
const [isNetworkInitFormValid, setIsNetworkInitFormValid] =
useState<boolean>(false);
const [isSubmittingForm, setIsSubmittingForm] = useState<boolean>(false);
const generalInitFormRef = useRef<GeneralInitFormForwardedRefContent>({});
const networkInitFormRef = useRef<NetworkInitFormForwardedRefContent>({});
const buildSubmitSection = useMemo(
() =>
isSubmittingForm ? (
<Spinner />
) : (
<FlexBox row sx={{ flexDirection: 'row-reverse' }}>
<ContainedButton
disabled={isDisableSubmit}
onClick={() => {
setRequestBody({
...(generalInitFormRef.current.get?.call(null) ?? {}),
...(networkInitFormRef.current.get?.call(null) ?? {
networks: [],
}),
});
setIsOpenConfirm(true);
}}
>
Initialize
</ContainedButton>
</FlexBox>
),
[isDisableSubmit, isSubmittingForm],
);
const toggleSubmitDisabled = useCallback((...testResults: boolean[]) => {
setIsDisableSubmit(!testResults.every((testResult) => testResult));
}, []);
return (
<>
<Panel>
<PanelHeader>
<HeaderText text="Initialize striker" />
</PanelHeader>
<FlexBox>
<GeneralInitForm
ref={generalInitFormRef}
toggleSubmitDisabled={(testResult) => {
if (testResult !== isGeneralInitFormValid) {
setIsGeneralInitFormValid(testResult);
toggleSubmitDisabled(testResult, isNetworkInitFormValid);
}
}}
/>
<NetworkInitForm
ref={networkInitFormRef}
toggleSubmitDisabled={(testResult) => {
if (testResult !== isNetworkInitFormValid) {
setIsNetworkInitFormValid(testResult);
toggleSubmitDisabled(isGeneralInitFormValid, testResult);
}
}}
/>
{submitMessage && (
<MessageBox
{...submitMessage}
onClose={() => setSubmitMessage(undefined)}
/>
)}
{buildSubmitSection}
</FlexBox>
</Panel>
<ConfirmDialog
actionProceedText="Initialize"
content={
<Grid container spacing=".6em" columns={{ xs: 2 }}>
<Grid item xs={1}>
<BodyText>Organization name</BodyText>
</Grid>
<Grid item xs={1}>
<MonoText>{requestBody?.organizationName}</MonoText>
</Grid>
<Grid item xs={1}>
<BodyText>Organization prefix</BodyText>
</Grid>
<Grid item xs={1}>
<MonoText>{requestBody?.organizationPrefix}</MonoText>
</Grid>
<Grid item xs={1}>
<BodyText>Striker number</BodyText>
</Grid>
<Grid item xs={1}>
<MonoText>{requestBody?.hostNumber}</MonoText>
</Grid>
<Grid item xs={1}>
<BodyText>Domain name</BodyText>
</Grid>
<Grid item xs={1}>
<MonoText>{requestBody?.domainName}</MonoText>
</Grid>
<Grid item xs={1}>
<BodyText>Host name</BodyText>
</Grid>
<Grid item xs={1}>
<MonoText>{requestBody?.hostName}</MonoText>
</Grid>
<Grid item sx={{ marginTop: '1.4em' }} xs={2}>
<BodyText>Networks</BodyText>
</Grid>
{requestBody?.networks.map(
({
inputUUID,
interfaces,
ipAddress,
name,
subnetMask,
type,
typeCount,
}) => (
<Grid key={`network-confirm-${inputUUID}`} item xs={1}>
<Grid container spacing=".6em" columns={{ xs: 2 }}>
<Grid item xs={2}>
<BodyText>
{name} (
<InlineMonoText>
{`${type.toUpperCase()}${typeCount}`}
</InlineMonoText>
)
</BodyText>
</Grid>
{interfaces.map((iface, ifaceIndex) => {
let key = `network-confirm-${inputUUID}-interface${ifaceIndex}`;
let ifaceName = 'none';
if (iface) {
const { networkInterfaceName, networkInterfaceUUID } =
iface;
key = `${key}-${networkInterfaceUUID}`;
ifaceName = networkInterfaceName;
}
return (
<Grid container key={key} item>
<Grid item xs={1}>
<BodyText>{`Link ${ifaceIndex + 1}`}</BodyText>
</Grid>
<Grid item xs={1}>
<MonoText>{ifaceName}</MonoText>
</Grid>
</Grid>
);
})}
<Grid item xs={2}>
<MonoText>{`${ipAddress}/${subnetMask}`}</MonoText>
</Grid>
</Grid>
</Grid>
),
)}
<Grid item sx={{ marginBottom: '1.4em' }} xs={2} />
<Grid item xs={1}>
<BodyText>Gateway</BodyText>
</Grid>
<Grid item xs={1}>
<MonoText>{requestBody?.gateway}</MonoText>
</Grid>
<Grid item xs={1}>
<BodyText>Gateway network</BodyText>
</Grid>
<Grid item xs={1}>
<MonoText>
{requestBody?.gatewayInterface?.toUpperCase()}
</MonoText>
</Grid>
<Grid item xs={1}>
<BodyText>Domain name server(s)</BodyText>
</Grid>
<Grid item xs={1}>
<MonoText>{requestBody?.domainNameServerCSV}</MonoText>
</Grid>
</Grid>
}
dialogProps={{ open: isOpenConfirm }}
onCancelAppend={() => {
setIsOpenConfirm(false);
}}
onProceedAppend={() => {
setSubmitMessage(undefined);
setIsSubmittingForm(true);
setIsOpenConfirm(false);
api
.post('/host', requestBody)
.then(() => {
setIsSubmittingForm(false);
})
.catch((error) => {
const errorMessage = handleAPIError(error);
setSubmitMessage(errorMessage);
setIsSubmittingForm(false);
});
}}
titleText="Confirm striker initialization"
/>
</>
);
};
export default StrikerInitForm;