@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,568 @@
1
+ import { BaseComponent, type ComponentConfig } from "../component";
2
+ import { type LLMConfig, type LLMInvokeRequest, type LLMInvokeResult, type LLMMessage, type ModelLimits } from "./types";
3
+ import { invoke, type InvokeConfig } from "./invoke";
4
+ import type { ConfigComponent } from "../../config/config-component";
5
+ import { createLogger } from "../log-trace/logger";
6
+ import { TracedAs } from "../log-trace/decorator";
7
+ import { toEnvKey, envKeyToConfigKey } from "../../config/env-key";
8
+ import { ContextError, ErrorCodes } from "../errors";
9
+
10
+ // 创建 llm 模块日志器
11
+ const logger = createLogger("llm");
12
+
13
+ /**
14
+ * LLM 调用上下文(共享数据)
15
+ */
16
+ interface InvokeContext {
17
+ providerId: string;
18
+ model: string;
19
+ invokeConfig: InvokeConfig;
20
+ messages: LLMMessage[];
21
+ temperature: number;
22
+ maxTokens: number;
23
+ topP: number;
24
+ abortSignal: AbortSignal | undefined;
25
+ }
26
+
27
+ /**
28
+ * LLMComponent 配置(通过 options.config 传递)
29
+ *
30
+ * 配置加载顺序(优先级从低到高):
31
+ * 1. File - 配置文件(通过 configPath 指定)
32
+ * 2. Env - 环境变量(通过 envPrefix 配置前缀)
33
+ * 3. Object - 直接传入的 config 对象(最高优先级)
34
+ *
35
+ * 环境变量转换规则:
36
+ * - 驼峰命名会被转换为下划线分隔的大写形式
37
+ * - 例如: "llm.defaultModel" + prefix "LLM" -> "LLM_LLM_DEFAULT_MODEL"
38
+ */
39
+ export interface LLMComponentOptions {
40
+ /** ConfigComponent 实例(必填) */
41
+ configComponent: ConfigComponent;
42
+
43
+ /** 配置文件相对路径(可选,基于 XDG_DATA_HOME) */
44
+ configPath?: string;
45
+
46
+ /** 环境变量前缀(可选,默认 "LLM") */
47
+ envPrefix?: string;
48
+
49
+ /** 直接传入的配置对象(可选,优先级最高) */
50
+ config?: Partial<LLMConfig>;
51
+ }
52
+
53
+ /**
54
+ * LLMComponent
55
+ * 管理 LLM 调用,使用 AI SDK 进行实际调用
56
+ *
57
+ * 配置通过 ConfigComponent 实时获取,支持配置热更新
58
+ *
59
+ * @example
60
+ * // 通过 ConfigComponent 加载配置
61
+ * const configComponent = new ConfigComponent();
62
+ * configComponent.setXdgDataHome(__dirname);
63
+ *
64
+ * const llmComponent = new LLMComponent();
65
+ * await llmComponent.init({
66
+ * env, // Environment 实例(必填)
67
+ * options: {
68
+ * configComponent, // ConfigComponent 实例
69
+ * configPath: "config.jsonc" // 配置文件路径
70
+ * }
71
+ * });
72
+ *
73
+ * // 配置更新后,下次调用会自动获取最新配置
74
+ * configComponent.set("llm.temperature", 0.5);
75
+ */
76
+ export class LLMComponent extends BaseComponent {
77
+ readonly name = "llm";
78
+ readonly version = "2.0.0";
79
+
80
+ private configComponent?: ConfigComponent;
81
+
82
+ /** 配置变更 watcher 清理函数 */
83
+ private configWatcher?: () => void;
84
+
85
+ constructor() {
86
+ super();
87
+ }
88
+
89
+ /**
90
+ * 初始化组件
91
+ *
92
+ * 配置加载优先级(从高到低):
93
+ * 1. Object - 直接传入的 config 对象
94
+ * 2. Env - 环境变量(通过 envPrefix 配置前缀)
95
+ * 3. File - 配置文件(通过 configPath 指定)
96
+ */
97
+ async init(config: ComponentConfig): Promise<void> {
98
+ // 调用基类 init,注入 env
99
+ await super.init(config);
100
+
101
+ // 从 options 获取 ConfigComponent
102
+ const options = config.options as unknown as LLMComponentOptions | undefined;
103
+ if (!options?.configComponent) {
104
+ throw new Error("ConfigComponent is required for LLMComponent initialization");
105
+ }
106
+
107
+ this.configComponent = options.configComponent;
108
+ await this.registerConfig(options);
109
+
110
+ this.setStatus("running");
111
+ }
112
+
113
+ /**
114
+ * 注册配置到 ConfigComponent
115
+ *
116
+ * 配置加载顺序(优先级从低到高):
117
+ * 1. FileSource - 从配置文件加载(最低)
118
+ * 2. EnvSource - 从环境变量加载(中等)
119
+ * 3. MemorySource - 直接配置对象(最高)
120
+ */
121
+ private async registerConfig(options: LLMComponentOptions): Promise<void> {
122
+ const configComponent = options.configComponent;
123
+ if (!configComponent) return;
124
+
125
+ const { configPath, envPrefix, config } = options;
126
+ const prefix = envPrefix !== undefined ? envPrefix : "LLM";
127
+
128
+ // 获取所有已注册的 keys
129
+ const llmKeys = [
130
+ "llm.defaultModel",
131
+ "llm.defaultProvider",
132
+ "llm.temperature",
133
+ "llm.maxTokens",
134
+ "llm.topP",
135
+ "llm.stream",
136
+ "llm.providers",
137
+ "llm.limits",
138
+ "llm.capabilities",
139
+ ];
140
+
141
+ // 1. 注册并加载 FileSource(如果提供了 configPath)
142
+ let fileSource: any = null;
143
+ if (configPath) {
144
+ configComponent.registerSource({
145
+ type: "file",
146
+ relativePath: configPath,
147
+ optional: true,
148
+ watch: false
149
+ });
150
+
151
+ // 确保 source 已创建
152
+ (configComponent as any).ensureSourcesCreated?.();
153
+
154
+ // 获取 file source
155
+ const sources = configComponent.getSources?.() || [];
156
+ fileSource = sources.find((s: any) => s.name === "file");
157
+
158
+ // 加载 file 配置到 memory
159
+ if (fileSource) {
160
+ for (const key of llmKeys) {
161
+ const value = fileSource.read?.(key);
162
+ if (value !== undefined) {
163
+ await configComponent.set(key, value);
164
+ }
165
+ }
166
+ }
167
+ }
168
+
169
+ // 2. 注册并加载 EnvSource
170
+ configComponent.registerSource({
171
+ type: "env",
172
+ envPrefix: prefix,
173
+ priority: 20,
174
+ watch: false
175
+ });
176
+
177
+ // 确保 source 已创建
178
+ (configComponent as any).ensureSourcesCreated?.();
179
+
180
+ // 获取 env source
181
+ const sourcesAfterEnv = configComponent.getSources?.() || [];
182
+ const envSource = sourcesAfterEnv.find((s: any) => s.name === "env");
183
+
184
+ // 加载 env 配置到 memory(覆盖 file)
185
+ if (envSource) {
186
+ for (const key of llmKeys) {
187
+ const value = envSource.read?.(key);
188
+ if (value !== undefined) {
189
+ await configComponent.set(key, value);
190
+ }
191
+ }
192
+ }
193
+
194
+ // 后备方案:直接读取环境变量
195
+ // 1. 先检查预定义的 llmKeys
196
+ for (const key of llmKeys) {
197
+ const envKey = toEnvKey(key, prefix);
198
+ const value = process.env[envKey];
199
+ if (value !== undefined) {
200
+ await configComponent.set(key, value);
201
+ }
202
+ }
203
+
204
+ // 2. 自动发现并加载所有以 prefix 开头的 llm.* 环境变量
205
+ const loadedKeys = new Set(llmKeys); // 记录已加载的 keys,避免重复
206
+ for (const envKey of Object.keys(process.env)) {
207
+ const configKey = envKeyToConfigKey(envKey, prefix, "llm");
208
+ if (!configKey) continue;
209
+
210
+ // 避免重复加载
211
+ if (loadedKeys.has(configKey)) continue;
212
+ loadedKeys.add(configKey);
213
+
214
+ const value = process.env[envKey];
215
+ if (value !== undefined) {
216
+ await configComponent.set(configKey, value);
217
+ }
218
+ }
219
+
220
+ // 3. 如果有直接配置对象,覆盖到 MemorySource(优先级最高)
221
+ if (config) {
222
+ const flatConfig = this.flattenConfig(config);
223
+ for (const [key, value] of Object.entries(flatConfig)) {
224
+ await configComponent.set(key, value);
225
+ }
226
+ }
227
+
228
+ // 4. 注册配置热更新监听
229
+ this.registerConfigWatcher(configComponent);
230
+ }
231
+
232
+ /**
233
+ * 将配置对象展平为点号路径
234
+ *
235
+ * @example
236
+ * flattenConfig({ providers: { openai: { apiKey: "xxx" } } })
237
+ * -> { "llm.providers.openai.apiKey": "xxx" }
238
+ */
239
+ private flattenConfig(
240
+ obj: Record<string, unknown>,
241
+ prefix = "llm"
242
+ ): Record<string, unknown> {
243
+ const result: Record<string, unknown> = {};
244
+ for (const [key, value] of Object.entries(obj)) {
245
+ const fullKey = `${prefix}.${key}`;
246
+ if (value && typeof value === "object" && !Array.isArray(value)) {
247
+ Object.assign(
248
+ result,
249
+ this.flattenConfig(value as Record<string, unknown>, fullKey)
250
+ );
251
+ } else {
252
+ result[fullKey] = value;
253
+ }
254
+ }
255
+ return result;
256
+ }
257
+
258
+ /**
259
+ * 注册配置热更新监听
260
+ */
261
+ private registerConfigWatcher(configComponent: ConfigComponent): void {
262
+ // 检查 watch 方法是否存在
263
+ if (typeof configComponent.watch !== "function") {
264
+ logger.debug("ConfigComponent does not support watch, hot reload disabled");
265
+ return;
266
+ }
267
+
268
+ // 注册热更新监听
269
+ this.configWatcher = configComponent.watch("llm.*", (event) => {
270
+ this.onConfigChange(event);
271
+ });
272
+
273
+ logger.debug("Config hot reload watcher registered for llm.*");
274
+ }
275
+
276
+ /**
277
+ * 处理配置变更(支持热更新)
278
+ */
279
+ protected onConfigChange(event: { key: string; oldValue?: unknown; newValue?: unknown }): void {
280
+ logger.info(`LLM config changed: ${event.key}`);
281
+
282
+ // 根据变更的 key 类型进行处理
283
+ if (event.key === "llm.providers" || event.key.startsWith("llm.providers.")) {
284
+ // Provider 配置变更,记录日志
285
+ // 注意:getProvider/getLLMConfig 会实时从 configComponent 读取,所以不需要显式重新加载
286
+ logger.info(`LLM provider config changed, will use new config on next call`);
287
+ } else if (event.key === "llm.defaultModel") {
288
+ logger.info(`LLM default model changed to: ${event.newValue}`);
289
+ } else if (event.key === "llm.defaultProvider") {
290
+ logger.info(`LLM default provider changed to: ${event.newValue}`);
291
+ } else {
292
+ logger.debug(`LLM config updated: ${event.key}`);
293
+ }
294
+ }
295
+
296
+ /**
297
+ * 启动组件
298
+ */
299
+ async onStart(): Promise<void> {
300
+ if (this._status === "stopped") {
301
+ this.setStatus("running");
302
+ }
303
+ }
304
+
305
+ /**
306
+ * 停止组件
307
+ */
308
+ async onStop(): Promise<void> {
309
+ // 清理配置热更新 watcher
310
+ this.configWatcher?.();
311
+ this.configWatcher = undefined;
312
+
313
+ this.setStatus("stopped");
314
+ }
315
+
316
+ /**
317
+ * 从配置源获取配置值
318
+ * 优先从 ConfigComponent 获取(支持热更新),否则返回默认值
319
+ */
320
+ private getLLMConfig<T>(key: string, defaultValue?: T): T {
321
+ // 优先从 ConfigComponent 获取(实时)
322
+ if (this.configComponent) {
323
+ const value = this.configComponent.get(key);
324
+ if (value !== undefined) {
325
+ return value as T;
326
+ }
327
+ }
328
+
329
+ return defaultValue!;
330
+ }
331
+
332
+ /**
333
+ * 解析请求,构建调用上下文(invoke 和 invokeStream 共享)
334
+ */
335
+ private resolveRequest(request: LLMInvokeRequest): InvokeContext {
336
+ // 实时获取配置
337
+ const defaultModel = this.getLLMConfig<string>("llm.defaultModel", "gpt-4o");
338
+ const defaultProvider = this.getLLMConfig<string>("llm.defaultProvider", "openai");
339
+ const defaultTemperature = this.getLLMConfig<number>("llm.temperature", 0.7);
340
+ const defaultMaxTokens = this.getLLMConfig<number>("llm.maxTokens", 4096);
341
+ const defaultTopP = this.getLLMConfig<number>("llm.topP", 1.0);
342
+
343
+ // 确定模型
344
+ const model = request.model || defaultModel;
345
+ const providerId = request.providerId || defaultProvider;
346
+
347
+ // 获取 API Key
348
+ const apiKey = this.getApiKey(providerId);
349
+ if (!apiKey) {
350
+ throw new Error(`API key not found for provider ${providerId}. Set providers.${providerId}.apiKey in config, or use \${env://...} protocol in config file.`);
351
+ }
352
+
353
+ // 构建消息
354
+ const messages: LLMMessage[] = request.messages.map((msg) => ({
355
+ role: msg.role,
356
+ content: msg.content,
357
+ name: msg.name,
358
+ toolCallId: (msg as any).toolCallId, // 保留 toolCallId 字段
359
+ toolCalls: (msg as any).toolCalls, // 保留 toolCalls 字段
360
+ }));
361
+
362
+ return {
363
+ providerId,
364
+ model,
365
+ invokeConfig: {
366
+ model: `${providerId}/${model}`,
367
+ apiKey,
368
+ baseURL: this.getProvider(providerId)?.baseURL,
369
+ },
370
+ messages,
371
+ temperature: request.temperature ?? defaultTemperature,
372
+ maxTokens: request.maxTokens ?? defaultMaxTokens,
373
+ topP: request.topP ?? defaultTopP,
374
+ abortSignal: request.abortSignal,
375
+ };
376
+ }
377
+
378
+ /**
379
+ * 调用 LLM
380
+ *
381
+ * 流式事件通过 env.pushEnvEvent() 发布
382
+ */
383
+ @TracedAs("llm.component.invoke", { recordParams: true, recordResult: true, log: true })
384
+ async invoke(request: LLMInvokeRequest): Promise<LLMInvokeResult> {
385
+ const startTime = Date.now();
386
+
387
+ if (this._status !== "running") {
388
+ throw new Error("LLMComponent is not running");
389
+ }
390
+
391
+ const ctx = this.resolveRequest(request);
392
+
393
+ let output: { content: string; reasoning?: string; finishReason: "length" | "stop" | "content-filter" | "tool-calls" | "function-call"; toolCalls?: any[]; usage?: { promptTokens: number; completionTokens: number; totalTokens: number } } = {
394
+ content: "",
395
+ reasoning: undefined,
396
+ finishReason: "stop",
397
+ };
398
+ let finishUsage: { promptTokens: number; completionTokens: number; totalTokens: number } | undefined;
399
+
400
+ const result = await invoke(
401
+ ctx.invokeConfig,
402
+ {
403
+ messages: ctx.messages,
404
+ model: ctx.invokeConfig.model,
405
+ temperature: ctx.temperature,
406
+ maxTokens: ctx.maxTokens,
407
+ tools: request.tools,
408
+ env: this.env,
409
+ context: {
410
+ sessionId: request.context?.sessionId,
411
+ messageId: request.context?.messageId,
412
+ },
413
+ },
414
+ { abort: ctx.abortSignal }
415
+ );
416
+
417
+ if (result.isError) {
418
+ throw new Error(result.result);
419
+ }
420
+
421
+ // 解析 ToolResult
422
+ try {
423
+ const parsed = JSON.parse(result.result);
424
+ output.content = parsed.content || "";
425
+ output.reasoning = parsed.reasoning || undefined; // 提取 reasoning 内容
426
+ output.finishReason = parsed.tool_calls?.length ? "tool-calls" : "stop";
427
+ output.toolCalls = parsed.tool_calls || [];
428
+ if (parsed.usage) {
429
+ finishUsage = {
430
+ promptTokens: parsed.usage.promptTokens ?? parsed.usage.inputTokens ?? 0,
431
+ completionTokens: parsed.usage.completionTokens ?? parsed.usage.outputTokens ?? 0,
432
+ totalTokens: parsed.usage.totalTokens ?? 0,
433
+ };
434
+ }
435
+ output.usage = finishUsage;
436
+ } catch {
437
+ output.content = result.result;
438
+ output.finishReason = "stop";
439
+ }
440
+
441
+ // 计算耗时
442
+ const latencyMs = Date.now() - startTime;
443
+
444
+ // 检测上下文阈值(除非 skipThresholdCheck 为 true)
445
+ if (!request.skipThresholdCheck) {
446
+ const sessionId = request.context?.sessionId;
447
+ const contextError = this.checkContextThreshold(
448
+ finishUsage,
449
+ ctx.providerId,
450
+ ctx.model,
451
+ sessionId
452
+ );
453
+
454
+ if (contextError) {
455
+ throw contextError;
456
+ }
457
+ }
458
+
459
+ return {
460
+ output,
461
+ providerId: ctx.providerId,
462
+ model: ctx.model,
463
+ latencyMs,
464
+ };
465
+ }
466
+
467
+ /**
468
+ * 获取 Provider API Key
469
+ */
470
+ private getApiKey(providerId: string): string | undefined {
471
+ const providers = this.getLLMConfig<LLMConfig["providers"]>("llm.providers", {});
472
+ return providers?.[providerId]?.apiKey;
473
+ }
474
+
475
+ /**
476
+ * 获取 Provider
477
+ */
478
+ getProvider(id: string) {
479
+ const providers = this.getLLMConfig<LLMConfig["providers"]>("llm.providers", {});
480
+ return providers?.[id];
481
+ }
482
+
483
+ /**
484
+ * 列出所有 Provider
485
+ */
486
+ listProviders() {
487
+ const providers = this.getLLMConfig<LLMConfig["providers"]>("llm.providers", {});
488
+ return Object.keys(providers || {});
489
+ }
490
+
491
+ /**
492
+ * 获取模型的限制配置
493
+ *
494
+ * @param providerId - Provider ID
495
+ * @param modelId - 模型 ID(可选,默认使用 provider 的 defaultModel)
496
+ * @returns 模型限制配置,如果没有配置则返回 undefined
497
+ */
498
+ getModelLimits(providerId: string, modelId?: string): ModelLimits | undefined {
499
+ const provider = this.getProvider(providerId);
500
+ if (!provider?.limits) return undefined;
501
+
502
+ // 如果没有指定 modelId,尝试获取 provider 的默认模型
503
+ const targetModel = modelId || provider.defaultModel;
504
+ if (!targetModel) return undefined;
505
+
506
+ return provider.limits[targetModel];
507
+ }
508
+
509
+ /**
510
+ * 获取上下文窗口阈值配置
511
+ *
512
+ * @param providerId - Provider ID
513
+ * @param modelId - 模型 ID
514
+ * @returns 包含 contextWindow 和 thresholdRatio 的对象
515
+ */
516
+ getContextThresholdConfig(providerId: string, modelId?: string): { contextWindow: number; thresholdRatio: number } {
517
+ // 默认值
518
+ const defaults = {
519
+ contextWindow: 128000, // 默认 128K 上下文窗口
520
+ thresholdRatio: 0.9, // 默认 90% 时触发
521
+ };
522
+
523
+ const limits = this.getModelLimits(providerId, modelId);
524
+ if (!limits) return defaults;
525
+
526
+ return {
527
+ contextWindow: limits.contextWindow || defaults.contextWindow,
528
+ thresholdRatio: limits.compactionThreshold ?? defaults.thresholdRatio,
529
+ };
530
+ }
531
+
532
+ /**
533
+ * 检测上下文阈值是否超过
534
+ *
535
+ * @param usage - LLM 使用量信息
536
+ * @param providerId - Provider ID
537
+ * @param modelId - 模型 ID
538
+ * @param sessionId - Session ID(用于错误信息)
539
+ * @returns 如果超过阈值,返回 ContextError;否则返回 undefined
540
+ */
541
+ checkContextThreshold(
542
+ usage: { totalTokens: number; promptTokens?: number; completionTokens?: number } | undefined,
543
+ providerId: string,
544
+ modelId: string | undefined,
545
+ sessionId?: string
546
+ ): ContextError | undefined {
547
+ if (!usage) return undefined;
548
+
549
+ const config = this.getContextThresholdConfig(providerId, modelId);
550
+ const threshold = config.contextWindow * config.thresholdRatio;
551
+
552
+ if (usage.totalTokens >= threshold) {
553
+ return new ContextError(
554
+ `Context threshold exceeded: ${usage.totalTokens}/${config.contextWindow} (${(usage.totalTokens / config.contextWindow * 100).toFixed(1)}%)`,
555
+ ErrorCodes.CONTEXT_THRESHOLD_EXCEEDED,
556
+ sessionId,
557
+ {
558
+ promptTokens: usage.promptTokens ?? 0,
559
+ completionTokens: usage.completionTokens ?? 0,
560
+ totalTokens: usage.totalTokens,
561
+ },
562
+ config.contextWindow
563
+ );
564
+ }
565
+
566
+ return undefined;
567
+ }
568
+ }