client/src/preview/route/digitaltwins/execute/pipelineUtils.ts
import { Dispatch, SetStateAction } from 'react';
import DigitalTwin, { formatName } from 'preview/util/digitalTwin';
import GitlabInstance from 'preview/util/gitlab';
import {
setJobLogs,
setPipelineCompleted,
setPipelineLoading,
} from 'preview/store/digitalTwin.slice';
import { useDispatch } from 'react-redux';
import { showSnackbar } from 'preview/store/snackbar.slice';
export const startPipeline = async (
digitalTwin: DigitalTwin,
dispatch: ReturnType<typeof useDispatch>,
setLogButtonDisabled: Dispatch<SetStateAction<boolean>>,
) => {
await digitalTwin.execute();
const executionStatusMessage =
digitalTwin.lastExecutionStatus === 'success'
? `Execution started successfully for ${formatName(digitalTwin.DTName)}. Wait until completion for the logs...`
: `Execution ${digitalTwin.lastExecutionStatus} for ${formatName(digitalTwin.DTName)}`;
dispatch(
showSnackbar({
message: executionStatusMessage,
severity:
digitalTwin.lastExecutionStatus === 'success' ? 'success' : 'error',
}),
);
setLogButtonDisabled(true);
};
export const updatePipelineState = (
digitalTwin: DigitalTwin,
dispatch: ReturnType<typeof useDispatch>,
) => {
dispatch(
setPipelineCompleted({
assetName: digitalTwin.DTName,
pipelineCompleted: false,
}),
);
dispatch(
setPipelineLoading({
assetName: digitalTwin.DTName,
pipelineLoading: true,
}),
);
};
export const updatePipelineStateOnCompletion = (
digitalTwin: DigitalTwin,
jobLogs: { jobName: string; log: string }[],
setButtonText: Dispatch<SetStateAction<string>>,
setLogButtonDisabled: Dispatch<SetStateAction<boolean>>,
dispatch: ReturnType<typeof useDispatch>,
) => {
dispatch(setJobLogs({ assetName: digitalTwin.DTName, jobLogs }));
dispatch(
setPipelineCompleted({
assetName: digitalTwin.DTName,
pipelineCompleted: true,
}),
);
dispatch(
setPipelineLoading({
assetName: digitalTwin.DTName,
pipelineLoading: false,
}),
);
setButtonText('Start');
setLogButtonDisabled(false);
};
export const updatePipelineStateOnStop = (
digitalTwin: DigitalTwin,
setButtonText: Dispatch<SetStateAction<string>>,
dispatch: ReturnType<typeof useDispatch>,
) => {
setButtonText('Start');
dispatch(
setPipelineCompleted({
assetName: digitalTwin.DTName,
pipelineCompleted: true,
}),
);
dispatch(
setPipelineLoading({
assetName: digitalTwin.DTName,
pipelineLoading: false,
}),
);
};
export const fetchJobLogs = async (
gitlabInstance: GitlabInstance,
pipelineId: number,
) => {
const jobs = await gitlabInstance.getPipelineJobs(
gitlabInstance.projectId!,
pipelineId,
);
const logPromises = jobs.map(async (job) => {
const log = await gitlabInstance.getJobTrace(
gitlabInstance.projectId!,
job.id,
);
if (typeof log === 'string') {
log
.replace(
// TODO: Fix ansi character stripping
// eslint-disable-next-line no-control-regex
/[\u001b\u009b][[()#;?]*(?:[0-9]{1,4}(?:;[0-9]{0,4})*)?[0-9A-ORZcf-nqry=><]/g,
'',
)
.split('\n')
.map((line: string) =>
line
.replace(/section_start:\d+:[^A-Z]*/, '')
.replace(/section_end:\d+:[^A-Z]*/, ''),
)
.join('\n');
}
return { jobName: job.name, log };
});
return (await Promise.all(logPromises)).reverse();
};