@ai-setting/roy-agent-core 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (378) hide show
  1. package/dist/index.js +99145 -0
  2. package/package.json +114 -0
  3. package/src/config/config-component.test.ts +627 -0
  4. package/src/config/config-component.ts +906 -0
  5. package/src/config/config-parser.test.ts +319 -0
  6. package/src/config/config-parser.ts +203 -0
  7. package/src/config/decentralized-config.test.ts +740 -0
  8. package/src/config/env-key.ts +210 -0
  9. package/src/config/env-source.test.ts +252 -0
  10. package/src/config/env-source.ts +301 -0
  11. package/src/config/file-source.test.ts +357 -0
  12. package/src/config/file-source.ts +421 -0
  13. package/src/config/index.ts +24 -0
  14. package/src/config/protocol-resolver.test.ts +217 -0
  15. package/src/config/protocol-resolver.ts +228 -0
  16. package/src/env/agent/agent-component.abort.test.ts +511 -0
  17. package/src/env/agent/agent-component.record-session.test.ts +349 -0
  18. package/src/env/agent/agent-component.test.ts +1389 -0
  19. package/src/env/agent/agent-component.tool-error.test.ts +327 -0
  20. package/src/env/agent/agent-component.ts +1711 -0
  21. package/src/env/agent/agent-config-registration.test.ts +226 -0
  22. package/src/env/agent/agent-config-registration.ts +46 -0
  23. package/src/env/agent/agent-reminder-plugin.integration.test.ts +243 -0
  24. package/src/env/agent/index.ts +10 -0
  25. package/src/env/agent/summary-agent.parse-hint.test.ts +360 -0
  26. package/src/env/agent/summary-agent.ts +508 -0
  27. package/src/env/agent/types.ts +536 -0
  28. package/src/env/commands/commands-component.test.ts +364 -0
  29. package/src/env/commands/commands-component.ts +604 -0
  30. package/src/env/commands/commands-config-registration.test.ts +198 -0
  31. package/src/env/commands/commands-config-registration.ts +38 -0
  32. package/src/env/commands/index.ts +21 -0
  33. package/src/env/commands/parser.test.ts +203 -0
  34. package/src/env/commands/parser.ts +115 -0
  35. package/src/env/commands/types.ts +184 -0
  36. package/src/env/commands-prompt-integration.test.ts +243 -0
  37. package/src/env/component-env.test.ts +119 -0
  38. package/src/env/component.ts +335 -0
  39. package/src/env/constants.test.ts +72 -0
  40. package/src/env/constants.ts +123 -0
  41. package/src/env/debug/debug-component.test.ts +114 -0
  42. package/src/env/debug/debug-component.ts +547 -0
  43. package/src/env/debug/formatters/index.ts +9 -0
  44. package/src/env/debug/formatters/repl-formatter.test.ts +139 -0
  45. package/src/env/debug/formatters/repl-formatter.ts +358 -0
  46. package/src/env/debug/formatters/trace-formatter.test.ts +119 -0
  47. package/src/env/debug/formatters/trace-formatter.ts +191 -0
  48. package/src/env/debug/formatters/tree-formatter.test.ts +107 -0
  49. package/src/env/debug/formatters/tree-formatter.ts +325 -0
  50. package/src/env/debug/index.ts +38 -0
  51. package/src/env/debug/parser/regex-parser.test.ts +201 -0
  52. package/src/env/debug/parser/regex-parser.ts +196 -0
  53. package/src/env/debug/parser/span-builder.test.ts +241 -0
  54. package/src/env/debug/parser/span-builder.ts +386 -0
  55. package/src/env/debug/reader/log-reader.test.ts +170 -0
  56. package/src/env/debug/reader/log-reader.ts +186 -0
  57. package/src/env/debug/reader/span-db-reader.test.ts +118 -0
  58. package/src/env/debug/reader/span-db-reader.ts +201 -0
  59. package/src/env/debug/types.test.ts +187 -0
  60. package/src/env/debug/types.ts +171 -0
  61. package/src/env/environment-init.test.ts +183 -0
  62. package/src/env/environment-lifecycle.test.ts +516 -0
  63. package/src/env/environment-service.test.ts +332 -0
  64. package/src/env/environment.handle-query.test.ts +96 -0
  65. package/src/env/environment.test.ts +232 -0
  66. package/src/env/environment.ts +708 -0
  67. package/src/env/errors.test.ts +165 -0
  68. package/src/env/errors.ts +157 -0
  69. package/src/env/event-source/event-source-agent-handler.test.ts +193 -0
  70. package/src/env/event-source/event-source-agent-handler.ts +111 -0
  71. package/src/env/event-source/event-source-component.process-cleanup.test.ts +236 -0
  72. package/src/env/event-source/event-source-component.stop.test.ts +346 -0
  73. package/src/env/event-source/event-source-component.test.ts +1207 -0
  74. package/src/env/event-source/event-source-component.ts +1379 -0
  75. package/src/env/event-source/event-source-config-registration.test.ts +242 -0
  76. package/src/env/event-source/event-source-config-registration.ts +37 -0
  77. package/src/env/event-source/event-source-integration.test.ts +320 -0
  78. package/src/env/event-source/event-source-platform.test.ts +630 -0
  79. package/src/env/event-source/types.ts +298 -0
  80. package/src/env/hook/global-hook-manager.ts +162 -0
  81. package/src/env/hook/hook-manager.test.ts +374 -0
  82. package/src/env/hook/hook-manager.ts +309 -0
  83. package/src/env/hook/index.ts +38 -0
  84. package/src/env/hook/types.ts +138 -0
  85. package/src/env/index.ts +144 -0
  86. package/src/env/interface.ts +203 -0
  87. package/src/env/llm/hooks.test.ts +293 -0
  88. package/src/env/llm/hooks.ts +316 -0
  89. package/src/env/llm/index.ts +61 -0
  90. package/src/env/llm/invoke-threshold-check.test.ts +88 -0
  91. package/src/env/llm/invoke-timeout.test.ts +54 -0
  92. package/src/env/llm/invoke.test.ts +71 -0
  93. package/src/env/llm/invoke.ts +1039 -0
  94. package/src/env/llm/llm-config.test.ts +523 -0
  95. package/src/env/llm/llm.test.ts +233 -0
  96. package/src/env/llm/llm.ts +568 -0
  97. package/src/env/llm/provider.test.ts +182 -0
  98. package/src/env/llm/provider.ts +108 -0
  99. package/src/env/llm/transform.test.ts +251 -0
  100. package/src/env/llm/transform.ts +286 -0
  101. package/src/env/llm/types.test.ts +580 -0
  102. package/src/env/llm/types.ts +424 -0
  103. package/src/env/log-trace/decorator-otel.test.ts +182 -0
  104. package/src/env/log-trace/decorator.ts +230 -0
  105. package/src/env/log-trace/index.ts +79 -0
  106. package/src/env/log-trace/log-trace-component.test.ts +242 -0
  107. package/src/env/log-trace/log-trace-component.ts +497 -0
  108. package/src/env/log-trace/log-trace-config-registration.test.ts +348 -0
  109. package/src/env/log-trace/log-trace-config-registration.ts +45 -0
  110. package/src/env/log-trace/logger.test.ts +149 -0
  111. package/src/env/log-trace/logger.ts +522 -0
  112. package/src/env/log-trace/opentelemetry/cli-propagation.test.ts +147 -0
  113. package/src/env/log-trace/opentelemetry/cli-propagation.ts +194 -0
  114. package/src/env/log-trace/opentelemetry/integration.test.ts +668 -0
  115. package/src/env/log-trace/opentelemetry/mod.ts +25 -0
  116. package/src/env/log-trace/opentelemetry/propagation-env.test.ts +181 -0
  117. package/src/env/log-trace/opentelemetry/propagation-env.ts +136 -0
  118. package/src/env/log-trace/opentelemetry/propagation.test.ts +259 -0
  119. package/src/env/log-trace/opentelemetry/propagation.ts +215 -0
  120. package/src/env/log-trace/opentelemetry/tracer-provider-context.test.ts +166 -0
  121. package/src/env/log-trace/opentelemetry/tracer-provider.test.ts +379 -0
  122. package/src/env/log-trace/opentelemetry/tracer-provider.ts +612 -0
  123. package/src/env/log-trace/span-storage.test.ts +145 -0
  124. package/src/env/log-trace/span-storage.ts +230 -0
  125. package/src/env/log-trace/trace-context.test.ts +187 -0
  126. package/src/env/log-trace/trace-context.ts +162 -0
  127. package/src/env/log-trace/types.test.ts +63 -0
  128. package/src/env/log-trace/types.ts +172 -0
  129. package/src/env/mcp/README.md +244 -0
  130. package/src/env/mcp/__integration__/mcp-component.integration.test.ts +373 -0
  131. package/src/env/mcp/config.test.ts +74 -0
  132. package/src/env/mcp/config.ts +116 -0
  133. package/src/env/mcp/index.ts +41 -0
  134. package/src/env/mcp/loader.test.ts +161 -0
  135. package/src/env/mcp/loader.ts +209 -0
  136. package/src/env/mcp/mcp-component.test.ts +111 -0
  137. package/src/env/mcp/mcp-component.ts +358 -0
  138. package/src/env/mcp/mcp-config-registration.test.ts +304 -0
  139. package/src/env/mcp/mcp-config-registration.ts +50 -0
  140. package/src/env/mcp/scanner.test.ts +170 -0
  141. package/src/env/mcp/scanner.ts +246 -0
  142. package/src/env/mcp/tool/adapter.test.ts +520 -0
  143. package/src/env/mcp/tool/adapter.ts +521 -0
  144. package/src/env/mcp/tool/index.ts +5 -0
  145. package/src/env/mcp/types.test.ts +171 -0
  146. package/src/env/mcp/types.ts +79 -0
  147. package/src/env/memory/README.md +177 -0
  148. package/src/env/memory/built-in/index.ts +59 -0
  149. package/src/env/memory/built-in/recall-memory.ts +103 -0
  150. package/src/env/memory/built-in/record-memory.ts +148 -0
  151. package/src/env/memory/index.ts +20 -0
  152. package/src/env/memory/memory-component.test.ts +239 -0
  153. package/src/env/memory/memory-component.ts +503 -0
  154. package/src/env/memory/memory-config-registration.test.ts +67 -0
  155. package/src/env/memory/memory-config-registration.ts +48 -0
  156. package/src/env/memory/memory-config.ts +45 -0
  157. package/src/env/memory/memory-file.test.ts +268 -0
  158. package/src/env/memory/plugin/index.ts +48 -0
  159. package/src/env/memory/plugin/memory-agent.test.ts +249 -0
  160. package/src/env/memory/plugin/memory-agent.ts +365 -0
  161. package/src/env/memory/plugin/memory-manager.ts +198 -0
  162. package/src/env/memory/plugin/memory-plugin-agent.test.ts +145 -0
  163. package/src/env/memory/plugin/memory-plugin.ts +210 -0
  164. package/src/env/memory/plugin/plugin-simplified.test.ts +51 -0
  165. package/src/env/memory/plugin/recall-memory.test.ts +106 -0
  166. package/src/env/memory/plugin/recall-memory.ts +53 -0
  167. package/src/env/memory/plugin/types.ts +101 -0
  168. package/src/env/memory/tools/memory-agent-tools.ts +228 -0
  169. package/src/env/memory/types.ts +85 -0
  170. package/src/env/paths.ts +118 -0
  171. package/src/env/prompt/index.ts +18 -0
  172. package/src/env/prompt/memory-prompts.test.ts +91 -0
  173. package/src/env/prompt/prompt-component.test.ts +491 -0
  174. package/src/env/prompt/prompt-component.ts +619 -0
  175. package/src/env/prompt/prompt-config-registration.test.ts +213 -0
  176. package/src/env/prompt/prompt-config-registration.ts +39 -0
  177. package/src/env/prompt/prompts-index.ts +504 -0
  178. package/src/env/prompt/renderer.ts +67 -0
  179. package/src/env/prompt/types.ts +136 -0
  180. package/src/env/session/hooks.ts +18 -0
  181. package/src/env/session/index.ts +37 -0
  182. package/src/env/session/search-query-parser.test.ts +425 -0
  183. package/src/env/session/search-query-parser.ts +171 -0
  184. package/src/env/session/session-checkpoint.test.ts +523 -0
  185. package/src/env/session/session-component.extract-recent-messages.test.ts +209 -0
  186. package/src/env/session/session-component.test.ts +132 -0
  187. package/src/env/session/session-component.ts +1249 -0
  188. package/src/env/session/session-config-registration.test.ts +138 -0
  189. package/src/env/session/session-config-registration.ts +52 -0
  190. package/src/env/session/session-message-converter.test.ts +763 -0
  191. package/src/env/session/session-message-converter.ts +415 -0
  192. package/src/env/session/session-message-e2e.test.ts +448 -0
  193. package/src/env/session/session-search.test.ts +391 -0
  194. package/src/env/session/session-store.test.ts +362 -0
  195. package/src/env/session/session-store.ts +141 -0
  196. package/src/env/session/storage/index.ts +6 -0
  197. package/src/env/session/storage/memory.ts +502 -0
  198. package/src/env/session/storage/sqlite.ts +794 -0
  199. package/src/env/session/types.ts +742 -0
  200. package/src/env/skill/config.ts +39 -0
  201. package/src/env/skill/index.ts +6 -0
  202. package/src/env/skill/parser.test.ts +116 -0
  203. package/src/env/skill/parser.ts +77 -0
  204. package/src/env/skill/scanner.test.ts +211 -0
  205. package/src/env/skill/scanner.ts +119 -0
  206. package/src/env/skill/skill-component.test.ts +234 -0
  207. package/src/env/skill/skill-component.ts +352 -0
  208. package/src/env/skill/skill-config-registration.test.ts +60 -0
  209. package/src/env/skill/skill-config-registration.ts +43 -0
  210. package/src/env/skill/tool/index.ts +1 -0
  211. package/src/env/skill/tool/skill-tool.test.ts +100 -0
  212. package/src/env/skill/tool/skill-tool.ts +72 -0
  213. package/src/env/skill/types.ts +64 -0
  214. package/src/env/task/delegate/delegate-tool.test.ts +498 -0
  215. package/src/env/task/delegate/delegate-tool.ts +1014 -0
  216. package/src/env/task/delegate/index.ts +18 -0
  217. package/src/env/task/delegate/stop-tool.test.ts +140 -0
  218. package/src/env/task/delegate/stop-tool.ts +119 -0
  219. package/src/env/task/delegate/task-events.test.ts +178 -0
  220. package/src/env/task/delegate/task-events.ts +143 -0
  221. package/src/env/task/hooks/contexts.test.ts +92 -0
  222. package/src/env/task/hooks/contexts.ts +192 -0
  223. package/src/env/task/hooks/index.ts +23 -0
  224. package/src/env/task/hooks/task-hook-points.test.ts +32 -0
  225. package/src/env/task/hooks/task-hook-points.ts +54 -0
  226. package/src/env/task/index.ts +7 -0
  227. package/src/env/task/plugins/index.ts +13 -0
  228. package/src/env/task/plugins/task-plugin.test.ts +74 -0
  229. package/src/env/task/plugins/task-plugin.ts +89 -0
  230. package/src/env/task/plugins/task-tag-plugin.test.ts +377 -0
  231. package/src/env/task/plugins/task-tag-plugin.ts +319 -0
  232. package/src/env/task/plugins/task-workflow-extractor.integration.test.ts +226 -0
  233. package/src/env/task/plugins/workflow-extractor-agent.test.ts +107 -0
  234. package/src/env/task/plugins/workflow-extractor-agent.ts +225 -0
  235. package/src/env/task/storage/index.ts +6 -0
  236. package/src/env/task/storage/sqlite-task-store.test.ts +283 -0
  237. package/src/env/task/storage/sqlite-task-store.ts +903 -0
  238. package/src/env/task/storage/task-search.test.ts +291 -0
  239. package/src/env/task/tag-service.test.ts +198 -0
  240. package/src/env/task/tag-service.ts +264 -0
  241. package/src/env/task/task-component.test.ts +193 -0
  242. package/src/env/task/task-component.ts +658 -0
  243. package/src/env/task/task-config-registration.test.ts +57 -0
  244. package/src/env/task/task-config-registration.ts +37 -0
  245. package/src/env/task/task-types.test.ts +137 -0
  246. package/src/env/task/tools/complete-tool.ts +44 -0
  247. package/src/env/task/tools/create-tool.ts +49 -0
  248. package/src/env/task/tools/delete-tool.ts +43 -0
  249. package/src/env/task/tools/get-tool.ts +59 -0
  250. package/src/env/task/tools/index.ts +10 -0
  251. package/src/env/task/tools/list-tool.ts +40 -0
  252. package/src/env/task/tools/operation/create-tool.ts +48 -0
  253. package/src/env/task/tools/operation/delete-tool.ts +43 -0
  254. package/src/env/task/tools/operation/get-tool.ts +43 -0
  255. package/src/env/task/tools/operation/index.ts +9 -0
  256. package/src/env/task/tools/operation/list-tool.ts +40 -0
  257. package/src/env/task/tools/operation/operation-tools.test.ts +274 -0
  258. package/src/env/task/tools/operation/operation-types.ts +75 -0
  259. package/src/env/task/tools/operation/update-tool.ts +47 -0
  260. package/src/env/task/tools/task-tools.test.ts +203 -0
  261. package/src/env/task/tools/task-types.test.ts +75 -0
  262. package/src/env/task/tools/task-types.ts +68 -0
  263. package/src/env/task/tools/update-tool.ts +70 -0
  264. package/src/env/task/types.ts +160 -0
  265. package/src/env/tool/built-in/bash.ts +201 -0
  266. package/src/env/tool/built-in/echo.ts +29 -0
  267. package/src/env/tool/built-in/edit-file.test.ts +136 -0
  268. package/src/env/tool/built-in/edit-file.ts +92 -0
  269. package/src/env/tool/built-in/glob.test.ts +94 -0
  270. package/src/env/tool/built-in/glob.ts +65 -0
  271. package/src/env/tool/built-in/grep.test.ts +122 -0
  272. package/src/env/tool/built-in/grep.ts +108 -0
  273. package/src/env/tool/built-in/index.ts +44 -0
  274. package/src/env/tool/built-in/read-file.test.ts +84 -0
  275. package/src/env/tool/built-in/read-file.ts +75 -0
  276. package/src/env/tool/built-in/write-file.test.ts +119 -0
  277. package/src/env/tool/built-in/write-file.ts +68 -0
  278. package/src/env/tool/index.ts +24 -0
  279. package/src/env/tool/registry.test.ts +257 -0
  280. package/src/env/tool/registry.ts +167 -0
  281. package/src/env/tool/tool-component.test.ts +559 -0
  282. package/src/env/tool/tool-component.ts +563 -0
  283. package/src/env/tool/tool-config-registration.test.ts +249 -0
  284. package/src/env/tool/tool-config-registration.ts +46 -0
  285. package/src/env/tool/types.ts +267 -0
  286. package/src/env/tool/validator.test.ts +143 -0
  287. package/src/env/tool/validator.ts +44 -0
  288. package/src/env/types.ts +180 -0
  289. package/src/env/workflow/ask-user-tool-registration.test.ts +216 -0
  290. package/src/env/workflow/complex-workflow.integration.test.ts +1900 -0
  291. package/src/env/workflow/decorators/decorator-node.ts +229 -0
  292. package/src/env/workflow/decorators/decorator.test.ts +196 -0
  293. package/src/env/workflow/decorators/edge.ts +82 -0
  294. package/src/env/workflow/decorators/index.ts +31 -0
  295. package/src/env/workflow/decorators/node-as.ts +98 -0
  296. package/src/env/workflow/decorators/workflow.ts +54 -0
  297. package/src/env/workflow/engine/dag-manager.test.ts +570 -0
  298. package/src/env/workflow/engine/dag-manager.ts +594 -0
  299. package/src/env/workflow/engine/engine.ts +1422 -0
  300. package/src/env/workflow/engine/event-bus.test.ts +359 -0
  301. package/src/env/workflow/engine/event-bus.ts +156 -0
  302. package/src/env/workflow/engine/executor-agent-session.test.ts +84 -0
  303. package/src/env/workflow/engine/executor.test.ts +619 -0
  304. package/src/env/workflow/engine/executor.ts +593 -0
  305. package/src/env/workflow/engine/index.ts +24 -0
  306. package/src/env/workflow/engine/node-registry.test.ts +560 -0
  307. package/src/env/workflow/engine/node-registry.ts +289 -0
  308. package/src/env/workflow/engine/resume-removed.test.ts +22 -0
  309. package/src/env/workflow/engine/scheduler.test.ts +715 -0
  310. package/src/env/workflow/engine/scheduler.ts +318 -0
  311. package/src/env/workflow/engine/workflow-engine.test.ts +815 -0
  312. package/src/env/workflow/extractor/workflow-converter.ts +306 -0
  313. package/src/env/workflow/fixtures.ts +380 -0
  314. package/src/env/workflow/index.ts +38 -0
  315. package/src/env/workflow/integration/run-resume-unified.test.ts +186 -0
  316. package/src/env/workflow/integration/service-integration.test.ts +267 -0
  317. package/src/env/workflow/metadata/keys.ts +12 -0
  318. package/src/env/workflow/nodes/agent-component-adapter.test.ts +318 -0
  319. package/src/env/workflow/nodes/agent-component-adapter.ts +448 -0
  320. package/src/env/workflow/nodes/agent-node.test.ts +371 -0
  321. package/src/env/workflow/nodes/agent-node.ts +598 -0
  322. package/src/env/workflow/nodes/ask-user-node.ts +113 -0
  323. package/src/env/workflow/nodes/condition-node.ts +200 -0
  324. package/src/env/workflow/nodes/index.ts +9 -0
  325. package/src/env/workflow/nodes/merge-node.ts +141 -0
  326. package/src/env/workflow/nodes/skill-node.test.ts +253 -0
  327. package/src/env/workflow/nodes/skill-node.ts +393 -0
  328. package/src/env/workflow/nodes/tool-node.test.ts +251 -0
  329. package/src/env/workflow/nodes/tool-node.ts +493 -0
  330. package/src/env/workflow/nodes/workflow-llm-history.test.ts +455 -0
  331. package/src/env/workflow/nodes/workflow-node.test.ts +315 -0
  332. package/src/env/workflow/nodes/workflow-node.ts +311 -0
  333. package/src/env/workflow/service/index.ts +27 -0
  334. package/src/env/workflow/service/registry.test.ts +133 -0
  335. package/src/env/workflow/service/registry.ts +71 -0
  336. package/src/env/workflow/service/workflow-service.test.ts +310 -0
  337. package/src/env/workflow/service/workflow-service.ts +393 -0
  338. package/src/env/workflow/storage/index.ts +28 -0
  339. package/src/env/workflow/storage/mock-repositories.ts +385 -0
  340. package/src/env/workflow/storage/sqlite.test.ts +179 -0
  341. package/src/env/workflow/storage/sqlite.ts +163 -0
  342. package/src/env/workflow/storage/workflow-repo.test.ts +780 -0
  343. package/src/env/workflow/storage/workflow-repo.ts +342 -0
  344. package/src/env/workflow/tools/ask-user-tool.ts +82 -0
  345. package/src/env/workflow/tools/index.ts +26 -0
  346. package/src/env/workflow/tools/run-workflow.test.ts +352 -0
  347. package/src/env/workflow/tools/run-workflow.ts +214 -0
  348. package/src/env/workflow/types/context.ts +18 -0
  349. package/src/env/workflow/types/decorators-types.ts +198 -0
  350. package/src/env/workflow/types/event.test.ts +515 -0
  351. package/src/env/workflow/types/event.ts +193 -0
  352. package/src/env/workflow/types/index.ts +49 -0
  353. package/src/env/workflow/types/run.test.ts +437 -0
  354. package/src/env/workflow/types/run.ts +173 -0
  355. package/src/env/workflow/types/workflow-hil.ts +114 -0
  356. package/src/env/workflow/types/workflow-message.test.ts +138 -0
  357. package/src/env/workflow/types/workflow-message.ts +196 -0
  358. package/src/env/workflow/types/workflow-session.test.ts +95 -0
  359. package/src/env/workflow/types/workflow-session.ts +59 -0
  360. package/src/env/workflow/types/workflow.test.ts +495 -0
  361. package/src/env/workflow/types/workflow.ts +195 -0
  362. package/src/env/workflow/types_compat.ts +51 -0
  363. package/src/env/workflow/utils/create-workflow.ts +47 -0
  364. package/src/env/workflow/utils/execution-state.ts +245 -0
  365. package/src/env/workflow/utils/index.ts +18 -0
  366. package/src/env/workflow/utils/node-registry-helper.ts +58 -0
  367. package/src/env/workflow/utils/recovery-validator.test.ts +460 -0
  368. package/src/env/workflow/utils/recovery-validator.ts +377 -0
  369. package/src/env/workflow/utils/session-parser.test.ts +111 -0
  370. package/src/env/workflow/utils/session-parser.ts +94 -0
  371. package/src/env/workflow/utils/session-recovery.test.ts +334 -0
  372. package/src/env/workflow/utils/session-recovery.ts +188 -0
  373. package/src/env/workflow/utils/template-resolver.test.ts +258 -0
  374. package/src/env/workflow/utils/template-resolver.ts +436 -0
  375. package/src/env/workflow/utils/validation-rules.ts +149 -0
  376. package/src/env/workflow/workflow-component.ts +544 -0
  377. package/src/index.ts +422 -0
  378. package/src/utils/id.ts +21 -0
