@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,906 @@
1
+ import { ZodType, ZodTypeDef } from "zod";
2
+ import {
3
+ BaseComponent,
4
+ type ComponentSchema,
5
+ type ComponentPaths,
6
+ type ComponentWatchCallback,
7
+ type ComponentConfigChangeEvent,
8
+ } from "../env/component";
9
+ import {
10
+ XDG_PATHS,
11
+ type XDGDirType,
12
+ } from "../env/paths";
13
+ import { parseJSONCWithProtocols, parseJSONCWithEnv } from "./config-parser";
14
+ import { ProtocolResolver } from "./protocol-resolver";
15
+ import { FileSource } from "./file-source";
16
+ import { EnvSource } from "./env-source";
17
+ import * as fsSync from "fs";
18
+ import * as fsPromises from "fs/promises";
19
+ import * as os from "os";
20
+ import * as path from "path";
21
+
22
+ /**
23
+ * ConfigSource 类型
24
+ */
25
+ export type ConfigSourceType = "memory" | "file" | "env" | "remote";
26
+
27
+ /**
28
+ * SourceRegistration - Source 注册配置
29
+ */
30
+ export interface SourceRegistration {
31
+ /** Source 类型 */
32
+ type: ConfigSourceType;
33
+ /** 相对路径(file 类型使用,基于 XDG_DATA_HOME) */
34
+ relativePath?: string;
35
+ /** 环境变量前缀(env 类型使用) */
36
+ envPrefix?: string;
37
+ /** 优先级,默认值:memory=0, file=10, env=20 */
38
+ priority?: number;
39
+ /** 是否可选(file 类型使用,文件不存在时不报错) */
40
+ optional?: boolean;
41
+ /** 是否启用文件监听(file 类型使用,默认 true) */
42
+ watch?: boolean;
43
+ }
44
+
45
+ /**
46
+ * KeyRegistration - 配置 Key 注册
47
+ */
48
+ export interface KeyRegistration {
49
+ /** 配置 Key(支持点号嵌套,如 "llm.provider") */
50
+ key: string;
51
+ /** 优先使用的 Source 列表(按优先级从低到高) */
52
+ sources: ConfigSourceType[];
53
+ /** 自定义优先级(覆盖默认优先级) */
54
+ priority?: number;
55
+ /** 默认值 */
56
+ default?: unknown;
57
+ }
58
+
59
+ /**
60
+ * ComponentRegistration - Component 注册配置
61
+ */
62
+ export interface ComponentRegistration {
63
+ /** Component 名称 */
64
+ name: string;
65
+ /** 声明需要的 Source */
66
+ sources: SourceRegistration[];
67
+ /** 声明需要的配置 Key */
68
+ keys: KeyRegistration[];
69
+ /** 配置 Schema(用于验证,可选) */
70
+ schema?: ZodType<unknown, ZodTypeDef, unknown>;
71
+ /** 默认值 */
72
+ defaults?: Record<string, unknown>;
73
+ }
74
+
75
+ /**
76
+ * 配置源不存在错误
77
+ */
78
+ export class ConfigSourceNotFoundError extends Error {
79
+ constructor(sourceType: string, path?: string) {
80
+ const message = path
81
+ ? `Config source not found: ${sourceType} at ${path}`
82
+ : `Config source not found: ${sourceType}`;
83
+ super(message);
84
+ this.name = "ConfigSourceNotFoundError";
85
+ }
86
+ }
87
+
88
+ /**
89
+ * 配置文件不存在错误
90
+ */
91
+ export class ConfigFileNotFoundError extends Error {
92
+ constructor(filePath: string) {
93
+ super(`Config file not found: ${filePath}`);
94
+ this.name = "ConfigFileNotFoundError";
95
+ }
96
+ }
97
+
98
+ /**
99
+ * ConfigSource 接口
100
+ */
101
+ export interface ConfigSource {
102
+ readonly name: ConfigSourceType;
103
+ readonly priority: number;
104
+ read(key: string): unknown | undefined;
105
+ write?(key: string, value: unknown): boolean;
106
+ delete?(key: string): boolean;
107
+ list(): Array<{ key: string; value: unknown }>;
108
+ watch?(callback: (event: ConfigChangeEvent) => void): () => void;
109
+ }
110
+
111
+ /**
112
+ * Config 变更事件
113
+ */
114
+ export interface ConfigChangeEvent {
115
+ type: "change" | "add" | "delete";
116
+ key: string;
117
+ oldValue: unknown;
118
+ newValue: unknown;
119
+ source: ConfigSourceType;
120
+ timestamp: number;
121
+ }
122
+
123
+ /**
124
+ * 异步持久化条目
125
+ */
126
+ interface PersistEntry {
127
+ key: string;
128
+ value: unknown;
129
+ timestamp: number;
130
+ }
131
+
132
+ /**
133
+ * MemorySource 实现
134
+ */
135
+ export class MemorySource implements ConfigSource {
136
+ readonly name: ConfigSourceType = "memory";
137
+ readonly priority = 0;
138
+
139
+ private store = new Map<string, { value: unknown; timestamp: number; version: number }>();
140
+ private watchers = new Set<(event: ConfigChangeEvent) => void>();
141
+
142
+ read(key: string): unknown | undefined {
143
+ return this.store.get(key)?.value;
144
+ }
145
+
146
+ write(key: string, value: unknown): boolean {
147
+ const oldEntry = this.store.get(key);
148
+ const entry = {
149
+ value,
150
+ timestamp: Date.now(),
151
+ version: (oldEntry?.version ?? 0) + 1,
152
+ };
153
+ this.store.set(key, entry);
154
+
155
+ this.notifyWatchers({
156
+ type: oldEntry ? "change" : "add",
157
+ key,
158
+ oldValue: oldEntry?.value,
159
+ newValue: value,
160
+ source: "memory",
161
+ timestamp: entry.timestamp,
162
+ });
163
+
164
+ return true;
165
+ }
166
+
167
+ delete(key: string): boolean {
168
+ const entry = this.store.get(key);
169
+ if (!entry) return false;
170
+
171
+ this.store.delete(key);
172
+
173
+ this.notifyWatchers({
174
+ type: "delete",
175
+ key,
176
+ oldValue: entry.value,
177
+ newValue: undefined,
178
+ source: "memory",
179
+ timestamp: Date.now(),
180
+ });
181
+
182
+ return true;
183
+ }
184
+
185
+ list(): Array<{ key: string; value: unknown }> {
186
+ return Array.from(this.store.entries()).map(([key, entry]) => ({
187
+ key,
188
+ value: entry.value,
189
+ }));
190
+ }
191
+
192
+ watch(callback: (event: ConfigChangeEvent) => void): () => void {
193
+ this.watchers.add(callback);
194
+ return () => this.watchers.delete(callback);
195
+ }
196
+
197
+ private notifyWatchers(event: ConfigChangeEvent): void {
198
+ this.watchers.forEach((cb) => cb(event));
199
+ }
200
+ }
201
+
202
+ /**
203
+ * 异步持久化队列
204
+ */
205
+ export class AsyncPersistQueue {
206
+ private queue: Map<string, PersistEntry> = new Map();
207
+ private timer?: NodeJS.Timeout;
208
+ private paused = false;
209
+
210
+ constructor(
211
+ private options: {
212
+ debounceMs: number;
213
+ persistFn: (entries: PersistEntry[]) => Promise<void>;
214
+ }
215
+ ) {}
216
+
217
+ add(key: string, value: unknown): void {
218
+ this.queue.set(key, { key, value, timestamp: Date.now() });
219
+ this.scheduleFlush();
220
+ }
221
+
222
+ private scheduleFlush(): void {
223
+ if (this.paused || this.timer) return;
224
+
225
+ this.timer = setTimeout(async () => {
226
+ this.timer = undefined;
227
+ await this.flush();
228
+ }, this.options.debounceMs);
229
+ }
230
+
231
+ async flush(): Promise<void> {
232
+ if (this.queue.size === 0) return;
233
+
234
+ const entries = Array.from(this.queue.values());
235
+ this.queue.clear();
236
+
237
+ try {
238
+ await this.options.persistFn(entries);
239
+ } catch (error) {
240
+ console.error("Failed to persist config:", error);
241
+ }
242
+ }
243
+
244
+ clear(): void {
245
+ this.queue.clear();
246
+ if (this.timer) {
247
+ clearTimeout(this.timer);
248
+ this.timer = undefined;
249
+ }
250
+ }
251
+
252
+ size(): number {
253
+ return this.queue.size;
254
+ }
255
+
256
+ pause(): void {
257
+ this.paused = true;
258
+ }
259
+
260
+ resume(): void {
261
+ this.paused = false;
262
+ if (this.queue.size > 0) {
263
+ this.scheduleFlush();
264
+ }
265
+ }
266
+ }
267
+
268
+ /**
269
+ * ConfigComponent 配置
270
+ */
271
+ export interface ConfigComponentOptions {
272
+ /** 持久化文件路径 */
273
+ persistFile?: string;
274
+ /** 防抖延迟(ms) */
275
+ debounceMs?: number;
276
+ /** 最大批量大小 */
277
+ maxBatchSize?: number;
278
+ }
279
+
280
+ /**
281
+ * ConfigComponent
282
+ */
283
+ export class ConfigComponent extends BaseComponent {
284
+ readonly name = "config";
285
+ readonly version = "1.0.0";
286
+
287
+ private memorySource = new MemorySource();
288
+ private sources: ConfigSource[] = [this.memorySource];
289
+ private componentSchemas = new Map<string, ComponentSchema>();
290
+ private componentPaths = new Map<string, Record<string, ComponentPaths>>();
291
+ private watchers = new Map<string, Set<ComponentWatchCallback>>();
292
+ /** Source watch unsubscribe 函数 */
293
+ private sourceUnwatchFns = new Map<string, () => void>();
294
+
295
+ private persistQueue: AsyncPersistQueue;
296
+ private persistFile?: string;
297
+
298
+ // === 去中心化配置注册 ===
299
+
300
+ /** Key 注册表:key -> KeyRegistration */
301
+ private keyRegistry = new Map<string, KeyRegistration>();
302
+
303
+ /** Component 注册表:name -> ComponentRegistration */
304
+ private componentRegistrations = new Map<string, ComponentRegistration>();
305
+
306
+ /** 协议解析器 */
307
+ private protocolResolver?: ProtocolResolver;
308
+
309
+ /** XDG_DATA_HOME 缓存(用于测试注入) */
310
+ private __xdgDataHome?: string;
311
+
312
+ /** Source 注册配置(延迟创建) */
313
+ private pendingSources: SourceRegistration[] = [];
314
+
315
+ /** Source 唯一标识缓存:type:relativePath -> true */
316
+ private sourceKeys = new Map<string, boolean>();
317
+
318
+ /**
319
+ * 获取 XDG_DATA_HOME(动态获取,支持运行时覆盖)
320
+ */
321
+ getXdgDataHome(): string {
322
+ return this.__xdgDataHome ?? process.env.XDG_DATA_HOME ??
323
+ path.join(os.homedir(), ".local", "share");
324
+ }
325
+
326
+ /**
327
+ * 设置 XDG_DATA_HOME(用于测试)
328
+ */
329
+ setXdgDataHome(path: string): void {
330
+ this.__xdgDataHome = path;
331
+ // 同时更新 ProtocolResolver 的 xdgDataHome
332
+ if (this.protocolResolver) {
333
+ this.protocolResolver.xdgDataHome = path;
334
+ }
335
+ }
336
+
337
+ constructor(options: ConfigComponentOptions = {}) {
338
+ super();
339
+ this.persistFile = options.persistFile;
340
+ this.persistQueue = new AsyncPersistQueue({
341
+ debounceMs: options.debounceMs ?? 300,
342
+ persistFn: this.persistToFile.bind(this),
343
+ });
344
+
345
+ // 初始化协议解析器(延迟初始化,依赖 getXdgDataHome)
346
+ this.protocolResolver = new ProtocolResolver();
347
+ }
348
+
349
+ // === Component 注册 ===
350
+
351
+ /**
352
+ * 注册 Component Schema
353
+ */
354
+ register(schema: ComponentSchema): void {
355
+ this.componentSchemas.set(schema.name, schema);
356
+
357
+ // 注册路径
358
+ if (schema.paths) {
359
+ this.componentPaths.set(schema.name, schema.paths);
360
+ }
361
+
362
+ // 初始化默认值到内存
363
+ for (const [key, value] of Object.entries(schema.defaults)) {
364
+ const fullKey = `${schema.name}.${key}`;
365
+ if (this.memorySource.read(fullKey) === undefined) {
366
+ this.memorySource.write(fullKey, value);
367
+ }
368
+ }
369
+ }
370
+
371
+ /**
372
+ * 注销 Component
373
+ */
374
+ unregister(name: string): boolean {
375
+ this.componentSchemas.delete(name);
376
+ this.componentPaths.delete(name);
377
+ return true;
378
+ }
379
+
380
+ /**
381
+ * 检查是否已注册
382
+ */
383
+ isRegistered(name: string): boolean {
384
+ return this.componentSchemas.has(name);
385
+ }
386
+
387
+ /**
388
+ * 列出所有注册的 Component
389
+ */
390
+ listComponents(): string[] {
391
+ return Array.from(this.componentSchemas.keys());
392
+ }
393
+
394
+ // === 层级 Key 访问 ===
395
+
396
+ /**
397
+ * 获取配置值(支持 xxx.yyy.zzz 嵌套访问)
398
+ * 支持两种模式:
399
+ * 1. 完整 key:直接查找,如 "component.key" -> 读取 "component.key"
400
+ * 2. 嵌套对象:查找 component 对象后从中获取属性,如 "component.nested.deep"
401
+ * -> 先找 "component.nested",再获取其 "deep" 属性
402
+ */
403
+ get(key: string): unknown {
404
+ // 先检查内存层
405
+ const memoryValue = this.memorySource.read(key);
406
+ if (memoryValue !== undefined) {
407
+ return memoryValue;
408
+ }
409
+
410
+ // 尝试解析嵌套 key(从短到长尝试)
411
+ if (key.includes(".")) {
412
+ const parts = key.split(".");
413
+
414
+ // 从第二部分开始尝试,逐级向上查找父对象
415
+ for (let i = parts.length - 1; i >= 1; i--) {
416
+ const componentParts = parts.slice(0, i);
417
+ const component = componentParts.join(".");
418
+ const restParts = parts.slice(i);
419
+ const restKey = restParts.join(".");
420
+
421
+ const componentValue = this.memorySource.read(component);
422
+ if (typeof componentValue === "object" && componentValue !== null) {
423
+ const nestedValue = this.getNestedValue(componentValue, restKey);
424
+ if (nestedValue !== undefined) {
425
+ return nestedValue;
426
+ }
427
+ }
428
+ }
429
+ }
430
+
431
+ return undefined;
432
+ }
433
+
434
+ /**
435
+ * 批量获取
436
+ */
437
+ getMany(keys: string[]): Record<string, unknown> {
438
+ const result: Record<string, unknown> = {};
439
+ for (const key of keys) {
440
+ result[key] = this.get(key);
441
+ }
442
+ return result;
443
+ }
444
+
445
+ /**
446
+ * 设置配置值(内存级别 + 异步持久化)
447
+ */
448
+ async set(key: string, value: unknown): Promise<void> {
449
+ const oldValue = this.get(key);
450
+
451
+ // 写入内存(立即生效)
452
+ this.memorySource.write(key, value);
453
+
454
+ // 触发监听器
455
+ const event: ComponentConfigChangeEvent = {
456
+ component: key.split(".")[0],
457
+ key,
458
+ oldValue,
459
+ newValue: value,
460
+ persisted: false,
461
+ timestamp: Date.now(),
462
+ };
463
+ this.notifyWatchers(key, event);
464
+
465
+ // 队列化持久化(异步)
466
+ this.persistQueue.add(key, value);
467
+ }
468
+
469
+ /**
470
+ * 批量设置
471
+ */
472
+ async setMany(config: Record<string, unknown>): Promise<void> {
473
+ for (const [key, value] of Object.entries(config)) {
474
+ await this.set(key, value);
475
+ }
476
+ }
477
+
478
+ // === 路径 ===
479
+
480
+ /**
481
+ * 获取 Component 的路径
482
+ */
483
+ getPath(component: string, subPath?: string): string {
484
+ const paths = this.componentPaths.get(component);
485
+
486
+ if (!paths) {
487
+ throw new Error(`Component not registered: ${component}`);
488
+ }
489
+
490
+ if (subPath && paths[subPath]) {
491
+ const { base, subPath: relativePath } = paths[subPath];
492
+ return XDG_PATHS.getComponentPath(base, relativePath);
493
+ }
494
+
495
+ // 返回第一个注册的路径
496
+ const firstKey = Object.keys(paths)[0];
497
+ if (!firstKey) {
498
+ throw new Error(`No paths registered for component: ${component}`);
499
+ }
500
+ const { base, subPath: relativePath } = paths[firstKey];
501
+ return XDG_PATHS.getComponentPath(base, relativePath);
502
+ }
503
+
504
+ // === 持久化 ===
505
+
506
+ /**
507
+ * 手动持久化
508
+ */
509
+ async save(key?: string): Promise<void> {
510
+ await this.persistQueue.flush();
511
+ }
512
+
513
+ /**
514
+ * 重置到默认值
515
+ */
516
+ async reset(key: string): Promise<void> {
517
+ const [component, ...rest] = key.split(".");
518
+ const schema = this.componentSchemas.get(component);
519
+ if (!schema) return;
520
+
521
+ const partialKey = rest.join(".");
522
+ const defaultValue = schema.defaults[partialKey] ?? schema.defaults[key];
523
+ if (defaultValue !== undefined) {
524
+ await this.set(key, defaultValue);
525
+ }
526
+ }
527
+
528
+ // === 监听 ===
529
+
530
+ /**
531
+ * 监听配置变更
532
+ */
533
+ watch(pattern: string, callback: ComponentWatchCallback): () => void {
534
+ if (!this.watchers.has(pattern)) {
535
+ this.watchers.set(pattern, new Set());
536
+ }
537
+ this.watchers.get(pattern)!.add(callback);
538
+
539
+ return () => {
540
+ this.watchers.get(pattern)?.delete(callback);
541
+ };
542
+ }
543
+
544
+ // === 配置源 ===
545
+
546
+ /**
547
+ * 添加配置源
548
+ */
549
+ addSource(source: ConfigSource): void {
550
+ this.sources.push(source);
551
+ this.sources.sort((a, b) => a.priority - b.priority);
552
+
553
+ // 启用监听
554
+ if (source.watch) {
555
+ const callback = (event: any) => {
556
+ this.notifyWatchers(event.key, {
557
+ component: event.key.split(".")[0],
558
+ key: event.key,
559
+ oldValue: event.oldValue,
560
+ newValue: event.newValue,
561
+ persisted: true,
562
+ timestamp: event.timestamp,
563
+ });
564
+ };
565
+
566
+ // 调用 watch 并忽略错误(确保初始化错误不会中断流程)
567
+ try {
568
+ const unwatch = source.watch(callback);
569
+ const sourceKey = this.getSourceKey(source);
570
+ this.sourceUnwatchFns.set(sourceKey, unwatch);
571
+ } catch {
572
+ // 忽略错误
573
+ }
574
+ }
575
+ }
576
+
577
+ /**
578
+ * 获取 Source 唯一标识
579
+ */
580
+ private getSourceKey(source: ConfigSource): string {
581
+ if ("filePath" in source) {
582
+ return `file:${source.filePath}`;
583
+ }
584
+ if ("prefix" in source) {
585
+ return `env:${source.prefix}`;
586
+ }
587
+ return source.name;
588
+ }
589
+
590
+ /**
591
+ * 移除配置源
592
+ */
593
+ removeSource(name: ConfigSourceType): boolean {
594
+ const index = this.sources.findIndex((s) => s.name === name);
595
+ if (index > 0) {
596
+ // 清理 watcher
597
+ const source = this.sources[index];
598
+ const sourceKey = this.getSourceKey(source);
599
+ const unwatch = this.sourceUnwatchFns.get(sourceKey);
600
+ if (unwatch) {
601
+ unwatch();
602
+ this.sourceUnwatchFns.delete(sourceKey);
603
+ }
604
+ // 不移除 memorySource
605
+ this.sources.splice(index, 1);
606
+ return true;
607
+ }
608
+ return false;
609
+ }
610
+
611
+ /**
612
+ * 关闭所有配置源的 watcher(用于优雅退出)
613
+ */
614
+ unwatchAll(): void {
615
+ for (const unwatch of this.sourceUnwatchFns.values()) {
616
+ unwatch();
617
+ }
618
+ this.sourceUnwatchFns.clear();
619
+ }
620
+
621
+ /**
622
+ * 获取配置源列表
623
+ */
624
+ getSources(): ConfigSource[] {
625
+ return [...this.sources];
626
+ }
627
+
628
+ // === 私有方法 ===
629
+
630
+ private getNestedValue(obj: unknown, key: string): unknown {
631
+ const keys = key.split(".");
632
+ let current: unknown = obj;
633
+
634
+ for (const k of keys) {
635
+ if (typeof current !== "object" || current === null) return undefined;
636
+ current = (current as Record<string, unknown>)[k];
637
+ }
638
+
639
+ return current;
640
+ }
641
+
642
+ private notifyWatchers(key: string, event: ComponentConfigChangeEvent): void {
643
+ // 精确匹配
644
+ const exactWatchers = this.watchers.get(key);
645
+ exactWatchers?.forEach((cb) => cb(event));
646
+
647
+ // 通配符匹配
648
+ const [component] = key.split(".");
649
+ const wildcardWatchers = this.watchers.get(`${component}.*`);
650
+ wildcardWatchers?.forEach((cb) => cb(event));
651
+
652
+ // 全局匹配
653
+ const globalWatchers = this.watchers.get("*");
654
+ globalWatchers?.forEach((cb) => cb(event));
655
+ }
656
+
657
+ private async persistToFile(entries: PersistEntry[]): Promise<void> {
658
+ if (!this.persistFile) return;
659
+
660
+ // 简化实现:追加到文件
661
+ // 实际应该读取文件、合并、更新
662
+ console.log(`[ConfigComponent] Persisting ${entries.length} entries to ${this.persistFile}`);
663
+ // TODO: 实现完整文件持久化
664
+ }
665
+
666
+ // === 去中心化配置注册 ===
667
+
668
+ /**
669
+ * 注册配置 Source
670
+ *
671
+ * Source 会在首次 load() 时被创建
672
+ *
673
+ * @example
674
+ * config.registerSource({ type: "file", relativePath: "llm-config.jsonc" });
675
+ * config.registerSource({ type: "env", envPrefix: "LLM" });
676
+ */
677
+ registerSource(registration: SourceRegistration): void {
678
+ const { type, relativePath, envPrefix } = registration;
679
+
680
+ // 生成唯一标识
681
+ const sourceKey = type === "file" && relativePath
682
+ ? `file:${relativePath}`
683
+ : `${type}:${envPrefix ?? ""}`;
684
+
685
+ // 检查是否已注册
686
+ if (this.sourceKeys.has(sourceKey)) {
687
+ return; // 忽略重复注册
688
+ }
689
+
690
+ // 延迟创建:只保存注册配置
691
+ this.pendingSources.push(registration);
692
+ this.sourceKeys.set(sourceKey, true);
693
+ }
694
+
695
+ /**
696
+ * 创建 Source 实例
697
+ */
698
+ private createSource(registration: SourceRegistration): ConfigSource {
699
+ const { type, relativePath, envPrefix, priority, optional, watch } = registration;
700
+
701
+ switch (type) {
702
+ case "memory": {
703
+ // MemorySource 是内置的,直接返回
704
+ return this.memorySource;
705
+ }
706
+
707
+ case "file": {
708
+ if (!relativePath) {
709
+ throw new Error("File source requires relativePath");
710
+ }
711
+
712
+ // 判断是绝对路径还是相对路径
713
+ // 注意:在 Bun 环境中,import.meta.url 返回的是 file:/// URL
714
+ const isAbsolutePath = path.isAbsolute(relativePath) || relativePath.startsWith("file://");
715
+
716
+ let filePath: string;
717
+ if (isAbsolutePath) {
718
+ // 绝对路径:直接使用
719
+ filePath = relativePath.startsWith("file://")
720
+ ? relativePath.replace(/^file:\/\//, "")
721
+ : relativePath;
722
+ } else {
723
+ // 相对路径:基于 XDG_DATA_HOME
724
+ // 更新 ProtocolResolver 的 XDG_DATA_HOME 为当前值
725
+ if (this.protocolResolver) {
726
+ this.protocolResolver.xdgDataHome = this.getXdgDataHome();
727
+ }
728
+ filePath = path.join(this.getXdgDataHome(), relativePath);
729
+ }
730
+
731
+ return new FileSource({
732
+ filePath,
733
+ optional: optional ?? false,
734
+ resolver: this.protocolResolver,
735
+ watch: watch,
736
+ });
737
+ }
738
+
739
+ case "env": {
740
+ return new EnvSource({
741
+ prefix: envPrefix ?? "",
742
+ priority: priority ?? 20,
743
+ watch: watch,
744
+ });
745
+ }
746
+
747
+ default:
748
+ throw new Error(`Unknown source type: ${type}`);
749
+ }
750
+ }
751
+
752
+ /**
753
+ * 确保所有待创建的 Source 已被创建
754
+ */
755
+ private ensureSourcesCreated(): void {
756
+ for (const pending of this.pendingSources) {
757
+ const source = this.createSource(pending);
758
+ this.addSource(source);
759
+ }
760
+ this.pendingSources = [];
761
+ }
762
+
763
+ /**
764
+ * 注册配置 Key
765
+ *
766
+ * @example
767
+ * config.registerKeys([
768
+ * { key: "llm.provider", sources: ["file", "memory"], default: "openai" },
769
+ * { key: "llm.model", sources: ["file"] },
770
+ * ]);
771
+ */
772
+ registerKeys(keys: KeyRegistration[]): void {
773
+ for (const keyReg of keys) {
774
+ // 如果已存在,跳过(保留原有注册)
775
+ if (this.keyRegistry.has(keyReg.key)) {
776
+ continue;
777
+ }
778
+ this.keyRegistry.set(keyReg.key, keyReg);
779
+
780
+ // 自动将默认值写入 memory source
781
+ if (keyReg.default !== undefined) {
782
+ if (this.memorySource.read(keyReg.key) === undefined) {
783
+ this.memorySource.write(keyReg.key, keyReg.default);
784
+ }
785
+ }
786
+ }
787
+ }
788
+
789
+ /**
790
+ * 注册 Component(一次性注册 Source 和 Keys)
791
+ *
792
+ * @example
793
+ * config.registerComponent({
794
+ * name: "llm",
795
+ * sources: [
796
+ * { type: "file", relativePath: "llm-config.jsonc" },
797
+ * { type: "env", envPrefix: "LLM" },
798
+ * ],
799
+ * keys: [
800
+ * { key: "llm.provider", sources: ["file", "env", "memory"] },
801
+ * ],
802
+ * });
803
+ */
804
+ registerComponent(registration: ComponentRegistration): void {
805
+ const { name, sources, keys, defaults } = registration;
806
+
807
+ // 存储 Component 注册信息
808
+ this.componentRegistrations.set(name, registration);
809
+
810
+ // 注册所有 Source
811
+ for (const sourceReg of sources) {
812
+ this.registerSource(sourceReg);
813
+ }
814
+
815
+ // 注册所有 Keys
816
+ this.registerKeys(keys);
817
+
818
+ // 注册默认值(如果提供)
819
+ if (defaults) {
820
+ for (const [key, value] of Object.entries(defaults)) {
821
+ const fullKey = `${name}.${key}`;
822
+ if (this.memorySource.read(fullKey) === undefined) {
823
+ this.memorySource.write(fullKey, value);
824
+ }
825
+ }
826
+ }
827
+ }
828
+
829
+ /**
830
+ * 按需加载配置
831
+ *
832
+ * 从注册的 Source 加载已注册 Keys 的配置值到内存中
833
+ *
834
+ * @param componentName - 可选,指定要加载的 Component 名称
835
+ *
836
+ * @example
837
+ * // 加载所有已注册的 Keys
838
+ * await config.load();
839
+ *
840
+ * // 只加载指定 Component 的 Keys
841
+ * await config.load("llm");
842
+ */
843
+ async load(componentName?: string): Promise<void> {
844
+ // 确保所有待创建的 Source 已被创建
845
+ this.ensureSourcesCreated();
846
+
847
+ // 确定要加载哪些 Keys
848
+ let keysToLoad: Array<{ key: string; sources: ConfigSourceType[] }>;
849
+
850
+ if (componentName) {
851
+ // 只加载指定 Component 的 Keys
852
+ const registration = this.componentRegistrations.get(componentName);
853
+ if (!registration) {
854
+ return; // Component 未注册,直接返回
855
+ }
856
+ keysToLoad = registration.keys.map(k => ({
857
+ key: k.key,
858
+ sources: k.sources,
859
+ }));
860
+ } else {
861
+ // 加载所有已注册的 Keys
862
+ keysToLoad = Array.from(this.keyRegistry.values()).map(k => ({
863
+ key: k.key,
864
+ sources: k.sources,
865
+ }));
866
+ }
867
+
868
+ // 按 Source 分组加载
869
+ const sourcesByType = new Map<ConfigSourceType, ConfigSource>();
870
+ for (const source of this.sources) {
871
+ sourcesByType.set(source.name, source);
872
+ }
873
+
874
+ // 加载每个 Key
875
+ for (const { key, sources: sourceTypes } of keysToLoad) {
876
+ // 检查内存中是否已有值
877
+ if (this.memorySource.read(key) !== undefined) {
878
+ continue; // 已有值,跳过
879
+ }
880
+
881
+ // 按优先级加载
882
+ // sources 参数指定了要查询的 Source 类型列表
883
+ // 我们需要从这些 Source 中找到第一个有值的
884
+ for (const sourceType of sourceTypes) {
885
+ const source = sourcesByType.get(sourceType);
886
+ if (!source) continue;
887
+
888
+ const value = source.read(key);
889
+ if (value !== undefined) {
890
+ // 加载到内存
891
+ this.memorySource.write(key, value);
892
+ break;
893
+ }
894
+ }
895
+ }
896
+ }
897
+
898
+ /**
899
+ * 获取已注册的 Source 列表
900
+ */
901
+ getRegisteredSources(): ConfigSource[] {
902
+ // 确保所有延迟创建的 Source 已被创建
903
+ this.ensureSourcesCreated();
904
+ return [...this.sources];
905
+ }
906
+ }