367 lines
10 KiB
TypeScript
367 lines
10 KiB
TypeScript
import { useRouter } from 'next/router';
|
|
import { FC, useCallback, useEffect, useMemo, useRef } from 'react';
|
|
|
|
import api from '../lib/api';
|
|
import ConfirmDialog from './ConfirmDialog';
|
|
import ContainedButton from './ContainedButton';
|
|
import FlexBox from './FlexBox';
|
|
import FormSummary from './FormSummary';
|
|
import getQueryParam from '../lib/getQueryParam';
|
|
import handleAPIError from '../lib/handleAPIError';
|
|
import InputWithRef from './InputWithRef';
|
|
import MessageBox, { Message } from './MessageBox';
|
|
import MessageGroup, { MessageGroupForwardedRefContent } from './MessageGroup';
|
|
import NetworkInitForm, {
|
|
NetworkInitFormForwardedRefContent,
|
|
} from './NetworkInitForm';
|
|
import OutlinedInputWithLabel from './OutlinedInputWithLabel';
|
|
import { Panel, PanelHeader } from './Panels';
|
|
import setMapNetwork from '../lib/setMapNetwork';
|
|
import Spinner from './Spinner';
|
|
import { buildPeacefulStringTestBatch } from '../lib/test_input';
|
|
import { HeaderText } from './Text';
|
|
import useConfirmDialogProps from '../hooks/useConfirmDialogProps';
|
|
import useFormUtils from '../hooks/useFormUtils';
|
|
import useProtectedState from '../hooks/useProtectedState';
|
|
|
|
const INPUT_ID_PREP_NET_HOST_NAME = 'prepare-network-host-name-input';
|
|
|
|
const INPUT_GROUP_ID_PREP_NET_NETCONF = 'prepare-network-netconf-input-group';
|
|
|
|
const INPUT_LABEL_PREP_NET_HOST_NAME = 'Host name';
|
|
|
|
const getFormData = (
|
|
{
|
|
netconf,
|
|
}: {
|
|
netconf: NetworkInitFormForwardedRefContent;
|
|
},
|
|
...[{ target }]: DivFormEventHandlerParameters
|
|
) => {
|
|
const { elements } = target as HTMLFormElement;
|
|
|
|
const { value: hostName } = elements.namedItem(
|
|
INPUT_ID_PREP_NET_HOST_NAME,
|
|
) as HTMLInputElement;
|
|
|
|
const data = {
|
|
hostName,
|
|
...netconf.get?.call(null),
|
|
};
|
|
|
|
return data;
|
|
};
|
|
|
|
const toFormEntries = (body: ReturnType<typeof getFormData>): FormEntries => {
|
|
const { networks: nets = [], ...rest } = body;
|
|
|
|
const networks = nets.reduce<FormEntries>((previous, network) => {
|
|
const {
|
|
interfaces: ifaces,
|
|
ipAddress,
|
|
name = '',
|
|
type,
|
|
typeCount,
|
|
subnetMask,
|
|
} = network;
|
|
const networkId = `${type}${typeCount}`;
|
|
|
|
const interfaces = ifaces.reduce<FormEntries>((pIfaces, iface, index) => {
|
|
if (iface) {
|
|
const { networkInterfaceName } = iface;
|
|
const linkNumber = index + 1;
|
|
|
|
pIfaces[`link${linkNumber}`] = networkInterfaceName;
|
|
}
|
|
|
|
return pIfaces;
|
|
}, {});
|
|
|
|
previous[networkId] = {
|
|
name,
|
|
network: `${ipAddress}/${subnetMask}`,
|
|
...interfaces,
|
|
};
|
|
|
|
return previous;
|
|
}, {});
|
|
|
|
return { ...rest, networks };
|
|
};
|
|
|
|
const PrepareNetworkForm: FC<PrepareNetworkFormProps> = ({
|
|
expectUUID: isExpectExternalHostUUID = false,
|
|
hostUUID,
|
|
}) => {
|
|
const {
|
|
isReady,
|
|
query: { host_uuid: queryHostUUID },
|
|
} = useRouter();
|
|
|
|
const confirmDialogRef = useRef<ConfirmDialogForwardedRefContent>({});
|
|
const netconfFormRef = useRef<NetworkInitFormForwardedRefContent>({});
|
|
|
|
const generalInputMessageGroupRef = useRef<MessageGroupForwardedRefContent>(
|
|
{},
|
|
);
|
|
|
|
const [confirmDialogProps, setConfirmDialogProps] = useConfirmDialogProps();
|
|
|
|
const [hostDetail, setHostDetail] = useProtectedState<
|
|
APIHostDetail | undefined
|
|
>(undefined);
|
|
const [fatalErrorMessage, setFatalErrorMessage] = useProtectedState<
|
|
Message | undefined
|
|
>(undefined);
|
|
const [isLoadingHostDetail, setIsLoadingHostDetail] =
|
|
useProtectedState<boolean>(true);
|
|
const [previousHostUUID, setPreviousHostUUID] =
|
|
useProtectedState<PrepareNetworkFormProps['hostUUID']>(undefined);
|
|
|
|
const {
|
|
buildFinishInputTestBatchFunction,
|
|
buildInputFirstRenderFunction,
|
|
isFormInvalid,
|
|
setMessage,
|
|
setValidity,
|
|
submitForm,
|
|
} = useFormUtils(
|
|
[INPUT_ID_PREP_NET_HOST_NAME, INPUT_GROUP_ID_PREP_NET_NETCONF],
|
|
generalInputMessageGroupRef,
|
|
);
|
|
|
|
const isDifferentHostUUID = useMemo(
|
|
() => hostUUID !== previousHostUUID,
|
|
[hostUUID, previousHostUUID],
|
|
);
|
|
const isReloadHostDetail = useMemo(
|
|
() => Boolean(hostUUID) && isDifferentHostUUID,
|
|
[hostUUID, isDifferentHostUUID],
|
|
);
|
|
|
|
const panelHeaderElement = useMemo(
|
|
() => (
|
|
<PanelHeader>
|
|
<HeaderText>Prepare network on {hostDetail?.shortHostName}</HeaderText>
|
|
</PanelHeader>
|
|
),
|
|
[hostDetail],
|
|
);
|
|
const netconfForm = useMemo(
|
|
() => (
|
|
<NetworkInitForm
|
|
expectHostDetail
|
|
hostDetail={hostDetail}
|
|
ref={netconfFormRef}
|
|
toggleSubmitDisabled={(valid) => {
|
|
setValidity(INPUT_GROUP_ID_PREP_NET_NETCONF, valid);
|
|
}}
|
|
/>
|
|
),
|
|
[hostDetail, setValidity],
|
|
);
|
|
const generalInputMessageArea = useMemo(
|
|
() => (
|
|
<MessageGroup
|
|
count={1}
|
|
defaultMessageType="warning"
|
|
ref={generalInputMessageGroupRef}
|
|
/>
|
|
),
|
|
[],
|
|
);
|
|
|
|
const contentElement = useMemo(() => {
|
|
let result;
|
|
|
|
if (isLoadingHostDetail) {
|
|
result = <Spinner mt={0} />;
|
|
} else if (fatalErrorMessage) {
|
|
result = <MessageBox {...fatalErrorMessage} />;
|
|
} else {
|
|
result = (
|
|
<>
|
|
{panelHeaderElement}
|
|
<FlexBox
|
|
component="form"
|
|
onSubmit={(...args) => {
|
|
const [event] = args;
|
|
|
|
event.preventDefault();
|
|
|
|
const body = getFormData(
|
|
{ netconf: netconfFormRef.current },
|
|
...args,
|
|
);
|
|
|
|
setConfirmDialogProps({
|
|
actionProceedText: 'Prepare',
|
|
content: (
|
|
<FormSummary
|
|
entries={toFormEntries(body)}
|
|
getEntryLabel={({ cap, key }) =>
|
|
/^(dns|[a-z]+n\d+)/.test(key)
|
|
? key.toUpperCase()
|
|
: cap(key)
|
|
}
|
|
/>
|
|
),
|
|
onProceedAppend: () => {
|
|
submitForm({
|
|
body,
|
|
getErrorMsg: (parentMsg) => (
|
|
<>Failed to prepare network. {parentMsg}</>
|
|
),
|
|
method: 'put',
|
|
onSuccess: () => {
|
|
setMapNetwork(0);
|
|
},
|
|
setMsg: netconfFormRef?.current?.setMessage,
|
|
successMsg: `Initiated prepare network on ${hostDetail?.shortHostName}`,
|
|
url: `/host/${hostUUID}?handler=subnode-network`,
|
|
});
|
|
},
|
|
titleText: `Prepare ${hostDetail?.shortHostName} network?`,
|
|
});
|
|
|
|
confirmDialogRef.current.setOpen?.call(null, true);
|
|
}}
|
|
>
|
|
<InputWithRef
|
|
input={
|
|
<OutlinedInputWithLabel
|
|
formControlProps={{ sx: { maxWidth: '20em' } }}
|
|
id={INPUT_ID_PREP_NET_HOST_NAME}
|
|
label={INPUT_LABEL_PREP_NET_HOST_NAME}
|
|
value={hostDetail?.hostName}
|
|
/>
|
|
}
|
|
inputTestBatch={buildPeacefulStringTestBatch(
|
|
INPUT_LABEL_PREP_NET_HOST_NAME,
|
|
() => {
|
|
setMessage(INPUT_ID_PREP_NET_HOST_NAME);
|
|
},
|
|
{
|
|
onFinishBatch: buildFinishInputTestBatchFunction(
|
|
INPUT_ID_PREP_NET_HOST_NAME,
|
|
),
|
|
},
|
|
(message) => {
|
|
setMessage(INPUT_ID_PREP_NET_HOST_NAME, {
|
|
children: message,
|
|
});
|
|
},
|
|
)}
|
|
onFirstRender={buildInputFirstRenderFunction(
|
|
INPUT_ID_PREP_NET_HOST_NAME,
|
|
)}
|
|
required
|
|
/>
|
|
{generalInputMessageArea}
|
|
{netconfForm}
|
|
<FlexBox row justifyContent="flex-end">
|
|
<ContainedButton disabled={isFormInvalid} type="submit">
|
|
Prepare network
|
|
</ContainedButton>
|
|
</FlexBox>
|
|
</FlexBox>
|
|
</>
|
|
);
|
|
}
|
|
|
|
return result;
|
|
}, [
|
|
isLoadingHostDetail,
|
|
fatalErrorMessage,
|
|
panelHeaderElement,
|
|
hostDetail?.hostName,
|
|
hostDetail?.shortHostName,
|
|
buildFinishInputTestBatchFunction,
|
|
buildInputFirstRenderFunction,
|
|
generalInputMessageArea,
|
|
netconfForm,
|
|
isFormInvalid,
|
|
setConfirmDialogProps,
|
|
submitForm,
|
|
hostUUID,
|
|
setMessage,
|
|
]);
|
|
|
|
const getHostDetail = useCallback(
|
|
(uuid: string) => {
|
|
setIsLoadingHostDetail(true);
|
|
|
|
if (isLoadingHostDetail) {
|
|
api
|
|
.get<APIHostDetail>(`/host/${uuid}`)
|
|
.then(({ data }) => {
|
|
setPreviousHostUUID(data.hostUUID);
|
|
setHostDetail(data);
|
|
})
|
|
.catch((error) => {
|
|
const { children } = handleAPIError(error);
|
|
|
|
setFatalErrorMessage({
|
|
children: `Failed to get target host information; cannot continue. ${children}`,
|
|
type: 'error',
|
|
});
|
|
})
|
|
.finally(() => {
|
|
setIsLoadingHostDetail(false);
|
|
});
|
|
}
|
|
},
|
|
[
|
|
setIsLoadingHostDetail,
|
|
isLoadingHostDetail,
|
|
setPreviousHostUUID,
|
|
setHostDetail,
|
|
setFatalErrorMessage,
|
|
],
|
|
);
|
|
|
|
useEffect(() => {
|
|
if (isExpectExternalHostUUID) {
|
|
if (isReloadHostDetail) {
|
|
getHostDetail(hostUUID as string);
|
|
}
|
|
} else if (isReady && !fatalErrorMessage) {
|
|
if (queryHostUUID) {
|
|
getHostDetail(getQueryParam(queryHostUUID));
|
|
} else {
|
|
setFatalErrorMessage({
|
|
children: `No host UUID provided; cannot continue.`,
|
|
type: 'error',
|
|
});
|
|
|
|
setIsLoadingHostDetail(false);
|
|
}
|
|
}
|
|
}, [
|
|
fatalErrorMessage,
|
|
getHostDetail,
|
|
hostUUID,
|
|
isExpectExternalHostUUID,
|
|
isReady,
|
|
queryHostUUID,
|
|
setFatalErrorMessage,
|
|
setHostDetail,
|
|
setIsLoadingHostDetail,
|
|
isReloadHostDetail,
|
|
]);
|
|
|
|
return (
|
|
<>
|
|
<Panel>{contentElement}</Panel>
|
|
<ConfirmDialog
|
|
closeOnProceed
|
|
scrollContent
|
|
{...confirmDialogProps}
|
|
ref={confirmDialogRef}
|
|
/>
|
|
</>
|
|
);
|
|
};
|
|
|
|
export default PrepareNetworkForm;
|