123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196 |
- import { IFlowData, IFlowDataNode, IFlowNode, IVariableItem } from '@/types/flow';
- import { Node } from 'reactflow';
- export const getUniqueNodeId = (nodeData: IFlowNode, nodes: Node[]) => {
- let count = 0;
- nodes.forEach(node => {
- if (node.data.name === nodeData.name) {
- count++;
- }
- });
- return `${nodeData.id}_${count}`;
- };
- // function getUniqueNodeId will add '_${count}' to id, so we need to remove it when we want to get the original id
- export const removeIndexFromNodeId = (id: string) => {
- const indexPattern = /_\d+$/;
- return id.replace(indexPattern, '');
- };
- // 驼峰转下划线,接口协议字段命名规范
- export const mapHumpToUnderline = (flowData: IFlowData) => {
- /**
- * sourceHandle -> source_handle,
- * targetHandle -> target_handle,
- * positionAbsolute -> position_absolute
- */
- const { nodes, edges, ...rest } = flowData;
- const newNodes = nodes.map(node => {
- const { positionAbsolute, ...rest } = node;
- return {
- position_absolute: positionAbsolute,
- ...rest,
- };
- });
- const newEdges = edges.map(edge => {
- const { sourceHandle, targetHandle, ...rest } = edge;
- return {
- source_handle: sourceHandle,
- target_handle: targetHandle,
- ...rest,
- };
- });
- return {
- nodes: newNodes,
- edges: newEdges,
- ...rest,
- };
- };
- export const mapUnderlineToHump = (flowData: IFlowData) => {
- /**
- * source_handle -> sourceHandle,
- * target_handle -> targetHandle,
- * position_absolute -> positionAbsolute
- */
- const { nodes, edges, ...rest } = flowData;
- const newNodes = nodes.map(node => {
- const { position_absolute, ...rest } = node;
- return {
- positionAbsolute: position_absolute,
- ...rest,
- };
- });
- const newEdges = edges.map(edge => {
- const { source_handle, target_handle, ...rest } = edge;
- return {
- sourceHandle: source_handle,
- targetHandle: target_handle,
- ...rest,
- };
- });
- return {
- nodes: newNodes,
- edges: newEdges,
- ...rest,
- };
- };
- export const checkFlowDataRequied = (flowData: IFlowData) => {
- const { nodes, edges } = flowData;
- // check the input, parameters that are required
- let result: [boolean, IFlowDataNode, string] = [true, nodes[0], ''];
- outerLoop: for (let i = 0; i < nodes.length; i++) {
- const node = nodes[i].data;
- const { inputs = [], parameters = [] } = node;
- // check inputs
- for (let j = 0; j < inputs.length; j++) {
- if (!edges.some(edge => edge.targetHandle === `${nodes[i].id}|inputs|${j}`)) {
- result = [false, nodes[i], `The input ${inputs[j].type_name} of node ${node.label} is required`];
- break outerLoop;
- }
- }
- // check parameters
- for (let k = 0; k < parameters.length; k++) {
- const parameter = parameters[k];
- if (
- !parameter.optional &&
- parameter.category === 'resource' &&
- !edges.some(edge => edge.targetHandle === `${nodes[i].id}|parameters|${k}`)
- ) {
- result = [false, nodes[i], `The parameter ${parameter.type_name} of node ${node.label} is required`];
- break outerLoop;
- } else if (
- !parameter.optional &&
- parameter.category === 'common' &&
- (parameter.value === undefined || parameter.value === null)
- ) {
- result = [false, nodes[i], `The parameter ${parameter.type_name} of node ${node.label} is required`];
- break outerLoop;
- }
- }
- }
- return result;
- };
- export const convertKeysToCamelCase = (obj: Record<string, any>): Record<string, any> => {
- function toCamelCase(str: string): string {
- return str.replace(/_([a-z])/g, (_, letter) => letter.toUpperCase());
- }
- function isObject(value: any): boolean {
- return value && typeof value === 'object' && !Array.isArray(value);
- }
- function convert(obj: any): any {
- if (Array.isArray(obj)) {
- return obj.map(item => convert(item));
- } else if (isObject(obj)) {
- const newObj: Record<string, any> = {};
- for (const key in obj) {
- if (Object.prototype.hasOwnProperty.call(obj, key)) {
- const newKey = toCamelCase(key);
- newObj[newKey] = convert(obj[key]);
- }
- }
- return newObj;
- }
- return obj;
- }
- return convert(obj);
- };
- function escapeVariable(value: string, enableEscape: boolean): string {
- if (!enableEscape) {
- return value;
- }
- return value.replace(/@/g, '\\@').replace(/#/g, '\\#').replace(/%/g, '\\%').replace(/:/g, '\\:');
- }
- export function buildVariableString(variableDict: IVariableItem): string {
- const scopeSig = '@';
- const sysCodeSig = '#';
- const userSig = '%';
- const kvSig = ':';
- const enableEscape = true;
- const specialChars = new Set([scopeSig, sysCodeSig, userSig, kvSig]);
- const newVariableDict: Partial<IVariableItem> = {
- key: variableDict.key || '',
- name: variableDict.name || '',
- scope: variableDict.scope || '',
- scope_key: variableDict.scope_key || '',
- sys_code: variableDict.sys_code || '',
- user_name: variableDict.user_name || '',
- };
- // Check for special characters in values
- for (const [key, value] of Object.entries(newVariableDict)) {
- if (value && [...specialChars].some(char => (value as string).includes(char))) {
- if (enableEscape) {
- newVariableDict[key] = escapeVariable(value as string, enableEscape);
- } else {
- throw new Error(
- `${key} contains special characters, error value: ${value}, special characters: ${[...specialChars].join(', ')}`,
- );
- }
- }
- }
- const { key, name, scope, scope_key, sys_code, user_name } = newVariableDict;
- let variableStr = `${key}`;
- if (name) variableStr += `${kvSig}${name}`;
- if (scope || scope_key) {
- variableStr += `${scopeSig}${scope}`;
- if (scope_key) {
- variableStr += `${kvSig}${scope_key}`;
- }
- }
- if (sys_code) variableStr += `${sysCodeSig}${sys_code}`;
- if (user_name) variableStr += `${userSig}${user_name}`;
- return `\${${variableStr}}`;
- }
|