flow.ts 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. import { IFlowData, IFlowDataNode, IFlowNode, IVariableItem } from '@/types/flow';
  2. import { Node } from 'reactflow';
  3. export const getUniqueNodeId = (nodeData: IFlowNode, nodes: Node[]) => {
  4. let count = 0;
  5. nodes.forEach(node => {
  6. if (node.data.name === nodeData.name) {
  7. count++;
  8. }
  9. });
  10. return `${nodeData.id}_${count}`;
  11. };
  12. // function getUniqueNodeId will add '_${count}' to id, so we need to remove it when we want to get the original id
  13. export const removeIndexFromNodeId = (id: string) => {
  14. const indexPattern = /_\d+$/;
  15. return id.replace(indexPattern, '');
  16. };
  17. // 驼峰转下划线,接口协议字段命名规范
  18. export const mapHumpToUnderline = (flowData: IFlowData) => {
  19. /**
  20. * sourceHandle -> source_handle,
  21. * targetHandle -> target_handle,
  22. * positionAbsolute -> position_absolute
  23. */
  24. const { nodes, edges, ...rest } = flowData;
  25. const newNodes = nodes.map(node => {
  26. const { positionAbsolute, ...rest } = node;
  27. return {
  28. position_absolute: positionAbsolute,
  29. ...rest,
  30. };
  31. });
  32. const newEdges = edges.map(edge => {
  33. const { sourceHandle, targetHandle, ...rest } = edge;
  34. return {
  35. source_handle: sourceHandle,
  36. target_handle: targetHandle,
  37. ...rest,
  38. };
  39. });
  40. return {
  41. nodes: newNodes,
  42. edges: newEdges,
  43. ...rest,
  44. };
  45. };
  46. export const mapUnderlineToHump = (flowData: IFlowData) => {
  47. /**
  48. * source_handle -> sourceHandle,
  49. * target_handle -> targetHandle,
  50. * position_absolute -> positionAbsolute
  51. */
  52. const { nodes, edges, ...rest } = flowData;
  53. const newNodes = nodes.map(node => {
  54. const { position_absolute, ...rest } = node;
  55. return {
  56. positionAbsolute: position_absolute,
  57. ...rest,
  58. };
  59. });
  60. const newEdges = edges.map(edge => {
  61. const { source_handle, target_handle, ...rest } = edge;
  62. return {
  63. sourceHandle: source_handle,
  64. targetHandle: target_handle,
  65. ...rest,
  66. };
  67. });
  68. return {
  69. nodes: newNodes,
  70. edges: newEdges,
  71. ...rest,
  72. };
  73. };
  74. export const checkFlowDataRequied = (flowData: IFlowData) => {
  75. const { nodes, edges } = flowData;
  76. // check the input, parameters that are required
  77. let result: [boolean, IFlowDataNode, string] = [true, nodes[0], ''];
  78. outerLoop: for (let i = 0; i < nodes.length; i++) {
  79. const node = nodes[i].data;
  80. const { inputs = [], parameters = [] } = node;
  81. // check inputs
  82. for (let j = 0; j < inputs.length; j++) {
  83. if (!edges.some(edge => edge.targetHandle === `${nodes[i].id}|inputs|${j}`)) {
  84. result = [false, nodes[i], `The input ${inputs[j].type_name} of node ${node.label} is required`];
  85. break outerLoop;
  86. }
  87. }
  88. // check parameters
  89. for (let k = 0; k < parameters.length; k++) {
  90. const parameter = parameters[k];
  91. if (
  92. !parameter.optional &&
  93. parameter.category === 'resource' &&
  94. !edges.some(edge => edge.targetHandle === `${nodes[i].id}|parameters|${k}`)
  95. ) {
  96. result = [false, nodes[i], `The parameter ${parameter.type_name} of node ${node.label} is required`];
  97. break outerLoop;
  98. } else if (
  99. !parameter.optional &&
  100. parameter.category === 'common' &&
  101. (parameter.value === undefined || parameter.value === null)
  102. ) {
  103. result = [false, nodes[i], `The parameter ${parameter.type_name} of node ${node.label} is required`];
  104. break outerLoop;
  105. }
  106. }
  107. }
  108. return result;
  109. };
  110. export const convertKeysToCamelCase = (obj: Record<string, any>): Record<string, any> => {
  111. function toCamelCase(str: string): string {
  112. return str.replace(/_([a-z])/g, (_, letter) => letter.toUpperCase());
  113. }
  114. function isObject(value: any): boolean {
  115. return value && typeof value === 'object' && !Array.isArray(value);
  116. }
  117. function convert(obj: any): any {
  118. if (Array.isArray(obj)) {
  119. return obj.map(item => convert(item));
  120. } else if (isObject(obj)) {
  121. const newObj: Record<string, any> = {};
  122. for (const key in obj) {
  123. if (Object.prototype.hasOwnProperty.call(obj, key)) {
  124. const newKey = toCamelCase(key);
  125. newObj[newKey] = convert(obj[key]);
  126. }
  127. }
  128. return newObj;
  129. }
  130. return obj;
  131. }
  132. return convert(obj);
  133. };
  134. function escapeVariable(value: string, enableEscape: boolean): string {
  135. if (!enableEscape) {
  136. return value;
  137. }
  138. return value.replace(/@/g, '\\@').replace(/#/g, '\\#').replace(/%/g, '\\%').replace(/:/g, '\\:');
  139. }
  140. export function buildVariableString(variableDict: IVariableItem): string {
  141. const scopeSig = '@';
  142. const sysCodeSig = '#';
  143. const userSig = '%';
  144. const kvSig = ':';
  145. const enableEscape = true;
  146. const specialChars = new Set([scopeSig, sysCodeSig, userSig, kvSig]);
  147. const newVariableDict: Partial<IVariableItem> = {
  148. key: variableDict.key || '',
  149. name: variableDict.name || '',
  150. scope: variableDict.scope || '',
  151. scope_key: variableDict.scope_key || '',
  152. sys_code: variableDict.sys_code || '',
  153. user_name: variableDict.user_name || '',
  154. };
  155. // Check for special characters in values
  156. for (const [key, value] of Object.entries(newVariableDict)) {
  157. if (value && [...specialChars].some(char => (value as string).includes(char))) {
  158. if (enableEscape) {
  159. newVariableDict[key] = escapeVariable(value as string, enableEscape);
  160. } else {
  161. throw new Error(
  162. `${key} contains special characters, error value: ${value}, special characters: ${[...specialChars].join(', ')}`,
  163. );
  164. }
  165. }
  166. }
  167. const { key, name, scope, scope_key, sys_code, user_name } = newVariableDict;
  168. let variableStr = `${key}`;
  169. if (name) variableStr += `${kvSig}${name}`;
  170. if (scope || scope_key) {
  171. variableStr += `${scopeSig}${scope}`;
  172. if (scope_key) {
  173. variableStr += `${kvSig}${scope_key}`;
  174. }
  175. }
  176. if (sys_code) variableStr += `${sysCodeSig}${sys_code}`;
  177. if (user_name) variableStr += `${userSig}${user_name}`;
  178. return `\${${variableStr}}`;
  179. }