@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,493 @@
1
+ /**
2
+ * Tool Registry interface for looking up tools
3
+ * Supports both the full ToolRegistration and simplified Tool interface
4
+ */
5
+ export interface ToolRegistry {
6
+ // Full registration interface (from ToolComponent)
7
+ getTool?(name: string): ToolRegistration | undefined;
8
+ // Simplified tool interface (from ToolNode's own interface)
9
+ getToolByName?(name: string): Tool | undefined;
10
+ hasTool?(name: string): boolean;
11
+ }
12
+
13
+ /**
14
+ * Tool Registration interface (from ToolComponent)
15
+ */
16
+ export interface ToolRegistration {
17
+ tool: Tool;
18
+ enabled: boolean;
19
+ source?: string;
20
+ }
21
+
22
+ /**
23
+ * Tool interface for execution
24
+ */
25
+ export interface Tool {
26
+ name: string;
27
+ execute(input: any, context?: any): Promise<any>;
28
+ }
29
+
30
+ /**
31
+ * ToolNode executes a registered Tool with the provided input.
32
+ *
33
+ * Input can contain template strings like {{nodeId.output.path}} that will be
34
+ * resolved from previousOutputs.
35
+ */
36
+ export class ToolNode implements Node {
37
+ readonly type = 'tool';
38
+ readonly id: string;
39
+
40
+ constructor(
41
+ private definition: NodeDefinition,
42
+ private toolRegistry: ToolRegistry
43
+ ) {
44
+ this.id = definition.id;
45
+ }
46
+
47
+ async execute(context: NodeExecutionContext): Promise<NodeExecutionResult> {
48
+ const startTime = Date.now();
49
+
50
+ try {
51
+ // 1. Get tool name from definition.config.tool or config.toolName
52
+ const toolName = (this.definition.config?.tool ?? this.definition.config?.toolName) as string | undefined;
53
+
54
+ if (!toolName) {
55
+ throw new Error('Tool name is required. Please specify config.tool or config.toolName in the node definition.');
56
+ }
57
+
58
+ // 2. Look up tool from registry (support both interfaces)
59
+ let tool: Tool | undefined;
60
+
61
+ if (this.toolRegistry.getTool) {
62
+ // getTool may return ToolRegistration or Tool depending on implementation
63
+ const result = this.toolRegistry.getTool(toolName);
64
+ if (!result) {
65
+ throw new Error(`Tool not found: ${toolName}`);
66
+ }
67
+ // Check if result has .tool property (ToolRegistration) or is a Tool directly
68
+ if ('tool' in result) {
69
+ tool = (result as ToolRegistration).tool;
70
+ } else {
71
+ tool = result as Tool;
72
+ }
73
+ } else if (this.toolRegistry.getToolByName) {
74
+ // Simplified interface: getToolByName returns Tool directly
75
+ tool = this.toolRegistry.getToolByName(toolName);
76
+ }
77
+
78
+ if (!tool) {
79
+ throw new Error(`Tool not found: ${toolName}`);
80
+ }
81
+
82
+ // 3. Prepare input (may reference previous outputs and workflow input)
83
+ // Support multiple formats:
84
+ // - config.input: { param1: value1, param2: value2 }
85
+ // - config.args: { param1: value1, param2: value2 } (alias)
86
+ // - config.command: "string command" (for tools like bash that use command field)
87
+ // - config.message: "string message" (for tools like echo that use message field)
88
+ let input: any;
89
+
90
+ if (this.definition.config?.command !== undefined) {
91
+ // Special case: single command parameter (e.g., bash tool)
92
+ // Resolve the command template and pass as { command: resolvedCommand }
93
+ const commandTemplate = this.definition.config.command;
94
+ const resolvedCommand = this.resolveTemplateReferences(commandTemplate, context.previousOutputs, context.input as any);
95
+ input = { command: resolvedCommand };
96
+ console.log(`[ToolNode] command input: ${resolvedCommand}`);
97
+ } else if (this.definition.config?.message !== undefined && Object.keys(this.definition.config).length === 1) {
98
+ // Special case: single message parameter (e.g., echo tool)
99
+ // Resolve the message template and pass as { message: resolvedMessage }
100
+ const messageTemplate = this.definition.config.message;
101
+ const resolvedMessage = this.resolveTemplateReferences(messageTemplate, context.previousOutputs, context.input as any);
102
+ input = { message: resolvedMessage };
103
+ console.log(`[ToolNode] message input: ${resolvedMessage}`);
104
+ } else {
105
+ // Normal case: multiple parameters
106
+ input = this.definition.config?.input ?? this.definition.config?.args ?? {};
107
+ input = this.resolveTemplateReferences(input, context.previousOutputs, context.input as any);
108
+ }
109
+
110
+ // 4. Execute tool
111
+ // Try to pass context if the tool supports it, otherwise just pass input
112
+ let output: any;
113
+ try {
114
+ // First try with context (for tools that expect it)
115
+ const toolContext = {
116
+ session_id: context.sessionId || `workflow_${context.runId}`,
117
+ message_id: context.nodeId,
118
+ metadata: {
119
+ workflowRunId: context.runId,
120
+ workflowName: context.workflowName,
121
+ nodeId: context.nodeId,
122
+ },
123
+ };
124
+
125
+ // Execute tool with timeout to detect hanging
126
+ const executePromise = tool.execute(input, toolContext as any);
127
+ const timeoutPromise = new Promise<never>((_, reject) => {
128
+ setTimeout(() => reject(new Error(`Tool ${toolName} execution timeout after 30s`)), 30000);
129
+ });
130
+
131
+ output = await Promise.race([executePromise, timeoutPromise]);
132
+ } catch (err) {
133
+ // If the tool doesn't support context parameter, try without it
134
+ console.log(`[ToolNode] Tool ${toolName} doesn't support context, retrying without context...`);
135
+ output = await tool.execute(input);
136
+ }
137
+
138
+ // 5. Return result
139
+ return {
140
+ output,
141
+ error: undefined,
142
+ durationMs: Date.now() - startTime,
143
+ };
144
+ } catch (error) {
145
+ return {
146
+ output: undefined,
147
+ error: error instanceof Error ? error : new Error(String(error)),
148
+ durationMs: Date.now() - startTime,
149
+ };
150
+ }
151
+ }
152
+
153
+ /**
154
+ * Resolve template references in input values.
155
+ * Templates can be in the format:
156
+ * - {{input.key}} - from workflow input
157
+ * - {{nodeId}} - entire previous node output
158
+ * - {{nodeId.path}} - nested path within previous node output
159
+ * - {{nodes.nodeId}} - entire previous node output (alias)
160
+ * - {{nodes.nodeId.path}} - nested path within previous node output
161
+ * - ${nodeId.output} - dollar-style template (same as {{nodeId.output}})
162
+ *
163
+ * Key behavior for template strings:
164
+ * - If the string is a pure template reference (e.g., "{{nodes.fetch}}"),
165
+ * the resolved value is returned as-is (preserving objects/arrays)
166
+ * - If the string contains text + templates (e.g., "Hello {{name}}"),
167
+ * returns a string with interpolated values
168
+ *
169
+ * @param input - The input object that may contain template strings
170
+ * @param previousOutputs - Map of previous node outputs
171
+ * @param workflowInput - The workflow input parameters
172
+ * @returns The input with template references resolved
173
+ */
174
+ private resolveTemplateReferences(
175
+ input: any,
176
+ previousOutputs: Map<string, any>,
177
+ workflowInput?: Record<string, unknown>
178
+ ): any {
179
+ if (typeof input === 'string') {
180
+ // Check if this is a pure template reference (no surrounding text)
181
+ // Support both {{nodeId.output}} and ${nodeId.output} formats
182
+ const pureTemplate = this.extractPureTemplate(input) || this.extractDollarTemplate(input);
183
+ if (pureTemplate) {
184
+ // Pure template reference - resolve and return value as-is (preserve objects/arrays)
185
+ return this.resolvePureTemplate(pureTemplate, previousOutputs, workflowInput);
186
+ }
187
+ // Mixed text + templates - return resolved string
188
+ return this.resolveStringTemplate(input, previousOutputs, workflowInput);
189
+ }
190
+
191
+ if (Array.isArray(input)) {
192
+ return input.map(item => this.resolveTemplateReferences(item, previousOutputs, workflowInput));
193
+ }
194
+
195
+ if (input !== null && typeof input === 'object') {
196
+ const resolved: Record<string, any> = {};
197
+ for (const [key, value] of Object.entries(input)) {
198
+ resolved[key] = this.resolveTemplateReferences(value, previousOutputs, workflowInput);
199
+ }
200
+ return resolved;
201
+ }
202
+
203
+ // Return primitive values as-is
204
+ return input;
205
+ }
206
+
207
+ /**
208
+ * Check if a string is a pure template reference (starts with {{ and ends with }})
209
+ * @param str - String to check
210
+ * @returns The template content if pure, null otherwise
211
+ */
212
+ private extractPureTemplate(str: string): string | null {
213
+ const trimmed = str.trim();
214
+ if (trimmed.startsWith('{{') && trimmed.endsWith('}}') && trimmed.length > 4) {
215
+ // Extract content between {{ and }}
216
+ const content = trimmed.slice(2, -2).trim();
217
+ // Check there's no other text (except the template patterns themselves)
218
+ if (!content.includes('{{') && !content.includes('}}')) {
219
+ return content;
220
+ }
221
+ }
222
+ return null;
223
+ }
224
+
225
+ /**
226
+ * Check if a string is a dollar-style template reference (starts with ${ and ends with })
227
+ * @param str - String to check
228
+ * @returns The template content if pure, null otherwise
229
+ */
230
+ private extractDollarTemplate(str: string): string | null {
231
+ const trimmed = str.trim();
232
+ if (trimmed.startsWith('${') && trimmed.endsWith('}') && trimmed.length > 3) {
233
+ // Extract content between ${ and }
234
+ const content = trimmed.slice(2, -1).trim();
235
+ // Check there's no other text (except the template patterns themselves)
236
+ if (!content.includes('${') && !content.includes('}')) {
237
+ return content;
238
+ }
239
+ }
240
+ return null;
241
+ }
242
+
243
+ /**
244
+ * Resolve a pure template reference and return the value as-is.
245
+ * @param templateContent - Content between {{ and }} or ${ and }
246
+ * @param previousOutputs - Map of previous node outputs
247
+ * @param workflowInput - The workflow input parameters
248
+ * @returns The resolved value (may be object, array, or primitive),
249
+ * or the original template string if unresolved
250
+ */
251
+ private resolvePureTemplate(
252
+ templateContent: string,
253
+ previousOutputs: Map<string, any>,
254
+ workflowInput?: Record<string, unknown>
255
+ ): any {
256
+ const trimmed = templateContent.trim();
257
+ const originalTemplate = `{{${templateContent}}}`;
258
+
259
+ // Pattern 1: {{input.key}} or ${input.key}
260
+ if (trimmed.startsWith('input.')) {
261
+ const value = this.getNestedValue(workflowInput, trimmed.slice(6));
262
+ return value !== undefined ? value : originalTemplate;
263
+ }
264
+
265
+ // Pattern 2: {{nodes.nodeId}} or ${nodes.nodeId}
266
+ if (trimmed.startsWith('nodes.')) {
267
+ const path = trimmed.slice(6);
268
+ const segments = path.split('.');
269
+ const nodeId = segments[0];
270
+ const restPath = segments.slice(1).join('.') || undefined;
271
+ const value = this.resolvePath(nodeId, restPath, previousOutputs);
272
+ return value !== undefined ? value : originalTemplate;
273
+ }
274
+
275
+ // Pattern 3: Bare nodeId (backward compatibility) - e.g., "node-1" or "node-1.output.task_id"
276
+ if (trimmed !== 'input' && trimmed !== 'nodes') {
277
+ const segments = trimmed.split('.');
278
+ const nodeId = segments[0];
279
+ const restPath = segments.slice(1).join('.') || undefined;
280
+ const value = this.resolvePath(nodeId, restPath, previousOutputs);
281
+ return value !== undefined ? value : originalTemplate;
282
+ }
283
+
284
+ return originalTemplate;
285
+ }
286
+
287
+ /**
288
+ * Resolve template references in a single string (for mixed text + templates).
289
+ * Supports formats:
290
+ * - {{input.key}} - from workflow input
291
+ * - {{nodeId}} - entire previous node output
292
+ * - {{nodeId.path}} - nested path within previous node output
293
+ * - {{nodes.nodeId}} - entire previous node output (alias)
294
+ * - {{nodes.nodeId.path}} - nested path within previous node output
295
+ * - ${nodeId.output} - dollar-style template
296
+ *
297
+ * Note: Returns string because the input contains text around the template.
298
+ * For pure template references, use resolvePureTemplate instead.
299
+ */
300
+ private resolveStringTemplate(
301
+ template: string,
302
+ previousOutputs: Map<string, any>,
303
+ workflowInput?: Record<string, unknown>
304
+ ): string {
305
+ let resolved = template;
306
+
307
+ // Pattern 1: {{input.key}} or ${input.key}
308
+ resolved = resolved.replace(/\{\{input\.([^}]+)\}\}/g, (match, path) => {
309
+ const value = this.getNestedValue(workflowInput, path.trim());
310
+ return this.stringifyValue(value, match);
311
+ });
312
+ resolved = resolved.replace(/\$\{input\.([^}]+)\}/g, (match, path) => {
313
+ const value = this.getNestedValue(workflowInput, path.trim());
314
+ return this.stringifyValue(value, match);
315
+ });
316
+
317
+ // Pattern 2: {{nodeId}} or {{nodeId.path}} (bare nodeId without 'nodes.' prefix)
318
+ resolved = resolved.replace(/\{\{([^:.{}][^}]*?)\}\}/g, (match, path) => {
319
+ // Skip if it starts with 'nodes.' (handled by pattern 3) or 'input.' (handled by pattern 1)
320
+ if (path.trim().startsWith('nodes.') || path.trim().startsWith('input.')) {
321
+ return match;
322
+ }
323
+ // Split into nodeId and optional path
324
+ const segments = path.trim().split('.');
325
+ const nodeId = segments[0];
326
+ const restPath = segments.slice(1).join('.') || undefined;
327
+ const value = this.resolvePath(nodeId, restPath, previousOutputs);
328
+ return this.stringifyValue(value, match);
329
+ });
330
+
331
+ // Pattern 3: ${nodeId.output} - dollar-style template
332
+ resolved = resolved.replace(/\$\{([^}]+)\}/g, (match, path) => {
333
+ // Skip if it starts with 'nodes.' or 'input.'
334
+ if (path.trim().startsWith('nodes.') || path.trim().startsWith('input.')) {
335
+ return match;
336
+ }
337
+ // Split into nodeId and optional path
338
+ const segments = path.trim().split('.');
339
+ const nodeId = segments[0];
340
+ const restPath = segments.slice(1).join('.') || undefined;
341
+ const value = this.resolvePath(nodeId, restPath, previousOutputs);
342
+ return this.stringifyValue(value, match);
343
+ });
344
+
345
+ // Pattern 4: {{nodes.nodeId}} or {{nodes.nodeId.path}}
346
+ resolved = resolved.replace(/\{\{nodes\.([^}]+)\}\}/g, (match, path) => {
347
+ const segments = path.trim().split('.');
348
+ const nodeId = segments[0];
349
+ const restPath = segments.slice(1).join('.') || undefined;
350
+ // Get raw node output and extract from wrapper format
351
+ const nodeOutput = this.resolvePath(nodeId, undefined, previousOutputs);
352
+ let extractedValue = this.extractFromWrapper(nodeOutput);
353
+
354
+ // If extracted value is a string and restPath is "output", the user is trying
355
+ // to access the output property of a tool result. In this case, the string
356
+ // itself IS the output, so return it directly.
357
+ if (typeof extractedValue === 'string' && restPath === 'output') {
358
+ return this.stringifyValue(extractedValue, match);
359
+ }
360
+
361
+ if (restPath) {
362
+ const value = this.getNestedValue(extractedValue, restPath);
363
+ return this.stringifyValue(value, match);
364
+ } else {
365
+ return this.stringifyValue(extractedValue, match);
366
+ }
367
+ });
368
+
369
+ return resolved;
370
+ }
371
+
372
+ /**
373
+ * Resolve a value from previous outputs.
374
+ * @param nodeId - The node ID to look up
375
+ * @param path - Optional dot-notation path within the node output
376
+ * @param previousOutputs - Map of previous node outputs
377
+ */
378
+ private resolvePath(nodeId: string, path: string | undefined, previousOutputs: Map<string, any>): any {
379
+ // Try exact match first, then try normalized versions (hyphen <-> underscore)
380
+ let nodeOutput = previousOutputs.get(nodeId);
381
+
382
+ if (nodeOutput === undefined) {
383
+ const normalizedUnderscore = nodeId.replace(/-/g, '_');
384
+ const normalizedHyphen = nodeId.replace(/_/g, '-');
385
+
386
+ if (normalizedUnderscore !== nodeId) {
387
+ nodeOutput = previousOutputs.get(normalizedUnderscore);
388
+ }
389
+ if (nodeOutput === undefined && normalizedHyphen !== nodeId) {
390
+ nodeOutput = previousOutputs.get(normalizedHyphen);
391
+ }
392
+ }
393
+
394
+ if (nodeOutput === undefined) {
395
+ return undefined;
396
+ }
397
+
398
+ if (path === undefined || path === '') {
399
+ return nodeOutput;
400
+ }
401
+
402
+ // Navigate through the path
403
+ return this.getNestedValue(nodeOutput, path);
404
+ }
405
+
406
+ /**
407
+ * Get a nested value from an object using dot notation.
408
+ * @param obj - The object to traverse
409
+ * @param path - Dot-notation path (e.g., "metadata.task_id")
410
+ */
411
+ private getNestedValue(obj: any, path: string): any {
412
+ if (obj === null || obj === undefined) {
413
+ return undefined;
414
+ }
415
+
416
+ const segments = path.split('.');
417
+ let current = obj;
418
+
419
+ for (const segment of segments) {
420
+ if (current === null || current === undefined) {
421
+ return undefined;
422
+ }
423
+ current = current[segment];
424
+ }
425
+
426
+ return current;
427
+ }
428
+
429
+ /**
430
+ * Extract the actual output value from wrapper formats.
431
+ * For strings, removes trailing whitespace (especially newlines from bash commands).
432
+ */
433
+ private extractFromWrapper(value: unknown): unknown {
434
+ if (value === null || value === undefined) {
435
+ return value;
436
+ }
437
+ if (typeof value !== 'object') {
438
+ return value;
439
+ }
440
+ let result: unknown;
441
+ if ('output' in value) {
442
+ result = (value as { output: unknown }).output;
443
+ } else if ('result' in value && !('output' in value)) {
444
+ result = (value as { result: unknown }).result;
445
+ } else {
446
+ return value;
447
+ }
448
+ // For string results from tools (like bash), trim trailing whitespace
449
+ // This prevents double newlines when embedding in shell commands
450
+ if (typeof result === 'string') {
451
+ return result.trimEnd();
452
+ }
453
+ return result;
454
+ }
455
+
456
+ /**
457
+ * Convert a value to string for template interpolation.
458
+ * @param value - The value to convert
459
+ * @param fallback - Fallback string if value is undefined
460
+ */
461
+ private stringifyValue(value: any, fallback: string): string {
462
+ if (value === undefined || value === null) {
463
+ return fallback;
464
+ }
465
+
466
+ let str: string;
467
+ if (typeof value === 'object') {
468
+ str = JSON.stringify(value);
469
+ } else {
470
+ str = String(value);
471
+ }
472
+
473
+ // Escape special characters for shell:
474
+ // - \ \ -> \\\ (for echo to show literal \)
475
+ // - " -> \"
476
+ // - ` -> \`
477
+ // - $ -> \$
478
+ // - \n -> \\n (for echo to show literal \n, not newline)
479
+ // - \r -> \\r
480
+ // - \t -> \\t
481
+ return str
482
+ .replace(/\\/g, '\\\\')
483
+ .replace(/\n/g, '\\n')
484
+ .replace(/\r/g, '\\r')
485
+ .replace(/\t/g, '\\t')
486
+ .replace(/"/g, '\\"')
487
+ .replace(/`/g, '\\`')
488
+ .replace(/\$/g, '\\$');
489
+ }
490
+ }
491
+
492
+ // Re-export types from workflow types
493
+ import type { Node, NodeDefinition, NodeExecutionContext, NodeExecutionResult } from '../types';