@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,594 @@
1
+ import type {
2
+ WorkflowDefinition,
3
+ NodeDefinition,
4
+ WorkflowDefinitionExtended
5
+ } from '../types';
6
+
7
+ /**
8
+ * DAG Analysis Result
9
+ */
10
+ export interface DAGAnalysis {
11
+ /** 入口节点(无依赖) */
12
+ entryNodes: string[];
13
+ /** 出口节点(无下游) */
14
+ exitNodes: string[];
15
+ /** 按层级组织的节点,用于并行执行 */
16
+ levels: string[][];
17
+ /** 每个节点的依赖 */
18
+ dependencies: Map<string, string[]>;
19
+ /** 每个节点的下游 */
20
+ dependents: Map<string, string[]>;
21
+ /** 关键路径 */
22
+ criticalPath: string[];
23
+ }
24
+
25
+ /**
26
+ * DAG Manager
27
+ *
28
+ * Manages Directed Acyclic Graph (DAG) operations for workflow execution.
29
+ * Provides analysis, topological sorting, and validation of workflow structures.
30
+ */
31
+ export class DAGManager {
32
+ private workflow: WorkflowDefinition;
33
+ private nodeMap: Map<string, NodeDefinition>;
34
+ private extendedDefinition: WorkflowDefinitionExtended | null = null;
35
+ private cachedAnalysis: DAGAnalysis | null = null;
36
+
37
+ constructor(workflow: WorkflowDefinition) {
38
+ // Validate basic structure
39
+ if (!workflow.nodes || workflow.nodes.length === 0) {
40
+ throw new Error('At least one node is required');
41
+ }
42
+
43
+ this.workflow = workflow;
44
+ this.nodeMap = new Map();
45
+
46
+ // Build node map
47
+ for (const node of workflow.nodes) {
48
+ if (node.id === '') {
49
+ throw new Error('Node ID cannot be empty');
50
+ }
51
+ if (this.nodeMap.has(node.id)) {
52
+ throw new Error(`Duplicate node ID: ${node.id}`);
53
+ }
54
+ this.nodeMap.set(node.id, node);
55
+ }
56
+ }
57
+
58
+ /**
59
+ * Get extended workflow definition with computed properties
60
+ */
61
+ getExtendedDefinition(): WorkflowDefinitionExtended {
62
+ if (this.extendedDefinition) {
63
+ return this.extendedDefinition;
64
+ }
65
+
66
+ const analysis = this.analyze();
67
+
68
+ this.extendedDefinition = {
69
+ ...this.workflow,
70
+ computedEntry: analysis.entryNodes,
71
+ nodeMap: this.nodeMap,
72
+ topologicalOrder: this.getTopologicalOrder(),
73
+ };
74
+
75
+ return this.extendedDefinition;
76
+ }
77
+
78
+ /**
79
+ * Analyze the DAG structure
80
+ */
81
+ analyze(): DAGAnalysis {
82
+ if (this.cachedAnalysis) {
83
+ return this.cachedAnalysis;
84
+ }
85
+
86
+ // Compute dependencies map
87
+ const dependencies = this.buildDependenciesMap();
88
+
89
+ // Compute dependents map (reverse of dependencies)
90
+ const dependents = this.buildDependentsMap(dependencies);
91
+
92
+ // Find entry nodes (no incoming edges)
93
+ const entryNodes = this.findEntryNodes(dependencies);
94
+
95
+ // Find exit nodes (no outgoing edges)
96
+ const exitNodes = this.findExitNodes(dependents);
97
+
98
+ // Compute levels for parallel execution
99
+ const levels = this.computeLevels(dependencies);
100
+
101
+ // Find critical path
102
+ const criticalPath = this.findCriticalPath(levels, dependencies);
103
+
104
+ this.cachedAnalysis = {
105
+ entryNodes,
106
+ exitNodes,
107
+ levels,
108
+ dependencies,
109
+ dependents,
110
+ criticalPath,
111
+ };
112
+
113
+ return this.cachedAnalysis;
114
+ }
115
+
116
+ /**
117
+ * Build a map of node dependencies
118
+ */
119
+ private buildDependenciesMap(): Map<string, string[]> {
120
+ const dependencies = new Map<string, string[]>();
121
+
122
+ for (const node of this.workflow.nodes) {
123
+ const deps = node.depends_on || [];
124
+ dependencies.set(node.id, [...deps]);
125
+ }
126
+
127
+ return dependencies;
128
+ }
129
+
130
+ /**
131
+ * Build a map of node dependents (reverse of dependencies)
132
+ */
133
+ private buildDependentsMap(dependencies: Map<string, string[]>): Map<string, string[]> {
134
+ const dependentsMap = new Map<string, string[]>();
135
+
136
+ // Initialize all nodes with empty dependents
137
+ for (const nodeId of dependencies.keys()) {
138
+ dependentsMap.set(nodeId, []);
139
+ }
140
+
141
+ // Populate dependents
142
+ for (const [nodeId, deps] of dependencies.entries()) {
143
+ for (const dep of deps) {
144
+ const nodeDependents = dependentsMap.get(dep) || [];
145
+ nodeDependents.push(nodeId);
146
+ dependentsMap.set(dep, nodeDependents);
147
+ }
148
+ }
149
+
150
+ return dependentsMap;
151
+ }
152
+
153
+ /**
154
+ * Find entry nodes (nodes with no dependencies)
155
+ */
156
+ private findEntryNodes(dependencies: Map<string, string[]>): string[] {
157
+ const entryNodes: string[] = [];
158
+
159
+ for (const [nodeId, deps] of dependencies.entries()) {
160
+ if (deps.length === 0) {
161
+ entryNodes.push(nodeId);
162
+ }
163
+ }
164
+
165
+ return entryNodes;
166
+ }
167
+
168
+ /**
169
+ * Find exit nodes (nodes with no dependents)
170
+ */
171
+ private findExitNodes(dependents: Map<string, string[]>): string[] {
172
+ const exitNodes: string[] = [];
173
+
174
+ for (const [nodeId, deps] of dependents.entries()) {
175
+ if (deps.length === 0) {
176
+ exitNodes.push(nodeId);
177
+ }
178
+ }
179
+
180
+ return exitNodes;
181
+ }
182
+
183
+ /**
184
+ * Compute levels for parallel execution
185
+ * Nodes at the same level can be executed in parallel
186
+ */
187
+ private computeLevels(dependencies: Map<string, string[]>): string[][] {
188
+ const levels: string[][] = [];
189
+ const assigned = new Set<string>();
190
+ const nodeLevels = new Map<string, number>();
191
+
192
+ // Build dependents map for efficient lookup
193
+ const dependentsMap = this.buildDependentsMap(dependencies);
194
+
195
+ // Calculate level for each node (BFS from entry nodes)
196
+ const entryNodes = this.findEntryNodes(dependencies);
197
+
198
+ // Initialize entry nodes at level 0
199
+ const queue: Array<{ nodeId: string; level: number }> = [];
200
+ for (const entry of entryNodes) {
201
+ queue.push({ nodeId: entry, level: 0 });
202
+ nodeLevels.set(entry, 0);
203
+ }
204
+
205
+ // BFS to compute levels
206
+ while (queue.length > 0) {
207
+ const { nodeId, level } = queue.shift()!;
208
+
209
+ if (assigned.has(nodeId)) {
210
+ // Update to maximum level if already assigned
211
+ const currentLevel = nodeLevels.get(nodeId) || 0;
212
+ if (level > currentLevel) {
213
+ nodeLevels.set(nodeId, level);
214
+ }
215
+ continue;
216
+ }
217
+
218
+ assigned.add(nodeId);
219
+ nodeLevels.set(nodeId, level);
220
+
221
+ // Add dependents to queue with level + 1
222
+ const dependentsOfNode = dependentsMap.get(nodeId) || [];
223
+ for (const dependent of dependentsOfNode) {
224
+ if (!assigned.has(dependent)) {
225
+ queue.push({ nodeId: dependent, level: level + 1 });
226
+ }
227
+ }
228
+ }
229
+
230
+ // Handle nodes not reachable from entry nodes (shouldn't happen in valid DAG)
231
+ for (const nodeId of dependencies.keys()) {
232
+ if (!nodeLevels.has(nodeId)) {
233
+ nodeLevels.set(nodeId, 0);
234
+ }
235
+ }
236
+
237
+ // Group nodes by level
238
+ const maxLevel = Math.max(...nodeLevels.values(), 0);
239
+ for (let i = 0; i <= maxLevel; i++) {
240
+ levels.push([]);
241
+ }
242
+
243
+ for (const [nodeId, level] of nodeLevels.entries()) {
244
+ levels[level].push(nodeId);
245
+ }
246
+
247
+ return levels;
248
+ }
249
+
250
+ /**
251
+ * Find critical path (longest path through the DAG)
252
+ */
253
+ private findCriticalPath(levels: string[][], dependencies: Map<string, string[]>): string[] {
254
+ if (levels.length === 0) {
255
+ return [];
256
+ }
257
+
258
+ // DP: longest path ending at each node
259
+ const longestPath = new Map<string, { length: number; path: string[] }>();
260
+
261
+ // Process nodes in reverse topological order
262
+ const allNodes: string[] = [];
263
+ for (const level of levels) {
264
+ allNodes.push(...level);
265
+ }
266
+
267
+ for (const nodeId of allNodes) {
268
+ const deps = dependencies.get(nodeId) || [];
269
+
270
+ if (deps.length === 0) {
271
+ longestPath.set(nodeId, { length: 1, path: [nodeId] });
272
+ } else {
273
+ let maxLength = 0;
274
+ let maxPath: string[] = [];
275
+
276
+ for (const dep of deps) {
277
+ const depInfo = longestPath.get(dep);
278
+ if (depInfo && depInfo.length > maxLength) {
279
+ maxLength = depInfo.length;
280
+ maxPath = depInfo.path;
281
+ }
282
+ }
283
+
284
+ longestPath.set(nodeId, {
285
+ length: maxLength + 1,
286
+ path: [...maxPath, nodeId]
287
+ });
288
+ }
289
+ }
290
+
291
+ // Find the longest path
292
+ let maxLength = 0;
293
+ let criticalPath: string[] = [];
294
+
295
+ for (const [, info] of longestPath.entries()) {
296
+ if (info.length > maxLength) {
297
+ maxLength = info.length;
298
+ criticalPath = info.path;
299
+ }
300
+ }
301
+
302
+ return criticalPath;
303
+ }
304
+
305
+ /**
306
+ * Get topological order using Kahn's algorithm
307
+ */
308
+ getTopologicalOrder(): string[] {
309
+ const dependencies = this.buildDependenciesMap();
310
+ const dependents = this.buildDependentsMap(dependencies);
311
+
312
+ // Calculate in-degree for each node
313
+ const inDegree = new Map<string, number>();
314
+ for (const nodeId of dependencies.keys()) {
315
+ inDegree.set(nodeId, dependencies.get(nodeId)?.length || 0);
316
+ }
317
+
318
+ // Start with nodes that have no dependencies
319
+ const queue: string[] = [];
320
+ for (const [nodeId, degree] of inDegree.entries()) {
321
+ if (degree === 0) {
322
+ queue.push(nodeId);
323
+ }
324
+ }
325
+
326
+ const result: string[] = [];
327
+
328
+ while (queue.length > 0) {
329
+ const nodeId = queue.shift()!;
330
+ result.push(nodeId);
331
+
332
+ // Reduce in-degree for dependents
333
+ const nodeDependents = dependents.get(nodeId) || [];
334
+ for (const dependent of nodeDependents) {
335
+ const currentDegree = inDegree.get(dependent) || 0;
336
+ inDegree.set(dependent, currentDegree - 1);
337
+
338
+ if (currentDegree - 1 === 0) {
339
+ queue.push(dependent);
340
+ }
341
+ }
342
+ }
343
+
344
+ return result;
345
+ }
346
+
347
+ /**
348
+ * Get nodes that are ready to execute (dependencies met and not completed)
349
+ */
350
+ getReadyNodes(completedNodes: Set<string>): string[] {
351
+ const dependencies = this.buildDependenciesMap();
352
+ const ready: string[] = [];
353
+
354
+ for (const nodeId of dependencies.keys()) {
355
+ // Skip already completed nodes
356
+ if (completedNodes.has(nodeId)) {
357
+ continue;
358
+ }
359
+
360
+ // Check if all dependencies are completed
361
+ if (this.areDependenciesMet(nodeId, completedNodes)) {
362
+ ready.push(nodeId);
363
+ }
364
+ }
365
+
366
+ return ready;
367
+ }
368
+
369
+ /**
370
+ * Check if all dependencies for a node are completed
371
+ */
372
+ areDependenciesMet(nodeId: string, completedNodes: Set<string>): boolean {
373
+ const node = this.nodeMap.get(nodeId);
374
+ if (!node) {
375
+ return false;
376
+ }
377
+
378
+ const dependencies = node.depends_on || [];
379
+
380
+ for (const dep of dependencies) {
381
+ if (!completedNodes.has(dep)) {
382
+ return false;
383
+ }
384
+ }
385
+
386
+ return true;
387
+ }
388
+
389
+ /**
390
+ * Check if the workflow contains a cycle
391
+ */
392
+ isCyclic(): boolean {
393
+ const dependencies = this.buildDependenciesMap();
394
+
395
+ // Try to get topological order
396
+ const order = this.getTopologicalOrderInternal(dependencies);
397
+
398
+ // If we can't include all nodes, there's a cycle
399
+ return order.length !== this.workflow.nodes.length;
400
+ }
401
+
402
+ /**
403
+ * Internal topological sort that detects cycles
404
+ */
405
+ private getTopologicalOrderInternal(dependencies: Map<string, string[]>): string[] {
406
+ const dependents = this.buildDependentsMap(dependencies);
407
+
408
+ // Calculate in-degree for each node
409
+ const inDegree = new Map<string, number>();
410
+ for (const nodeId of dependencies.keys()) {
411
+ inDegree.set(nodeId, dependencies.get(nodeId)?.length || 0);
412
+ }
413
+
414
+ // Start with nodes that have no dependencies
415
+ const queue: string[] = [];
416
+ for (const [nodeId, degree] of inDegree.entries()) {
417
+ if (degree === 0) {
418
+ queue.push(nodeId);
419
+ }
420
+ }
421
+
422
+ const result: string[] = [];
423
+
424
+ while (queue.length > 0) {
425
+ const nodeId = queue.shift()!;
426
+ result.push(nodeId);
427
+
428
+ // Reduce in-degree for dependents
429
+ const nodeDependents = dependents.get(nodeId) || [];
430
+ for (const dependent of nodeDependents) {
431
+ const currentDegree = inDegree.get(dependent) || 0;
432
+ inDegree.set(dependent, currentDegree - 1);
433
+
434
+ if (currentDegree - 1 === 0) {
435
+ queue.push(dependent);
436
+ }
437
+ }
438
+ }
439
+
440
+ return result;
441
+ }
442
+
443
+ /**
444
+ * Validate the workflow
445
+ */
446
+ validate(): { valid: boolean; errors: string[] } {
447
+ const errors: string[] = [];
448
+
449
+ // Check for duplicate node IDs
450
+ const nodeIds = new Set<string>();
451
+ for (const node of this.workflow.nodes) {
452
+ if (node.id === '') {
453
+ errors.push('Node ID cannot be empty');
454
+ continue;
455
+ }
456
+
457
+ if (nodeIds.has(node.id)) {
458
+ errors.push(`Duplicate node ID: ${node.id}`);
459
+ }
460
+ nodeIds.add(node.id);
461
+ }
462
+
463
+ // Check for missing dependencies
464
+ for (const node of this.workflow.nodes) {
465
+ const deps = node.depends_on || [];
466
+ for (const dep of deps) {
467
+ if (!nodeIds.has(dep)) {
468
+ errors.push(`Node '${node.id}' depends on non-existent node '${dep}'`);
469
+ }
470
+ }
471
+ }
472
+
473
+ // Check for cycles
474
+ if (this.isCyclic()) {
475
+ errors.push('Workflow contains a cycle');
476
+ }
477
+
478
+ return {
479
+ valid: errors.length === 0,
480
+ errors,
481
+ };
482
+ }
483
+
484
+ /**
485
+ * Add a new node to the workflow dynamically
486
+ *
487
+ * @param node - The node definition to add
488
+ * @throws Error if a node with the same ID already exists
489
+ * @throws Error if the node depends on a non-existent node
490
+ * @throws Error if adding this node would create a cycle
491
+ */
492
+ addNode(node: NodeDefinition): void {
493
+ // Validate node ID
494
+ if (!node.id || node.id.trim() === '') {
495
+ throw new Error('Node ID cannot be empty');
496
+ }
497
+
498
+ // Check for duplicate node ID
499
+ if (this.nodeMap.has(node.id)) {
500
+ throw new Error(`Node with ID '${node.id}' already exists`);
501
+ }
502
+
503
+ // Check for missing dependencies
504
+ const deps = node.depends_on || [];
505
+ for (const dep of deps) {
506
+ if (!this.nodeMap.has(dep)) {
507
+ throw new Error(`Cannot add node '${node.id}': depends on non-existent node '${dep}'`);
508
+ }
509
+ }
510
+
511
+ // Temporarily add the node and check for cycles
512
+ const originalNodes = [...this.workflow.nodes];
513
+ this.workflow.nodes.push(node);
514
+
515
+ if (this.isCyclic()) {
516
+ // Revert if it creates a cycle
517
+ this.workflow.nodes = originalNodes;
518
+ throw new Error(`Cannot add node '${node.id}': would create a cycle in the workflow`);
519
+ }
520
+
521
+ // Add to node map
522
+ this.nodeMap.set(node.id, node);
523
+
524
+ // Clear cached analysis since the DAG has changed
525
+ this.cachedAnalysis = null;
526
+ this.extendedDefinition = null;
527
+ }
528
+
529
+ /**
530
+ * Remove a node from the workflow dynamically
531
+ *
532
+ * @param nodeId - The ID of the node to remove
533
+ * @throws Error if the node doesn't exist
534
+ * @throws Error if other nodes depend on this node
535
+ */
536
+ removeNode(nodeId: string): void {
537
+ // Check if node exists
538
+ if (!this.nodeMap.has(nodeId)) {
539
+ throw new Error(`Node '${nodeId}' does not exist`);
540
+ }
541
+
542
+ // Check if any nodes depend on this node
543
+ const dependents = this.getDependents(nodeId);
544
+ if (dependents.length > 0) {
545
+ throw new Error(`Cannot remove node '${nodeId}': other nodes depend on it: ${dependents.join(', ')}`);
546
+ }
547
+
548
+ // Remove from workflow nodes array
549
+ this.workflow.nodes = this.workflow.nodes.filter(n => n.id !== nodeId);
550
+
551
+ // Remove from node map
552
+ this.nodeMap.delete(nodeId);
553
+
554
+ // Clear cached analysis since the DAG has changed
555
+ this.cachedAnalysis = null;
556
+ this.extendedDefinition = null;
557
+ }
558
+
559
+ /**
560
+ * Get direct dependents of a node (nodes that depend on this node)
561
+ */
562
+ getDependents(nodeId: string): string[] {
563
+ const analysis = this.analyze();
564
+ return analysis.dependents.get(nodeId) || [];
565
+ }
566
+
567
+ /**
568
+ * Get the workflow definition (current state)
569
+ */
570
+ getDefinition(): WorkflowDefinition {
571
+ return this.workflow;
572
+ }
573
+
574
+ /**
575
+ * Get a node by ID
576
+ */
577
+ getNode(nodeId: string): NodeDefinition | undefined {
578
+ return this.nodeMap.get(nodeId);
579
+ }
580
+
581
+ /**
582
+ * Get all node IDs
583
+ */
584
+ getNodeIds(): string[] {
585
+ return Array.from(this.nodeMap.keys());
586
+ }
587
+
588
+ /**
589
+ * Get the number of nodes in the workflow
590
+ */
591
+ getNodeCount(): number {
592
+ return this.nodeMap.size;
593
+ }
594
+ }