@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,315 @@
1
+ import { describe, it, expect, beforeEach, jest } from 'vitest';
2
+ import { WorkflowNode, WorkflowRunner, WorkflowRunResult } from './workflow-node';
3
+ import type { NodeDefinition, NodeExecutionContext } from '../types';
4
+
5
+ // Mock EventBus
6
+ const mockEventBus = {
7
+ emit: jest.fn(),
8
+ on: jest.fn(),
9
+ off: jest.fn(),
10
+ } as any;
11
+
12
+ // Mock WorkflowRunner
13
+ const createMockRunner = () => ({
14
+ run: jest.fn<(workflowName: string, input?: any) => Promise<WorkflowRunResult>>(),
15
+ getRun: jest.fn<(runId: string) => Promise<WorkflowRunResult | null>>(),
16
+ });
17
+
18
+ describe('WorkflowNode', () => {
19
+ let nodeDefinition: NodeDefinition;
20
+ let executionContext: NodeExecutionContext;
21
+ let mockRunner: ReturnType<typeof createMockRunner>;
22
+
23
+ beforeEach(() => {
24
+ jest.clearAllMocks();
25
+
26
+ nodeDefinition = {
27
+ id: 'workflow-node-1',
28
+ type: 'workflow',
29
+ name: 'Nested Workflow Node',
30
+ config: {
31
+ workflow_name: 'sub-workflow',
32
+ input: { message: 'hello from parent' },
33
+ wait_complete: true,
34
+ },
35
+ };
36
+
37
+ executionContext = {
38
+ sessionId: 'session-1',
39
+ nodeOutputs: {},
40
+ input: {},
41
+ metadata: {},
42
+ };
43
+
44
+ mockRunner = createMockRunner();
45
+ });
46
+
47
+ describe('constructor', () => {
48
+ it('should create a WorkflowNode with correct type', () => {
49
+ const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
50
+ expect(workflowNode.type).toBe('workflow');
51
+ expect(workflowNode.id).toBe('workflow-node-1');
52
+ });
53
+
54
+ it('should create WorkflowNode with minimal config', () => {
55
+ const minimalDef: NodeDefinition = {
56
+ id: 'minimal-node',
57
+ type: 'workflow',
58
+ config: {
59
+ workflow_name: 'simple-workflow',
60
+ },
61
+ };
62
+
63
+ const workflowNode = new WorkflowNode(minimalDef, mockRunner);
64
+ expect(workflowNode.type).toBe('workflow');
65
+ expect(workflowNode.id).toBe('minimal-node');
66
+ });
67
+ });
68
+
69
+ describe('execute', () => {
70
+ it('should run sub-workflow successfully', async () => {
71
+ const completedResult: WorkflowRunResult = {
72
+ runId: 'run_123',
73
+ status: 'completed',
74
+ output: { result: 'nested workflow done' },
75
+ };
76
+
77
+ mockRunner.run.mockResolvedValue(completedResult);
78
+
79
+ const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
80
+ const result = await workflowNode.execute(executionContext);
81
+
82
+ expect(result.success).toBe(true);
83
+ expect(result.output).toEqual({ result: 'nested workflow done' });
84
+ expect(result.error).toBeUndefined();
85
+ expect(mockRunner.run).toHaveBeenCalledWith('sub-workflow', {
86
+ message: 'hello from parent',
87
+ });
88
+ });
89
+
90
+ it('should pass input to sub-workflow', async () => {
91
+ const customInput = { param1: 'value1', param2: 123, nested: { data: 'test' } };
92
+ nodeDefinition.config.input = customInput;
93
+
94
+ mockRunner.run.mockResolvedValue({
95
+ runId: 'run_456',
96
+ status: 'completed',
97
+ output: { processed: true },
98
+ });
99
+
100
+ const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
101
+ await workflowNode.execute(executionContext);
102
+
103
+ expect(mockRunner.run).toHaveBeenCalledWith('sub-workflow', customInput);
104
+ });
105
+
106
+ it('should wait for completion when wait_complete=true', async () => {
107
+ nodeDefinition.config.wait_complete = true;
108
+
109
+ mockRunner.run.mockResolvedValue({
110
+ runId: 'run_wait',
111
+ status: 'completed',
112
+ output: { waited: true },
113
+ });
114
+
115
+ const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
116
+ const result = await workflowNode.execute(executionContext);
117
+
118
+ expect(result.success).toBe(true);
119
+ expect(result.output).toEqual({ waited: true });
120
+ // When wait_complete=true, run should return completed status
121
+ expect(mockRunner.run).toHaveBeenCalled();
122
+ });
123
+
124
+ it('should not wait for completion when wait_complete=false', async () => {
125
+ nodeDefinition.config.wait_complete = false;
126
+
127
+ mockRunner.run.mockResolvedValue({
128
+ runId: 'run_fire',
129
+ status: 'completed', // Fire-and-forget returns immediately
130
+ output: { runId: 'run_fire' },
131
+ });
132
+
133
+ const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
134
+ const result = await workflowNode.execute(executionContext);
135
+
136
+ expect(result.success).toBe(true);
137
+ expect(result.output).toBeDefined();
138
+ // Should have called run
139
+ expect(mockRunner.run).toHaveBeenCalled();
140
+ });
141
+
142
+ it('should handle sub-workflow failure', async () => {
143
+ mockRunner.run.mockResolvedValue({
144
+ runId: 'run_fail',
145
+ status: 'failed',
146
+ error: 'Sub-workflow execution failed',
147
+ });
148
+
149
+ const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
150
+ const result = await workflowNode.execute(executionContext);
151
+
152
+ expect(result.success).toBe(false);
153
+ expect(result.error).toBeDefined();
154
+ expect(result.error).toContain('Sub-workflow execution failed');
155
+ });
156
+
157
+ it('should handle sub-workflow not found', async () => {
158
+ mockRunner.run.mockRejectedValue(new Error('Workflow not found: non-existent-workflow'));
159
+
160
+ nodeDefinition.config.workflow_name = 'non-existent-workflow';
161
+
162
+ const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
163
+ const result = await workflowNode.execute(executionContext);
164
+
165
+ expect(result.success).toBe(false);
166
+ expect(result.error).toBeDefined();
167
+ expect(result.error.message).toContain('Workflow not found');
168
+ });
169
+
170
+ it('should record execution duration', async () => {
171
+ mockRunner.run.mockImplementation(async () => {
172
+ await new Promise(resolve => setTimeout(resolve, 10));
173
+ return {
174
+ runId: 'run_duration',
175
+ status: 'completed',
176
+ output: { done: true },
177
+ };
178
+ });
179
+
180
+ const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
181
+ const result = await workflowNode.execute(executionContext);
182
+
183
+ expect(result.duration).toBeGreaterThanOrEqual(0);
184
+ });
185
+
186
+ it('should resolve template references from previous outputs', async () => {
187
+ // Set up previous outputs from other nodes
188
+ // Note: The Map format stores raw outputs directly, so path is {{nodeId.property}}
189
+ executionContext.previousOutputs = new Map([
190
+ ['node-0', { value: 'from-node-0' }]
191
+ ]);
192
+
193
+ // Create node with template reference input
194
+ const templateNodeDef: NodeDefinition = {
195
+ id: 'workflow-node-1',
196
+ type: 'workflow',
197
+ name: 'Nested Workflow Node',
198
+ config: {
199
+ workflow_name: 'sub-workflow',
200
+ input: {
201
+ message: '{{node-0.value}}',
202
+ },
203
+ },
204
+ };
205
+
206
+ mockRunner.run.mockResolvedValue({
207
+ runId: 'run_template',
208
+ status: 'completed',
209
+ output: { received: 'from-node-0' },
210
+ });
211
+
212
+ const workflowNode = new WorkflowNode(templateNodeDef, mockRunner);
213
+ await workflowNode.execute(executionContext);
214
+
215
+ // The input should have template references resolved
216
+ expect(mockRunner.run).toHaveBeenCalledWith('sub-workflow', {
217
+ message: 'from-node-0',
218
+ });
219
+ });
220
+
221
+ it('should handle unresolved template references gracefully', async () => {
222
+ nodeDefinition.config.input = {
223
+ message: '{{non-existent.value}}',
224
+ };
225
+
226
+ mockRunner.run.mockResolvedValue({
227
+ runId: 'run_unresolved',
228
+ status: 'completed',
229
+ output: {},
230
+ });
231
+
232
+ const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
233
+ await workflowNode.execute(executionContext);
234
+
235
+ // Should pass through unresolved template as-is
236
+ expect(mockRunner.run).toHaveBeenCalledWith('sub-workflow', {
237
+ message: '{{non-existent.value}}',
238
+ });
239
+ });
240
+
241
+ it('should throw error when workflow_name is missing', async () => {
242
+ nodeDefinition.config = {}; // No workflow_name specified
243
+
244
+ const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
245
+ const result = await workflowNode.execute(executionContext);
246
+
247
+ expect(result.success).toBe(false);
248
+ expect(result.error).toBeDefined();
249
+ expect(result.error.message).toContain('workflow_name');
250
+ });
251
+
252
+ it('should pass metadata to sub-workflow', async () => {
253
+ executionContext.metadata = {
254
+ parentRunId: 'parent-123',
255
+ correlationId: 'corr-456',
256
+ };
257
+
258
+ mockRunner.run.mockResolvedValue({
259
+ runId: 'run_meta',
260
+ status: 'completed',
261
+ output: {},
262
+ });
263
+
264
+ const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
265
+ await workflowNode.execute(executionContext);
266
+
267
+ expect(mockRunner.run).toHaveBeenCalledWith('sub-workflow', {
268
+ message: 'hello from parent',
269
+ });
270
+ });
271
+
272
+ it('should handle stopped sub-workflow', async () => {
273
+ mockRunner.run.mockResolvedValue({
274
+ runId: 'run_stopped',
275
+ status: 'stopped',
276
+ error: 'Workflow was stopped',
277
+ });
278
+
279
+ const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
280
+ const result = await workflowNode.execute(executionContext);
281
+
282
+ expect(result.success).toBe(false);
283
+ expect(result.error).toBeDefined();
284
+ });
285
+ });
286
+
287
+ describe('WorkflowRunner interface', () => {
288
+ it('should use run method to start workflows', async () => {
289
+ mockRunner.run.mockResolvedValue({
290
+ runId: 'run_interface',
291
+ status: 'completed',
292
+ output: { test: true },
293
+ });
294
+
295
+ const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
296
+ await workflowNode.execute(executionContext);
297
+
298
+ expect(mockRunner.run).toHaveBeenCalledTimes(1);
299
+ });
300
+
301
+ it('should use getRun method for status checking', async () => {
302
+ // This test verifies the interface contract
303
+ mockRunner.getRun.mockResolvedValue({
304
+ runId: 'run_status',
305
+ status: 'completed',
306
+ output: {},
307
+ });
308
+
309
+ // The interface supports getRun for status checking
310
+ const status = await mockRunner.getRun('run_status');
311
+ expect(status).toBeDefined();
312
+ expect(status?.runId).toBe('run_status');
313
+ });
314
+ });
315
+ });
@@ -0,0 +1,311 @@
1
+ import type { Node, NodeDefinition, NodeExecutionContext, NodeExecutionResult } from '../types';
2
+
3
+ /**
4
+ * WorkflowRunner interface for running sub-workflows
5
+ */
6
+ export interface WorkflowRunner {
7
+ /**
8
+ * Run a workflow by name
9
+ * @param workflowName - Name of the workflow to run
10
+ * @param input - Optional input to pass to the workflow
11
+ * @returns Promise resolving to the workflow run result
12
+ */
13
+ run(workflowName: string, input?: any): Promise<WorkflowRunResult>;
14
+
15
+ /**
16
+ * Get the status of a workflow run
17
+ * @param runId - The run ID to check
18
+ * @returns Promise resolving to the run result or null if not found
19
+ */
20
+ getRun(runId: string): Promise<WorkflowRunResult | null>;
21
+ }
22
+
23
+ /**
24
+ * Result from a workflow run
25
+ */
26
+ export interface WorkflowRunResult {
27
+ /** Unique run ID */
28
+ runId: string;
29
+ /** Final status of the run */
30
+ status: 'completed' | 'failed' | 'stopped';
31
+ /** Output from the workflow (if completed) */
32
+ output?: any;
33
+ /** Error message (if failed or stopped) */
34
+ error?: string;
35
+ }
36
+
37
+ /**
38
+ * WorkflowNode configuration
39
+ */
40
+ export interface WorkflowNodeConfig {
41
+ /** Name of the workflow to invoke */
42
+ workflow_name: string;
43
+ /** Input to pass to the sub-workflow (supports template strings) */
44
+ input?: Record<string, unknown>;
45
+ /** Whether to wait for completion (default: true) */
46
+ wait_complete?: boolean;
47
+ }
48
+
49
+ /**
50
+ * WorkflowNode executes a nested workflow as a node.
51
+ *
52
+ * This allows one workflow to invoke another workflow as a single node,
53
+ * enabling workflow composition and reusability.
54
+ *
55
+ * Features:
56
+ * - Pass input to sub-workflow (with template resolution)
57
+ * - Wait for completion or fire-and-forget
58
+ * - Handle sub-workflow errors
59
+ * - Track execution duration
60
+ */
61
+ export class WorkflowNode implements Node {
62
+ readonly type = 'workflow';
63
+ readonly id: string;
64
+
65
+ private readonly config: WorkflowNodeConfig;
66
+
67
+ constructor(
68
+ definition: NodeDefinition,
69
+ private workflowRunner: WorkflowRunner
70
+ ) {
71
+ this.id = definition.id;
72
+ this.config = (definition.config ?? {}) as any as WorkflowNodeConfig;
73
+ }
74
+
75
+ async execute(context: NodeExecutionContext): Promise<NodeExecutionResult> {
76
+ const startTime = Date.now();
77
+
78
+ try {
79
+ // 1. Get workflow name from config
80
+ const workflowName = this.config.workflow_name;
81
+
82
+ if (!workflowName) {
83
+ throw new Error('workflow_name is required in config. Please specify config.workflow_name in the node definition.');
84
+ }
85
+
86
+ // 2. Resolve input template (may reference previous outputs)
87
+ // Support both Map (previousOutputs) and Record (nodeOutputs) formats
88
+ const previousOutputs = context.previousOutputs ?? this.convertToMap((context.nodeOutputs ?? {}) as any);
89
+ const resolvedInput = this.resolveInput(
90
+ this.config.input ?? {},
91
+ previousOutputs
92
+ );
93
+
94
+ // 3. Run sub-workflow via workflowRunner
95
+ const result = await this.workflowRunner.run(workflowName, resolvedInput);
96
+
97
+ // 4. Determine success based on status
98
+ const success = result.status === 'completed';
99
+
100
+ // 5. Return result
101
+ return {
102
+ success,
103
+ output: result.output,
104
+ error: success ? undefined : result.error,
105
+ duration: Date.now() - startTime,
106
+ };
107
+ } catch (error) {
108
+ return {
109
+ success: false,
110
+ output: undefined,
111
+ error: error instanceof Error ? error : new Error(String(error)),
112
+ duration: Date.now() - startTime,
113
+ };
114
+ }
115
+ }
116
+
117
+ /**
118
+ * Convert Record<string, NodeExecutionResult> to Map<string, any>
119
+ * for compatibility with resolvePath that expects Map format.
120
+ *
121
+ * In the Record format, the value is NodeExecutionResult (with output field).
122
+ * In the Map format, the value is the raw output directly.
123
+ */
124
+ private convertToMap(nodeOutputs: Record<string, NodeExecutionResult>): Map<string, any> {
125
+ const map = new Map<string, any>();
126
+ for (const [nodeId, result] of Object.entries(nodeOutputs)) {
127
+ // Store the entire result for access to both output and success
128
+ map.set(nodeId, result.output);
129
+ }
130
+ return map;
131
+ }
132
+
133
+ /**
134
+ * Resolve input template references.
135
+ * Templates are in the format {{nodeId.output.path}}
136
+ *
137
+ * @param input - The input object that may contain template strings
138
+ * @param previousOutputs - Map of previous node outputs (value is raw output)
139
+ * @returns The input with template references resolved
140
+ */
141
+ private resolveInput(
142
+ input: Record<string, unknown>,
143
+ previousOutputs: Map<string, any>
144
+ ): Record<string, unknown> {
145
+ const resolved: Record<string, unknown> = {};
146
+
147
+ for (const [key, value] of Object.entries(input)) {
148
+ resolved[key] = this.resolveValue(value, previousOutputs);
149
+ }
150
+
151
+ return resolved;
152
+ }
153
+
154
+ /**
155
+ * Resolve a value that may contain template references.
156
+ */
157
+ private resolveValue(value: unknown, previousOutputs: Map<string, any>): unknown {
158
+ if (typeof value === 'string') {
159
+ return this.resolveStringTemplate(value, previousOutputs);
160
+ }
161
+
162
+ if (Array.isArray(value)) {
163
+ return value.map(item => this.resolveValue(item, previousOutputs));
164
+ }
165
+
166
+ if (value !== null && typeof value === 'object') {
167
+ const resolved: Record<string, unknown> = {};
168
+ for (const [k, v] of Object.entries(value as Record<string, unknown>)) {
169
+ resolved[k] = this.resolveValue(v, previousOutputs);
170
+ }
171
+ return resolved;
172
+ }
173
+
174
+ // Return primitive values as-is
175
+ return value;
176
+ }
177
+
178
+ /**
179
+ * Resolve template references in a single string.
180
+ * Supports format: {{nodeId.output.path}}
181
+ */
182
+ private resolveStringTemplate(
183
+ template: string,
184
+ previousOutputs: Map<string, any>
185
+ ): string {
186
+ // Match template patterns like {{nodeId}} or {{nodeId.output.value}}
187
+ const templatePattern = /\{\{([^}]+)\}\}/g;
188
+
189
+ return template.replace(templatePattern, (match, path) => {
190
+ const trimmedPath = path.trim();
191
+
192
+ // Try to resolve the path from previous outputs
193
+ const value = this.resolvePath(trimmedPath, previousOutputs);
194
+
195
+ if (value !== undefined) {
196
+ return this.stringifyValue(value, match);
197
+ }
198
+
199
+ // If not found, return the original template string
200
+ return match;
201
+ });
202
+ }
203
+
204
+ /**
205
+ * Resolve a dot-notation path from previous outputs.
206
+ * First segment is the node ID, rest is the path within that node's output.
207
+ */
208
+ private resolvePath(
209
+ path: string,
210
+ previousOutputs: Map<string, any>
211
+ ): unknown {
212
+ const segments = path.split('.');
213
+
214
+ if (segments.length === 0) {
215
+ return undefined;
216
+ }
217
+
218
+ const nodeId = segments[0];
219
+ let nodeOutput = previousOutputs.get(nodeId);
220
+
221
+ if (nodeOutput === undefined) {
222
+ return undefined;
223
+ }
224
+
225
+ // Extract from wrapper formats (DecoratorNode: {success, output} and AgentNode: {result, metadata})
226
+ nodeOutput = this.extractFromWrapper(nodeOutput);
227
+
228
+ // If no path beyond node ID, return the extracted value
229
+ if (segments.length === 1) {
230
+ return nodeOutput;
231
+ }
232
+
233
+ // Navigate the path within the node's output
234
+ // Skip first segment if it's a wrapper property since extractFromWrapper already extracted it
235
+ let current: any = nodeOutput;
236
+ let startIndex = 1;
237
+ if (segments.length > 1 &&
238
+ (segments[1] === 'output' || segments[1] === 'result' || segments[1] === 'metadata')) {
239
+ startIndex = 2;
240
+ }
241
+
242
+ for (let i = startIndex; i < segments.length; i++) {
243
+ if (current === null || current === undefined) {
244
+ return undefined;
245
+ }
246
+ current = current[segments[i]];
247
+ }
248
+
249
+ return current;
250
+ }
251
+
252
+ /**
253
+ * Extract the actual output value from wrapper formats.
254
+ *
255
+ * Handles:
256
+ * - AgentNode format: { result, metadata } -> returns result
257
+ * - DecoratorNode format: { success, output, ... } -> returns output
258
+ * - Other formats: returns value as-is
259
+ *
260
+ * @param value - The value to extract from
261
+ * @returns The extracted value
262
+ */
263
+ private extractFromWrapper(value: unknown): unknown {
264
+ if (value === null || value === undefined) {
265
+ return value;
266
+ }
267
+
268
+ if (typeof value !== 'object') {
269
+ return value;
270
+ }
271
+
272
+ // Check for AgentNode wrapper: { result, metadata }
273
+ // This is the signature of AgentNode output
274
+ if ('result' in value && 'metadata' in value) {
275
+ return (value as { result: unknown }).result;
276
+ }
277
+
278
+ // Check for DecoratorNode wrapper: { success, output, ... }
279
+ // This has 'output' but NOT 'result' (that's AgentNode's signature)
280
+ if ('output' in value) {
281
+ return (value as { output: unknown }).output;
282
+ }
283
+
284
+ return value;
285
+ }
286
+
287
+ /**
288
+ * Convert a value to string for template replacement.
289
+ * - If value is undefined, return the original template
290
+ * - If value is a string, return it directly
291
+ * - If value is null, return 'null'
292
+ * - If value is an object or array, serialize to JSON
293
+ *
294
+ * @param value - The value to stringify
295
+ * @param originalTemplate - The original template string (returned if value is undefined)
296
+ * @returns String representation of the value
297
+ */
298
+ private stringifyValue(value: unknown, originalTemplate: string): string {
299
+ if (value === undefined) {
300
+ return originalTemplate;
301
+ }
302
+ if (value === null) {
303
+ return 'null';
304
+ }
305
+ if (typeof value === 'string') {
306
+ return value;
307
+ }
308
+ // For objects and arrays, serialize to JSON
309
+ return JSON.stringify(value);
310
+ }
311
+ }
@@ -0,0 +1,27 @@
1
+ /**
2
+ * Service Index - Session-based Workflow Service
3
+ *
4
+ * After refactoring, RunRepository is removed. Workflow runs are managed via SessionComponent.
5
+ */
6
+
7
+ export { WorkflowService } from './workflow-service';
8
+ export type {
9
+ WorkflowRepository,
10
+ WorkflowEngine,
11
+ CreateOptions,
12
+ ListOptions,
13
+ WorkflowUpdates,
14
+ WorkflowEngineFactory,
15
+ SessionInfo,
16
+ WorkflowSessionMetadata,
17
+ AgentSessionRef,
18
+ } from './workflow-service';
19
+
20
+ export {
21
+ WorkflowServiceRegistry,
22
+ getWorkflowService,
23
+ registerWorkflowService,
24
+ } from './registry';
25
+
26
+ // Re-export WorkflowDefinition from types
27
+ export type { WorkflowDefinition } from '../types/workflow';