@ai-setting/roy-agent-core 1.3.10 → 1.3.14

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 (379) hide show
  1. package/dist/config/index.js +1647 -0
  2. package/dist/index.js +12579 -89691
  3. package/package.json +19 -56
  4. package/src/config/config-component.test.ts +0 -627
  5. package/src/config/config-component.ts +0 -906
  6. package/src/config/config-parser.test.ts +0 -319
  7. package/src/config/config-parser.ts +0 -203
  8. package/src/config/decentralized-config.test.ts +0 -740
  9. package/src/config/env-key.ts +0 -210
  10. package/src/config/env-source.test.ts +0 -252
  11. package/src/config/env-source.ts +0 -301
  12. package/src/config/file-source.test.ts +0 -357
  13. package/src/config/file-source.ts +0 -421
  14. package/src/config/index.ts +0 -24
  15. package/src/config/protocol-resolver.test.ts +0 -217
  16. package/src/config/protocol-resolver.ts +0 -228
  17. package/src/env/agent/agent-component.abort.test.ts +0 -511
  18. package/src/env/agent/agent-component.record-session.test.ts +0 -349
  19. package/src/env/agent/agent-component.test.ts +0 -1389
  20. package/src/env/agent/agent-component.tool-error.test.ts +0 -327
  21. package/src/env/agent/agent-component.ts +0 -1711
  22. package/src/env/agent/agent-config-registration.test.ts +0 -226
  23. package/src/env/agent/agent-config-registration.ts +0 -46
  24. package/src/env/agent/agent-reminder-plugin.integration.test.ts +0 -243
  25. package/src/env/agent/index.ts +0 -10
  26. package/src/env/agent/summary-agent.parse-hint.test.ts +0 -360
  27. package/src/env/agent/summary-agent.ts +0 -508
  28. package/src/env/agent/types.ts +0 -536
  29. package/src/env/commands/commands-component.test.ts +0 -364
  30. package/src/env/commands/commands-component.ts +0 -604
  31. package/src/env/commands/commands-config-registration.test.ts +0 -198
  32. package/src/env/commands/commands-config-registration.ts +0 -38
  33. package/src/env/commands/index.ts +0 -21
  34. package/src/env/commands/parser.test.ts +0 -203
  35. package/src/env/commands/parser.ts +0 -115
  36. package/src/env/commands/types.ts +0 -184
  37. package/src/env/commands-prompt-integration.test.ts +0 -243
  38. package/src/env/component-env.test.ts +0 -119
  39. package/src/env/component.ts +0 -335
  40. package/src/env/constants.test.ts +0 -72
  41. package/src/env/constants.ts +0 -123
  42. package/src/env/debug/debug-component.test.ts +0 -114
  43. package/src/env/debug/debug-component.ts +0 -547
  44. package/src/env/debug/formatters/index.ts +0 -9
  45. package/src/env/debug/formatters/repl-formatter.test.ts +0 -139
  46. package/src/env/debug/formatters/repl-formatter.ts +0 -358
  47. package/src/env/debug/formatters/trace-formatter.test.ts +0 -119
  48. package/src/env/debug/formatters/trace-formatter.ts +0 -191
  49. package/src/env/debug/formatters/tree-formatter.test.ts +0 -107
  50. package/src/env/debug/formatters/tree-formatter.ts +0 -325
  51. package/src/env/debug/index.ts +0 -38
  52. package/src/env/debug/parser/regex-parser.test.ts +0 -201
  53. package/src/env/debug/parser/regex-parser.ts +0 -196
  54. package/src/env/debug/parser/span-builder.test.ts +0 -241
  55. package/src/env/debug/parser/span-builder.ts +0 -386
  56. package/src/env/debug/reader/log-reader.test.ts +0 -170
  57. package/src/env/debug/reader/log-reader.ts +0 -186
  58. package/src/env/debug/reader/span-db-reader.test.ts +0 -118
  59. package/src/env/debug/reader/span-db-reader.ts +0 -201
  60. package/src/env/debug/types.test.ts +0 -187
  61. package/src/env/debug/types.ts +0 -171
  62. package/src/env/environment-init.test.ts +0 -183
  63. package/src/env/environment-lifecycle.test.ts +0 -516
  64. package/src/env/environment-service.test.ts +0 -332
  65. package/src/env/environment.handle-query.test.ts +0 -96
  66. package/src/env/environment.test.ts +0 -232
  67. package/src/env/environment.ts +0 -708
  68. package/src/env/errors.test.ts +0 -165
  69. package/src/env/errors.ts +0 -157
  70. package/src/env/event-source/event-source-agent-handler.test.ts +0 -193
  71. package/src/env/event-source/event-source-agent-handler.ts +0 -111
  72. package/src/env/event-source/event-source-component.process-cleanup.test.ts +0 -236
  73. package/src/env/event-source/event-source-component.stop.test.ts +0 -346
  74. package/src/env/event-source/event-source-component.test.ts +0 -1207
  75. package/src/env/event-source/event-source-component.ts +0 -1379
  76. package/src/env/event-source/event-source-config-registration.test.ts +0 -242
  77. package/src/env/event-source/event-source-config-registration.ts +0 -37
  78. package/src/env/event-source/event-source-integration.test.ts +0 -320
  79. package/src/env/event-source/event-source-platform.test.ts +0 -630
  80. package/src/env/event-source/types.ts +0 -298
  81. package/src/env/hook/global-hook-manager.ts +0 -162
  82. package/src/env/hook/hook-manager.test.ts +0 -374
  83. package/src/env/hook/hook-manager.ts +0 -309
  84. package/src/env/hook/index.ts +0 -38
  85. package/src/env/hook/types.ts +0 -138
  86. package/src/env/index.ts +0 -144
  87. package/src/env/interface.ts +0 -203
  88. package/src/env/llm/hooks.test.ts +0 -293
  89. package/src/env/llm/hooks.ts +0 -316
  90. package/src/env/llm/index.ts +0 -61
  91. package/src/env/llm/invoke-threshold-check.test.ts +0 -88
  92. package/src/env/llm/invoke-timeout.test.ts +0 -54
  93. package/src/env/llm/invoke.test.ts +0 -71
  94. package/src/env/llm/invoke.ts +0 -1039
  95. package/src/env/llm/llm-config.test.ts +0 -523
  96. package/src/env/llm/llm.test.ts +0 -233
  97. package/src/env/llm/llm.ts +0 -568
  98. package/src/env/llm/provider.test.ts +0 -182
  99. package/src/env/llm/provider.ts +0 -108
  100. package/src/env/llm/transform.test.ts +0 -251
  101. package/src/env/llm/transform.ts +0 -286
  102. package/src/env/llm/types.test.ts +0 -580
  103. package/src/env/llm/types.ts +0 -424
  104. package/src/env/log-trace/decorator-otel.test.ts +0 -182
  105. package/src/env/log-trace/decorator.ts +0 -230
  106. package/src/env/log-trace/index.ts +0 -79
  107. package/src/env/log-trace/log-trace-component.test.ts +0 -242
  108. package/src/env/log-trace/log-trace-component.ts +0 -497
  109. package/src/env/log-trace/log-trace-config-registration.test.ts +0 -348
  110. package/src/env/log-trace/log-trace-config-registration.ts +0 -45
  111. package/src/env/log-trace/logger.test.ts +0 -149
  112. package/src/env/log-trace/logger.ts +0 -522
  113. package/src/env/log-trace/opentelemetry/cli-propagation.test.ts +0 -147
  114. package/src/env/log-trace/opentelemetry/cli-propagation.ts +0 -194
  115. package/src/env/log-trace/opentelemetry/integration.test.ts +0 -668
  116. package/src/env/log-trace/opentelemetry/mod.ts +0 -25
  117. package/src/env/log-trace/opentelemetry/propagation-env.test.ts +0 -181
  118. package/src/env/log-trace/opentelemetry/propagation-env.ts +0 -136
  119. package/src/env/log-trace/opentelemetry/propagation.test.ts +0 -259
  120. package/src/env/log-trace/opentelemetry/propagation.ts +0 -215
  121. package/src/env/log-trace/opentelemetry/tracer-provider-context.test.ts +0 -166
  122. package/src/env/log-trace/opentelemetry/tracer-provider.test.ts +0 -379
  123. package/src/env/log-trace/opentelemetry/tracer-provider.ts +0 -612
  124. package/src/env/log-trace/span-storage.test.ts +0 -145
  125. package/src/env/log-trace/span-storage.ts +0 -230
  126. package/src/env/log-trace/trace-context.test.ts +0 -187
  127. package/src/env/log-trace/trace-context.ts +0 -162
  128. package/src/env/log-trace/types.test.ts +0 -63
  129. package/src/env/log-trace/types.ts +0 -172
  130. package/src/env/mcp/README.md +0 -244
  131. package/src/env/mcp/__integration__/mcp-component.integration.test.ts +0 -373
  132. package/src/env/mcp/config.test.ts +0 -74
  133. package/src/env/mcp/config.ts +0 -116
  134. package/src/env/mcp/index.ts +0 -41
  135. package/src/env/mcp/loader.test.ts +0 -161
  136. package/src/env/mcp/loader.ts +0 -209
  137. package/src/env/mcp/mcp-component.test.ts +0 -111
  138. package/src/env/mcp/mcp-component.ts +0 -358
  139. package/src/env/mcp/mcp-config-registration.test.ts +0 -304
  140. package/src/env/mcp/mcp-config-registration.ts +0 -50
  141. package/src/env/mcp/scanner.test.ts +0 -170
  142. package/src/env/mcp/scanner.ts +0 -246
  143. package/src/env/mcp/tool/adapter.test.ts +0 -520
  144. package/src/env/mcp/tool/adapter.ts +0 -521
  145. package/src/env/mcp/tool/index.ts +0 -5
  146. package/src/env/mcp/types.test.ts +0 -171
  147. package/src/env/mcp/types.ts +0 -79
  148. package/src/env/memory/README.md +0 -177
  149. package/src/env/memory/built-in/index.ts +0 -59
  150. package/src/env/memory/built-in/recall-memory.ts +0 -103
  151. package/src/env/memory/built-in/record-memory.ts +0 -148
  152. package/src/env/memory/index.ts +0 -20
  153. package/src/env/memory/memory-component.test.ts +0 -239
  154. package/src/env/memory/memory-component.ts +0 -503
  155. package/src/env/memory/memory-config-registration.test.ts +0 -67
  156. package/src/env/memory/memory-config-registration.ts +0 -48
  157. package/src/env/memory/memory-config.ts +0 -45
  158. package/src/env/memory/memory-file.test.ts +0 -268
  159. package/src/env/memory/plugin/index.ts +0 -48
  160. package/src/env/memory/plugin/memory-agent.test.ts +0 -249
  161. package/src/env/memory/plugin/memory-agent.ts +0 -365
  162. package/src/env/memory/plugin/memory-manager.ts +0 -198
  163. package/src/env/memory/plugin/memory-plugin-agent.test.ts +0 -145
  164. package/src/env/memory/plugin/memory-plugin.ts +0 -210
  165. package/src/env/memory/plugin/plugin-simplified.test.ts +0 -51
  166. package/src/env/memory/plugin/recall-memory.test.ts +0 -106
  167. package/src/env/memory/plugin/recall-memory.ts +0 -53
  168. package/src/env/memory/plugin/types.ts +0 -101
  169. package/src/env/memory/tools/memory-agent-tools.ts +0 -228
  170. package/src/env/memory/types.ts +0 -85
  171. package/src/env/paths.ts +0 -118
  172. package/src/env/prompt/index.ts +0 -18
  173. package/src/env/prompt/memory-prompts.test.ts +0 -91
  174. package/src/env/prompt/prompt-component.test.ts +0 -491
  175. package/src/env/prompt/prompt-component.ts +0 -619
  176. package/src/env/prompt/prompt-config-registration.test.ts +0 -213
  177. package/src/env/prompt/prompt-config-registration.ts +0 -39
  178. package/src/env/prompt/prompts-index.ts +0 -504
  179. package/src/env/prompt/renderer.ts +0 -67
  180. package/src/env/prompt/types.ts +0 -136
  181. package/src/env/session/hooks.ts +0 -18
  182. package/src/env/session/index.ts +0 -37
  183. package/src/env/session/search-query-parser.test.ts +0 -425
  184. package/src/env/session/search-query-parser.ts +0 -171
  185. package/src/env/session/session-checkpoint.test.ts +0 -523
  186. package/src/env/session/session-component.extract-recent-messages.test.ts +0 -209
  187. package/src/env/session/session-component.test.ts +0 -132
  188. package/src/env/session/session-component.ts +0 -1249
  189. package/src/env/session/session-config-registration.test.ts +0 -138
  190. package/src/env/session/session-config-registration.ts +0 -52
  191. package/src/env/session/session-message-converter.test.ts +0 -763
  192. package/src/env/session/session-message-converter.ts +0 -415
  193. package/src/env/session/session-message-e2e.test.ts +0 -448
  194. package/src/env/session/session-search.test.ts +0 -391
  195. package/src/env/session/session-store.test.ts +0 -362
  196. package/src/env/session/session-store.ts +0 -141
  197. package/src/env/session/storage/index.ts +0 -6
  198. package/src/env/session/storage/memory.ts +0 -502
  199. package/src/env/session/storage/sqlite.ts +0 -794
  200. package/src/env/session/types.ts +0 -742
  201. package/src/env/skill/config.ts +0 -39
  202. package/src/env/skill/index.ts +0 -6
  203. package/src/env/skill/parser.test.ts +0 -116
  204. package/src/env/skill/parser.ts +0 -77
  205. package/src/env/skill/scanner.test.ts +0 -211
  206. package/src/env/skill/scanner.ts +0 -119
  207. package/src/env/skill/skill-component.test.ts +0 -234
  208. package/src/env/skill/skill-component.ts +0 -352
  209. package/src/env/skill/skill-config-registration.test.ts +0 -60
  210. package/src/env/skill/skill-config-registration.ts +0 -43
  211. package/src/env/skill/tool/index.ts +0 -1
  212. package/src/env/skill/tool/skill-tool.test.ts +0 -100
  213. package/src/env/skill/tool/skill-tool.ts +0 -72
  214. package/src/env/skill/types.ts +0 -64
  215. package/src/env/task/delegate/delegate-tool.test.ts +0 -498
  216. package/src/env/task/delegate/delegate-tool.ts +0 -1014
  217. package/src/env/task/delegate/index.ts +0 -18
  218. package/src/env/task/delegate/stop-tool.test.ts +0 -140
  219. package/src/env/task/delegate/stop-tool.ts +0 -119
  220. package/src/env/task/delegate/task-events.test.ts +0 -178
  221. package/src/env/task/delegate/task-events.ts +0 -143
  222. package/src/env/task/hooks/contexts.test.ts +0 -92
  223. package/src/env/task/hooks/contexts.ts +0 -192
  224. package/src/env/task/hooks/index.ts +0 -23
  225. package/src/env/task/hooks/task-hook-points.test.ts +0 -32
  226. package/src/env/task/hooks/task-hook-points.ts +0 -54
  227. package/src/env/task/index.ts +0 -7
  228. package/src/env/task/plugins/index.ts +0 -13
  229. package/src/env/task/plugins/task-plugin.test.ts +0 -74
  230. package/src/env/task/plugins/task-plugin.ts +0 -89
  231. package/src/env/task/plugins/task-tag-plugin.test.ts +0 -377
  232. package/src/env/task/plugins/task-tag-plugin.ts +0 -319
  233. package/src/env/task/plugins/task-workflow-extractor.integration.test.ts +0 -226
  234. package/src/env/task/plugins/workflow-extractor-agent.test.ts +0 -107
  235. package/src/env/task/plugins/workflow-extractor-agent.ts +0 -225
  236. package/src/env/task/storage/index.ts +0 -6
  237. package/src/env/task/storage/sqlite-task-store.test.ts +0 -283
  238. package/src/env/task/storage/sqlite-task-store.ts +0 -903
  239. package/src/env/task/storage/task-search.test.ts +0 -291
  240. package/src/env/task/tag-service.test.ts +0 -198
  241. package/src/env/task/tag-service.ts +0 -264
  242. package/src/env/task/task-component.test.ts +0 -193
  243. package/src/env/task/task-component.ts +0 -658
  244. package/src/env/task/task-config-registration.test.ts +0 -57
  245. package/src/env/task/task-config-registration.ts +0 -37
  246. package/src/env/task/task-types.test.ts +0 -137
  247. package/src/env/task/tools/complete-tool.ts +0 -44
  248. package/src/env/task/tools/create-tool.ts +0 -49
  249. package/src/env/task/tools/delete-tool.ts +0 -43
  250. package/src/env/task/tools/get-tool.ts +0 -59
  251. package/src/env/task/tools/index.ts +0 -10
  252. package/src/env/task/tools/list-tool.ts +0 -40
  253. package/src/env/task/tools/operation/create-tool.ts +0 -48
  254. package/src/env/task/tools/operation/delete-tool.ts +0 -43
  255. package/src/env/task/tools/operation/get-tool.ts +0 -43
  256. package/src/env/task/tools/operation/index.ts +0 -9
  257. package/src/env/task/tools/operation/list-tool.ts +0 -40
  258. package/src/env/task/tools/operation/operation-tools.test.ts +0 -274
  259. package/src/env/task/tools/operation/operation-types.ts +0 -75
  260. package/src/env/task/tools/operation/update-tool.ts +0 -47
  261. package/src/env/task/tools/task-tools.test.ts +0 -203
  262. package/src/env/task/tools/task-types.test.ts +0 -75
  263. package/src/env/task/tools/task-types.ts +0 -68
  264. package/src/env/task/tools/update-tool.ts +0 -70
  265. package/src/env/task/types.ts +0 -160
  266. package/src/env/tool/built-in/bash.ts +0 -201
  267. package/src/env/tool/built-in/echo.ts +0 -29
  268. package/src/env/tool/built-in/edit-file.test.ts +0 -136
  269. package/src/env/tool/built-in/edit-file.ts +0 -92
  270. package/src/env/tool/built-in/glob.test.ts +0 -94
  271. package/src/env/tool/built-in/glob.ts +0 -65
  272. package/src/env/tool/built-in/grep.test.ts +0 -122
  273. package/src/env/tool/built-in/grep.ts +0 -108
  274. package/src/env/tool/built-in/index.ts +0 -44
  275. package/src/env/tool/built-in/read-file.test.ts +0 -84
  276. package/src/env/tool/built-in/read-file.ts +0 -75
  277. package/src/env/tool/built-in/write-file.test.ts +0 -119
  278. package/src/env/tool/built-in/write-file.ts +0 -68
  279. package/src/env/tool/index.ts +0 -24
  280. package/src/env/tool/registry.test.ts +0 -257
  281. package/src/env/tool/registry.ts +0 -167
  282. package/src/env/tool/tool-component.test.ts +0 -559
  283. package/src/env/tool/tool-component.ts +0 -563
  284. package/src/env/tool/tool-config-registration.test.ts +0 -249
  285. package/src/env/tool/tool-config-registration.ts +0 -46
  286. package/src/env/tool/types.ts +0 -267
  287. package/src/env/tool/validator.test.ts +0 -143
  288. package/src/env/tool/validator.ts +0 -44
  289. package/src/env/types.ts +0 -180
  290. package/src/env/workflow/ask-user-tool-registration.test.ts +0 -216
  291. package/src/env/workflow/complex-workflow.integration.test.ts +0 -1900
  292. package/src/env/workflow/decorators/decorator-node.ts +0 -229
  293. package/src/env/workflow/decorators/decorator.test.ts +0 -196
  294. package/src/env/workflow/decorators/edge.ts +0 -82
  295. package/src/env/workflow/decorators/index.ts +0 -31
  296. package/src/env/workflow/decorators/node-as.ts +0 -98
  297. package/src/env/workflow/decorators/workflow.ts +0 -54
  298. package/src/env/workflow/engine/dag-manager.test.ts +0 -570
  299. package/src/env/workflow/engine/dag-manager.ts +0 -594
  300. package/src/env/workflow/engine/engine.ts +0 -1422
  301. package/src/env/workflow/engine/event-bus.test.ts +0 -359
  302. package/src/env/workflow/engine/event-bus.ts +0 -156
  303. package/src/env/workflow/engine/executor-agent-session.test.ts +0 -84
  304. package/src/env/workflow/engine/executor.test.ts +0 -619
  305. package/src/env/workflow/engine/executor.ts +0 -593
  306. package/src/env/workflow/engine/index.ts +0 -24
  307. package/src/env/workflow/engine/node-registry.test.ts +0 -560
  308. package/src/env/workflow/engine/node-registry.ts +0 -289
  309. package/src/env/workflow/engine/resume-removed.test.ts +0 -22
  310. package/src/env/workflow/engine/scheduler.test.ts +0 -715
  311. package/src/env/workflow/engine/scheduler.ts +0 -318
  312. package/src/env/workflow/engine/workflow-engine.test.ts +0 -815
  313. package/src/env/workflow/extractor/workflow-converter.ts +0 -306
  314. package/src/env/workflow/fixtures.ts +0 -380
  315. package/src/env/workflow/index.ts +0 -38
  316. package/src/env/workflow/integration/run-resume-unified.test.ts +0 -186
  317. package/src/env/workflow/integration/service-integration.test.ts +0 -267
  318. package/src/env/workflow/metadata/keys.ts +0 -12
  319. package/src/env/workflow/nodes/agent-component-adapter.test.ts +0 -318
  320. package/src/env/workflow/nodes/agent-component-adapter.ts +0 -448
  321. package/src/env/workflow/nodes/agent-node.test.ts +0 -371
  322. package/src/env/workflow/nodes/agent-node.ts +0 -598
  323. package/src/env/workflow/nodes/ask-user-node.ts +0 -113
  324. package/src/env/workflow/nodes/condition-node.ts +0 -200
  325. package/src/env/workflow/nodes/index.ts +0 -9
  326. package/src/env/workflow/nodes/merge-node.ts +0 -141
  327. package/src/env/workflow/nodes/skill-node.test.ts +0 -253
  328. package/src/env/workflow/nodes/skill-node.ts +0 -393
  329. package/src/env/workflow/nodes/tool-node.test.ts +0 -251
  330. package/src/env/workflow/nodes/tool-node.ts +0 -493
  331. package/src/env/workflow/nodes/workflow-llm-history.test.ts +0 -455
  332. package/src/env/workflow/nodes/workflow-node.test.ts +0 -315
  333. package/src/env/workflow/nodes/workflow-node.ts +0 -311
  334. package/src/env/workflow/service/index.ts +0 -27
  335. package/src/env/workflow/service/registry.test.ts +0 -133
  336. package/src/env/workflow/service/registry.ts +0 -71
  337. package/src/env/workflow/service/workflow-service.test.ts +0 -310
  338. package/src/env/workflow/service/workflow-service.ts +0 -393
  339. package/src/env/workflow/storage/index.ts +0 -28
  340. package/src/env/workflow/storage/mock-repositories.ts +0 -385
  341. package/src/env/workflow/storage/sqlite.test.ts +0 -179
  342. package/src/env/workflow/storage/sqlite.ts +0 -163
  343. package/src/env/workflow/storage/workflow-repo.test.ts +0 -780
  344. package/src/env/workflow/storage/workflow-repo.ts +0 -342
  345. package/src/env/workflow/tools/ask-user-tool.ts +0 -82
  346. package/src/env/workflow/tools/index.ts +0 -26
  347. package/src/env/workflow/tools/run-workflow.test.ts +0 -352
  348. package/src/env/workflow/tools/run-workflow.ts +0 -214
  349. package/src/env/workflow/types/context.ts +0 -18
  350. package/src/env/workflow/types/decorators-types.ts +0 -198
  351. package/src/env/workflow/types/event.test.ts +0 -515
  352. package/src/env/workflow/types/event.ts +0 -193
  353. package/src/env/workflow/types/index.ts +0 -49
  354. package/src/env/workflow/types/run.test.ts +0 -437
  355. package/src/env/workflow/types/run.ts +0 -173
  356. package/src/env/workflow/types/workflow-hil.ts +0 -114
  357. package/src/env/workflow/types/workflow-message.test.ts +0 -138
  358. package/src/env/workflow/types/workflow-message.ts +0 -196
  359. package/src/env/workflow/types/workflow-session.test.ts +0 -95
  360. package/src/env/workflow/types/workflow-session.ts +0 -59
  361. package/src/env/workflow/types/workflow.test.ts +0 -495
  362. package/src/env/workflow/types/workflow.ts +0 -195
  363. package/src/env/workflow/types_compat.ts +0 -51
  364. package/src/env/workflow/utils/create-workflow.ts +0 -47
  365. package/src/env/workflow/utils/execution-state.ts +0 -245
  366. package/src/env/workflow/utils/index.ts +0 -18
  367. package/src/env/workflow/utils/node-registry-helper.ts +0 -58
  368. package/src/env/workflow/utils/recovery-validator.test.ts +0 -460
  369. package/src/env/workflow/utils/recovery-validator.ts +0 -377
  370. package/src/env/workflow/utils/session-parser.test.ts +0 -111
  371. package/src/env/workflow/utils/session-parser.ts +0 -94
  372. package/src/env/workflow/utils/session-recovery.test.ts +0 -334
  373. package/src/env/workflow/utils/session-recovery.ts +0 -188
  374. package/src/env/workflow/utils/template-resolver.test.ts +0 -258
  375. package/src/env/workflow/utils/template-resolver.ts +0 -436
  376. package/src/env/workflow/utils/validation-rules.ts +0 -149
  377. package/src/env/workflow/workflow-component.ts +0 -544
  378. package/src/index.ts +0 -422
  379. package/src/utils/id.ts +0 -21
@@ -1,1249 +0,0 @@
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
- }