@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.
- package/dist/index.js +99145 -0
- package/package.json +114 -0
- package/src/config/config-component.test.ts +627 -0
- package/src/config/config-component.ts +906 -0
- package/src/config/config-parser.test.ts +319 -0
- package/src/config/config-parser.ts +203 -0
- package/src/config/decentralized-config.test.ts +740 -0
- package/src/config/env-key.ts +210 -0
- package/src/config/env-source.test.ts +252 -0
- package/src/config/env-source.ts +301 -0
- package/src/config/file-source.test.ts +357 -0
- package/src/config/file-source.ts +421 -0
- package/src/config/index.ts +24 -0
- package/src/config/protocol-resolver.test.ts +217 -0
- package/src/config/protocol-resolver.ts +228 -0
- package/src/env/agent/agent-component.abort.test.ts +511 -0
- package/src/env/agent/agent-component.record-session.test.ts +349 -0
- package/src/env/agent/agent-component.test.ts +1389 -0
- package/src/env/agent/agent-component.tool-error.test.ts +327 -0
- package/src/env/agent/agent-component.ts +1711 -0
- package/src/env/agent/agent-config-registration.test.ts +226 -0
- package/src/env/agent/agent-config-registration.ts +46 -0
- package/src/env/agent/agent-reminder-plugin.integration.test.ts +243 -0
- package/src/env/agent/index.ts +10 -0
- package/src/env/agent/summary-agent.parse-hint.test.ts +360 -0
- package/src/env/agent/summary-agent.ts +508 -0
- package/src/env/agent/types.ts +536 -0
- package/src/env/commands/commands-component.test.ts +364 -0
- package/src/env/commands/commands-component.ts +604 -0
- package/src/env/commands/commands-config-registration.test.ts +198 -0
- package/src/env/commands/commands-config-registration.ts +38 -0
- package/src/env/commands/index.ts +21 -0
- package/src/env/commands/parser.test.ts +203 -0
- package/src/env/commands/parser.ts +115 -0
- package/src/env/commands/types.ts +184 -0
- package/src/env/commands-prompt-integration.test.ts +243 -0
- package/src/env/component-env.test.ts +119 -0
- package/src/env/component.ts +335 -0
- package/src/env/constants.test.ts +72 -0
- package/src/env/constants.ts +123 -0
- package/src/env/debug/debug-component.test.ts +114 -0
- package/src/env/debug/debug-component.ts +547 -0
- package/src/env/debug/formatters/index.ts +9 -0
- package/src/env/debug/formatters/repl-formatter.test.ts +139 -0
- package/src/env/debug/formatters/repl-formatter.ts +358 -0
- package/src/env/debug/formatters/trace-formatter.test.ts +119 -0
- package/src/env/debug/formatters/trace-formatter.ts +191 -0
- package/src/env/debug/formatters/tree-formatter.test.ts +107 -0
- package/src/env/debug/formatters/tree-formatter.ts +325 -0
- package/src/env/debug/index.ts +38 -0
- package/src/env/debug/parser/regex-parser.test.ts +201 -0
- package/src/env/debug/parser/regex-parser.ts +196 -0
- package/src/env/debug/parser/span-builder.test.ts +241 -0
- package/src/env/debug/parser/span-builder.ts +386 -0
- package/src/env/debug/reader/log-reader.test.ts +170 -0
- package/src/env/debug/reader/log-reader.ts +186 -0
- package/src/env/debug/reader/span-db-reader.test.ts +118 -0
- package/src/env/debug/reader/span-db-reader.ts +201 -0
- package/src/env/debug/types.test.ts +187 -0
- package/src/env/debug/types.ts +171 -0
- package/src/env/environment-init.test.ts +183 -0
- package/src/env/environment-lifecycle.test.ts +516 -0
- package/src/env/environment-service.test.ts +332 -0
- package/src/env/environment.handle-query.test.ts +96 -0
- package/src/env/environment.test.ts +232 -0
- package/src/env/environment.ts +708 -0
- package/src/env/errors.test.ts +165 -0
- package/src/env/errors.ts +157 -0
- package/src/env/event-source/event-source-agent-handler.test.ts +193 -0
- package/src/env/event-source/event-source-agent-handler.ts +111 -0
- package/src/env/event-source/event-source-component.process-cleanup.test.ts +236 -0
- package/src/env/event-source/event-source-component.stop.test.ts +346 -0
- package/src/env/event-source/event-source-component.test.ts +1207 -0
- package/src/env/event-source/event-source-component.ts +1379 -0
- package/src/env/event-source/event-source-config-registration.test.ts +242 -0
- package/src/env/event-source/event-source-config-registration.ts +37 -0
- package/src/env/event-source/event-source-integration.test.ts +320 -0
- package/src/env/event-source/event-source-platform.test.ts +630 -0
- package/src/env/event-source/types.ts +298 -0
- package/src/env/hook/global-hook-manager.ts +162 -0
- package/src/env/hook/hook-manager.test.ts +374 -0
- package/src/env/hook/hook-manager.ts +309 -0
- package/src/env/hook/index.ts +38 -0
- package/src/env/hook/types.ts +138 -0
- package/src/env/index.ts +144 -0
- package/src/env/interface.ts +203 -0
- package/src/env/llm/hooks.test.ts +293 -0
- package/src/env/llm/hooks.ts +316 -0
- package/src/env/llm/index.ts +61 -0
- package/src/env/llm/invoke-threshold-check.test.ts +88 -0
- package/src/env/llm/invoke-timeout.test.ts +54 -0
- package/src/env/llm/invoke.test.ts +71 -0
- package/src/env/llm/invoke.ts +1039 -0
- package/src/env/llm/llm-config.test.ts +523 -0
- package/src/env/llm/llm.test.ts +233 -0
- package/src/env/llm/llm.ts +568 -0
- package/src/env/llm/provider.test.ts +182 -0
- package/src/env/llm/provider.ts +108 -0
- package/src/env/llm/transform.test.ts +251 -0
- package/src/env/llm/transform.ts +286 -0
- package/src/env/llm/types.test.ts +580 -0
- package/src/env/llm/types.ts +424 -0
- package/src/env/log-trace/decorator-otel.test.ts +182 -0
- package/src/env/log-trace/decorator.ts +230 -0
- package/src/env/log-trace/index.ts +79 -0
- package/src/env/log-trace/log-trace-component.test.ts +242 -0
- package/src/env/log-trace/log-trace-component.ts +497 -0
- package/src/env/log-trace/log-trace-config-registration.test.ts +348 -0
- package/src/env/log-trace/log-trace-config-registration.ts +45 -0
- package/src/env/log-trace/logger.test.ts +149 -0
- package/src/env/log-trace/logger.ts +522 -0
- package/src/env/log-trace/opentelemetry/cli-propagation.test.ts +147 -0
- package/src/env/log-trace/opentelemetry/cli-propagation.ts +194 -0
- package/src/env/log-trace/opentelemetry/integration.test.ts +668 -0
- package/src/env/log-trace/opentelemetry/mod.ts +25 -0
- package/src/env/log-trace/opentelemetry/propagation-env.test.ts +181 -0
- package/src/env/log-trace/opentelemetry/propagation-env.ts +136 -0
- package/src/env/log-trace/opentelemetry/propagation.test.ts +259 -0
- package/src/env/log-trace/opentelemetry/propagation.ts +215 -0
- package/src/env/log-trace/opentelemetry/tracer-provider-context.test.ts +166 -0
- package/src/env/log-trace/opentelemetry/tracer-provider.test.ts +379 -0
- package/src/env/log-trace/opentelemetry/tracer-provider.ts +612 -0
- package/src/env/log-trace/span-storage.test.ts +145 -0
- package/src/env/log-trace/span-storage.ts +230 -0
- package/src/env/log-trace/trace-context.test.ts +187 -0
- package/src/env/log-trace/trace-context.ts +162 -0
- package/src/env/log-trace/types.test.ts +63 -0
- package/src/env/log-trace/types.ts +172 -0
- package/src/env/mcp/README.md +244 -0
- package/src/env/mcp/__integration__/mcp-component.integration.test.ts +373 -0
- package/src/env/mcp/config.test.ts +74 -0
- package/src/env/mcp/config.ts +116 -0
- package/src/env/mcp/index.ts +41 -0
- package/src/env/mcp/loader.test.ts +161 -0
- package/src/env/mcp/loader.ts +209 -0
- package/src/env/mcp/mcp-component.test.ts +111 -0
- package/src/env/mcp/mcp-component.ts +358 -0
- package/src/env/mcp/mcp-config-registration.test.ts +304 -0
- package/src/env/mcp/mcp-config-registration.ts +50 -0
- package/src/env/mcp/scanner.test.ts +170 -0
- package/src/env/mcp/scanner.ts +246 -0
- package/src/env/mcp/tool/adapter.test.ts +520 -0
- package/src/env/mcp/tool/adapter.ts +521 -0
- package/src/env/mcp/tool/index.ts +5 -0
- package/src/env/mcp/types.test.ts +171 -0
- package/src/env/mcp/types.ts +79 -0
- package/src/env/memory/README.md +177 -0
- package/src/env/memory/built-in/index.ts +59 -0
- package/src/env/memory/built-in/recall-memory.ts +103 -0
- package/src/env/memory/built-in/record-memory.ts +148 -0
- package/src/env/memory/index.ts +20 -0
- package/src/env/memory/memory-component.test.ts +239 -0
- package/src/env/memory/memory-component.ts +503 -0
- package/src/env/memory/memory-config-registration.test.ts +67 -0
- package/src/env/memory/memory-config-registration.ts +48 -0
- package/src/env/memory/memory-config.ts +45 -0
- package/src/env/memory/memory-file.test.ts +268 -0
- package/src/env/memory/plugin/index.ts +48 -0
- package/src/env/memory/plugin/memory-agent.test.ts +249 -0
- package/src/env/memory/plugin/memory-agent.ts +365 -0
- package/src/env/memory/plugin/memory-manager.ts +198 -0
- package/src/env/memory/plugin/memory-plugin-agent.test.ts +145 -0
- package/src/env/memory/plugin/memory-plugin.ts +210 -0
- package/src/env/memory/plugin/plugin-simplified.test.ts +51 -0
- package/src/env/memory/plugin/recall-memory.test.ts +106 -0
- package/src/env/memory/plugin/recall-memory.ts +53 -0
- package/src/env/memory/plugin/types.ts +101 -0
- package/src/env/memory/tools/memory-agent-tools.ts +228 -0
- package/src/env/memory/types.ts +85 -0
- package/src/env/paths.ts +118 -0
- package/src/env/prompt/index.ts +18 -0
- package/src/env/prompt/memory-prompts.test.ts +91 -0
- package/src/env/prompt/prompt-component.test.ts +491 -0
- package/src/env/prompt/prompt-component.ts +619 -0
- package/src/env/prompt/prompt-config-registration.test.ts +213 -0
- package/src/env/prompt/prompt-config-registration.ts +39 -0
- package/src/env/prompt/prompts-index.ts +504 -0
- package/src/env/prompt/renderer.ts +67 -0
- package/src/env/prompt/types.ts +136 -0
- package/src/env/session/hooks.ts +18 -0
- package/src/env/session/index.ts +37 -0
- package/src/env/session/search-query-parser.test.ts +425 -0
- package/src/env/session/search-query-parser.ts +171 -0
- package/src/env/session/session-checkpoint.test.ts +523 -0
- package/src/env/session/session-component.extract-recent-messages.test.ts +209 -0
- package/src/env/session/session-component.test.ts +132 -0
- package/src/env/session/session-component.ts +1249 -0
- package/src/env/session/session-config-registration.test.ts +138 -0
- package/src/env/session/session-config-registration.ts +52 -0
- package/src/env/session/session-message-converter.test.ts +763 -0
- package/src/env/session/session-message-converter.ts +415 -0
- package/src/env/session/session-message-e2e.test.ts +448 -0
- package/src/env/session/session-search.test.ts +391 -0
- package/src/env/session/session-store.test.ts +362 -0
- package/src/env/session/session-store.ts +141 -0
- package/src/env/session/storage/index.ts +6 -0
- package/src/env/session/storage/memory.ts +502 -0
- package/src/env/session/storage/sqlite.ts +794 -0
- package/src/env/session/types.ts +742 -0
- package/src/env/skill/config.ts +39 -0
- package/src/env/skill/index.ts +6 -0
- package/src/env/skill/parser.test.ts +116 -0
- package/src/env/skill/parser.ts +77 -0
- package/src/env/skill/scanner.test.ts +211 -0
- package/src/env/skill/scanner.ts +119 -0
- package/src/env/skill/skill-component.test.ts +234 -0
- package/src/env/skill/skill-component.ts +352 -0
- package/src/env/skill/skill-config-registration.test.ts +60 -0
- package/src/env/skill/skill-config-registration.ts +43 -0
- package/src/env/skill/tool/index.ts +1 -0
- package/src/env/skill/tool/skill-tool.test.ts +100 -0
- package/src/env/skill/tool/skill-tool.ts +72 -0
- package/src/env/skill/types.ts +64 -0
- package/src/env/task/delegate/delegate-tool.test.ts +498 -0
- package/src/env/task/delegate/delegate-tool.ts +1014 -0
- package/src/env/task/delegate/index.ts +18 -0
- package/src/env/task/delegate/stop-tool.test.ts +140 -0
- package/src/env/task/delegate/stop-tool.ts +119 -0
- package/src/env/task/delegate/task-events.test.ts +178 -0
- package/src/env/task/delegate/task-events.ts +143 -0
- package/src/env/task/hooks/contexts.test.ts +92 -0
- package/src/env/task/hooks/contexts.ts +192 -0
- package/src/env/task/hooks/index.ts +23 -0
- package/src/env/task/hooks/task-hook-points.test.ts +32 -0
- package/src/env/task/hooks/task-hook-points.ts +54 -0
- package/src/env/task/index.ts +7 -0
- package/src/env/task/plugins/index.ts +13 -0
- package/src/env/task/plugins/task-plugin.test.ts +74 -0
- package/src/env/task/plugins/task-plugin.ts +89 -0
- package/src/env/task/plugins/task-tag-plugin.test.ts +377 -0
- package/src/env/task/plugins/task-tag-plugin.ts +319 -0
- package/src/env/task/plugins/task-workflow-extractor.integration.test.ts +226 -0
- package/src/env/task/plugins/workflow-extractor-agent.test.ts +107 -0
- package/src/env/task/plugins/workflow-extractor-agent.ts +225 -0
- package/src/env/task/storage/index.ts +6 -0
- package/src/env/task/storage/sqlite-task-store.test.ts +283 -0
- package/src/env/task/storage/sqlite-task-store.ts +903 -0
- package/src/env/task/storage/task-search.test.ts +291 -0
- package/src/env/task/tag-service.test.ts +198 -0
- package/src/env/task/tag-service.ts +264 -0
- package/src/env/task/task-component.test.ts +193 -0
- package/src/env/task/task-component.ts +658 -0
- package/src/env/task/task-config-registration.test.ts +57 -0
- package/src/env/task/task-config-registration.ts +37 -0
- package/src/env/task/task-types.test.ts +137 -0
- package/src/env/task/tools/complete-tool.ts +44 -0
- package/src/env/task/tools/create-tool.ts +49 -0
- package/src/env/task/tools/delete-tool.ts +43 -0
- package/src/env/task/tools/get-tool.ts +59 -0
- package/src/env/task/tools/index.ts +10 -0
- package/src/env/task/tools/list-tool.ts +40 -0
- package/src/env/task/tools/operation/create-tool.ts +48 -0
- package/src/env/task/tools/operation/delete-tool.ts +43 -0
- package/src/env/task/tools/operation/get-tool.ts +43 -0
- package/src/env/task/tools/operation/index.ts +9 -0
- package/src/env/task/tools/operation/list-tool.ts +40 -0
- package/src/env/task/tools/operation/operation-tools.test.ts +274 -0
- package/src/env/task/tools/operation/operation-types.ts +75 -0
- package/src/env/task/tools/operation/update-tool.ts +47 -0
- package/src/env/task/tools/task-tools.test.ts +203 -0
- package/src/env/task/tools/task-types.test.ts +75 -0
- package/src/env/task/tools/task-types.ts +68 -0
- package/src/env/task/tools/update-tool.ts +70 -0
- package/src/env/task/types.ts +160 -0
- package/src/env/tool/built-in/bash.ts +201 -0
- package/src/env/tool/built-in/echo.ts +29 -0
- package/src/env/tool/built-in/edit-file.test.ts +136 -0
- package/src/env/tool/built-in/edit-file.ts +92 -0
- package/src/env/tool/built-in/glob.test.ts +94 -0
- package/src/env/tool/built-in/glob.ts +65 -0
- package/src/env/tool/built-in/grep.test.ts +122 -0
- package/src/env/tool/built-in/grep.ts +108 -0
- package/src/env/tool/built-in/index.ts +44 -0
- package/src/env/tool/built-in/read-file.test.ts +84 -0
- package/src/env/tool/built-in/read-file.ts +75 -0
- package/src/env/tool/built-in/write-file.test.ts +119 -0
- package/src/env/tool/built-in/write-file.ts +68 -0
- package/src/env/tool/index.ts +24 -0
- package/src/env/tool/registry.test.ts +257 -0
- package/src/env/tool/registry.ts +167 -0
- package/src/env/tool/tool-component.test.ts +559 -0
- package/src/env/tool/tool-component.ts +563 -0
- package/src/env/tool/tool-config-registration.test.ts +249 -0
- package/src/env/tool/tool-config-registration.ts +46 -0
- package/src/env/tool/types.ts +267 -0
- package/src/env/tool/validator.test.ts +143 -0
- package/src/env/tool/validator.ts +44 -0
- package/src/env/types.ts +180 -0
- package/src/env/workflow/ask-user-tool-registration.test.ts +216 -0
- package/src/env/workflow/complex-workflow.integration.test.ts +1900 -0
- package/src/env/workflow/decorators/decorator-node.ts +229 -0
- package/src/env/workflow/decorators/decorator.test.ts +196 -0
- package/src/env/workflow/decorators/edge.ts +82 -0
- package/src/env/workflow/decorators/index.ts +31 -0
- package/src/env/workflow/decorators/node-as.ts +98 -0
- package/src/env/workflow/decorators/workflow.ts +54 -0
- package/src/env/workflow/engine/dag-manager.test.ts +570 -0
- package/src/env/workflow/engine/dag-manager.ts +594 -0
- package/src/env/workflow/engine/engine.ts +1422 -0
- package/src/env/workflow/engine/event-bus.test.ts +359 -0
- package/src/env/workflow/engine/event-bus.ts +156 -0
- package/src/env/workflow/engine/executor-agent-session.test.ts +84 -0
- package/src/env/workflow/engine/executor.test.ts +619 -0
- package/src/env/workflow/engine/executor.ts +593 -0
- package/src/env/workflow/engine/index.ts +24 -0
- package/src/env/workflow/engine/node-registry.test.ts +560 -0
- package/src/env/workflow/engine/node-registry.ts +289 -0
- package/src/env/workflow/engine/resume-removed.test.ts +22 -0
- package/src/env/workflow/engine/scheduler.test.ts +715 -0
- package/src/env/workflow/engine/scheduler.ts +318 -0
- package/src/env/workflow/engine/workflow-engine.test.ts +815 -0
- package/src/env/workflow/extractor/workflow-converter.ts +306 -0
- package/src/env/workflow/fixtures.ts +380 -0
- package/src/env/workflow/index.ts +38 -0
- package/src/env/workflow/integration/run-resume-unified.test.ts +186 -0
- package/src/env/workflow/integration/service-integration.test.ts +267 -0
- package/src/env/workflow/metadata/keys.ts +12 -0
- package/src/env/workflow/nodes/agent-component-adapter.test.ts +318 -0
- package/src/env/workflow/nodes/agent-component-adapter.ts +448 -0
- package/src/env/workflow/nodes/agent-node.test.ts +371 -0
- package/src/env/workflow/nodes/agent-node.ts +598 -0
- package/src/env/workflow/nodes/ask-user-node.ts +113 -0
- package/src/env/workflow/nodes/condition-node.ts +200 -0
- package/src/env/workflow/nodes/index.ts +9 -0
- package/src/env/workflow/nodes/merge-node.ts +141 -0
- package/src/env/workflow/nodes/skill-node.test.ts +253 -0
- package/src/env/workflow/nodes/skill-node.ts +393 -0
- package/src/env/workflow/nodes/tool-node.test.ts +251 -0
- package/src/env/workflow/nodes/tool-node.ts +493 -0
- package/src/env/workflow/nodes/workflow-llm-history.test.ts +455 -0
- package/src/env/workflow/nodes/workflow-node.test.ts +315 -0
- package/src/env/workflow/nodes/workflow-node.ts +311 -0
- package/src/env/workflow/service/index.ts +27 -0
- package/src/env/workflow/service/registry.test.ts +133 -0
- package/src/env/workflow/service/registry.ts +71 -0
- package/src/env/workflow/service/workflow-service.test.ts +310 -0
- package/src/env/workflow/service/workflow-service.ts +393 -0
- package/src/env/workflow/storage/index.ts +28 -0
- package/src/env/workflow/storage/mock-repositories.ts +385 -0
- package/src/env/workflow/storage/sqlite.test.ts +179 -0
- package/src/env/workflow/storage/sqlite.ts +163 -0
- package/src/env/workflow/storage/workflow-repo.test.ts +780 -0
- package/src/env/workflow/storage/workflow-repo.ts +342 -0
- package/src/env/workflow/tools/ask-user-tool.ts +82 -0
- package/src/env/workflow/tools/index.ts +26 -0
- package/src/env/workflow/tools/run-workflow.test.ts +352 -0
- package/src/env/workflow/tools/run-workflow.ts +214 -0
- package/src/env/workflow/types/context.ts +18 -0
- package/src/env/workflow/types/decorators-types.ts +198 -0
- package/src/env/workflow/types/event.test.ts +515 -0
- package/src/env/workflow/types/event.ts +193 -0
- package/src/env/workflow/types/index.ts +49 -0
- package/src/env/workflow/types/run.test.ts +437 -0
- package/src/env/workflow/types/run.ts +173 -0
- package/src/env/workflow/types/workflow-hil.ts +114 -0
- package/src/env/workflow/types/workflow-message.test.ts +138 -0
- package/src/env/workflow/types/workflow-message.ts +196 -0
- package/src/env/workflow/types/workflow-session.test.ts +95 -0
- package/src/env/workflow/types/workflow-session.ts +59 -0
- package/src/env/workflow/types/workflow.test.ts +495 -0
- package/src/env/workflow/types/workflow.ts +195 -0
- package/src/env/workflow/types_compat.ts +51 -0
- package/src/env/workflow/utils/create-workflow.ts +47 -0
- package/src/env/workflow/utils/execution-state.ts +245 -0
- package/src/env/workflow/utils/index.ts +18 -0
- package/src/env/workflow/utils/node-registry-helper.ts +58 -0
- package/src/env/workflow/utils/recovery-validator.test.ts +460 -0
- package/src/env/workflow/utils/recovery-validator.ts +377 -0
- package/src/env/workflow/utils/session-parser.test.ts +111 -0
- package/src/env/workflow/utils/session-parser.ts +94 -0
- package/src/env/workflow/utils/session-recovery.test.ts +334 -0
- package/src/env/workflow/utils/session-recovery.ts +188 -0
- package/src/env/workflow/utils/template-resolver.test.ts +258 -0
- package/src/env/workflow/utils/template-resolver.ts +436 -0
- package/src/env/workflow/utils/validation-rules.ts +149 -0
- package/src/env/workflow/workflow-component.ts +544 -0
- package/src/index.ts +422 -0
- 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
|
+
}
|