@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,1249 @@
1
+ /**
2
+ * @fileoverview Session Component
3
+ *
4
+ * Component for managing session lifecycle
5
+ */
6
+
7
+ import { BaseComponent, type ComponentConfig } from "../component";
8
+ import {
9
+ type SessionConfig,
10
+ type Session,
11
+ type CreateSessionOptions,
12
+ type UpdateSessionOptions,
13
+ type ListSessionsOptions,
14
+ type SessionMessage,
15
+ type AddMessageOptions,
16
+ type SessionCheckpoint, // (new)
17
+ type CompactOptions, // (new)
18
+ type CompactResult, // (new)
19
+ type CompactPreview, // (new)
20
+ type GetContextOptions, // (new)
21
+ type GetContextResult, // (new)
22
+ type SearchMessagesOptions, // (new)
23
+ type SearchResult, // (new)
24
+ type MessageMatch, // (new)
25
+ } from "./types";
26
+ import type { SessionHookContext } from "../types";
27
+ import type { SessionStore } from "./session-store";
28
+ import { MemorySessionStore, SQLiteSessionStore } from "./storage";
29
+ import { createLogger } from "../log-trace/logger";
30
+ import type { ConfigComponent } from "../../config/config-component";
31
+ import { envKeyToConfigKey } from "../../config/env-key";
32
+ import { TracedAs } from "../log-trace/decorator";
33
+ import { SummaryAgent } from "../agent/summary-agent"; // (new)
34
+ import { SESSION_CONFIG_REGISTRATION, SESSION_DEFAULTS } from "./session-config-registration";
35
+ import { parseSearchQuery, matchesQuery } from "./search-query-parser";
36
+ import path from "path";
37
+
38
+ // Re-export for convenience
39
+ export type {
40
+ SessionConfig,
41
+ Session,
42
+ SessionMessage,
43
+ CreateSessionOptions,
44
+ UpdateSessionOptions,
45
+ ListSessionsOptions,
46
+ AddMessageOptions,
47
+ SessionCheckpoint, // (new)
48
+ CompactOptions, // (new)
49
+ CompactResult, // (new)
50
+ CompactPreview, // (new)
51
+ GetContextOptions, // (new)
52
+ GetContextResult, // (new)
53
+ SearchMessagesOptions, // (new) Search
54
+ SearchResult, // (new) Search
55
+ MessageMatch, // (new) Search
56
+ } from "./types";
57
+
58
+ // Create logger for session component
59
+ const logger = createLogger("session");
60
+
61
+ // ============================================================================
62
+ // SessionComponentOptions (统一配置机制)
63
+ // ============================================================================
64
+
65
+ /**
66
+ * SessionComponent 配置选项(通过 options 传递)
67
+ *
68
+ * 配置加载顺序(优先级从低到高):
69
+ * 1. File - 配置文件(通过 configPath 指定)
70
+ * 2. Env - 环境变量(通过 envPrefix 配置前缀)
71
+ * 3. Object - 直接传入的 config 对象(最高优先级)
72
+ */
73
+ export interface SessionComponentOptions {
74
+ /** ConfigComponent 实例(必填) */
75
+ configComponent: ConfigComponent;
76
+
77
+ /** 配置文件相对路径(可选,基于 XDG_DATA_HOME) */
78
+ configPath?: string;
79
+
80
+ /** 环境变量前缀(可选,默认 "SESSION") */
81
+ envPrefix?: string;
82
+
83
+ /** 直接传入的配置对象(可选,优先级最高) */
84
+ config?: Partial<SessionConfig>;
85
+ }
86
+
87
+ /**
88
+ * Session component hooks configuration
89
+ */
90
+ interface SessionComponentHooks {
91
+ "before.session.create"?: import("./types").SessionHook[];
92
+ "after.session.create"?: import("./types").SessionHook[];
93
+ "before.session.get"?: import("./types").SessionHook[];
94
+ "after.session.get"?: import("./types").SessionHook[];
95
+ "before.session.list"?: import("./types").SessionHook[];
96
+ "after.session.list"?: import("./types").SessionHook[];
97
+ "before.session.update"?: import("./types").SessionHook[];
98
+ "after.session.update"?: import("./types").SessionHook[];
99
+ "before.session.delete"?: import("./types").SessionHook[];
100
+ "after.session.delete"?: import("./types").SessionHook[];
101
+ }
102
+
103
+ /**
104
+ * SessionComponent
105
+ *
106
+ * Manages session lifecycle including:
107
+ * - Session CRUD operations
108
+ * - Message management
109
+ * - Active session tracking
110
+ */
111
+ export class SessionComponent extends BaseComponent {
112
+ readonly name = "session";
113
+ readonly version = "1.1.0"; // (new)
114
+
115
+ private config?: SessionConfig;
116
+ private store?: SessionStore;
117
+ private activeSessionId?: string;
118
+ private hooksConfig: SessionComponentHooks = {};
119
+
120
+ /** ConfigComponent 用于配置管理 */
121
+ private configComponent?: ConfigComponent;
122
+
123
+ /** 配置变更 watcher 清理函数 */
124
+ private configWatcher?: () => void;
125
+
126
+ /** Summary Agent for checkpoint generation [new] */
127
+ private summaryAgent?: SummaryAgent;
128
+
129
+ /** PromptComponent reference for SummaryAgent */
130
+ private promptComponent?: any;
131
+
132
+ /** LLMComponent reference for SummaryAgent */
133
+ private llmComponent?: any;
134
+
135
+ constructor() {
136
+ super();
137
+ }
138
+
139
+ /**
140
+ * Initialize the component
141
+ *
142
+ * 配置加载优先级(从高到低):
143
+ * 1. Object - 直接传入的 config 对象
144
+ * 2. Env - 环境变量(通过 envPrefix 配置前缀)
145
+ * 3. File - 配置文件(通过 configPath 指定)
146
+ */
147
+ async init(config?: ComponentConfig): Promise<void> {
148
+ // 调用基类 init,注入 env
149
+ await super.init(config);
150
+
151
+ // 从 options 获取 ConfigComponent
152
+ const options = config?.options as unknown as SessionComponentOptions | undefined;
153
+ if (!options?.configComponent) {
154
+ throw new Error("ConfigComponent is required for SessionComponent initialization");
155
+ }
156
+
157
+ this.configComponent = options.configComponent;
158
+ await this.registerConfig(options);
159
+
160
+ // Initialize store based on config
161
+ const storageConfig = this.config?.storage;
162
+ if (storageConfig?.type === "sqlite") {
163
+ this.store = new SQLiteSessionStore(storageConfig.dbPath);
164
+ logger.info(`[SessionComponent] Using SQLite storage: ${storageConfig.dbPath || "default path"}`);
165
+ } else {
166
+ this.store = new MemorySessionStore();
167
+ logger.info("[SessionComponent] Using memory storage");
168
+ }
169
+
170
+ this.setStatus("running");
171
+ logger.info("[SessionComponent] Initialized");
172
+ }
173
+
174
+ /**
175
+ * 获取默认 session 数据库路径
176
+ */
177
+ private getDefaultSessionDbPath(): string {
178
+ const home = process.env.HOME || process.env.USERPROFILE || "/tmp";
179
+ return path.join(home, ".local", "share", "roy-agent", "sessions.db");
180
+ }
181
+
182
+ /**
183
+ * 注册配置到 ConfigComponent
184
+ *
185
+ * 配置加载顺序(优先级从低到高):
186
+ * 1. Defaults - 默认值(最低)
187
+ * 2. FileSource - 从配置文件加载
188
+ * 3. EnvSource - 从环境变量加载
189
+ * 4. config 对象 - 直接传入的配置(最高)
190
+ */
191
+ private async registerConfig(options: SessionComponentOptions): Promise<void> {
192
+ const configComponent = options.configComponent;
193
+ if (!configComponent) return;
194
+
195
+ const { configPath, envPrefix, config } = options;
196
+ const prefix = envPrefix !== undefined ? envPrefix : "SESSION";
197
+
198
+ // 1. 使用 registerComponent 注册配置结构(keys 和 sources,不含 defaults)
199
+ configComponent.registerComponent(SESSION_CONFIG_REGISTRATION);
200
+
201
+ // 2. 注册 FileSource(如果提供了 configPath)
202
+ if (configPath) {
203
+ configComponent.registerSource({
204
+ type: "file",
205
+ relativePath: configPath,
206
+ optional: true,
207
+ watch: false
208
+ });
209
+ }
210
+
211
+ // 3. 注册 EnvSource
212
+ configComponent.registerSource({
213
+ type: "env",
214
+ envPrefix: prefix,
215
+ priority: 20,
216
+ watch: false
217
+ });
218
+
219
+ // 4. 从 sources 加载配置(FileSource 和 EnvSource)
220
+ await configComponent.load("session");
221
+
222
+ // 5. 后备方案:直接读取环境变量(确保所有环境变量都被处理)
223
+ for (const envKey of Object.keys(process.env)) {
224
+ const configKey = envKeyToConfigKey(envKey, prefix, "session");
225
+ if (!configKey) continue;
226
+ const value = process.env[envKey];
227
+ if (value !== undefined) {
228
+ await configComponent.set(configKey, value);
229
+ }
230
+ }
231
+
232
+ // 6. 设置默认值(只有当配置不存在时)
233
+ for (const [key, value] of Object.entries(SESSION_DEFAULTS)) {
234
+ if (configComponent.get(key) === undefined) {
235
+ await configComponent.set(key, value);
236
+ }
237
+ }
238
+
239
+ // 7. 特殊处理 dbPath 动态默认值(优先级高于 SESSION_DEFAULTS)
240
+ if (configComponent.get("session.storage.dbPath") === undefined) {
241
+ await configComponent.set("session.storage.dbPath", this.getDefaultSessionDbPath());
242
+ }
243
+
244
+ // 8. 直接配置对象(最高优先级)
245
+ if (config) {
246
+ const flatConfig = this.flattenConfig(config);
247
+ for (const [key, value] of Object.entries(flatConfig)) {
248
+ await configComponent.set(key, value);
249
+ }
250
+ }
251
+
252
+ // 9. 注册配置热更新监听
253
+ this.registerConfigWatcher(configComponent);
254
+
255
+ // 10. 解析最终配置
256
+ this.config = this.buildConfig(configComponent);
257
+ }
258
+
259
+ /**
260
+ * 构建最终配置对象
261
+ */
262
+ private buildConfig(configComponent: ConfigComponent): SessionConfig {
263
+ return {
264
+ defaultDirectory: configComponent.get("session.defaultDirectory") as string,
265
+ defaultTitleTemplate: configComponent.get("session.defaultTitleTemplate") as string,
266
+ maxMessages: configComponent.get("session.maxMessages") as number,
267
+ compactionThreshold: configComponent.get("session.compactionThreshold") as number,
268
+ autoCompact: configComponent.get("session.autoCompact") as boolean,
269
+ storage: {
270
+ type: configComponent.get("session.storage.type") as "memory" | "sqlite",
271
+ dbPath: configComponent.get("session.storage.dbPath") as string | undefined,
272
+ },
273
+ };
274
+ }
275
+
276
+ /**
277
+ * 将配置对象展平为点号路径
278
+ */
279
+ private flattenConfig(
280
+ obj: Record<string, unknown>,
281
+ prefix = "session"
282
+ ): Record<string, unknown> {
283
+ const result: Record<string, unknown> = {};
284
+ for (const [key, value] of Object.entries(obj)) {
285
+ const fullKey = `${prefix}.${key}`;
286
+ if (value && typeof value === "object" && !Array.isArray(value)) {
287
+ Object.assign(
288
+ result,
289
+ this.flattenConfig(value as Record<string, unknown>, fullKey)
290
+ );
291
+ } else {
292
+ result[fullKey] = value;
293
+ }
294
+ }
295
+ return result;
296
+ }
297
+
298
+ /**
299
+ * 注册配置热更新监听
300
+ */
301
+ private registerConfigWatcher(configComponent: ConfigComponent): void {
302
+ if (typeof configComponent.watch !== "function") {
303
+ return;
304
+ }
305
+
306
+ this.configWatcher = configComponent.watch("session.*", (event) => {
307
+ this.onConfigChange(event);
308
+ });
309
+ }
310
+
311
+ /**
312
+ * 处理配置变更
313
+ */
314
+ protected onConfigChange(event: { key: string; oldValue?: unknown; newValue?: unknown }): void {
315
+ logger.info(`Session config changed: ${event.key}`, {
316
+ oldValue: event.oldValue,
317
+ newValue: event.newValue
318
+ });
319
+ }
320
+
321
+ /**
322
+ * Start the component
323
+ */
324
+ async onStart(): Promise<void> {
325
+ logger.info("[SessionComponent] Started");
326
+ }
327
+
328
+ /**
329
+ * Stop the component
330
+ */
331
+ async onStop(): Promise<void> {
332
+ // 清理配置 watcher
333
+ this.configWatcher?.();
334
+ this.configWatcher = undefined;
335
+
336
+ // 清理资源
337
+ await this.store?.close();
338
+ this.store = undefined;
339
+ this.setStatus("stopped");
340
+ logger.info("[SessionComponent] Stopped");
341
+ }
342
+
343
+ // ============ Session CRUD ============
344
+
345
+ /**
346
+ * Create a new session
347
+ */
348
+ async create(options: CreateSessionOptions = {}): Promise<Session> {
349
+ // Execute before hooks
350
+ await this.executeBeforeHooks("create", { options });
351
+
352
+ // Generate title if not provided
353
+ const title = options.title ?? this.generateTitle();
354
+
355
+ // Create session
356
+ const session = await this.store!.create({
357
+ ...options,
358
+ title,
359
+ });
360
+
361
+ logger.info(`[SessionComponent] Created session: ${session.id}`, { title });
362
+
363
+ // Execute after hooks
364
+ await this.executeAfterHooks("create", { session, options });
365
+
366
+ return session;
367
+ }
368
+
369
+ /**
370
+ * Get a session by ID
371
+ */
372
+ async get(id: string): Promise<Session | undefined> {
373
+ await this.executeBeforeHooks("session.get", { id });
374
+
375
+ const session = await this.store?.get(id);
376
+
377
+ await this.executeAfterHooks("session.get", { session });
378
+
379
+ return session;
380
+ }
381
+
382
+ /**
383
+ * Get a session with full context (for Memory Agent)
384
+ */
385
+ async getSession(id: string): Promise<{ sessionId: string; title: string; messages: any[] } | null> {
386
+ const session = await this.get(id);
387
+ if (!session) {
388
+ return null;
389
+ }
390
+
391
+ const messages = await this.getMessages(id);
392
+ return {
393
+ sessionId: session.id,
394
+ title: session.title || "Untitled",
395
+ messages: messages.map(m => ({
396
+ role: m.role,
397
+ content: m.content,
398
+ timestamp: m.timestamp,
399
+ })),
400
+ };
401
+ }
402
+
403
+ /**
404
+ * Search sessions by query (for Memory Agent)
405
+ *
406
+ * Simple implementation: list all sessions and return them for the agent to filter
407
+ */
408
+ async searchSessions(
409
+ query: string,
410
+ options?: { session_id?: string; limit?: number }
411
+ ): Promise<Array<{ sessionId: string; title: string; preview: string; timestamp: number }>> {
412
+ const limit = options?.limit ?? 20;
413
+ type SearchResult = { sessionId: string; title: string; preview: string; timestamp: number };
414
+
415
+ // 如果指定了 session_id,只返回该会话
416
+ if (options?.session_id) {
417
+ const session = await this.get(options.session_id);
418
+ if (session) {
419
+ const messages = await this.getMessages(options.session_id);
420
+ const lastMessage = messages[messages.length - 1];
421
+ return [{
422
+ sessionId: session.id,
423
+ title: session.title || "Untitled",
424
+ preview: lastMessage?.content?.substring(0, 200) || "",
425
+ timestamp: session.updatedAt || session.createdAt,
426
+ }];
427
+ }
428
+ return [];
429
+ }
430
+
431
+ // 否则列出所有会话
432
+ const sessions = await this.list({ limit });
433
+
434
+ // 对每个会话获取预览
435
+ const results: SearchResult[] = [];
436
+ for (const session of sessions) {
437
+ if (results.length >= limit) break;
438
+
439
+ const messages = await this.getMessages(session.id);
440
+ const lastMessage = messages[messages.length - 1];
441
+ const preview = lastMessage?.content?.substring(0, 200) || "";
442
+
443
+ // 简单过滤:如果 query 不为空,检查标题或预览是否包含 query
444
+ if (query) {
445
+ const lowerQuery = query.toLowerCase();
446
+ const titleMatch = (session.title || "").toLowerCase().includes(lowerQuery);
447
+ const previewMatch = preview.toLowerCase().includes(lowerQuery);
448
+ if (!titleMatch && !previewMatch) {
449
+ continue;
450
+ }
451
+ }
452
+
453
+ results.push({
454
+ sessionId: session.id,
455
+ title: session.title || "Untitled",
456
+ preview,
457
+ timestamp: session.updatedAt || session.createdAt,
458
+ });
459
+ }
460
+
461
+ return results;
462
+ }
463
+
464
+ /**
465
+ * List all sessions
466
+ */
467
+ async list(options?: ListSessionsOptions): Promise<Session[]> {
468
+ await this.executeBeforeHooks("session.list", { options });
469
+
470
+ const sessions = (await this.store?.list(options)) ?? [];
471
+
472
+ await this.executeAfterHooks("session.list", { sessions });
473
+
474
+ return sessions;
475
+ }
476
+
477
+ /**
478
+ * Get total session count
479
+ */
480
+ async getCount(): Promise<number> {
481
+ return (await this.store?.getCount()) ?? 0;
482
+ }
483
+
484
+ /**
485
+ * Update a session
486
+ */
487
+ async update(id: string, updates: UpdateSessionOptions): Promise<boolean> {
488
+ await this.executeBeforeHooks("session.update", { id, updates });
489
+
490
+ const success = await this.store?.update(id, updates);
491
+
492
+ if (success) {
493
+ logger.info(`[SessionComponent] Updated session: ${id}`, { updates });
494
+ await this.executeAfterHooks("session.update", { id, updates });
495
+ }
496
+
497
+ return success ?? false;
498
+ }
499
+
500
+ /**
501
+ * Delete a session
502
+ */
503
+ async delete(id: string): Promise<boolean> {
504
+ await this.executeBeforeHooks("session.delete", { id });
505
+
506
+ const success = await this.store?.delete(id);
507
+
508
+ if (success) {
509
+ if (this.activeSessionId === id) {
510
+ this.activeSessionId = undefined;
511
+ }
512
+ logger.info(`[SessionComponent] Deleted session: ${id}`);
513
+ await this.executeAfterHooks("session.delete", { id });
514
+ }
515
+
516
+ return success ?? false;
517
+ }
518
+
519
+ // ============ Active Session ============
520
+
521
+ /**
522
+ * Get the current active session
523
+ */
524
+ getActiveSession(): Session | undefined {
525
+ if (!this.activeSessionId) return undefined;
526
+ // Synchronous getter, so we need to handle this carefully
527
+ // In a real implementation, we'd cache the active session
528
+ return undefined; // Will be populated by callers who need it
529
+ }
530
+
531
+ /**
532
+ * Get the active session ID
533
+ */
534
+ getActiveSessionId(): string | undefined {
535
+ return this.activeSessionId;
536
+ }
537
+
538
+ /**
539
+ * Set the active session
540
+ */
541
+ async setActiveSession(id: string): Promise<boolean> {
542
+ const session = await this.store?.get(id);
543
+ if (!session) return false;
544
+
545
+ this.activeSessionId = id;
546
+ logger.info(`[SessionComponent] Active session set: ${id}`);
547
+ return true;
548
+ }
549
+
550
+ // ============ Messages ============
551
+
552
+ /**
553
+ * Add a message to a session
554
+ */
555
+ async addMessage(sessionId: string, message: AddMessageOptions): Promise<string | undefined> {
556
+ // DEBUG: 记录添加消息的调用栈
557
+ const contentPreview = typeof message.content === "string"
558
+ ? message.content.substring(0, 100)
559
+ : JSON.stringify(message.content).substring(0, 100);
560
+ logger.info(`[SessionComponent.addMessage] sessionId=${sessionId}, role=${message.role}, content="${contentPreview}"`);
561
+
562
+ try {
563
+ const messageId = await this.store?.addMessage(sessionId, message);
564
+ logger.info(`[SessionComponent.addMessage] Success: messageId=${messageId}`);
565
+ return messageId;
566
+ } catch (error) {
567
+ logger.error(`[SessionComponent] Failed to add message: ${error}`);
568
+ return undefined;
569
+ }
570
+ }
571
+
572
+ /**
573
+ * Get messages from a session
574
+ */
575
+ async getMessages(sessionId: string, options?: { offset?: number; limit?: number }): Promise<SessionMessage[]> {
576
+ const messages = await this.store?.getMessages(
577
+ sessionId,
578
+ options?.offset,
579
+ options?.limit
580
+ );
581
+ return messages ?? [];
582
+ }
583
+
584
+ /**
585
+ * Get message count for a session
586
+ */
587
+ async getMessageCount(sessionId: string, includeArchived?: boolean): Promise<number> {
588
+ return this.store?.getMessageCount(sessionId, includeArchived) ?? 0;
589
+ }
590
+
591
+ /**
592
+ * Get indexes of all messages with the specified role
593
+ * @param sessionId - Session ID
594
+ * @param role - Role to filter by (e.g., "user", "assistant")
595
+ * @returns Array of message indexes (0-indexed, in chronological order)
596
+ */
597
+ async getMessageIndexes(sessionId: string, role: string): Promise<number[]> {
598
+ return this.store?.getMessageIndexes(sessionId, role) ?? [];
599
+ }
600
+
601
+ // ============ Hooks ============
602
+
603
+ private async executeBeforeHooks(
604
+ operation: string,
605
+ context: Record<string, unknown>
606
+ ): Promise<void> {
607
+ // operation is like "create", so key should be "before.session.create"
608
+ const hookKey = `before.session.${operation}` as keyof SessionComponentHooks;
609
+ const hooks = this.hooksConfig[hookKey] ?? [];
610
+ const hookContext = context as unknown as SessionHookContext;
611
+
612
+ // Execute registered hooks first
613
+ for (const hook of hooks) {
614
+ if (hook.before) {
615
+ await hook.before(hookContext);
616
+ }
617
+ }
618
+
619
+ // Also execute from hooks config if passed in options
620
+ const options = context.options as CreateSessionOptions | undefined;
621
+ if (options?.hooks) {
622
+ // options.hooks has keys like "before.session.create"
623
+ const hookArray = (options.hooks as Record<string, unknown>)[hookKey] as SessionComponentHooks[typeof hookKey] ?? [];
624
+ for (const hook of hookArray) {
625
+ if (hook.before) {
626
+ await hook.before(hookContext);
627
+ }
628
+ }
629
+ }
630
+ }
631
+
632
+ private async executeAfterHooks(
633
+ operation: string,
634
+ context: Record<string, unknown>
635
+ ): Promise<void> {
636
+ const hookKey = `after.session.${operation}` as keyof SessionComponentHooks;
637
+ const hooks = this.hooksConfig[hookKey] ?? [];
638
+ const hookContext = context as unknown as SessionHookContext;
639
+
640
+ // Execute registered hooks first
641
+ for (const hook of hooks) {
642
+ if (hook.after) {
643
+ await hook.after(hookContext);
644
+ }
645
+ }
646
+
647
+ // Also execute from hooks config if passed in options
648
+ const options = context.options as CreateSessionOptions | undefined;
649
+ if (options?.hooks) {
650
+ // options.hooks has keys like "after.session.create"
651
+ const hookArray = (options.hooks as Record<string, unknown>)[hookKey] as SessionComponentHooks[typeof hookKey] ?? [];
652
+ for (const hook of hookArray) {
653
+ if (hook.after) {
654
+ await hook.after(hookContext);
655
+ }
656
+ }
657
+ }
658
+ }
659
+
660
+ // ============ Helper Methods ============
661
+
662
+ private generateTitle(): string {
663
+ const template = this.config?.defaultTitleTemplate ?? "Session - {date}";
664
+ const date = new Date().toISOString().replace("T", " ").substring(0, 19);
665
+ return template.replace("{date}", date);
666
+ }
667
+
668
+ // ============ Checkpoint Management [new] ============
669
+
670
+ /**
671
+ * Set components for SummaryAgent
672
+ *
673
+ * Call this after initialization if you need compact functionality
674
+ */
675
+ setSummaryComponents(promptComponent: any, llmComponent: any): void {
676
+ this.promptComponent = promptComponent;
677
+ this.llmComponent = llmComponent;
678
+ this.summaryAgent = new SummaryAgent(promptComponent, llmComponent);
679
+ }
680
+
681
+ /**
682
+ * 初始化 SummaryAgent(如未初始化)
683
+ */
684
+ private ensureSummaryAgent(): void {
685
+ if (!this.summaryAgent) {
686
+ if (!this.promptComponent || !this.llmComponent) {
687
+ throw new Error("SummaryAgent components not initialized. Call setSummaryComponents() first.");
688
+ }
689
+ this.summaryAgent = new SummaryAgent(this.promptComponent, this.llmComponent);
690
+ }
691
+ }
692
+
693
+ /**
694
+ * 生成场景化压缩提示 (Compact Hint)
695
+ *
696
+ * 两阶段自动压缩的第一阶段:
697
+ * 根据会话历史生成场景化的压缩提示
698
+ *
699
+ * @param sessionId 会话 ID
700
+ * @returns 生成的场景化提示
701
+ */
702
+ @TracedAs("session.generateCompactHint")
703
+ async generateCompactHint(sessionId: string): Promise<string> {
704
+ const session = await this.get(sessionId);
705
+ if (!session) {
706
+ throw new Error(`Session not found: ${sessionId}`);
707
+ }
708
+
709
+ // 获取最新 checkpoint 之后的所有消息
710
+ const latestCheckpoint = this.getLatestCheckpoint(session);
711
+ const startIndex = latestCheckpoint?.messageIndex ?? 0;
712
+ const MAX_SAFE_MESSAGES = Number.MAX_SAFE_INTEGER;
713
+ const messages = await this.store!.getMessages(
714
+ sessionId,
715
+ startIndex,
716
+ MAX_SAFE_MESSAGES,
717
+ { includeArchived: false }
718
+ );
719
+
720
+ if (messages.length < 3) {
721
+ logger.warn("[SessionComponent] Not enough messages for hint generation");
722
+ return "";
723
+ }
724
+
725
+ // 确保 SummaryAgent 已初始化
726
+ this.ensureSummaryAgent();
727
+
728
+ // 生成提示
729
+ const result = await this.summaryAgent!.generateCompactHint({
730
+ messages: messages.map((m) => ({ role: m.role, content: m.content })),
731
+ sessionContext: {
732
+ // 可以扩展更多上下文信息
733
+ },
734
+ });
735
+
736
+ logger.info("[SessionComponent] Compact hint generated", {
737
+ sessionId,
738
+ hintLength: result.hint.length,
739
+ });
740
+
741
+ return result.hint;
742
+ }
743
+
744
+ /**
745
+ * Compact a session
746
+ *
747
+ * 1. Get messages from latest checkpoint to current
748
+ * 2. Call Summary Agent to generate checkpoint
749
+ * 3. Save checkpoint and archive messages
750
+ *
751
+ * 支持 scenarioHint 参数,用于指导不同场景下的压缩重点提取
752
+ */
753
+ @TracedAs("session.compact", { recordParams: true, recordResult: true })
754
+ async compact(sessionId: string, options?: CompactOptions): Promise<CompactResult> {
755
+ const session = await this.get(sessionId);
756
+ if (!session) {
757
+ throw new Error(`Session not found: ${sessionId}`);
758
+ }
759
+
760
+ // 1. Determine start position
761
+ const latestCheckpoint = this.getLatestCheckpoint(session);
762
+ const startIndex = latestCheckpoint?.messageIndex ?? 0;
763
+
764
+ // 2. Get messages to compact
765
+ // Use MAX_SAFE_INTEGER to ensure we get ALL messages (default limit=50 can cause issues)
766
+ const MAX_SAFE_MESSAGES = Number.MAX_SAFE_INTEGER;
767
+ const messages = await this.store!.getMessages(
768
+ sessionId,
769
+ startIndex,
770
+ MAX_SAFE_MESSAGES,
771
+ { includeArchived: false }
772
+ );
773
+
774
+ if (messages.length < 3) {
775
+ throw new Error("Not enough messages to compact (minimum 3 required)");
776
+ }
777
+
778
+ // 3. Extract recent messages for checkpoint context
779
+ // We need the last 2 turns of user query + assistant text response (no tool calls/results)
780
+ const recentMessages = this.extractRecentMessages(messages, 2);
781
+
782
+ // 4. Ensure Summary Agent is initialized
783
+ this.ensureSummaryAgent();
784
+
785
+ // 5. Execute before hooks
786
+ await this.executeBeforeHooks("compact", { session, options });
787
+
788
+ // 6. Call Summary Agent (with scenarioHint if provided)
789
+ const summaryResult = await this.summaryAgent!.run({
790
+ messages: messages.map((m) => ({ role: m.role, content: m.content })),
791
+ userContext: options?.summary,
792
+ outputFormat: "json",
793
+ scenarioHint: options?.scenarioHint, // Pass scenario hint to SummaryAgent
794
+ });
795
+
796
+ // 7. Generate checkpoint ID
797
+ const checkpointId = `cp_${Date.now().toString(36)}_${Math.random().toString(36).substring(2, 8)}`;
798
+
799
+ // 8. Create checkpoint object (include recentMessages for fresh history context)
800
+ const checkpoint: SessionCheckpoint = {
801
+ id: checkpointId,
802
+ messageIndex: startIndex,
803
+ title: summaryResult.title,
804
+ summary: [
805
+ `Process: ${summaryResult.processKeyPoints.join("; ")}`,
806
+ `State: ${summaryResult.currentState}`,
807
+ ].join("\n"),
808
+ processKeyPoints: summaryResult.processKeyPoints,
809
+ currentState: summaryResult.currentState,
810
+ nextSteps: summaryResult.nextSteps,
811
+ userIntents: summaryResult.userIntents,
812
+ messageCountBefore: messages.length,
813
+ createdAt: Date.now(),
814
+ type: "compact",
815
+ recentMessages, // Preserve recent messages for context
816
+ };
817
+
818
+ // 9. Save checkpoint
819
+ await this.store!.saveCheckpoint(sessionId, checkpoint);
820
+
821
+ // 10. Archive messages
822
+ await this.store!.archiveMessages(sessionId, checkpointId, messages.length);
823
+
824
+ // 11. Create and add checkpoint message (user role)
825
+ const checkpointMessage = this.createCheckpointMessage(checkpoint);
826
+ await this.store!.addMessage(sessionId, checkpointMessage);
827
+
828
+ // 12. Update session
829
+ const updatedSession = await this.get(sessionId);
830
+ const remainingMessageCount = updatedSession?.messageCount ?? 0;
831
+ const checkpointCount = updatedSession?.metadata?.checkpoints?.checkpoints?.length ?? 0;
832
+
833
+ logger.info(`[SessionComponent] Compacted session: ${sessionId}`, {
834
+ checkpointId,
835
+ archivedCount: messages.length,
836
+ remainingCount: remainingMessageCount,
837
+ hasScenarioHint: !!options?.scenarioHint,
838
+ recentMessagesCount: recentMessages.length,
839
+ });
840
+
841
+ // 13. Execute after hooks
842
+ await this.executeAfterHooks("compact", { session, checkpoint, options });
843
+
844
+ return {
845
+ checkpoint,
846
+ deletedMessageCount: messages.length,
847
+ remainingMessageCount,
848
+ checkpointCount,
849
+ };
850
+ }
851
+
852
+ /**
853
+ * Extract recent messages from the conversation for checkpoint context
854
+ *
855
+ * Extracts the last N turns of user query + assistant text response,
856
+ * excluding tool calls and tool results.
857
+ *
858
+ * @param messages - All messages in the conversation
859
+ * @param turnCount - Number of turns to extract (default: 2)
860
+ * @returns Array of recent messages formatted as { role, content }
861
+ */
862
+ @TracedAs("session.extractRecentMessages")
863
+ private extractRecentMessages(
864
+ messages: SessionMessage[],
865
+ turnCount: number = 2
866
+ ): { role: string; content: string }[] {
867
+ const recentMessages: { role: string; content: string }[] = [];
868
+ let turnsExtracted = 0;
869
+
870
+ // Iterate from the most recent messages backwards
871
+ // We need to find user + assistant pairs
872
+ let i = messages.length - 1;
873
+
874
+ while (i >= 0 && turnsExtracted < turnCount) {
875
+ const msg = messages[i];
876
+
877
+ // Skip checkpoint messages
878
+ if (msg.metadata?.isCheckpoint) {
879
+ i--;
880
+ continue;
881
+ }
882
+
883
+ // Check if this is a text response (no tool calls/results)
884
+ const hasToolParts = msg.parts?.some(
885
+ p => p.type === 'tool-call' || p.type === 'tool-result'
886
+ );
887
+
888
+ if (msg.role === 'assistant' && !hasToolParts && msg.content.trim()) {
889
+ // Found an assistant text response
890
+ recentMessages.unshift({
891
+ role: 'assistant',
892
+ content: msg.content,
893
+ });
894
+
895
+ // Look for the preceding user message
896
+ let j = i - 1;
897
+ while (j >= 0) {
898
+ const prevMsg = messages[j];
899
+ if (prevMsg.metadata?.isCheckpoint) {
900
+ j--;
901
+ continue;
902
+ }
903
+
904
+ if (prevMsg.role === 'user') {
905
+ recentMessages.unshift({
906
+ role: 'user',
907
+ content: prevMsg.content,
908
+ });
909
+ turnsExtracted++;
910
+ break;
911
+ }
912
+ j--;
913
+ }
914
+
915
+ i = j;
916
+ } else {
917
+ i--;
918
+ }
919
+ }
920
+
921
+ logger.debug("[SessionComponent] Extracted recent messages", {
922
+ turnCount,
923
+ messagesExtracted: recentMessages.length,
924
+ });
925
+
926
+ return recentMessages;
927
+ }
928
+
929
+ /**
930
+ * Preview compact operation
931
+ */
932
+ async previewCompact(sessionId: string): Promise<CompactPreview> {
933
+ const session = await this.get(sessionId);
934
+ if (!session) {
935
+ throw new Error(`Session not found: ${sessionId}`);
936
+ }
937
+
938
+ const latestCheckpoint = this.getLatestCheckpoint(session);
939
+ const startIndex = latestCheckpoint?.messageIndex ?? 0;
940
+ // Use MAX_SAFE_INTEGER to ensure we get ALL messages (default limit=50 can cause issues)
941
+ const MAX_SAFE_MESSAGES = Number.MAX_SAFE_INTEGER;
942
+ const messages = await this.store!.getMessages(sessionId, startIndex, MAX_SAFE_MESSAGES, { includeArchived: false });
943
+
944
+ // Rough estimate: ~4 tokens per character
945
+ const estimatedTokens = messages.reduce((sum, m) => sum + m.content.length * 0.25, 0);
946
+
947
+ return {
948
+ messageCountToCompact: messages.length,
949
+ estimatedCheckpointTokens: Math.round(estimatedTokens * 0.1), // Checkpoint is ~10% of original
950
+ wouldTrigger: messages.length >= (this.config?.maxMessages ?? 50),
951
+ };
952
+ }
953
+
954
+ /**
955
+ * Get all checkpoints for a session
956
+ */
957
+ async getCheckpoints(sessionId: string): Promise<SessionCheckpoint[]> {
958
+ return await this.store!.getCheckpoints(sessionId);
959
+ }
960
+
961
+ /**
962
+ * Get a specific checkpoint
963
+ */
964
+ async getCheckpoint(sessionId: string, checkpointId: string): Promise<SessionCheckpoint | undefined> {
965
+ return await this.store!.getCheckpoint(sessionId, checkpointId);
966
+ }
967
+
968
+ /**
969
+ * Delete a checkpoint and restore archived messages
970
+ */
971
+ async deleteCheckpoint(sessionId: string, checkpointId: string): Promise<boolean> {
972
+ await this.executeBeforeHooks("deleteCheckpoint", { sessionId, checkpointId });
973
+
974
+ const success = await this.store!.deleteCheckpoint(sessionId, checkpointId);
975
+
976
+ if (success) {
977
+ logger.info(`[SessionComponent] Deleted checkpoint: ${checkpointId}`);
978
+ await this.executeAfterHooks("deleteCheckpoint", { sessionId, checkpointId });
979
+ }
980
+
981
+ return success;
982
+ }
983
+
984
+ /**
985
+ * Get session context
986
+ *
987
+ * Returns messages starting from the latest checkpoint by default
988
+ */
989
+ async getContext(sessionId: string, options?: GetContextOptions): Promise<GetContextResult> {
990
+ const session = await this.get(sessionId);
991
+ if (!session) {
992
+ throw new Error(`Session not found`);
993
+ }
994
+
995
+ let contextMessages: SessionMessage[] = [];
996
+ let startCheckpoint: SessionCheckpoint | undefined;
997
+
998
+ if (options?.fullHistory) {
999
+ // Get ALL messages including archived ones (for full history view)
1000
+ const MAX_SAFE_MESSAGES = Number.MAX_SAFE_INTEGER;
1001
+ contextMessages = await this.store!.getMessages(sessionId, 0, MAX_SAFE_MESSAGES, {
1002
+ includeArchived: true,
1003
+ });
1004
+ } else {
1005
+ // Optimization: Start from latest checkpoint instead of reading all messages
1006
+ // This avoids reading archived messages that have been compacted
1007
+ const latestCheckpointId = session.metadata?.checkpoints?.latestCheckpointId;
1008
+
1009
+ if (latestCheckpointId) {
1010
+ // Get checkpoint details to find messageIndex
1011
+ startCheckpoint = await this.store!.getCheckpoint(sessionId, latestCheckpointId);
1012
+ if (startCheckpoint) {
1013
+ // Read messages from checkpoint position onwards
1014
+ const MAX_SAFE_MESSAGES = Number.MAX_SAFE_INTEGER;
1015
+ contextMessages = await this.store!.getMessages(
1016
+ sessionId,
1017
+ startCheckpoint.messageIndex,
1018
+ MAX_SAFE_MESSAGES,
1019
+ { includeArchived: false }
1020
+ );
1021
+ }
1022
+ }
1023
+
1024
+ // If no checkpoint found, read from beginning (first session)
1025
+ if (contextMessages.length === 0) {
1026
+ const MAX_SAFE_MESSAGES = Number.MAX_SAFE_INTEGER;
1027
+ contextMessages = await this.store!.getMessages(sessionId, 0, MAX_SAFE_MESSAGES, {
1028
+ includeArchived: false,
1029
+ });
1030
+ }
1031
+ }
1032
+
1033
+ // Ensure starting from user message boundary (avoids tool_call/tool_result mismatch)
1034
+ if (options?.minUserMessages && options.minUserMessages > 0) {
1035
+ const userIndexes: number[] = [];
1036
+ for (let i = 0; i < contextMessages.length; i++) {
1037
+ if (contextMessages[i].role === "user") {
1038
+ userIndexes.push(i);
1039
+ }
1040
+ }
1041
+
1042
+ if (userIndexes.length > options.minUserMessages) {
1043
+ const targetIndex = userIndexes[userIndexes.length - options.minUserMessages];
1044
+ contextMessages = contextMessages.slice(targetIndex);
1045
+ }
1046
+ }
1047
+
1048
+ // Apply message limit (after minUserMessages processing)
1049
+ if (options?.messageLimit) {
1050
+ contextMessages = contextMessages.slice(-options.messageLimit);
1051
+ }
1052
+
1053
+ // Extract checkpoint info
1054
+ const checkpoints = contextMessages
1055
+ .filter((m) => m.metadata?.isCheckpoint)
1056
+ .map((m) => m.metadata!.checkpointMeta as SessionCheckpoint);
1057
+
1058
+ // Estimate tokens
1059
+ const estimatedTokens = contextMessages.reduce(
1060
+ (sum, m) => sum + m.content.length * 0.25,
1061
+ 0
1062
+ );
1063
+
1064
+ return {
1065
+ session,
1066
+ checkpoints,
1067
+ startCheckpoint: startCheckpoint ?? checkpoints[checkpoints.length - 1],
1068
+ messages: contextMessages,
1069
+ totalMessageCount: contextMessages.length,
1070
+ activeMessageCount: contextMessages.filter((m) => !m.isArchived).length,
1071
+ archivedMessageCount: contextMessages.filter((m) => m.isArchived).length,
1072
+ estimatedTokens: Math.round(estimatedTokens),
1073
+ };
1074
+ }
1075
+
1076
+ /**
1077
+ * Check if session needs compaction
1078
+ */
1079
+ async shouldCompact(sessionId: string): Promise<boolean> {
1080
+ const session = await this.get(sessionId);
1081
+ if (!session) return false;
1082
+
1083
+ const preview = await this.previewCompact(sessionId);
1084
+ return preview.wouldTrigger;
1085
+ }
1086
+
1087
+ // ============ Search [new] ============
1088
+
1089
+ /**
1090
+ * Search messages across sessions
1091
+ *
1092
+ * Searches all messages matching the query, aggregated by session.
1093
+ * Supports SQLite FTS5 for fast full-text search.
1094
+ */
1095
+ async searchMessages(options: SearchMessagesOptions): Promise<SearchResult[]> {
1096
+ const {
1097
+ query,
1098
+ sessionId,
1099
+ limit = 10,
1100
+ maxResults = 100,
1101
+ beforeTime,
1102
+ afterTime,
1103
+ includeArchived = false,
1104
+ includeContext = false,
1105
+ contextLines = 2,
1106
+ } = options;
1107
+
1108
+ // Delegate to store for actual search
1109
+ if (!this.store) {
1110
+ return [];
1111
+ }
1112
+
1113
+ // Get all sessions or specific session
1114
+ let sessions: Session[];
1115
+ if (sessionId) {
1116
+ const session = await this.get(sessionId);
1117
+ sessions = session ? [session] : [];
1118
+ } else {
1119
+ sessions = await this.list({ limit: 1000 }); // Get all sessions
1120
+ }
1121
+
1122
+ const results: SearchResult[] = [];
1123
+ let totalMatches = 0;
1124
+
1125
+ for (const session of sessions) {
1126
+ // Check time constraints
1127
+ if (beforeTime && session.updatedAt > beforeTime) continue;
1128
+ if (afterTime && session.updatedAt < afterTime) continue;
1129
+
1130
+ // Search messages in this session
1131
+ const messages = await this.store.getMessages(session.id, 0, 1000, { includeArchived });
1132
+
1133
+ // Simple text search (case-insensitive, supports AND/OR/NOT)
1134
+ const queryTerms = parseSearchQuery(query);
1135
+ const matches: MessageMatch[] = [];
1136
+
1137
+ for (const msg of messages) {
1138
+ // Time constraints
1139
+ if (beforeTime && msg.timestamp > beforeTime) continue;
1140
+ if (afterTime && msg.timestamp < afterTime) continue;
1141
+
1142
+ // Check if message matches query
1143
+ if (matchesQuery(msg.content, queryTerms)) {
1144
+ const match: MessageMatch = {
1145
+ messageId: msg.id,
1146
+ sessionId: session.id,
1147
+ role: msg.role,
1148
+ content: msg.content,
1149
+ timestamp: msg.timestamp,
1150
+ };
1151
+
1152
+ // Add context if requested
1153
+ if (includeContext) {
1154
+ const msgIndex = messages.indexOf(msg);
1155
+ const beforeMsgs = messages.slice(Math.max(0, msgIndex - contextLines), msgIndex);
1156
+ const afterMsgs = messages.slice(msgIndex + 1, msgIndex + 1 + contextLines);
1157
+
1158
+ match.contextBefore = beforeMsgs.map(m => m.content).join("\n");
1159
+ match.contextAfter = afterMsgs.map(m => m.content).join("\n");
1160
+ }
1161
+
1162
+ matches.push(match);
1163
+
1164
+ if (matches.length >= limit) break;
1165
+ }
1166
+ }
1167
+
1168
+ if (matches.length > 0) {
1169
+ results.push({
1170
+ sessionId: session.id,
1171
+ sessionTitle: session.title,
1172
+ sessionDirectory: session.directory,
1173
+ updatedAt: session.updatedAt,
1174
+ messageCount: session.messageCount,
1175
+ matches,
1176
+ });
1177
+ totalMatches += matches.length;
1178
+
1179
+ if (totalMatches >= maxResults) break;
1180
+ }
1181
+ }
1182
+
1183
+ return results;
1184
+ }
1185
+
1186
+ // ============ Private Helper Methods [new] ============
1187
+
1188
+ /**
1189
+ * Get latest checkpoint from session
1190
+ */
1191
+ private getLatestCheckpoint(session: Session): SessionCheckpoint | undefined {
1192
+ const latestId = session.metadata?.checkpoints?.latestCheckpointId;
1193
+ if (!latestId) return undefined;
1194
+
1195
+ return session.metadata?.checkpointDetails?.[latestId];
1196
+ }
1197
+
1198
+ /**
1199
+ * Create checkpoint message (user role)
1200
+ *
1201
+ * Generates a markdown-formatted checkpoint summary as a user message,
1202
+ * allowing it to appear in the message list and provide context for
1203
+ * subsequent LLM calls.
1204
+ */
1205
+ private createCheckpointMessage(checkpoint: SessionCheckpoint): AddMessageOptions {
1206
+ const lines = [
1207
+ `# Checkpoint: ${checkpoint.title}`,
1208
+ "",
1209
+ "## Process Key Points",
1210
+ ...checkpoint.processKeyPoints.map((p, i) => `${i + 1}. ${p}`),
1211
+ "",
1212
+ "## Current State",
1213
+ checkpoint.currentState,
1214
+ "",
1215
+ ];
1216
+
1217
+ if (checkpoint.nextSteps?.length) {
1218
+ lines.push("## Next Steps");
1219
+ checkpoint.nextSteps.forEach((s, i) => lines.push(`${i + 1}. ${s}`));
1220
+ lines.push("");
1221
+ }
1222
+
1223
+ lines.push("## Messages Compacted");
1224
+ lines.push(`${checkpoint.messageCountBefore} messages compacted at ${new Date(checkpoint.createdAt).toISOString()}`);
1225
+
1226
+ const content = lines.join("\n");
1227
+
1228
+ return {
1229
+ role: "user",
1230
+ content,
1231
+ parts: [{
1232
+ type: "checkpoint" as const,
1233
+ checkpointId: checkpoint.id,
1234
+ content,
1235
+ title: checkpoint.title,
1236
+ processKeyPoints: checkpoint.processKeyPoints,
1237
+ currentState: checkpoint.currentState,
1238
+ nextSteps: checkpoint.nextSteps,
1239
+ messageCountBefore: checkpoint.messageCountBefore,
1240
+ createdAt: checkpoint.createdAt,
1241
+ }],
1242
+ metadata: {
1243
+ isCheckpoint: true,
1244
+ checkpointId: checkpoint.id,
1245
+ checkpointMeta: checkpoint,
1246
+ },
1247
+ };
1248
+ }
1249
+ }