@@ -0,0 +1,229 @@
1
+ /**
2
+ * DecoratorNode - 执行装饰器类方法的节点
3
+ *
4
+ * 实现 Node 接口,用于执行使用 @Workflow/@NodeAs 装饰的类方法
5
+ */
6
+
7
+ import type { Node, NodeDefinition, NodeExecutionContext, NodeExecutionResult } from '../types';
8
+
9
+ /**
10
+ * DecoratorNode 节点配置
11
+ */
12
+ export interface DecoratorNodeConfig extends NodeDefinition {
13
+ config: {
14
+ /** 方法名称 */
15
+ _methodName: string;
16
+ /** 类实例 */
17
+ _instance: any;
18
+ /** 用户自定义配置 */
19
+ [key: string]: unknown;
20
+ };
21
+ }
22
+
23
+ /**
24
+ * DecoratorNode - 执行装饰器类方法的节点
25
+ *
26
+ * 用于执行使用 @Workflow/@NodeAs 装饰器定义的类方法
27
+ *
28
+ * @example
29
+ * ```typescript
30
+ * const definition: DecoratorNodeConfig = {
31
+ * id: 'myNode',
32
+ * type: 'decorator',
33
+ * config: {
34
+ * _methodName: 'myMethod',
35
+ * _instance: myWorkflowInstance,
36
+ * }
37
+ * };
38
+ *
39
+ * const node = new DecoratorNode(definition);
40
+ * const result = await node.execute(context);
41
+ * ```
42
+ */
43
+ export class DecoratorNode implements Node {
44
+ readonly type = 'decorator';
45
+ readonly id: string;
46
+
47
+ private readonly methodName: string;
48
+ private readonly instance: any;
49
+ private readonly dependsOn: string[];
50
+ private readonly timeout?: number;
51
+ private readonly retryConfig?: {
52
+ max_attempts?: number;
53
+ backoff?: 'fixed' | 'exponential';
54
+ initial_delay?: number;
55
+ };
56
+
57
+ constructor(definition: NodeDefinition) {
58
+ const config = definition.config as DecoratorNodeConfig['config'];
59
+
60
+ this.id = definition.id;
61
+ this.methodName = config._methodName;
62
+ this.instance = config._instance;
63
+ this.dependsOn = definition.depends_on || [];
64
+ this.timeout = definition.timeout;
65
+ this.retryConfig = definition.retry;
66
+ }
67
+
68
+ /**
69
+ * 执行装饰器方法
70
+ */
71
+ async execute(context: NodeExecutionContext): Promise<NodeExecutionResult> {
72
+ const startTime = Date.now();
73
+
74
+ // 获取方法
75
+ const method = this.instance[this.methodName];
76
+ if (typeof method !== 'function') {
77
+ return {
78
+ success: false,
79
+ error: `Method '${this.methodName}' is not a function`,
80
+ duration: Date.now() - startTime,
81
+ };
82
+ }
83
+
84
+ // 准备输入参数
85
+ const input = this.prepareInput(context);
86
+
87
+ // 执行(带重试和超时)
88
+ const result = await this.executeWithRetry(input, (context as any).abort);
89
+
90
+ return {
91
+ success: !result.error,
92
+ output: result.output,
93
+ error: result.error,
94
+ duration: Date.now() - startTime,
95
+ metadata: {
96
+ methodName: this.methodName,
97
+ },
98
+ };
99
+ }
100
+
101
+ /**
102
+ * 准备输入参数
103
+ *
104
+ * 根据依赖的节点 ID,从 nodeOutputs 中提取对应的输出
105
+ * 返回格式: { [depNodeId]: depOutput }
106
+ */
107
+ private prepareInput(context: NodeExecutionContext): any {
108
+ // nodeOutputs 可以是 Map 或普通对象
109
+ const nodeOutputs = context.nodeOutputs;
110
+
111
+ // 如果没有依赖节点,直接使用全局 input
112
+ if (this.dependsOn.length === 0) {
113
+ return context.input || {};
114
+ }
115
+
116
+ // 根据依赖节点 ID 提取输出,构建输入对象
117
+ const input: Record<string, unknown> = {};
118
+ for (const depId of this.dependsOn) {
119
+ // 支持 Map 和普通对象
120
+ const depOutput = nodeOutputs instanceof Map
121
+ ? nodeOutputs.get(depId)
122
+ : nodeOutputs?.[depId];
123
+
124
+ if (depOutput && typeof depOutput === 'object') {
125
+ // 如果是完整的 NodeExecutionResult,提取 output 字段
126
+ if ('output' in depOutput) {
127
+ input[depId] = (depOutput as any).output;
128
+ } else {
129
+ // 已经是裸对象
130
+ input[depId] = depOutput;
131
+ }
132
+ }
133
+ }
134
+
135
+ return input;
136
+ }
137
+
138
+ /**
139
+ * 带重试的执行
140
+ */
141
+ private async executeWithRetry(
142
+ input: any,
143
+ abort?: AbortSignal
144
+ ): Promise<{ output?: unknown; error?: string }> {
145
+ const maxAttempts = this.retryConfig?.max_attempts ?? 1;
146
+ const backoff = this.retryConfig?.backoff ?? 'exponential';
147
+ const initialDelay = this.retryConfig?.initial_delay ?? 1000;
148
+
149
+ let lastError: Error | undefined;
150
+
151
+ for (let attempt = 1; attempt <= maxAttempts; attempt++) {
152
+ // 检查中断信号
153
+ if (abort?.aborted) {
154
+ return { error: 'Execution aborted' };
155
+ }
156
+
157
+ try {
158
+ // 执行方法
159
+ const output = await this.executeWithTimeout(input, abort);
160
+ return { output };
161
+ } catch (error) {
162
+ lastError = error instanceof Error ? error : new Error(String(error));
163
+
164
+ // 如果不是最后一次尝试,等待后重试
165
+ if (attempt < maxAttempts) {
166
+ const delay = this.calculateBackoff(backoff, initialDelay, attempt - 1);
167
+ await this.sleep(delay, abort);
168
+ }
169
+ }
170
+ }
171
+
172
+ return { error: lastError?.message || 'Unknown error' };
173
+ }
174
+
175
+ /**
176
+ * 带超时的执行
177
+ */
178
+ private async executeWithTimeout(
179
+ input: any,
180
+ abort?: AbortSignal
181
+ ): Promise<unknown> {
182
+ if (!this.timeout) {
183
+ // 无超时限制
184
+ return this.instance[this.methodName](input);
185
+ }
186
+
187
+ // 创建超时 Promise
188
+ const timeoutPromise = new Promise<never>((_, reject) => {
189
+ setTimeout(() => {
190
+ reject(new Error(`Execution timeout after ${this.timeout}ms`));
191
+ }, this.timeout);
192
+ });
193
+
194
+ // 竞速执行
195
+ return Promise.race([
196
+ this.instance[this.methodName](input),
197
+ timeoutPromise,
198
+ ]);
199
+ }
200
+
201
+ /**
202
+ * 计算退避延迟
203
+ */
204
+ private calculateBackoff(
205
+ backoff: 'fixed' | 'exponential',
206
+ initialDelay: number,
207
+ attempt: number
208
+ ): number {
209
+ if (backoff === 'exponential') {
210
+ return initialDelay * Math.pow(2, attempt);
211
+ }
212
+ return initialDelay;
213
+ }
214
+
215
+ /**
216
+ * 睡眠
217
+ */
218
+ private sleep(ms: number, abort?: AbortSignal): Promise<void> {
219
+ return new Promise((resolve) => {
220
+ const timeout = setTimeout(resolve, ms);
221
+
222
+ // 如果被中断,清除定时器
223
+ abort?.addEventListener('abort', () => {
224
+ clearTimeout(timeout);
225
+ resolve();
226
+ });
227
+ });
228
+ }
229
+ }
@@ -0,0 +1,196 @@
1
+ /**
2
+ * @fileoverview Workflow Decorator Integration Tests
3
+ */
4
+
5
+ import { describe, test, expect } from 'bun:test';
6
+ import 'reflect-metadata';
7
+ import { Workflow, NodeAs, Edge, WorkflowEdges } from './index';
8
+ import { WorkflowConverter } from '../extractor/workflow-converter';
9
+ import { createWorkflowFromClass } from '../utils/create-workflow';
10
+ import { getWorkflowMetadata, getNodeMetadatas, getEdgeMetadatas } from './index';
11
+
12
+ describe('Workflow Decorators', () => {
13
+
14
+ // ============ Test Workflow Class ============
15
+
16
+ @Workflow({
17
+ name: 'test-workflow',
18
+ version: '1.0',
19
+ description: 'Test workflow for decorator functionality',
20
+ entry: 'step1',
21
+ config: { parallel_limit: 3 }
22
+ })
23
+ @WorkflowEdges([
24
+ { from: 'step2', to: 'step3' },
25
+ { from: 'step3', to: 'step4' }
26
+ ])
27
+ class TestWorkflow {
28
+ @NodeAs({ nodeId: 'step1', nodeType: 'tool' })
29
+ async step1(input: { message: string }) {
30
+ return { result: `Step1: ${input.message}` };
31
+ }
32
+
33
+ @NodeAs({ nodeId: 'step2', dependsOn: 'step1' })
34
+ async step2(input: { step1: any }) {
35
+ return { result: `Step2: ${input.step1.result}` };
36
+ }
37
+
38
+ @NodeAs({ nodeId: 'step3', dependsOn: 'step1' })
39
+ async step3(input: { step1: any }) {
40
+ return { result: `Step3: ${input.step1.result}` };
41
+ }
42
+
43
+ @NodeAs({ nodeId: 'step4', dependsOn: ['step2', 'step3'] })
44
+ async step4(input: { step2: any; step3: any }) {
45
+ return {
46
+ result: `Step4: ${input.step2.result} & ${input.step3.result}`
47
+ };
48
+ }
49
+ }
50
+
51
+ // ============ Tests ============
52
+
53
+ test('should extract workflow metadata', () => {
54
+ const metadata = getWorkflowMetadata(TestWorkflow);
55
+
56
+ expect(metadata).toBeDefined();
57
+ expect(metadata?.name).toBe('test-workflow');
58
+ expect(metadata?.version).toBe('1.0');
59
+ expect(metadata?.description).toBe('Test workflow for decorator functionality');
60
+ expect(metadata?.entry).toBe('step1');
61
+ });
62
+
63
+ test('should extract node metadatas', () => {
64
+ const nodes = getNodeMetadatas(TestWorkflow.prototype);
65
+
66
+ expect(nodes).toHaveLength(4);
67
+
68
+ // Check step1
69
+ const step1 = nodes.find(n => n.nodeId === 'step1');
70
+ expect(step1).toBeDefined();
71
+ expect(step1?.nodeType).toBe('tool');
72
+ expect(step1?.dependsOn).toEqual([]);
73
+ expect(step1?.methodName).toBe('step1');
74
+
75
+ // Check step2 (depends on step1)
76
+ const step2 = nodes.find(n => n.nodeId === 'step2');
77
+ expect(step2).toBeDefined();
78
+ expect(step2?.dependsOn).toEqual(['step1']);
79
+
80
+ // Check step4 (depends on step2 and step3)
81
+ const step4 = nodes.find(n => n.nodeId === 'step4');
82
+ expect(step4).toBeDefined();
83
+ expect(step4?.dependsOn).toEqual(['step2', 'step3']);
84
+ });
85
+
86
+ test('should extract edge metadatas', () => {
87
+ const edges = getEdgeMetadatas(TestWorkflow);
88
+
89
+ expect(edges).toHaveLength(2);
90
+ expect(edges).toContainEqual({ from: 'step2', to: 'step3' });
91
+ expect(edges).toContainEqual({ from: 'step3', to: 'step4' });
92
+ });
93
+
94
+ test('should validate workflow class', () => {
95
+ const result = WorkflowConverter.validateWorkflowClass(TestWorkflow);
96
+
97
+ expect(result.valid).toBe(true);
98
+ expect(result.errors).toHaveLength(0);
99
+ });
100
+
101
+ test('should convert to WorkflowDefinition', () => {
102
+ const workflow = createWorkflowFromClass(TestWorkflow);
103
+
104
+ expect(workflow.name).toBe('test-workflow');
105
+ expect(workflow.version).toBe('1.0');
106
+ expect(workflow.nodes).toHaveLength(4);
107
+
108
+ // Check nodes
109
+ const step1Node = workflow.nodes.find(n => n.id === 'step1');
110
+ expect(step1Node).toBeDefined();
111
+ expect(step1Node?.type).toBe('tool');
112
+ expect(step1Node?.depends_on).toEqual([]);
113
+
114
+ const step4Node = workflow.nodes.find(n => n.id === 'step4');
115
+ expect(step4Node).toBeDefined();
116
+ expect(step4Node?.depends_on).toContain('step2');
117
+ expect(step4Node?.depends_on).toContain('step3');
118
+ });
119
+
120
+ test('should validate missing entry node', () => {
121
+ @Workflow({ name: 'broken-workflow', entry: 'nonexistent' })
122
+ class BrokenWorkflow {
123
+ @NodeAs({ nodeId: 'step1' })
124
+ async step1() { return {}; }
125
+ }
126
+
127
+ const result = WorkflowConverter.validateWorkflowClass(BrokenWorkflow);
128
+
129
+ expect(result.valid).toBe(false);
130
+ expect(result.errors.some(e => e.includes("Entry node 'nonexistent' not found"))).toBe(true);
131
+ });
132
+
133
+ test('should validate missing dependency', () => {
134
+ @Workflow({ name: 'broken-workflow-2' })
135
+ class BrokenWorkflow2 {
136
+ @NodeAs({ nodeId: 'step1', dependsOn: 'nonexistent' })
137
+ async step1() { return {}; }
138
+ }
139
+
140
+ const result = WorkflowConverter.validateWorkflowClass(BrokenWorkflow2);
141
+
142
+ expect(result.valid).toBe(false);
143
+ expect(result.errors.some(e => e.includes("depends on non-existent node"))).toBe(true);
144
+ });
145
+
146
+ test('should validate cycle detection', () => {
147
+ @Workflow({ name: 'cyclic-workflow' })
148
+ class CyclicWorkflow {
149
+ @NodeAs({ nodeId: 'a', dependsOn: 'c' })
150
+ async a() { return {}; }
151
+
152
+ @NodeAs({ nodeId: 'b', dependsOn: 'a' })
153
+ async b() { return {}; }
154
+
155
+ @NodeAs({ nodeId: 'c', dependsOn: 'b' })
156
+ async c() { return {}; }
157
+ }
158
+
159
+ const result = WorkflowConverter.validateWorkflowClass(CyclicWorkflow);
160
+
161
+ expect(result.valid).toBe(false);
162
+ expect(result.errors.some(e => e.includes('cycle'))).toBe(true);
163
+ });
164
+
165
+ test('should execute decorator node method', async () => {
166
+ const workflow = createWorkflowFromClass(TestWorkflow);
167
+ const step1Node = workflow.nodes.find(n => n.id === 'step1')!;
168
+
169
+ const result = await WorkflowConverter.executeNode(
170
+ step1Node,
171
+ { input: { message: 'hello' }, previousOutputs: new Map() }
172
+ );
173
+
174
+ expect(result.output).toEqual({ result: 'Step1: hello' });
175
+ expect(result.error).toBeUndefined();
176
+ });
177
+
178
+ test('should handle node execution with dependencies', async () => {
179
+ const workflow = createWorkflowFromClass(TestWorkflow);
180
+ const step4Node = workflow.nodes.find(n => n.id === 'step4')!;
181
+
182
+ const previousOutputs = new Map([
183
+ ['step2', { result: 'Step2: data2' }],
184
+ ['step3', { result: 'Step3: data3' }]
185
+ ]);
186
+
187
+ const result = await WorkflowConverter.executeNode(
188
+ step4Node,
189
+ { input: {}, previousOutputs }
190
+ );
191
+
192
+ expect(result.output).toEqual({
193
+ result: 'Step4: Step2: data2 & Step3: data3'
194
+ });
195
+ });
196
+ });
@@ -0,0 +1,82 @@
1
+ /**
2
+ * @Edge 和 @WorkflowEdges 装饰器
3
+ *
4
+ * 用于显式声明节点之间的边关系
5
+ */
6
+
7
+ import 'reflect-metadata';
8
+ import { EDGE_METADATA_KEY } from '../metadata/keys';
9
+ import type { EdgeMetadata, EdgeOptions } from '../types/decorators-types';
10
+
11
+ /**
12
+ * 声明一条边 (from -> to)
13
+ *
14
+ * 这是一个方法装饰器,用于在方法上声明边关系
15
+ *
16
+ * @param from - 起始节点 ID
17
+ * @param to - 目标节点 ID
18
+ *
19
+ * @example
20
+ * ```typescript
21
+ * @Edge('nodeA', 'nodeB')
22
+ * static __edge_nodeA_to_nodeB__ = null;
23
+ * ```
24
+ */
25
+ export function Edge(from: string, to: string): MethodDecorator {
26
+ return function (
27
+ target: Object,
28
+ propertyKey: string | symbol,
29
+ descriptor: PropertyDescriptor
30
+ ): PropertyDescriptor {
31
+ const edge: EdgeMetadata = { from, to };
32
+
33
+ // 获取已存在的边元数据列表
34
+ const existingEdges: EdgeMetadata[] = Reflect.getMetadata(EDGE_METADATA_KEY, target) || [];
35
+
36
+ // 添加新边
37
+ existingEdges.push(edge);
38
+
39
+ // 存储元数据
40
+ Reflect.defineMetadata(EDGE_METADATA_KEY, existingEdges, target);
41
+
42
+ return descriptor;
43
+ };
44
+ }
45
+
46
+ /**
47
+ * 类级别装饰器,声明多条边
48
+ *
49
+ * @param edges - 边数组
50
+ *
51
+ * @example
52
+ * ```typescript
53
+ * @WorkflowEdges([
54
+ * { from: 'A', to: 'B' },
55
+ * { from: 'B', to: 'C' }
56
+ * ])
57
+ * @Workflow({ name: 'test' })
58
+ * class TestWorkflow { }
59
+ * ```
60
+ */
61
+ export function WorkflowEdges(edges: EdgeOptions[]): ClassDecorator {
62
+ return function <T extends Function>(target: T): T {
63
+ // 将 EdgeOptions[] 转换为 EdgeMetadata[]
64
+ const edgeMetadatas: EdgeMetadata[] = edges.map(edge => ({
65
+ from: edge.from,
66
+ to: edge.to,
67
+ condition: edge.condition,
68
+ }));
69
+
70
+ // 存储元数据
71
+ Reflect.defineMetadata(EDGE_METADATA_KEY, edgeMetadatas, target);
72
+
73
+ return target;
74
+ };
75
+ }
76
+
77
+ /**
78
+ * 获取类上的所有边元数据
79
+ */
80
+ export function getEdgeMetadatas(target: Function): EdgeMetadata[] {
81
+ return Reflect.getMetadata(EDGE_METADATA_KEY, target) || [];
82
+ }
@@ -0,0 +1,31 @@
1
+ /**
2
+ * @fileoverview Workflow Decorators Index
3
+ *
4
+ * 导出所有装饰器相关的 API
5
+ */
6
+
7
+ // Decorators
8
+ export { Workflow, getWorkflowMetadata } from './workflow';
9
+ export { NodeAs, getNodeMetadatas } from './node-as';
10
+ export { Edge, WorkflowEdges, getEdgeMetadatas } from './edge';
11
+
12
+ // Node
13
+ export { DecoratorNode } from './decorator-node';
14
+ export type { DecoratorNodeConfig } from './decorator-node';
15
+
16
+ // Converter (需要单独导入,避免循环依赖)
17
+ export { WorkflowConverter } from '../extractor/workflow-converter';
18
+ export { createWorkflowFromClass } from '../utils/create-workflow';
19
+
20
+ // Types
21
+ export type {
22
+ WorkflowOptions,
23
+ NodeAsOptions,
24
+ EdgeOptions,
25
+ WorkflowMetadata,
26
+ NodeMetadata,
27
+ EdgeMetadata,
28
+ ValidationResult,
29
+ ConvertedWorkflowDefinition,
30
+ ConvertedNodeDefinition,
31
+ } from '../types/decorators-types';
@@ -0,0 +1,98 @@
1
+ /**
2
+ * @NodeAs 装饰器
3
+ *
4
+ * 用于将类方法声明为 Workflow 的 Node
5
+ */
6
+
7
+ import 'reflect-metadata';
8
+ import { NODE_METADATA_KEY } from '../metadata/keys';
9
+ import type { NodeAsOptions, NodeMetadata } from '../types/decorators-types';
10
+
11
+ /**
12
+ * 将方法声明为 Workflow 的 Node
13
+ *
14
+ * @param options - Node 配置选项
15
+ *
16
+ * @example
17
+ * ```typescript
18
+ * @Workflow({ name: 'my-workflow', entry: 'fetch' })
19
+ * class MyWorkflow {
20
+ * @NodeAs({
21
+ * nodeId: 'fetch',
22
+ * nodeType: 'tool',
23
+ * config: { tool: 'web_search' }
24
+ * })
25
+ * async fetch(input: { query: string }) {
26
+ * return { data: [] };
27
+ * }
28
+ *
29
+ * @NodeAs({
30
+ * nodeId: 'process',
31
+ * dependsOn: 'fetch'
32
+ * })
33
+ * async process(input: { data: any }) {
34
+ * return { result: {} };
35
+ * }
36
+ * }
37
+ * ```
38
+ */
39
+ export function NodeAs(options: NodeAsOptions = {}): MethodDecorator {
40
+ return function (
41
+ target: Object,
42
+ propertyKey: string | symbol,
43
+ descriptor: PropertyDescriptor
44
+ ): PropertyDescriptor {
45
+ const methodName = String(propertyKey);
46
+
47
+ // 构建节点元数据
48
+ const nodeMetadata: NodeMetadata = {
49
+ nodeId: options.nodeId || methodName, // 默认使用方法名作为 nodeId
50
+ nodeType: options.nodeType || 'tool', // 默认类型为 tool
51
+ name: options.name,
52
+ // 标准化 dependsOn 为数组
53
+ dependsOn: normalizeDependsOn(options.dependsOn),
54
+ condition: options.condition,
55
+ retry: options.retry,
56
+ timeout: options.timeout,
57
+ config: options.config || {},
58
+ methodName,
59
+ method: descriptor.value, // 存储方法引用
60
+ };
61
+
62
+ // 获取已存在的节点元数据列表
63
+ const existingNodes: NodeMetadata[] = Reflect.getMetadata(NODE_METADATA_KEY, target) || [];
64
+
65
+ // 检查是否已存在同名节点(同一方法不能被多次装饰)
66
+ const existingIndex = existingNodes.findIndex(n => n.methodName === methodName);
67
+
68
+ if (existingIndex >= 0) {
69
+ // 更新已存在的节点元数据
70
+ existingNodes[existingIndex] = nodeMetadata;
71
+ } else {
72
+ // 添加新节点元数据
73
+ existingNodes.push(nodeMetadata);
74
+ }
75
+
76
+ // 存储元数据
77
+ Reflect.defineMetadata(NODE_METADATA_KEY, existingNodes, target);
78
+
79
+ return descriptor;
80
+ };
81
+ }
82
+
83
+ /**
84
+ * 标准化 dependsOn 为数组
85
+ */
86
+ function normalizeDependsOn(dependsOn?: string | string[] | undefined): string[] {
87
+ if (!dependsOn) {
88
+ return [];
89
+ }
90
+ return Array.isArray(dependsOn) ? dependsOn : [dependsOn];
91
+ }
92
+
93
+ /**
94
+ * 获取类上所有方法的 Node 元数据
95
+ */
96
+ export function getNodeMetadatas(target: Object): NodeMetadata[] {
97
+ return Reflect.getMetadata(NODE_METADATA_KEY, target) || [];
98
+ }
@@ -0,0 +1,54 @@
1
+ /**
2
+ * @Workflow 装饰器
3
+ *
4
+ * 用于声明一个类为 Workflow 类
5
+ */
6
+
7
+ import 'reflect-metadata';
8
+ import { WORKFLOW_METADATA_KEY } from '../metadata/keys';
9
+ import type { WorkflowOptions, WorkflowMetadata } from '../types/decorators-types';
10
+
11
+ /**
12
+ * 声明一个类为 Workflow 类
13
+ *
14
+ * @param options - Workflow 配置选项
15
+ *
16
+ * @example
17
+ * ```typescript
18
+ * @Workflow({
19
+ * name: 'my-workflow',
20
+ * version: '1.0',
21
+ * entry: 'start'
22
+ * })
23
+ * class MyWorkflow {
24
+ * @NodeAs({ nodeId: 'start' })
25
+ * async start() { }
26
+ * }
27
+ * ```
28
+ */
29
+ export function Workflow(options: WorkflowOptions): ClassDecorator {
30
+ return function <T extends Function>(target: T): T {
31
+ // 构建元数据对象
32
+ const metadata: WorkflowMetadata = {
33
+ name: options.name,
34
+ version: options.version || '1.0',
35
+ description: options.description,
36
+ entry: options.entry,
37
+ config: options.config,
38
+ tags: options.tags || [],
39
+ author: options.author,
40
+ };
41
+
42
+ // 使用 Reflect.metadata 存储元数据
43
+ Reflect.defineMetadata(WORKFLOW_METADATA_KEY, metadata, target);
44
+
45
+ return target;
46
+ };
47
+ }
48
+
49
+ /**
50
+ * 获取类上的 Workflow 元数据
51
+ */
52
+ export function getWorkflowMetadata(target: Function): WorkflowMetadata | undefined {
53
+ return Reflect.getMetadata(WORKFLOW_METADATA_KEY, target);
54
+ }