@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,604 @@
1
+ /**
2
+ * @fileoverview CommandsComponent
3
+ *
4
+ * 命令管理核心组件(收藏式)
5
+ *
6
+ * 功能:
7
+ * - 管理 user/project 多路径命令目录
8
+ * - 提供命令发现、添加、移除、执行能力
9
+ * - 支持 File/Env/Object 三种配置来源
10
+ */
11
+
12
+ import path from "path";
13
+ import os from "os";
14
+ import { BaseComponent, type ComponentConfig } from "../component";
15
+ import { createLogger } from "../log-trace/logger";
16
+ import * as fs from "fs/promises";
17
+ import { exec } from "child_process";
18
+ import { promisify } from "util";
19
+
20
+ import type {
21
+ CommandsComponentOptions,
22
+ CommandsComponentConfig,
23
+ CommandIndex,
24
+ CommandMeta,
25
+ CommandSource,
26
+ ScanResult,
27
+ DiscoverOptions,
28
+ AddCommandOptions,
29
+ RemoveCommandOptions,
30
+ ExecutionResult,
31
+ } from "./types";
32
+ import { parseMetaFile, scanMetaFiles } from "./parser";
33
+ import type { ConfigComponent } from "../../config/config-component";
34
+ import { toEnvKey, envKeyToConfigKey } from "../../config/env-key";
35
+ import { COMMANDS_CONFIG_REGISTRATION, COMMANDS_DEFAULTS } from "./commands-config-registration";
36
+
37
+ const execAsync = promisify(exec);
38
+ const logger = createLogger("commands");
39
+
40
+ // 默认配置
41
+ const DEFAULT_USER_DIR = ".roy-agent/commands";
42
+ const DEFAULT_PROJECT_DIR = ".roy/commands";
43
+ const DEFAULT_CACHE_DIR = ".cache/roy-agent/commands";
44
+ const DESCRIPTIONS_FILE = "descriptions.json";
45
+
46
+ // 配置键名列表
47
+ const CONFIG_KEYS = [
48
+ "command.userCommandsDir",
49
+ "command.projectCommandsDir",
50
+ "command.cacheDir",
51
+ ];
52
+
53
+ export class CommandsComponent extends BaseComponent {
54
+ readonly name = "commands";
55
+ readonly version = "2.0.0";
56
+
57
+ private config?: CommandsComponentConfig;
58
+ private configComponent?: ConfigComponent;
59
+ private userCommandsDir: string = "";
60
+ private projectCommandsDir: string = "";
61
+ private descriptionCache: Map<string, string> = new Map();
62
+ private descriptionCacheFile: string = "";
63
+
64
+ constructor() {
65
+ super();
66
+ this.initDefaultDirs();
67
+ }
68
+
69
+ private initDefaultDirs(): void {
70
+ const homeDir = os.homedir();
71
+ this.userCommandsDir = path.join(homeDir, DEFAULT_USER_DIR);
72
+ this.projectCommandsDir = DEFAULT_PROJECT_DIR;
73
+ this.descriptionCacheFile = path.join(homeDir, DEFAULT_CACHE_DIR, DESCRIPTIONS_FILE);
74
+ }
75
+
76
+ // ========================================================================
77
+ // Component Lifecycle
78
+ // ========================================================================
79
+
80
+ /**
81
+ * 初始化组件
82
+ *
83
+ * 配置加载顺序(优先级从低到高):
84
+ * 1. File - 配置文件
85
+ * 2. Env - 环境变量
86
+ * 3. Object - 直接传入的 config 对象
87
+ */
88
+ async init(config?: ComponentConfig): Promise<void> {
89
+ await super.init(config);
90
+
91
+ const options = config?.options as unknown as CommandsComponentOptions | undefined;
92
+ if (!options?.configComponent) {
93
+ throw new Error("ConfigComponent is required for CommandsComponent initialization");
94
+ }
95
+
96
+ this.configComponent = options.configComponent;
97
+ await this.registerConfig(options);
98
+
99
+ // 确保 user 目录存在
100
+ await fs.mkdir(this.userCommandsDir, { recursive: true });
101
+
102
+ // 加载描述缓存
103
+ await this.loadDescriptionCache();
104
+
105
+ logger.info(`CommandsComponent initialized: user=${this.userCommandsDir}, project=${this.projectCommandsDir}`);
106
+ }
107
+
108
+ /**
109
+ * 注册配置到 ConfigComponent
110
+ *
111
+ * 使用 SessionComponent 的正确模式:
112
+ * 1. 使用 registerComponent() 注册配置(包含 sources、keys、defaults)
113
+ * 2. 使用 load("command") 加载配置值
114
+ * 3. 处理环境变量后备方案
115
+ * 4. 处理 config 对象(最高优先级)
116
+ */
117
+ private async registerConfig(options: CommandsComponentOptions): Promise<void> {
118
+ const configComponent = this.configComponent;
119
+ if (!configComponent) return;
120
+
121
+ const { configPath, envPrefix, config } = options;
122
+
123
+ // 1. 注册 COMMANDS_CONFIG_REGISTRATION(包含 defaults、sources、keys)
124
+ configComponent.registerComponent(COMMANDS_CONFIG_REGISTRATION);
125
+
126
+ // 2. 如果提供了 configPath,添加额外的 file source
127
+ if (configPath) {
128
+ configComponent.registerSource({
129
+ type: "file",
130
+ relativePath: configPath,
131
+ optional: true,
132
+ watch: false,
133
+ });
134
+ }
135
+
136
+ // 3. 使用 load 加载配置(会使用 defaults 填充,然后被环境变量覆盖)
137
+ await configComponent.load("command");
138
+
139
+ // 4. 环境变量后备方案(确保环境变量能够覆盖配置)
140
+ const prefix = envPrefix !== undefined ? envPrefix : "COMMAND";
141
+ for (const key of CONFIG_KEYS) {
142
+ const envKey = toEnvKey(key, prefix);
143
+ const value = process.env[envKey];
144
+ if (value !== undefined) {
145
+ await configComponent.set(key, value);
146
+ }
147
+ }
148
+
149
+ // 5. 自动发现其他相关环境变量
150
+ const loadedKeys = new Set(CONFIG_KEYS);
151
+ for (const envKey of Object.keys(process.env)) {
152
+ const configKey = envKeyToConfigKey(envKey, prefix, "command");
153
+ if (!configKey) continue;
154
+ if (loadedKeys.has(configKey)) continue;
155
+ loadedKeys.add(configKey);
156
+ const value = process.env[envKey];
157
+ if (value !== undefined) {
158
+ await configComponent.set(configKey, value);
159
+ }
160
+ }
161
+
162
+ // 6. 直接配置对象(最高优先级)
163
+ if (config) {
164
+ const flatConfig = this.flattenConfig(config);
165
+ for (const [key, value] of Object.entries(flatConfig)) {
166
+ if (value !== undefined) {
167
+ await configComponent.set(key, value);
168
+ }
169
+ }
170
+ }
171
+
172
+ // 7. 设置默认值(只有当配置不存在时)
173
+ // 注意:commands 使用基于 HOME 的绝对路径作为默认值
174
+ const homeDir = os.homedir();
175
+ const defaultDirs = {
176
+ "command.userCommandsDir": path.join(homeDir, DEFAULT_USER_DIR),
177
+ "command.projectCommandsDir": DEFAULT_PROJECT_DIR,
178
+ "command.cacheDir": path.join(homeDir, DEFAULT_CACHE_DIR),
179
+ };
180
+
181
+ for (const [key, value] of Object.entries(defaultDirs)) {
182
+ if (configComponent.get(key) === undefined) {
183
+ await configComponent.set(key, value);
184
+ }
185
+ }
186
+
187
+ // 应用最终配置
188
+ this.applyConfig(configComponent);
189
+ }
190
+
191
+ /**
192
+ * 将嵌套配置扁平化
193
+ */
194
+ private flattenConfig(config: CommandsComponentConfig): Record<string, string> {
195
+ const result: Record<string, string> = {};
196
+
197
+ if (config.userCommandsDir !== undefined) {
198
+ result["command.userCommandsDir"] = config.userCommandsDir;
199
+ }
200
+ if (config.projectCommandsDir !== undefined) {
201
+ result["command.projectCommandsDir"] = config.projectCommandsDir;
202
+ }
203
+ if (config.cacheDir !== undefined) {
204
+ result["command.cacheDir"] = config.cacheDir;
205
+ }
206
+
207
+ return result;
208
+ }
209
+
210
+ /**
211
+ * 从 ConfigComponent 应用配置
212
+ */
213
+ private applyConfig(configComponent: ConfigComponent): void {
214
+ const userDir = configComponent.get("command.userCommandsDir") as string | undefined;
215
+ const projectDir = configComponent.get("command.projectCommandsDir") as string | undefined;
216
+ const cacheDir = configComponent.get("command.cacheDir") as string | undefined;
217
+
218
+ this.userCommandsDir = userDir || path.join(os.homedir(), DEFAULT_USER_DIR);
219
+ this.projectCommandsDir = projectDir || DEFAULT_PROJECT_DIR;
220
+ this.descriptionCacheFile = path.join(
221
+ cacheDir || path.join(os.homedir(), DEFAULT_CACHE_DIR),
222
+ DESCRIPTIONS_FILE
223
+ );
224
+
225
+ this.config = {
226
+ userCommandsDir: this.userCommandsDir,
227
+ projectCommandsDir: this.projectCommandsDir,
228
+ cacheDir: cacheDir,
229
+ };
230
+ }
231
+
232
+ async start(): Promise<void> {
233
+ await super.start();
234
+ logger.debug("CommandsComponent started");
235
+ }
236
+
237
+ async stop(): Promise<void> {
238
+ await this.saveDescriptionCache();
239
+ await super.stop();
240
+ logger.debug("CommandsComponent stopped");
241
+ }
242
+
243
+ /**
244
+ * 获取命令目录列表
245
+ */
246
+ async getCommandDirs(): Promise<{ user: string; project: string }> {
247
+ return {
248
+ user: this.userCommandsDir,
249
+ project: this.projectCommandsDir,
250
+ };
251
+ }
252
+
253
+ /**
254
+ * 发现命令
255
+ */
256
+ async discover(options?: DiscoverOptions): Promise<ScanResult> {
257
+ const commands: CommandIndex[] = [];
258
+ let userCount = 0;
259
+ let projectCount = 0;
260
+
261
+ // 先扫描 project 目录(优先)
262
+ try {
263
+ const projectCommands = await this.scanDir(this.projectCommandsDir, "project", options?.pattern);
264
+ commands.push(...projectCommands);
265
+ projectCount = projectCommands.length;
266
+ } catch (error) {
267
+ logger.debug(`Failed to scan project dir: ${error}`);
268
+ }
269
+
270
+ // 再扫描 user 目录(去重)
271
+ try {
272
+ const userCommands = await this.scanDir(this.userCommandsDir, "user", options?.pattern);
273
+ for (const cmd of userCommands) {
274
+ const exists = commands.find((c) => c.name === cmd.name);
275
+ if (!exists) {
276
+ commands.push(cmd);
277
+ userCount++;
278
+ }
279
+ }
280
+ } catch (error) {
281
+ logger.debug(`Failed to scan user dir: ${error}`);
282
+ }
283
+
284
+ commands.sort((a, b) => a.name.localeCompare(b.name));
285
+
286
+ return {
287
+ commands,
288
+ stats: { userCount, projectCount },
289
+ };
290
+ }
291
+
292
+ /**
293
+ * 添加命令
294
+ */
295
+ async add(options: AddCommandOptions): Promise<void> {
296
+ const { name, target, source = "user", description, tips } = options;
297
+ const dir = source === "user" ? this.userCommandsDir : this.projectCommandsDir;
298
+ const targetPath = path.resolve(target);
299
+
300
+ await fs.mkdir(dir, { recursive: true });
301
+
302
+ const commandPath = path.join(dir, name);
303
+
304
+ try {
305
+ await fs.access(commandPath);
306
+ throw new Error(`Command '${name}' already exists`);
307
+ } catch (error: any) {
308
+ if (error.code !== "ENOENT") {
309
+ throw error;
310
+ }
311
+ }
312
+
313
+ // 创建符号链接
314
+ await fs.symlink(targetPath, commandPath);
315
+ logger.info(`Added command '${name}' -> ${targetPath} (${source})`);
316
+
317
+ // 创建 meta.md(用于 Prompt 注入)
318
+ const commandDir = path.join(dir, name);
319
+ await this.createCommandMeta(commandDir, {
320
+ name,
321
+ target: targetPath,
322
+ description: description || `Command: ${name}`,
323
+ tips,
324
+ source,
325
+ });
326
+ }
327
+
328
+ /**
329
+ * 创建命令的 meta.md 文件
330
+ */
331
+ private async createCommandMeta(
332
+ commandDir: string,
333
+ options: {
334
+ name: string;
335
+ target: string;
336
+ description: string;
337
+ tips?: string;
338
+ source: CommandSource;
339
+ }
340
+ ): Promise<void> {
341
+ // 确保命令目录存在(符号链接本身不能作为目录)
342
+ // meta.md 存在命令的父目录中
343
+ const parentDir = path.dirname(commandDir);
344
+ const metaPath = path.join(parentDir, `${options.name}.meta.md`);
345
+
346
+ const tipsContent = options.tips
347
+ ? `\ntips: "${options.tips}"`
348
+ : "";
349
+
350
+ const content = `---
351
+ name: ${options.name}
352
+ description: ${options.description}${tipsContent}
353
+ source: ${options.source}
354
+ ---
355
+
356
+ # ${options.name} Command
357
+
358
+ ## Target
359
+ \`${options.target}\`
360
+
361
+ ## 功能
362
+ ${options.description}
363
+
364
+ ## 探索提示
365
+ Run \`${options.name} --help\` to explore full capabilities.
366
+ `;
367
+
368
+ await fs.writeFile(metaPath, content, "utf-8");
369
+ logger.info(`[CommandsComponent] Created meta.md for '${options.name}'`);
370
+ }
371
+
372
+ /**
373
+ * 获取所有收藏命令的元信息(用于 Prompt 注入)
374
+ */
375
+ async getCommandsMeta(): Promise<CommandMeta[]> {
376
+ const metaList: CommandMeta[] = [];
377
+
378
+ // 扫描 user 目录
379
+ try {
380
+ const entries = await fs.readdir(this.userCommandsDir, { withFileTypes: true });
381
+ for (const entry of entries) {
382
+ if (entry.isFile() && entry.name.endsWith(".meta.md")) {
383
+ const metaPath = path.join(this.userCommandsDir, entry.name);
384
+ const meta = await parseMetaFile(metaPath, "user");
385
+ if (meta) {
386
+ metaList.push(meta);
387
+ }
388
+ }
389
+ }
390
+ } catch (error) {
391
+ // 目录不存在时忽略
392
+ if ((error as NodeJS.ErrnoException).code !== "ENOENT") {
393
+ logger.debug(`Failed to scan user commands dir: ${error}`);
394
+ }
395
+ }
396
+
397
+ // 扫描 project 目录
398
+ try {
399
+ const entries = await fs.readdir(this.projectCommandsDir, { withFileTypes: true });
400
+ for (const entry of entries) {
401
+ if (entry.isFile() && entry.name.endsWith(".meta.md")) {
402
+ const metaPath = path.join(this.projectCommandsDir, entry.name);
403
+ const meta = await parseMetaFile(metaPath, "project");
404
+ if (meta) {
405
+ metaList.push(meta);
406
+ }
407
+ }
408
+ }
409
+ } catch (error) {
410
+ // 目录不存在时忽略
411
+ if ((error as NodeJS.ErrnoException).code !== "ENOENT") {
412
+ logger.debug(`Failed to scan project commands dir: ${error}`);
413
+ }
414
+ }
415
+
416
+ return metaList;
417
+ }
418
+
419
+ /**
420
+ * 移除命令
421
+ */
422
+ async remove(options: RemoveCommandOptions): Promise<void> {
423
+ const { name, source } = options;
424
+
425
+ const dirsToCheck = source
426
+ ? [source === "user" ? this.userCommandsDir : this.projectCommandsDir]
427
+ : [this.userCommandsDir, this.projectCommandsDir];
428
+
429
+ for (const dir of dirsToCheck) {
430
+ const commandPath = path.join(dir, name);
431
+ try {
432
+ await fs.unlink(commandPath);
433
+ logger.info(`Removed command '${name}' from ${dir}`);
434
+ return;
435
+ } catch (error: any) {
436
+ if (error.code !== "ENOENT") {
437
+ throw error;
438
+ }
439
+ }
440
+ }
441
+
442
+ throw new Error(`Command '${name}' not found`);
443
+ }
444
+
445
+ /**
446
+ * 获取命令信息
447
+ */
448
+ async getInfo(name: string): Promise<CommandIndex | null> {
449
+ const dirsToCheck = [this.projectCommandsDir, this.userCommandsDir];
450
+
451
+ for (const dir of dirsToCheck) {
452
+ const commandPath = path.join(dir, name);
453
+ try {
454
+ const stat = await fs.lstat(commandPath);
455
+ const realPath = stat.isSymbolicLink() ? await fs.readlink(commandPath) : commandPath;
456
+ const cmdSource: CommandSource = dir === this.projectCommandsDir ? "project" : "user";
457
+
458
+ const shortDescription = await this.getDescription(realPath);
459
+
460
+ return {
461
+ name,
462
+ path: realPath,
463
+ source: cmdSource,
464
+ shortDescription,
465
+ };
466
+ } catch (error: any) {
467
+ if (error.code !== "ENOENT") {
468
+ throw error;
469
+ }
470
+ }
471
+ }
472
+
473
+ return null;
474
+ }
475
+
476
+ /**
477
+ * 执行命令
478
+ */
479
+ async exec(options: { name: string; args?: string[]; cwd?: string }): Promise<ExecutionResult> {
480
+ const { name, args = [], cwd } = options;
481
+
482
+ const info = await this.getInfo(name);
483
+ if (!info) {
484
+ return {
485
+ stdout: "",
486
+ stderr: `Command '${name}' not found`,
487
+ exitCode: 127,
488
+ success: false,
489
+ };
490
+ }
491
+
492
+ const cmd = `${info.path} ${args.join(" ")}`;
493
+ try {
494
+ const { stdout, stderr } = await execAsync(cmd, {
495
+ cwd: cwd || process.cwd(),
496
+ timeout: 30000,
497
+ });
498
+ return {
499
+ stdout,
500
+ stderr,
501
+ exitCode: 0,
502
+ success: true,
503
+ };
504
+ } catch (error: any) {
505
+ return {
506
+ stdout: error.stdout || "",
507
+ stderr: error.stderr || error.message,
508
+ exitCode: error.code || 1,
509
+ success: false,
510
+ };
511
+ }
512
+ }
513
+
514
+ // ========================================================================
515
+ // Private Methods
516
+ // ========================================================================
517
+
518
+ private async scanDir(dir: string, source: CommandSource, pattern?: string): Promise<CommandIndex[]> {
519
+ const commands: CommandIndex[] = [];
520
+
521
+ try {
522
+ const entries = await fs.readdir(dir);
523
+
524
+ for (const entry of entries) {
525
+ if (entry.startsWith(".")) continue;
526
+ if (pattern && !this.matchPattern(entry, pattern)) continue;
527
+
528
+ const fullPath = path.join(dir, entry);
529
+
530
+ try {
531
+ const stat = await fs.lstat(fullPath);
532
+
533
+ if (!stat.isSymbolicLink() && !stat.isFile()) continue;
534
+
535
+ if (stat.isFile()) {
536
+ try {
537
+ await fs.access(fullPath, fs.constants.X_OK);
538
+ } catch {
539
+ continue;
540
+ }
541
+ }
542
+
543
+ const realPath = stat.isSymbolicLink() ? await fs.readlink(fullPath) : fullPath;
544
+ const shortDescription = await this.getDescription(realPath);
545
+
546
+ commands.push({
547
+ name: entry,
548
+ path: realPath,
549
+ source,
550
+ shortDescription,
551
+ });
552
+ } catch (error) {
553
+ logger.debug(`Failed to stat ${fullPath}: ${error}`);
554
+ }
555
+ }
556
+ } catch (error) {
557
+ logger.debug(`Failed to scan dir ${dir}: ${error}`);
558
+ }
559
+
560
+ return commands;
561
+ }
562
+
563
+ private matchPattern(name: string, pattern: string): boolean {
564
+ const regex = new RegExp("^" + pattern.replace(/\*/g, ".*").replace(/\?/g, ".") + "$");
565
+ return regex.test(name);
566
+ }
567
+
568
+ private async getDescription(commandPath: string): Promise<string> {
569
+ const cached = this.descriptionCache.get(commandPath);
570
+ if (cached !== undefined) return cached;
571
+
572
+ try {
573
+ const { stdout } = await execAsync(`${commandPath} --help 2>/dev/null | head -1`, {
574
+ timeout: 2000,
575
+ });
576
+ const description = stdout.trim().split("\n")[0] || "";
577
+
578
+ this.descriptionCache.set(commandPath, description);
579
+ return description;
580
+ } catch {
581
+ return "";
582
+ }
583
+ }
584
+
585
+ private async loadDescriptionCache(): Promise<void> {
586
+ try {
587
+ const content = await fs.readFile(this.descriptionCacheFile, "utf-8");
588
+ const data = JSON.parse(content);
589
+ this.descriptionCache = new Map(Object.entries(data));
590
+ } catch {
591
+ // 缓存文件不存在,忽略
592
+ }
593
+ }
594
+
595
+ private async saveDescriptionCache(): Promise<void> {
596
+ try {
597
+ await fs.mkdir(path.dirname(this.descriptionCacheFile), { recursive: true });
598
+ const data = Object.fromEntries(this.descriptionCache);
599
+ await fs.writeFile(this.descriptionCacheFile, JSON.stringify(data, null, 2));
600
+ } catch (error) {
601
+ logger.debug(`Failed to save description cache: ${error}`);
602
+ }
603
+ }
604
+ }