@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,612 @@
1
+ /**
2
+ * @fileoverview OpenTelemetry TracerProvider 实现
3
+ *
4
+ * 提供 OpenTelemetry 风格的追踪能力:
5
+ * - TracerProvider: 全局 Tracer 管理器
6
+ * - OTelTracer: 具体 Tracer 实现
7
+ * - OTelSpan: Span 实现
8
+ *
9
+ * 支持 W3C TraceContext 格式的跨进程上下文传递
10
+ */
11
+
12
+ import type { SpanContext, SpanAttributes } from "../types";
13
+ import { SQLiteSpanStorage, type SpanStorage } from "../span-storage";
14
+ import { propagation, type ExtractedContext } from "./propagation";
15
+ import { SpanStatus } from "../types";
16
+
17
+ // ============================================================================
18
+ // Types
19
+ // ============================================================================
20
+
21
+ /**
22
+ * Span 选项
23
+ */
24
+ export interface SpanOptions {
25
+ /** 父 Span 上下文(可选) */
26
+ parent?: SpanContext;
27
+ /** Span 属性 */
28
+ attributes?: SpanAttributes;
29
+ /** Span 类型 */
30
+ kind?: string;
31
+ /** 指定 traceId(可选,如果不提供则自动生成) */
32
+ traceId?: string;
33
+ }
34
+
35
+ /**
36
+ * OTel Span 接口
37
+ */
38
+ export interface OTelSpan {
39
+ readonly name: string;
40
+ readonly kind: string;
41
+ readonly spanContext: SpanContext;
42
+ readonly attributes: SpanAttributes;
43
+ startTime?: number;
44
+ endTime?: number;
45
+ error?: string;
46
+
47
+ setAttribute(key: string, value: unknown): void;
48
+ addEvent(name: string, attributes?: Record<string, unknown>): void;
49
+ end(result?: unknown, error?: Error): void;
50
+ }
51
+
52
+ /**
53
+ * 结束 span 并返回清理函数(用于恢复父上下文)
54
+ */
55
+ export interface OTelSpanEndResult {
56
+ /** 结束后的父上下文 */
57
+ parentContext?: SpanContext;
58
+ /** traceId(用于显示) */
59
+ traceId: string;
60
+ }
61
+
62
+ /**
63
+ * OTel Span 接口扩展 - 提供 endSpan 方法
64
+ */
65
+ export interface OTelSpanWithEndSpan extends OTelSpan {
66
+ /**
67
+ * 结束 span 并返回父上下文
68
+ * 用于结束根 span 后恢复之前的上下文
69
+ */
70
+ endSpan(): OTelSpanEndResult;
71
+ }
72
+
73
+ /**
74
+ * Span 结束时的回调函数类型
75
+ */
76
+ type SpanEndCallback = (spanId: string, parentSpanId?: string) => void;
77
+
78
+ /**
79
+ * OTel Tracer 接口
80
+ */
81
+ export interface OTelTracer {
82
+ readonly name: string;
83
+ readonly version?: string;
84
+
85
+ startSpan(name: string, options?: SpanOptions): OTelSpan;
86
+ /**
87
+ * 结束指定 span 并恢复父上下文
88
+ * @param span 要结束的 span
89
+ * @returns 结束后的父上下文(用于恢复之前的上下文)
90
+ */
91
+ endSpan(span: OTelSpan): SpanContext | undefined;
92
+ injectToEnv(env: Record<string, string | undefined>): void;
93
+ getCurrentContext(): SpanContext | undefined;
94
+ setCurrentContext(context: SpanContext | undefined): void;
95
+
96
+ /**
97
+ * 生成符合 OTel 格式的 traceId(32 字符 16 进制)
98
+ */
99
+ generateTraceId(): string;
100
+
101
+ /**
102
+ * 生成符合 OTel 格式的 spanId(16 字符 16 进制)
103
+ */
104
+ generateSpanId(): string;
105
+ }
106
+
107
+ // ============================================================================
108
+ // Span Implementation
109
+ // ============================================================================
110
+
111
+ /**
112
+ * OTel Span 实现
113
+ */
114
+ class OTelSpanImpl implements OTelSpan {
115
+ readonly name: string;
116
+ readonly kind: string;
117
+ readonly spanContext: SpanContext;
118
+ readonly attributes: SpanAttributes = {};
119
+ startTime?: number;
120
+ endTime?: number;
121
+ error?: string;
122
+
123
+ private storage: SpanStorage;
124
+ private parentSpanContext?: SpanContext;
125
+ private onEnd?: () => void;
126
+
127
+ constructor(
128
+ name: string,
129
+ spanContext: SpanContext,
130
+ storage: SpanStorage,
131
+ parentSpanContext?: SpanContext,
132
+ onEnd?: () => void
133
+ ) {
134
+ this.name = name;
135
+ this.kind = "internal";
136
+ this.spanContext = spanContext;
137
+ this.parentSpanContext = parentSpanContext;
138
+ this.startTime = Date.now();
139
+ this.storage = storage;
140
+ this.onEnd = onEnd;
141
+ }
142
+
143
+ setAttribute(key: string, value: unknown): void {
144
+ this.attributes[key] = value as any;
145
+ }
146
+
147
+ addEvent(name: string, attributes?: Record<string, unknown>): void {
148
+ // 事件暂时存储在 attributes 中
149
+ const events = (this.attributes as any)._events as Array<{ name: string; attributes?: Record<string, unknown> }> || [];
150
+ events.push({ name, attributes });
151
+ (this.attributes as any)._events = events;
152
+ }
153
+
154
+ end(result?: unknown, error?: Error): void {
155
+ this.endTime = Date.now();
156
+ if (error) {
157
+ this.error = error.message;
158
+ }
159
+
160
+ // 保存到存储
161
+ // 注意:parentSpanId 来自 spanContext.parentSpanId,这是 startSpan 时设置的正确值
162
+ this.storage.save({
163
+ traceId: this.spanContext.traceId,
164
+ spanId: this.spanContext.spanId,
165
+ parentSpanId: this.spanContext.parentSpanId,
166
+ name: this.name,
167
+ kind: this.kind as any,
168
+ status: error ? SpanStatus.ERROR : SpanStatus.OK,
169
+ startTime: this.startTime!,
170
+ endTime: this.endTime,
171
+ attributes: this.attributes,
172
+ result,
173
+ error: this.error,
174
+ });
175
+
176
+ // 调用回调(用于上下文恢复等)
177
+ this.onEnd?.();
178
+ }
179
+ }
180
+
181
+ // ============================================================================
182
+ // Tracer Implementation
183
+ // ============================================================================
184
+
185
+ /**
186
+ * OTel Tracer 实现
187
+ */
188
+ class OTelTracerImpl implements OTelTracer {
189
+ readonly name: string;
190
+ readonly version?: string;
191
+
192
+ private storage: SpanStorage;
193
+ private currentContext: SpanContext | undefined;
194
+ private activeSpans: Map<string, OTelSpan> = new Map();
195
+ private onSpanEndCallback?: SpanEndCallback;
196
+
197
+ // 引用到 Provider,用于访问全局 context
198
+ private provider: OTelTracerProvider;
199
+
200
+ constructor(name: string, version: string | undefined, storage: SpanStorage, provider: OTelTracerProvider) {
201
+ this.name = name;
202
+ this.version = version;
203
+ this.storage = storage;
204
+ this.provider = provider;
205
+ }
206
+
207
+ /**
208
+ * 设置 span 结束时的回调
209
+ */
210
+ setOnSpanEndCallback(callback: SpanEndCallback): void {
211
+ this.onSpanEndCallback = callback;
212
+ }
213
+
214
+ /**
215
+ * 获取当前 tracer 的活跃 span 数量(用于调试)
216
+ */
217
+ getActiveSpanCount(): number {
218
+ return this.activeSpans.size;
219
+ }
220
+
221
+ /**
222
+ * 启动新 Span
223
+ *
224
+ * parentSpanId 确定逻辑:
225
+ * 1. 跨进程场景(options.parent 显式指定):
226
+ * → options.parent.parentSpanId || options.parent.spanId
227
+ * 2. 同进程嵌套(隐式继承 currentContext):
228
+ * → currentContext.spanId(忽略 currentContext.parentSpanId)
229
+ * 3. 无父上下文:parentSpanId = undefined
230
+ *
231
+ * 关键:currentContext.parentSpanId 是当前 span 的父 span,不应该作为新 span 的父 span
232
+ */
233
+ startSpan(name: string, options?: SpanOptions): OTelSpan {
234
+ // 确定 traceId 和 parentContext
235
+ // 优先级:
236
+ // 1. options.parent 显式传入(即使值是 undefined)
237
+ // 2. 当前 tracer 的 currentContext
238
+ // 3. Provider 全局 context(所有 tracer 共享)
239
+ // 4. 自动生成新的 traceId
240
+ const parentFromOptions = options?.parent;
241
+ // 使用 'in' 操作符检查 parent 是否在 options 中被显式设置
242
+ const hasExplicitParent = options && 'parent' in options;
243
+ const globalContext = this.provider.getGlobalContext();
244
+
245
+ // 确定有效的 parent context
246
+ // 如果显式传入了 parent(即使值是 undefined),优先使用
247
+ // 否则使用当前 tracer 的 currentContext
248
+ // 如果都没有,使用全局 context
249
+ let effectiveParentContext: SpanContext | undefined;
250
+ if (hasExplicitParent) {
251
+ // parent 显式传入,使用它(可能是 undefined)
252
+ effectiveParentContext = parentFromOptions;
253
+ } else if (this.currentContext) {
254
+ effectiveParentContext = this.currentContext;
255
+ } else if (globalContext) {
256
+ // 全局 context 存在但当前 tracer 没有 context
257
+ // 使用全局 context 作为 parent
258
+ effectiveParentContext = globalContext;
259
+ }
260
+
261
+ const traceId = effectiveParentContext?.traceId || this.generateTraceId();
262
+ const spanId = this.generateSpanId();
263
+
264
+ // parentSpanId 确定:
265
+ // 1. 显式 options.parent(跨进程场景):优先用 parentSpanId
266
+ // 2. 隐式继承 currentContext(同进程嵌套):只用 effectiveParentContext.spanId
267
+ // 3. 无父上下文:undefined
268
+ let parentSpanId: string | undefined;
269
+ if (hasExplicitParent && parentFromOptions) {
270
+ // 跨进程场景:显式传递 parent
271
+ parentSpanId = parentFromOptions.parentSpanId || parentFromOptions.spanId;
272
+ } else if (this.currentContext) {
273
+ // 同进程嵌套:继承 currentContext,忽略 currentContext.parentSpanId
274
+ parentSpanId = this.currentContext.spanId;
275
+ } else if (globalContext) {
276
+ // 使用全局 context 作为 parent
277
+ parentSpanId = globalContext.spanId;
278
+ }
279
+
280
+ // ⚠️ 关键:创建独立的 spanContext 对象,而不是修改现有对象
281
+ // 因为 parent?.spanId 可能指向 currentContext.spanContext(span 的 spanContext)
282
+ // 如果直接赋值,parentContext.spanContext.spanId 会被修改
283
+ // 使用对象展开确保创建新对象
284
+ const spanContext = {
285
+ traceId,
286
+ spanId,
287
+ parentSpanId,
288
+ } as SpanContext;
289
+
290
+ // ⚠️ 关键:在创建 Span 之前,先更新 currentContext
291
+ // 这样子 span 获取 parent 时,currentContext 已经是当前 span 的上下文
292
+ // 而不是被新 span 的上下文覆盖
293
+ this.currentContext = {
294
+ traceId,
295
+ spanId,
296
+ parentSpanId,
297
+ };
298
+
299
+ // 创建 Span,传递完整的父上下文(用于存储和上下文恢复)
300
+ const span = new OTelSpanImpl(
301
+ name,
302
+ spanContext,
303
+ this.storage,
304
+ effectiveParentContext, // 传递完整父上下文
305
+ () => this.handleSpanEnd(spanContext.spanId, effectiveParentContext?.spanId)
306
+ );
307
+
308
+ // 设置初始属性
309
+ if (options?.attributes) {
310
+ for (const [key, value] of Object.entries(options.attributes)) {
311
+ span.setAttribute(key, value);
312
+ }
313
+ }
314
+
315
+ this.activeSpans.set(spanId, span);
316
+
317
+ return span;
318
+ }
319
+
320
+ /**
321
+ * 将当前上下文注入到环境变量
322
+ */
323
+ injectToEnv(env: Record<string, string | undefined>): void {
324
+ if (this.currentContext) {
325
+ propagation.inject(env, this.currentContext);
326
+ }
327
+ }
328
+
329
+ /**
330
+ * 获取当前上下文
331
+ */
332
+ getCurrentContext(): SpanContext | undefined {
333
+ return this.currentContext;
334
+ }
335
+
336
+ /**
337
+ * 设置当前上下文(用于从外部恢复上下文)
338
+ */
339
+ setCurrentContext(context: SpanContext | undefined): void {
340
+ this.currentContext = context;
341
+ }
342
+
343
+ /**
344
+ * 结束 Span 并恢复父上下文
345
+ *
346
+ * 关键:需要正确恢复 parentSpanId,使得子 span 可以正确关联到父 span
347
+ *
348
+ * 跨进程场景处理:
349
+ * - 如果父 span 不存在(可能在另一个进程中),保留 traceId
350
+ * - 这样子进程创建的 span 能继承 traceId,而不是创建新的根节点
351
+ *
352
+ * @returns 结束后的父上下文(用于恢复之前的上下文)
353
+ */
354
+ endSpan(span: OTelSpan): SpanContext | undefined {
355
+ const spanContext = span.spanContext;
356
+ this.activeSpans.delete(spanContext.spanId);
357
+
358
+ // 保存当前 span 的 traceId(如果存在)
359
+ const currentTraceId = this.currentContext?.traceId;
360
+
361
+ if (spanContext.parentSpanId) {
362
+ // 尝试找到父 Span 并恢复其上下文
363
+ const parentSpan = this.activeSpans.get(spanContext.parentSpanId);
364
+ if (parentSpan) {
365
+ // 正确恢复完整上下文:包含 parentSpanId(父 span 的 spanId)
366
+ const parentContext: SpanContext = {
367
+ traceId: spanContext.traceId,
368
+ spanId: parentSpan.spanContext.spanId,
369
+ parentSpanId: parentSpan.spanContext.parentSpanId,
370
+ };
371
+ this.currentContext = parentContext;
372
+ return parentContext;
373
+ }
374
+
375
+ // 父 span 不存在(跨进程场景)
376
+ // 保留 traceId,让后续 span 能继承,而不是创建新的根
377
+ if (currentTraceId) {
378
+ const parentContext: SpanContext = {
379
+ traceId: currentTraceId,
380
+ spanId: spanContext.parentSpanId,
381
+ parentSpanId: undefined,
382
+ };
383
+ this.currentContext = parentContext;
384
+ return parentContext;
385
+ }
386
+ }
387
+
388
+ // 当前 span 没有父 span,清除上下文
389
+ this.currentContext = undefined;
390
+ return undefined;
391
+ }
392
+
393
+ generateTraceId(): string {
394
+ // 生成 32 字符的十六进制 traceId (128 位)
395
+ const timestamp = Date.now().toString(16).padStart(12, "0");
396
+ const random = Array.from({ length: 5 }, () =>
397
+ Math.floor(Math.random() * 0xFFFFFFFF).toString(16).padStart(8, "0")
398
+ ).join("");
399
+ return (timestamp + random).slice(0, 32).padStart(32, "0");
400
+ }
401
+
402
+ generateSpanId(): string {
403
+ // 生成 16 字符的十六进制 spanId (64 位)
404
+ return Math.floor(Math.random() * 0xFFFFFFFFFFFFFFFF)
405
+ .toString(16)
406
+ .padStart(16, "0");
407
+ }
408
+
409
+ /**
410
+ * 处理 span 结束
411
+ *
412
+ * 当 span 结束时,尝试恢复父 span 的上下文。
413
+ * 如果当前 span 的父 span 仍然活跃,则恢复父上下文。
414
+ *
415
+ * 跨进程场景处理:
416
+ * - 如果父 span 不存在(可能在另一个进程中),保留 traceId
417
+ * - 这样子进程创建的 span 能继承 traceId,而不是创建新的根节点
418
+ *
419
+ * 关键:需要正确恢复 parentSpanId,使得后续 span 可以正确关联
420
+ */
421
+ private handleSpanEnd(spanId: string, parentSpanId?: string): void {
422
+ this.activeSpans.delete(spanId);
423
+
424
+ // 保存当前 span 的 traceId(如果存在)
425
+ const currentTraceId = this.currentContext?.traceId;
426
+
427
+ if (parentSpanId) {
428
+ // 如果父 span 仍然活跃,恢复父上下文
429
+ const parentSpan = this.activeSpans.get(parentSpanId);
430
+ if (parentSpan) {
431
+ this.currentContext = {
432
+ traceId: parentSpan.spanContext.traceId,
433
+ spanId: parentSpan.spanContext.spanId,
434
+ parentSpanId: parentSpan.spanContext.parentSpanId,
435
+ };
436
+ return;
437
+ }
438
+
439
+ // 父 span 不存在(跨进程场景)
440
+ // 保留 traceId,让后续 span 能继承,而不是创建新的根
441
+ // parentSpanId 设为 undefined,因为父 span 已经结束
442
+ if (currentTraceId) {
443
+ this.currentContext = {
444
+ traceId: currentTraceId,
445
+ spanId: parentSpanId, // 保留父 span ID 作为参考
446
+ parentSpanId: undefined,
447
+ };
448
+ return;
449
+ }
450
+ }
451
+
452
+ // 当前 span 没有父 span,清除上下文
453
+ this.currentContext = undefined;
454
+ }
455
+ }
456
+
457
+ // ============================================================================
458
+ // TracerProvider Implementation
459
+ // ============================================================================
460
+
461
+ /**
462
+ * OpenTelemetry TracerProvider
463
+ *
464
+ * 全局 Tracer 管理器,负责:
465
+ * - 维护 Tracer 缓存
466
+ * - 管理存储
467
+ * - 处理上下文初始化
468
+ * - 提供全局 context 供所有 tracer 共享
469
+ */
470
+ export class OTelTracerProvider {
471
+ private tracers: Map<string, OTelTracerImpl> = new Map();
472
+ private storage: SpanStorage;
473
+ private initialized = false;
474
+
475
+ // 全局 context - 所有 tracer 共享
476
+ // 用于在不同的 tracer 实例之间传递 traceId
477
+ private globalContext: SpanContext | undefined;
478
+
479
+ constructor(storage?: SpanStorage, dbPath?: string) {
480
+ this.storage = storage || new SQLiteSpanStorage(dbPath || getDefaultDbPath());
481
+ }
482
+
483
+ /**
484
+ * 初始化 Provider
485
+ *
486
+ * 从环境变量恢复上下文(如 TRACEPARENT)
487
+ */
488
+ async initialize(): Promise<void> {
489
+ if (this.initialized) return;
490
+
491
+ await this.storage.initialize();
492
+ this.initialized = true;
493
+ }
494
+
495
+ /**
496
+ * 检查是否已初始化
497
+ */
498
+ isInitialized(): boolean {
499
+ return this.initialized;
500
+ }
501
+
502
+ /**
503
+ * 获取全局 context(所有 tracer 共享)
504
+ */
505
+ getGlobalContext(): SpanContext | undefined {
506
+ return this.globalContext;
507
+ }
508
+
509
+ /**
510
+ * 设置全局 context(所有 tracer 共享)
511
+ */
512
+ setGlobalContext(context: SpanContext | undefined): void {
513
+ this.globalContext = context;
514
+ }
515
+
516
+ /**
517
+ * 获取 Tracer
518
+ */
519
+ getTracer(name: string, version?: string): OTelTracer {
520
+ const key = `${name}@${version || "0.0.0"}`;
521
+
522
+ let tracer = this.tracers.get(key);
523
+ if (!tracer) {
524
+ tracer = new OTelTracerImpl(name, version, this.storage, this);
525
+ this.tracers.set(key, tracer);
526
+
527
+ // 检查环境变量中是否有父上下文(仅在初始化后)
528
+ if (this.initialized) {
529
+ this.restoreFromEnv(tracer);
530
+ }
531
+ }
532
+
533
+ return tracer;
534
+ }
535
+
536
+ /**
537
+ * 从环境变量恢复上下文
538
+ */
539
+ private restoreFromEnv(tracer: OTelTracerImpl): void {
540
+ const extracted = propagation.extract(process.env as Record<string, string | undefined>);
541
+ if (extracted) {
542
+ // 恢复上下文:设置 spanId 为发送方的 spanId,作为后续 span 的 parentSpanId
543
+ tracer.setCurrentContext({
544
+ traceId: extracted.traceId,
545
+ spanId: extracted.spanId, // 发送方 span ID,用于后续 span 的 parentSpanId
546
+ parentSpanId: undefined, // 根 span 没有父
547
+ });
548
+ }
549
+ }
550
+
551
+ /**
552
+ * 关闭 Provider
553
+ */
554
+ shutdown(): void {
555
+ for (const tracer of this.tracers.values()) {
556
+ // 结束所有活跃的 span
557
+ }
558
+ this.tracers.clear();
559
+ this.storage.close();
560
+ this.initialized = false;
561
+ // 清除全局 context,确保下次创建 tracer 时不会继承旧上下文
562
+ this.globalContext = undefined;
563
+ // 清除环境变量中的 trace context,避免影响后续测试
564
+ delete process.env['TRACEPARENT'];
565
+ delete process.env['TRACE_ID'];
566
+ delete process.env['LOG_TRACE_REQUEST_ID'];
567
+ }
568
+
569
+ /**
570
+ * 获取存储
571
+ */
572
+ getStorage(): SpanStorage {
573
+ return this.storage;
574
+ }
575
+ }
576
+
577
+ // ============================================================================
578
+ // Singleton
579
+ // ============================================================================
580
+
581
+ let providerInstance: OTelTracerProvider | null = null;
582
+
583
+ /**
584
+ * 获取 TracerProvider 单例
585
+ */
586
+ export function getTracerProvider(): OTelTracerProvider {
587
+ if (!providerInstance) {
588
+ providerInstance = new OTelTracerProvider();
589
+ }
590
+ return providerInstance;
591
+ }
592
+
593
+ /**
594
+ * 重置 TracerProvider(用于测试)
595
+ */
596
+ export function resetTracerProvider(): void {
597
+ if (providerInstance) {
598
+ providerInstance.shutdown();
599
+ providerInstance = null;
600
+ }
601
+ }
602
+
603
+ /**
604
+ * 获取默认的 SQLite 数据库路径
605
+ */
606
+ function getDefaultDbPath(): string {
607
+ const os = require("os");
608
+ const path = require("path");
609
+ const home = os.homedir();
610
+ const dataHome = process.env.XDG_DATA_HOME || path.join(home, ".local", "share");
611
+ return path.join(dataHome, "roy-agent", "traces.db");
612
+ }