DB-GPT/web/utils/flow.ts
Dreammy23 746e4fda37
feat(web): Add DAG variables to web flow (#1981)
Co-authored-by: Fangyin Cheng <staneyffer@gmail.com>
Co-authored-by: 谨欣 <echo.cmy@antgroup.com>
Co-authored-by: yanzhiyong <932374019@qq.com>
Co-authored-by: 严志勇 <yanzhiyong@tiansuixiansheng.com>
2024-09-10 09:39:40 +08:00

197 lines
5.8 KiB
TypeScript

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}}`;
}