@xopcai/xopc 0.0.95 → 0.0.97
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/browser-ext/manifest.json +1 -1
- package/dist/extensions/feishu/src/outbound/media-load.js +1 -1
- package/dist/extensions/feishu/src/workflow-progress.js +1 -1
- package/dist/extensions/telegram/src/plugin.js +1 -1
- package/dist/extensions/telegram/src/routing-integration.js +2 -2
- package/dist/extensions/telegram/src/workflow-progress.js +1 -1
- package/dist/extensions/telegram/xopc.extension.json +1 -1
- package/dist/extensions/weixin/src/api/api.js +2 -2
- package/dist/extensions/weixin/src/auth/accounts.js +1 -1
- package/dist/extensions/weixin/src/cdn/upload.js +1 -1
- package/dist/extensions/weixin/src/media/data-url.js +1 -1
- package/dist/extensions/weixin/src/messaging/debug-mode.js +1 -1
- package/dist/extensions/weixin/src/messaging/inbound.js +1 -1
- package/dist/extensions/weixin/src/messaging/process-message.js +1 -1
- package/dist/extensions/weixin/src/plugin.js +1 -1
- package/dist/extensions/weixin/src/storage/sync-buf.js +1 -1
- package/dist/extensions/weixin/src/workflow-progress.js +1 -1
- package/dist/gateway/static/root/assets/{agents-CKe2LMnz.js → agents-B_YUvNi6.js} +2 -2
- package/dist/gateway/static/root/assets/{apps-page-Mi9mMIZ1.js → apps-page-BmwG5aur.js} +1 -1
- package/dist/gateway/static/root/assets/{channels-settings-BrdyC101.js → channels-settings-BiwkeKPb.js} +1 -1
- package/dist/gateway/static/root/assets/{channels-status-swr-D55Bu0nn.js → channels-status-swr-ChyN473C.js} +1 -1
- package/dist/gateway/static/root/assets/{cron-api-CPpx2l-E.js → cron-api-CvSifIfJ.js} +1 -1
- package/dist/gateway/static/root/assets/{cron-page-Bx2jB0YN.js → cron-page-BDqTDFy6.js} +1 -1
- package/dist/gateway/static/root/assets/{dist-D_AiG_Kg.js → dist-DxsUrjpy.js} +1 -1
- package/dist/gateway/static/root/assets/{extension-debug-page-6ieHsxRE.js → extension-debug-page-DV_Av5Jq.js} +1 -1
- package/dist/gateway/static/root/assets/{extension-page-B8nywHRO.js → extension-page-CwZwRhWw.js} +1 -1
- package/dist/gateway/static/root/assets/{extension-settings-page-DrskdEIV.js → extension-settings-page-Bb7TR1Se.js} +1 -1
- package/dist/gateway/static/root/assets/{fetch-B0aeeY0q.js → fetch-BLLOP2CM.js} +1 -1
- package/dist/gateway/static/root/assets/{field-primitives--9ooY8Xl.js → field-primitives-CyqVu1QR.js} +1 -1
- package/dist/gateway/static/root/assets/{heartbeat-config-api-DUZ_W1w-.js → heartbeat-config-api-Cd4M1eHt.js} +1 -1
- package/dist/gateway/static/root/assets/{index-Dj9FuxCm.js → index-0tS9lV85.js} +74 -74
- package/dist/gateway/static/root/assets/index-BJDmBCSl.css +1 -0
- package/dist/gateway/static/root/assets/{logs-page-CaXqhpKf.js → logs-page-BsAOSowN.js} +1 -1
- package/dist/gateway/static/root/assets/{note-detail-page-B91pLkEI.css → note-detail-page-D4ZIVQbk.css} +1 -1
- package/dist/gateway/static/root/assets/{note-detail-page-DYzym2B0.js → note-detail-page-Dlxoy6Ap.js} +54 -53
- package/dist/gateway/static/root/assets/{note-time-B-vSi2dR.js → note-time-B-r8yTpQ.js} +1 -1
- package/dist/gateway/static/root/assets/{notes-page-BkhWdGiT.js → notes-page-CHFcyqYW.js} +1 -1
- package/dist/gateway/static/root/assets/{sessions-page-53YFokoe.js → sessions-page-Ctu0kgt7.js} +1 -1
- package/dist/gateway/static/root/assets/{settings-advanced-gate-BaZmaklx.js → settings-advanced-gate-Dh0TyOOg.js} +1 -1
- package/dist/gateway/static/root/assets/{settings-form-section-DIJPKpTR.js → settings-form-section-DXMCEW1d.js} +1 -1
- package/dist/gateway/static/root/assets/{settings-page-Dvb230FF.js → settings-page-CIkZ7233.js} +1 -1
- package/dist/gateway/static/root/assets/{share-preview-page-CRyjTAG6.js → share-preview-page-7RV65xhJ.js} +1 -1
- package/dist/gateway/static/root/assets/{skills-page-C5ZJbfAe.js → skills-page-D_Az1SlU.js} +1 -1
- package/dist/gateway/static/root/assets/{theme-store-Cg_SuBw0.js → theme-store-e2q2yjs4.js} +1 -1
- package/dist/gateway/static/root/assets/url-DpFBIyN9.js +3 -0
- package/dist/gateway/static/root/assets/{utils-lMYoWhqo.js → utils-OA_b1q0Q.js} +1 -1
- package/dist/gateway/static/root/assets/{voice-api-key-field-Dda2pcUU.js → voice-api-key-field-SJml1hAt.js} +1 -1
- package/dist/gateway/static/root/assets/{workflow-page.utils-KIladUrU.js → workflow-page.utils-D90VVCzC.js} +1 -1
- package/dist/gateway/static/root/assets/{workflows-page-BTis4Z7Y.js → workflows-page-y7Btji0J.js} +1 -1
- package/dist/gateway/static/root/index.html +5 -5
- package/dist/package.js +1 -1
- package/dist/src/agent/agent-manager.js +12 -8
- package/dist/src/agent/agent-manager.js.map +1 -1
- package/dist/src/agent/agent-scope.d.ts +0 -1
- package/dist/src/agent/agent-scope.js +2 -5
- package/dist/src/agent/agent-scope.js.map +1 -1
- package/dist/src/agent/bootstrap/bootstrap-cache.d.ts +2 -0
- package/dist/src/agent/bootstrap/bootstrap-cache.js +10 -1
- package/dist/src/agent/bootstrap/bootstrap-cache.js.map +1 -1
- package/dist/src/agent/bootstrap/bootstrap-files.d.ts +2 -1
- package/dist/src/agent/bootstrap/bootstrap-files.js +34 -12
- package/dist/src/agent/bootstrap/bootstrap-files.js.map +1 -1
- package/dist/src/agent/bootstrap/load-bootstrap-files.d.ts +1 -2
- package/dist/src/agent/bootstrap/load-bootstrap-files.js +6 -12
- package/dist/src/agent/bootstrap/load-bootstrap-files.js.map +1 -1
- package/dist/src/agent/bootstrap/types.d.ts +5 -5
- package/dist/src/agent/context/workspace-seed.js +6 -6
- package/dist/src/agent/context/workspace-seed.js.map +1 -1
- package/dist/src/agent/context/workspace-state.d.ts +20 -0
- package/dist/src/agent/context/workspace-state.js +57 -0
- package/dist/src/agent/context/workspace-state.js.map +1 -0
- package/dist/src/agent/context/workspace-templates/AGENTS.md +0 -4
- package/dist/src/agent/embedded/index.d.ts +2 -2
- package/dist/src/agent/embedded/index.js +3 -3
- package/dist/src/agent/embedded/run-turn.js +0 -3
- package/dist/src/agent/embedded/run-turn.js.map +1 -1
- package/dist/src/agent/embedded/session-manager-init.d.ts +0 -17
- package/dist/src/agent/embedded/session-manager-init.js +1 -36
- package/dist/src/agent/embedded/session-manager-init.js.map +1 -1
- package/dist/src/agent/embedded/session-runner.d.ts +3 -12
- package/dist/src/agent/embedded/session-runner.js +12 -26
- package/dist/src/agent/embedded/session-runner.js.map +1 -1
- package/dist/src/agent/embedded/session-tool-result-guard.js +2 -4
- package/dist/src/agent/embedded/session-tool-result-guard.js.map +1 -1
- package/dist/src/agent/embedded/sqlite-hydrating-session-manager.d.ts +10 -0
- package/dist/src/agent/embedded/sqlite-hydrating-session-manager.js +34 -0
- package/dist/src/agent/embedded/sqlite-hydrating-session-manager.js.map +1 -0
- package/dist/src/agent/goals/goal-run-store.js +4 -4
- package/dist/src/agent/goals/persistent-goal-service.js +8 -15
- package/dist/src/agent/goals/persistent-goal-service.js.map +1 -1
- package/dist/src/agent/goals/post-turn.js +2 -2
- package/dist/src/agent/image/load-image-media.js +2 -2
- package/dist/src/agent/ipc/bus.js +1 -1
- package/dist/src/agent/ipc/inbox.js +2 -2
- package/dist/src/agent/ipc/socket.js +1 -1
- package/dist/src/agent/mcp/bundle-mcp-materialize.js +1 -1
- package/dist/src/agent/mcp/bundle-mcp-runtime.js +2 -2
- package/dist/src/agent/mcp/mcp-transport-config.js +1 -1
- package/dist/src/agent/mcp/mcp-transport.js +1 -1
- package/dist/src/agent/memory/builtin-memory-store.js +1 -1
- package/dist/src/agent/memory/dreaming/deep-promotion.js +1 -1
- package/dist/src/agent/memory/dreaming/events.js +1 -1
- package/dist/src/agent/memory/dreaming/last-run.js +1 -1
- package/dist/src/agent/memory/dreaming/light-sweep.js +1 -1
- package/dist/src/agent/memory/dreaming/preview.js +1 -1
- package/dist/src/agent/memory/dreaming/rem-patterns.js +1 -1
- package/dist/src/agent/memory/dreaming/short-term-store.js +1 -1
- package/dist/src/agent/memory/dreaming/utils.js +1 -1
- package/dist/src/agent/memory/plugin-discovery.js +1 -1
- package/dist/src/agent/models/manager.js +1 -1
- package/dist/src/agent/prompt/memory/index.d.ts +1 -0
- package/dist/src/agent/prompt/memory/index.js +34 -80
- package/dist/src/agent/prompt/memory/index.js.map +1 -1
- package/dist/src/agent/prompt/service-prompt-builder.js +2 -2
- package/dist/src/agent/prompt/system-prompt.js +0 -1
- package/dist/src/agent/prompt/system-prompt.js.map +1 -1
- package/dist/src/agent/reply/post-compaction-context.js +1 -1
- package/dist/src/agent/reply/workspace-boundary-read.js +1 -1
- package/dist/src/agent/sandbox/path-policy.js +2 -2
- package/dist/src/agent/service/build-direct-message-content.js +1 -1
- package/dist/src/agent/service/process-direct-one-shot.js +8 -17
- package/dist/src/agent/service/process-direct-one-shot.js.map +1 -1
- package/dist/src/agent/service/process-direct-streaming.js +14 -23
- package/dist/src/agent/service/process-direct-streaming.js.map +1 -1
- package/dist/src/agent/service.js +7 -11
- package/dist/src/agent/service.js.map +1 -1
- package/dist/src/agent/session/session-inspector.js +1 -1
- package/dist/src/agent/skills/config.js +1 -1
- package/dist/src/agent/skills/hub-hash.js +2 -2
- package/dist/src/agent/skills/hub-lock.js +1 -1
- package/dist/src/agent/skills/hub-pull.js +3 -3
- package/dist/src/agent/skills/index.js +1 -1
- package/dist/src/agent/skills/managed-store.js +1 -1
- package/dist/src/agent/skills/scanner.js +1 -1
- package/dist/src/agent/skills/skill-manage-ops.js +1 -1
- package/dist/src/agent/skills/skill-manager.js +1 -1
- package/dist/src/agent/tools/dreaming-tool.js +1 -1
- package/dist/src/agent/tools/factory.js +1 -1
- package/dist/src/agent/tools/image-generate-tool.js +1 -1
- package/dist/src/agent/tools/index.d.ts +0 -1
- package/dist/src/agent/tools/index.js +1 -2
- package/dist/src/agent/tools/send-media.js +1 -1
- package/dist/src/agent/tools/session-search-tool.d.ts +0 -1
- package/dist/src/agent/tools/session-search-tool.js +11 -6
- package/dist/src/agent/tools/session-search-tool.js.map +1 -1
- package/dist/src/agent/tools/skill-manage-tool.js +1 -1
- package/dist/src/agent/tools/tool-paths.js +1 -3
- package/dist/src/agent/tools/tool-paths.js.map +1 -1
- package/dist/src/agent/tools/workflow-tool.js +1 -1
- package/dist/src/agent/tools/write.js +1 -1
- package/dist/src/agent/workflow/catalog.js +1 -1
- package/dist/src/auth/credentials.js +3 -3
- package/dist/src/auth/profiles/store.js +1 -1
- package/dist/src/auth/sync-provider-auth.js +1 -1
- package/dist/src/browser/cache-dir-policy.js +1 -1
- package/dist/src/browser/cdp-local-launcher.js +2 -2
- package/dist/src/browser/providers/browser-ext-install.js +4 -4
- package/dist/src/browser/providers/cloakbrowser.js +4 -4
- package/dist/src/browser/providers/playwright-doctor.js +1 -1
- package/dist/src/browser/stealth.js +1 -1
- package/dist/src/channels/attachments/inbound-persist.js +1 -1
- package/dist/src/channels/attachments/outbound-tts-persist.js +1 -1
- package/dist/src/channels/outbound/persist-store.js +1 -1
- package/dist/src/channels/pairing/allow-from-file.js +1 -1
- package/dist/src/channels/pairing/pairing-store.js +2 -2
- package/dist/src/chat-commands/agent-edit.js +3 -4
- package/dist/src/chat-commands/agent-edit.js.map +1 -1
- package/dist/src/chat-commands/builtins/config.js +2 -2
- package/dist/src/chat-commands/context.js +1 -1
- package/dist/src/cli/commands/config.js +1 -1
- package/dist/src/cli/commands/doctor/checks/config-health.js +1 -1
- package/dist/src/cli/commands/doctor/checks/provider-auth.js +1 -1
- package/dist/src/cli/commands/doctor/checks/session-integrity.js +32 -95
- package/dist/src/cli/commands/doctor/checks/session-integrity.js.map +1 -1
- package/dist/src/cli/commands/doctor/checks/state-integrity.js +1 -1
- package/dist/src/cli/commands/doctor/checks/workspace-status.js +1 -1
- package/dist/src/cli/commands/extension-dev.js +1 -1
- package/dist/src/cli/commands/extension-marketplace.js +1 -1
- package/dist/src/cli/commands/extension-pack.js +1 -1
- package/dist/src/cli/commands/gateway/logs.js +1 -1
- package/dist/src/cli/commands/image.js +1 -1
- package/dist/src/cli/commands/init.js +5 -7
- package/dist/src/cli/commands/init.js.map +1 -1
- package/dist/src/cli/commands/onboard.js +0 -8
- package/dist/src/cli/commands/onboard.js.map +1 -1
- package/dist/src/cli/templates.d.ts +3 -10
- package/dist/src/cli/templates.js +4 -32
- package/dist/src/cli/templates.js.map +1 -1
- package/dist/src/cli/utils/init-workspace-core.js +2 -2
- package/dist/src/commands/agents.config.js +1 -1
- package/dist/src/config/agent-profile.js +1 -1
- package/dist/src/config/gateway-bind.js +1 -1
- package/dist/src/config/index.js +7 -8
- package/dist/src/config/index.js.map +1 -1
- package/dist/src/config/loader.js +2 -2
- package/dist/src/config/models-json.js +2 -2
- package/dist/src/config/paths-state.d.ts +3 -0
- package/dist/src/config/paths-state.js +7 -3
- package/dist/src/config/paths-state.js.map +1 -1
- package/dist/src/config/paths.d.ts +5 -36
- package/dist/src/config/paths.js +7 -52
- package/dist/src/config/paths.js.map +1 -1
- package/dist/src/config/profile.js +2 -2
- package/dist/src/config/schema.d.ts +15 -0
- package/dist/src/config/schema.js +11 -0
- package/dist/src/config/schema.js.map +1 -1
- package/dist/src/config/workspace-path.js +1 -1
- package/dist/src/cron/execution-types.d.ts +42 -0
- package/dist/src/cron/executor.js +2 -2
- package/dist/src/cron/persistence.js +1 -1
- package/dist/src/cron/run-log-store.d.ts +4 -8
- package/dist/src/cron/run-log-store.js +26 -78
- package/dist/src/cron/run-log-store.js.map +1 -1
- package/dist/src/cron/service.d.ts +3 -3
- package/dist/src/cron/service.js +2 -2
- package/dist/src/cron/service.js.map +1 -1
- package/dist/src/cron/types.d.ts +1 -42
- package/dist/src/daemon/constants.js +1 -1
- package/dist/src/daemon/install-plan.js +2 -2
- package/dist/src/daemon/launchd.js +2 -2
- package/dist/src/daemon/schtasks.js +2 -2
- package/dist/src/daemon/systemd.js +2 -2
- package/dist/src/extensions/bundle-mcp.js +1 -1
- package/dist/src/extensions/discover-extensions.js +1 -1
- package/dist/src/extensions/health.js +1 -1
- package/dist/src/extensions/loader.js +1 -1
- package/dist/src/extensions/lockfile.js +2 -2
- package/dist/src/extensions/update.js +1 -1
- package/dist/src/gateway/agents-admin.js +4 -4
- package/dist/src/gateway/agents-admin.js.map +1 -1
- package/dist/src/gateway/file-path-classifier.d.ts +0 -1
- package/dist/src/gateway/file-path-classifier.js +2 -8
- package/dist/src/gateway/file-path-classifier.js.map +1 -1
- package/dist/src/gateway/heartbeat/service.js +1 -1
- package/dist/src/gateway/hono/lib/config-payload.js +1 -1
- package/dist/src/gateway/hono/lib/extension-store.js +2 -2
- package/dist/src/gateway/hono/lib/static-ui.js +2 -2
- package/dist/src/gateway/hono/oauth.js +1 -1
- package/dist/src/gateway/hono/routes/agents.js +1 -1
- package/dist/src/gateway/hono/routes/auth-registry-extensions.js +1 -1
- package/dist/src/gateway/hono/routes/config-patch/misc.js +1 -1
- package/dist/src/gateway/hono/routes/dreaming.js +1 -1
- package/dist/src/gateway/hono/routes/host-fs.js +2 -2
- package/dist/src/gateway/hono/routes/models.js +1 -1
- package/dist/src/gateway/hono/routes/shares.js +1 -1
- package/dist/src/gateway/hono/routes/workspace.js +2 -2
- package/dist/src/gateway/lock.js +3 -3
- package/dist/src/gateway/ports.js +1 -1
- package/dist/src/gateway/service/agent-runner.js +2 -2
- package/dist/src/gateway/service/marketplace-service.js +2 -2
- package/dist/src/gateway/service.js +5 -1
- package/dist/src/gateway/service.js.map +1 -1
- package/dist/src/gateway/session-reset-service.d.ts +1 -1
- package/dist/src/gateway/session-reset-service.js +1 -1
- package/dist/src/gateway/session-reset-service.js.map +1 -1
- package/dist/src/gateway/workspace-fs-file-list.js +1 -1
- package/dist/src/heartbeat/index.js +1 -1
- package/dist/src/infra/brew.js +1 -1
- package/dist/src/infra/node-sqlite.d.ts +1 -0
- package/dist/src/infra/node-sqlite.js +17 -0
- package/dist/src/infra/node-sqlite.js.map +1 -0
- package/dist/src/infra/package-json.js +1 -1
- package/dist/src/infra/package-update-steps.js +1 -1
- package/dist/src/infra/path-env.js +2 -2
- package/dist/src/infra/restart.js +2 -2
- package/dist/src/infra/sqlite-errors.d.ts +1 -0
- package/dist/src/infra/sqlite-errors.js +77 -0
- package/dist/src/infra/sqlite-errors.js.map +1 -0
- package/dist/src/infra/stable-node-path.js +1 -1
- package/dist/src/infra/unhandled-rejections.d.ts +1 -0
- package/dist/src/infra/unhandled-rejections.js +25 -0
- package/dist/src/infra/unhandled-rejections.js.map +1 -0
- package/dist/src/infra/update-check.js +1 -1
- package/dist/src/infra/update-global.js +1 -1
- package/dist/src/infra/update-lock.js +3 -3
- package/dist/src/infra/update-runner.js +1 -1
- package/dist/src/infra/update-startup.js +2 -2
- package/dist/src/infra/warning-filter.d.ts +7 -0
- package/dist/src/infra/warning-filter.js +59 -0
- package/dist/src/infra/warning-filter.js.map +1 -0
- package/dist/src/infra/write-file-atomic.js +2 -2
- package/dist/src/notes/store.d.ts +3 -9
- package/dist/src/notes/store.js +22 -196
- package/dist/src/notes/store.js.map +1 -1
- package/dist/src/providers/auth-runtime/auth-profile-store.js +1 -1
- package/dist/src/providers/index.js +2 -2
- package/dist/src/providers/model-registry.js +1 -1
- package/dist/src/session/config-store.d.ts +6 -75
- package/dist/src/session/config-store.js +38 -144
- package/dist/src/session/config-store.js.map +1 -1
- package/dist/src/session/config-types.d.ts +15 -0
- package/dist/src/session/config-types.js +1 -0
- package/dist/src/session/index.d.ts +1 -3
- package/dist/src/session/index.js +3 -5
- package/dist/src/session/init-session-turn.d.ts +0 -6
- package/dist/src/session/init-session-turn.js +18 -18
- package/dist/src/session/init-session-turn.js.map +1 -1
- package/dist/src/session/lifecycle-timestamps.d.ts +5 -2
- package/dist/src/session/lifecycle-timestamps.js.map +1 -1
- package/dist/src/session/{parity/load-jsonl-entries.js → load-jsonl-entries.js} +1 -1
- package/dist/src/session/load-jsonl-entries.js.map +1 -0
- package/dist/src/session/manager.d.ts +5 -3
- package/dist/src/session/manager.js +1 -5
- package/dist/src/session/manager.js.map +1 -1
- package/dist/src/session/resolve-session.d.ts +3 -6
- package/dist/src/session/resolve-session.js +26 -31
- package/dist/src/session/resolve-session.js.map +1 -1
- package/dist/src/session/session-context-for-llm.js +5 -1
- package/dist/src/session/session-context-for-llm.js.map +1 -1
- package/dist/src/session/session-id.js +12 -0
- package/dist/src/session/session-id.js.map +1 -0
- package/dist/src/session/session-title.js +2 -2
- package/dist/src/session/session-workspace.d.ts +1 -1
- package/dist/src/session/session-workspace.js.map +1 -1
- package/dist/src/session/store.d.ts +14 -63
- package/dist/src/session/store.js +172 -847
- package/dist/src/session/store.js.map +1 -1
- package/dist/src/session/stored-rows-to-file-entries.d.ts +11 -0
- package/dist/src/session/stored-rows-to-file-entries.js +95 -0
- package/dist/src/session/stored-rows-to-file-entries.js.map +1 -0
- package/dist/src/session/transcript-events.d.ts +1 -2
- package/dist/src/session/transcript-events.js +5 -12
- package/dist/src/session/transcript-events.js.map +1 -1
- package/dist/src/session/transcript-format.d.ts +1 -1
- package/dist/src/session/transcript-format.js.map +1 -1
- package/dist/src/session/transcript-stats.d.ts +1 -0
- package/dist/src/session/transcript-stats.js +10 -0
- package/dist/src/session/transcript-stats.js.map +1 -0
- package/dist/src/share/share-auto.js +2 -2
- package/dist/src/share/share-store.js +3 -3
- package/dist/src/share/share-thumbnail.js +2 -2
- package/dist/src/share/share-zip.js +1 -1
- package/dist/src/share/site-share-store.js +3 -3
- package/dist/src/share/site-static-serve.js +1 -1
- package/dist/src/storage/sqlite/config-repository.d.ts +6 -0
- package/dist/src/storage/sqlite/config-repository.js +56 -0
- package/dist/src/storage/sqlite/config-repository.js.map +1 -0
- package/dist/src/storage/sqlite/connection.d.ts +38 -0
- package/dist/src/storage/sqlite/connection.js +258 -0
- package/dist/src/storage/sqlite/connection.js.map +1 -0
- package/dist/src/storage/sqlite/cron-run-repository.d.ts +5 -0
- package/dist/src/storage/sqlite/cron-run-repository.js +97 -0
- package/dist/src/storage/sqlite/cron-run-repository.js.map +1 -0
- package/dist/src/storage/sqlite/fts.d.ts +2 -0
- package/dist/src/storage/sqlite/fts.js +11 -0
- package/dist/src/storage/sqlite/fts.js.map +1 -0
- package/dist/src/storage/sqlite/index.d.ts +12 -0
- package/dist/src/storage/sqlite/index.js +13 -0
- package/dist/src/storage/sqlite/memory-index-repository.d.ts +18 -0
- package/dist/src/storage/sqlite/memory-index-repository.js +132 -0
- package/dist/src/storage/sqlite/memory-index-repository.js.map +1 -0
- package/dist/src/storage/sqlite/notes-repository.d.ts +11 -0
- package/dist/src/storage/sqlite/notes-repository.js +191 -0
- package/dist/src/storage/sqlite/notes-repository.js.map +1 -0
- package/dist/src/storage/sqlite/paths.d.ts +1 -0
- package/dist/src/storage/sqlite/paths.js +7 -0
- package/dist/src/storage/sqlite/paths.js.map +1 -0
- package/dist/src/storage/sqlite/row-mappers.d.ts +82 -0
- package/dist/src/storage/sqlite/row-mappers.js +164 -0
- package/dist/src/storage/sqlite/row-mappers.js.map +1 -0
- package/dist/src/storage/sqlite/schema.d.ts +5 -0
- package/dist/src/storage/sqlite/schema.js +43 -0
- package/dist/src/storage/sqlite/schema.js.map +1 -0
- package/dist/src/storage/sqlite/schema.sql +195 -0
- package/dist/src/storage/sqlite/session-metadata.d.ts +8 -0
- package/dist/src/storage/sqlite/session-metadata.js +83 -0
- package/dist/src/storage/sqlite/session-metadata.js.map +1 -0
- package/dist/src/storage/sqlite/session-repository.d.ts +29 -0
- package/dist/src/storage/sqlite/session-repository.js +268 -0
- package/dist/src/storage/sqlite/session-repository.js.map +1 -0
- package/dist/src/storage/sqlite/transaction.d.ts +11 -0
- package/dist/src/storage/sqlite/transaction.js +115 -0
- package/dist/src/storage/sqlite/transaction.js.map +1 -0
- package/dist/src/storage/sqlite/transcript-repository.d.ts +34 -0
- package/dist/src/storage/sqlite/transcript-repository.js +241 -0
- package/dist/src/storage/sqlite/transcript-repository.js.map +1 -0
- package/dist/src/tui/clipboard-image.js +3 -3
- package/dist/src/tui/theme-manager.js +1 -1
- package/dist/src/tui/tui-keybindings-file.js +1 -1
- package/dist/src/tui/tui-scoped-models.js +2 -2
- package/dist/src/tui/tui-settings.js +1 -1
- package/dist/src/tui/tui.js +3 -3
- package/dist/src/tunnel/frpc-binary.js +3 -3
- package/dist/src/tunnel/frpc-config.js +1 -1
- package/dist/src/tunnel/frpc-extract.js +1 -1
- package/dist/src/tunnel/tunnel-state.js +1 -1
- package/dist/src/utils/logger/audit.js +1 -1
- package/dist/src/utils/logger/log-store.js +1 -1
- package/dist/src/utils/logger/rotation.js +1 -1
- package/dist/src/voice/tts/audio.js +1 -1
- package/dist/src/voice/tts/providers/edge-speech.js +2 -2
- package/dist/src/workflows/service/workflow-session-bridge.js +41 -64
- package/dist/src/workflows/service/workflow-session-bridge.js.map +1 -1
- package/dist/src/workflows/store/event-store.js +1 -1
- package/dist/src/workflows/store/run-store.js +1 -1
- package/package.json +2 -2
- package/dist/gateway/static/root/assets/index-Bj_l8QDp.css +0 -1
- package/dist/gateway/static/root/assets/url-BHHmdJYc.js +0 -3
- package/dist/src/agent/context/workspace-templates/BOOTSTRAP.md +0 -61
- package/dist/src/agent/embedded/session-manager-cache.d.ts +0 -19
- package/dist/src/agent/embedded/session-manager-cache.js +0 -48
- package/dist/src/agent/embedded/session-manager-cache.js.map +0 -1
- package/dist/src/session/parity/artifacts.d.ts +0 -16
- package/dist/src/session/parity/artifacts.js +0 -80
- package/dist/src/session/parity/artifacts.js.map +0 -1
- package/dist/src/session/parity/jsonl-transcript-io.d.ts +0 -54
- package/dist/src/session/parity/jsonl-transcript-io.js +0 -236
- package/dist/src/session/parity/jsonl-transcript-io.js.map +0 -1
- package/dist/src/session/parity/load-jsonl-entries.js.map +0 -1
- package/dist/src/session/parity/session-id.js +0 -18
- package/dist/src/session/parity/session-id.js.map +0 -1
- package/dist/src/session/parity/sessions-json-cache.d.ts +0 -14
- package/dist/src/session/parity/sessions-json-cache.js +0 -98
- package/dist/src/session/parity/sessions-json-cache.js.map +0 -1
- package/dist/src/session/parity/sessions-json-file-read.d.ts +0 -6
- package/dist/src/session/parity/sessions-json-file-read.js +0 -19
- package/dist/src/session/parity/sessions-json-file-read.js.map +0 -1
- package/dist/src/session/parity/sessions-json-file.d.ts +0 -11
- package/dist/src/session/parity/sessions-json-file.js +0 -52
- package/dist/src/session/parity/sessions-json-file.js.map +0 -1
- package/dist/src/session/parity/sessions-json-patch.d.ts +0 -14
- package/dist/src/session/parity/sessions-json-patch.js +0 -40
- package/dist/src/session/parity/sessions-json-patch.js.map +0 -1
- package/dist/src/session/parity/transcript-file-lock.d.ts +0 -22
- package/dist/src/session/parity/transcript-file-lock.js +0 -142
- package/dist/src/session/parity/transcript-file-lock.js.map +0 -1
- package/dist/src/session/parity/transcript-pagination.d.ts +0 -29
- package/dist/src/session/parity/transcript-pagination.js +0 -132
- package/dist/src/session/parity/transcript-pagination.js.map +0 -1
- package/dist/src/session/parity/transcript-paths.d.ts +0 -13
- package/dist/src/session/parity/transcript-paths.js +0 -64
- package/dist/src/session/parity/transcript-paths.js.map +0 -1
- package/dist/src/session/parity/xopc-session-disk-entry.d.ts +0 -22
- package/dist/src/session/search-index-cache.d.ts +0 -6
- package/dist/src/session/search-index-cache.js +0 -44
- package/dist/src/session/search-index-cache.js.map +0 -1
- package/dist/src/session/search-index.d.ts +0 -20
- package/dist/src/session/search-index.js +0 -124
- package/dist/src/session/search-index.js.map +0 -1
- /package/dist/src/{session/parity/xopc-session-disk-entry.js → cron/execution-types.js} +0 -0
- /package/dist/src/session/{parity/load-jsonl-entries.d.ts → load-jsonl-entries.d.ts} +0 -0
- /package/dist/src/session/{parity/session-id.d.ts → session-id.d.ts} +0 -0
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"store.js","names":["parseRoutingSessionKey"],"sources":["../../../src/session/store.ts"],"sourcesContent":["import { randomUUID } from 'node:crypto';\nimport { performance } from 'node:perf_hooks';\nimport { copyFile, mkdir, readdir, stat, unlink } from 'fs/promises';\nimport { join } from 'path';\nimport { existsSync } from 'fs';\n\nimport type { AgentMessage } from '@earendil-works/pi-agent-core';\nimport type { CompactionEntry } from '@earendil-works/pi-coding-agent';\n\nimport { loadEntriesFromFile } from './parity/load-jsonl-entries.js';\n\nimport type { Config } from '../config/schema.js';\nimport { resolveSessionsDir, resolveStateDir, FILENAMES } from '../config/paths.js';\nimport { resolveDefaultAgentId, listAgentEntries } from '../agent/agent-scope.js';\nimport { resolveEffectiveAgentProfileForSession } from '../config/agent-profile.js';\nimport { readPostCompactionContext } from '../agent/reply/post-compaction-context.js';\nimport { parseSessionKey as parseRoutingSessionKey } from '../routing/session-key.js';\nimport { createLogger } from '../utils/logger.js';\nimport { SessionCompactor, type CompactionConfig, type CompactionResult } from '../agent/memory/compaction.js';\nimport { SlidingWindow, type WindowConfig } from '../agent/memory/window.js';\nimport { invalidateSessionSearchIndexCache } from './search-index-cache.js';\nimport type { TranscriptCompactionRecord, XopcSessionTranscriptV1 } from './transcript-format.js';\nimport {\n mergeLlmMessagesPreservingContextRows,\n type TranscriptStoredRow,\n type XopcTranscriptContextEntry,\n} from './session-context-for-llm.js';\nimport { normalizeCompactionCheckpointId } from './compaction-checkpoints.js';\nimport type {\n SessionMetadata,\n SessionDetail,\n SessionListQuery,\n PaginatedResult,\n GlobalSessionStats,\n ExportFormat,\n SessionExport,\n SessionTranscriptSummary,\n CompactionCheckpointSummary,\n CompactionCheckpointDetail,\n} from './types.js';\nimport { SessionStatus } from './types.js';\nimport type { Message } from './types.js';\nimport { parseCompactionCheckpointTranscriptFileName } from './parity/artifacts.js';\nimport { archiveFileOnDisk, resolveSessionFilePath, resolveSessionTranscriptPathInDir } from './parity/transcript-paths.js';\nimport { validateSessionId } from './parity/session-id.js';\nimport { readSessionsJsonFile, withSessionsJsonLock } from './parity/sessions-json-file.js';\nimport {\n buildSessionsJsonStatsPatch,\n incrementSessionsJsonStatsForAppend,\n isAppendOnlyLlmTranscriptMessage,\n patchSessionsJsonEntryStats,\n} from './parity/sessions-json-patch.js';\nimport {\n countTranscriptMessageRows,\n readDisplayMessagePageFromTranscriptFile,\n} from './parity/transcript-pagination.js';\nimport type { XopcSessionDiskEntry } from './parity/xopc-session-disk-entry.js';\nimport {\n appendPiTranscriptContextEntry,\n persistMergedTranscriptRows,\n readTranscriptRowsFromFile,\n rowsToLlmMessages,\n writeTranscriptJsonl,\n} from './parity/jsonl-transcript-io.js';\nimport type { SessionTranscriptUpdate } from './transcript-events.js';\n\nconst log = createLogger('SessionStore');\n\nconst INDEX_VERSION = '1.0';\nconst DELETED_MARKER = '.jsonl.deleted.';\nconst ALL_SESSIONS_MAP_CACHE_TTL_MS = 2_000;\n\nexport interface SessionStoreOptions {\n config: Config;\n agentId?: string;\n sessionsDir?: string;\n}\n\nexport class SessionStore {\n private sessionsDir: string;\n private archiveDir: string;\n private storePath: string;\n private window: SlidingWindow;\n private compactor: SessionCompactor;\n private storeMutationDepth = 0;\n private storeMutationChain: Promise<void> = Promise.resolve();\n private allSessionsMapCache?: {\n expiresAtMs: number;\n map: Record<string, XopcSessionDiskEntry>;\n };\n /** Cache of per-agent sessions dirs to avoid re-resolution on every call. */\n private agentSessionsDirCache = new Map<string, string>();\n\n constructor(\n private options: SessionStoreOptions,\n windowConfig?: Partial<WindowConfig>,\n compactionConfig?: Partial<CompactionConfig>,\n ) {\n const agentId = options.agentId ?? resolveDefaultAgentId(options.config);\n this.sessionsDir = options.sessionsDir ?? resolveSessionsDir(options.config, agentId);\n this.archiveDir = join(this.sessionsDir, 'archive');\n this.storePath = join(this.sessionsDir, FILENAMES.SESSIONS_MAP);\n this.window = new SlidingWindow(windowConfig);\n this.compactor = new SessionCompactor(compactionConfig);\n }\n\n getSessionsRoot(): string {\n return this.sessionsDir;\n }\n\n /**\n * OpenClaw-aligned: resolve the sessions directory for a given session key.\n * Extracts agentId from the session key and routes to `agents/<agentId>/sessions/`.\n * Falls back to the default sessions directory when agentId cannot be parsed\n * or when `sessionsDir` was explicitly provided in options.\n */\n private resolveSessionsDirForKey(sessionKey: string): string {\n if (this.options.sessionsDir) {\n return this.sessionsDir;\n }\n const parsed = parseRoutingSessionKey(sessionKey);\n if (!parsed) {\n return this.sessionsDir;\n }\n const agentId = parsed.agentId;\n const cached = this.agentSessionsDirCache.get(agentId);\n if (cached) {\n return cached;\n }\n const resolved = resolveSessionsDir(this.options.config, agentId);\n this.agentSessionsDirCache.set(agentId, resolved);\n return resolved;\n }\n\n private resolveStorePathForKey(sessionKey: string): string {\n return join(this.resolveSessionsDirForKey(sessionKey), FILENAMES.SESSIONS_MAP);\n }\n\n private async runStoreMutation<T>(fn: () => Promise<T>): Promise<T> {\n if (this.storeMutationDepth > 0) {\n return fn();\n }\n const run = this.storeMutationChain.then(async () => {\n this.storeMutationDepth++;\n try {\n return await fn();\n } finally {\n this.storeMutationDepth--;\n }\n });\n this.storeMutationChain = run.then(() => undefined).catch(() => undefined);\n return run as Promise<T>;\n }\n\n async initialize(): Promise<void> {\n await mkdir(this.sessionsDir, { recursive: true });\n await mkdir(this.archiveDir, { recursive: true });\n if (!existsSync(this.storePath)) {\n await withSessionsJsonLock(this.storePath, async () => undefined);\n }\n log.debug('Session store initialized (sessions.json + JSONL)');\n }\n\n private transcriptPathForEntry(entry: XopcSessionDiskEntry, sessionsDir?: string): string {\n return resolveSessionFilePath(entry.sessionId, entry, { sessionsDir: sessionsDir ?? this.sessionsDir });\n }\n\n private async readMapForKey(sessionKey: string): Promise<Record<string, XopcSessionDiskEntry>> {\n const storePath = this.resolveStorePathForKey(sessionKey);\n return readSessionsJsonFile<XopcSessionDiskEntry>(storePath);\n }\n\n private async readMap(): Promise<Record<string, XopcSessionDiskEntry>> {\n return readSessionsJsonFile<XopcSessionDiskEntry>(this.storePath);\n }\n\n private invalidateAllSessionsMapCache(): void {\n this.allSessionsMapCache = undefined;\n }\n\n private async discoverSessionMapPaths(): Promise<Array<{ agentId: string; mapPath: string }>> {\n const agents = listAgentEntries(this.options.config);\n const defaultId = resolveDefaultAgentId(this.options.config);\n const agentIds = new Set<string>([defaultId, ...agents.map((agent) => agent.id)]);\n\n const agentsRoot = join(resolveStateDir(process.env), 'agents');\n if (existsSync(agentsRoot)) {\n const entries = await readdir(agentsRoot, { withFileTypes: true }).catch(() => []);\n for (const entry of entries) {\n if (entry.isDirectory()) {\n agentIds.add(entry.name);\n }\n }\n }\n\n return [...agentIds].map((agentId) => ({\n agentId,\n mapPath: join(resolveSessionsDir(this.options.config, agentId), FILENAMES.SESSIONS_MAP),\n }));\n }\n\n /**\n * Unified cross-agent aggregation entry for global session views.\n * Reads configured agents plus existing per-agent session maps under the state directory.\n */\n private async readAllMaps(): Promise<Record<string, XopcSessionDiskEntry>> {\n if (this.options.sessionsDir) {\n return this.readMap();\n }\n\n const nowMs = Date.now();\n if (this.allSessionsMapCache && this.allSessionsMapCache.expiresAtMs > nowMs) {\n log.debug(\n { sessionCount: Object.keys(this.allSessionsMapCache.map).length },\n 'All session maps cache hit',\n );\n return this.allSessionsMapCache.map;\n }\n\n const startedAt = performance.now();\n const paths = await this.discoverSessionMapPaths();\n const merged: Record<string, XopcSessionDiskEntry> = {};\n let scannedMapCount = 0;\n for (const { mapPath } of paths) {\n if (!existsSync(mapPath)) {\n continue;\n }\n const map = await readSessionsJsonFile<XopcSessionDiskEntry>(mapPath);\n Object.assign(merged, map);\n scannedMapCount++;\n }\n\n this.allSessionsMapCache = {\n expiresAtMs: nowMs + ALL_SESSIONS_MAP_CACHE_TTL_MS,\n map: merged,\n };\n log.debug(\n {\n candidateAgentCount: paths.length,\n scannedMapCount,\n sessionCount: Object.keys(merged).length,\n durationMs: Math.round(performance.now() - startedAt),\n },\n 'All session maps scanned',\n );\n return merged;\n }\n\n private async getDiskEntry(sessionKey: string): Promise<XopcSessionDiskEntry | undefined> {\n const map = await this.readMapForKey(sessionKey);\n return map[sessionKey];\n }\n\n private buildDefaultMetadata(key: string): SessionMetadata {\n const { channel, chatId } = this.parseSessionKey(key);\n const routing = this.extractRoutingFromKey(key);\n const isCronSession = channel === 'cron';\n const isHeartbeatSession = channel === 'heartbeat';\n const now = new Date().toISOString();\n return {\n key,\n status: SessionStatus.ACTIVE,\n tags: [],\n createdAt: now,\n updatedAt: now,\n lastAccessedAt: now,\n messageCount: 0,\n estimatedTokens: 0,\n compactedCount: 0,\n sourceChannel: channel,\n sourceChatId: chatId,\n routing,\n ...(isCronSession\n ? { sessionType: 'cron', customData: { cronJobId: chatId } }\n : {}),\n ...(isHeartbeatSession\n ? { sessionType: 'heartbeat', customData: { heartbeatTarget: chatId } }\n : {}),\n stats: { messageCount: 0, tokenCount: 0 },\n };\n }\n\n private parseSessionKey(key: string): { channel: string; chatId: string } {\n const parts = key.split(':');\n if (parts.length >= 2 && parts[0] === 'heartbeat') {\n return { channel: 'heartbeat', chatId: parts.slice(1).join(':') };\n }\n const parsed = parseRoutingSessionKey(key);\n if (parsed) {\n if (parsed.source === 'cron') {\n return { channel: 'cron', chatId: parsed.peerId };\n }\n return {\n channel: parsed.source,\n chatId: [parsed.accountId, parsed.peerKind, parsed.peerId].join(':'),\n };\n }\n return { channel: 'unknown', chatId: key };\n }\n\n private extractRoutingFromKey(key: string): SessionMetadata['routing'] {\n const parsed = parseRoutingSessionKey(key);\n if (!parsed) {\n return undefined;\n }\n return {\n agentId: parsed.agentId?.toLowerCase() || 'main',\n source: parsed.source?.toLowerCase() || 'unknown',\n accountId: parsed.accountId?.toLowerCase() || 'default',\n peerKind: parsed.peerKind?.toLowerCase() || 'dm',\n peerId: parsed.peerId?.toLowerCase() || 'unknown',\n threadId: parsed.threadId,\n scopeId: parsed.scopeId,\n };\n }\n\n /** Resolve on-disk transcript path; creates session row + empty JSONL when missing. */\n async resolveTranscriptPath(\n sessionKey: string,\n ): Promise<{ sessionId: string; absPath: string; sessionsDir: string }> {\n const entry = await this.ensureSession(sessionKey);\n const sessionsDir = this.resolveSessionsDirForKey(sessionKey);\n const absPath = this.transcriptPathForEntry(entry, sessionsDir);\n return { sessionId: entry.sessionId, absPath, sessionsDir };\n }\n\n /** Ensure sessions.json has an entry and transcript file exist for `sessionKey`. */\n private async ensureSession(sessionKey: string): Promise<XopcSessionDiskEntry> {\n const keyStorePath = this.resolveStorePathForKey(sessionKey);\n const keySessionsDir = this.resolveSessionsDirForKey(sessionKey);\n await mkdir(keySessionsDir, { recursive: true });\n let changed = false;\n const entry = await withSessionsJsonLock(keyStorePath, async (map) => {\n const existing = map[sessionKey] as XopcSessionDiskEntry | undefined;\n if (existing?.pluginExtensions?.xopc?.metadata) {\n return existing;\n }\n let nextEntry = existing;\n if (!nextEntry) {\n const sessionId = randomUUID();\n validateSessionId(sessionId);\n const sessionFile = `${sessionId}.jsonl`;\n const now = Date.now();\n const metadata = this.buildDefaultMetadata(sessionKey);\n metadata.transcriptId = sessionId;\n nextEntry = {\n sessionId,\n updatedAt: now,\n sessionStartedAt: now,\n sessionFile,\n pluginExtensions: { xopc: { metadata } },\n };\n map[sessionKey] = nextEntry as Record<string, unknown>;\n const abs = resolveSessionTranscriptPathInDir(sessionId, keySessionsDir);\n await writeTranscriptJsonl({\n absPath: abs,\n sessionId,\n cwd: process.cwd(),\n rows: [],\n });\n changed = true;\n } else if (!nextEntry.pluginExtensions?.xopc?.metadata) {\n const metadata = this.buildDefaultMetadata(sessionKey);\n metadata.transcriptId = nextEntry.sessionId;\n nextEntry.pluginExtensions = { xopc: { metadata } };\n map[sessionKey] = nextEntry as Record<string, unknown>;\n changed = true;\n }\n return nextEntry!;\n });\n if (changed) {\n this.invalidateAllSessionsMapCache();\n }\n return entry;\n }\n\n private metadataFromEntry(sessionKey: string, entry: XopcSessionDiskEntry): SessionMetadata {\n const base = entry.pluginExtensions?.xopc?.metadata ?? this.buildDefaultMetadata(sessionKey);\n const { channel: keySource, chatId: keyChatId } = this.parseSessionKey(sessionKey);\n const diskSc = typeof base.sourceChannel === 'string' ? base.sourceChannel.trim() : '';\n const diskChat = typeof base.sourceChatId === 'string' ? base.sourceChatId.trim() : '';\n return {\n ...base,\n key: sessionKey,\n transcriptId: entry.sessionId,\n sourceChannel: diskSc || keySource,\n sourceChatId: diskChat || keyChatId,\n };\n }\n\n async getByAgent(agentId: string): Promise<SessionMetadata[]> {\n const map = await this.readAllMaps();\n const out: SessionMetadata[] = [];\n for (const [key, e] of Object.entries(map)) {\n const m = this.metadataFromEntry(key, e);\n if (m.routing?.agentId?.toLowerCase() === agentId.toLowerCase()) {\n out.push(m);\n }\n }\n return out;\n }\n\n async getByAccount(accountId: string): Promise<SessionMetadata[]> {\n const map = await this.readAllMaps();\n const out: SessionMetadata[] = [];\n for (const [key, e] of Object.entries(map)) {\n const m = this.metadataFromEntry(key, e);\n if (m.routing?.accountId === accountId) {\n out.push(m);\n }\n }\n return out;\n }\n\n async getByPeer(peerKind: string, peerId: string): Promise<SessionMetadata[]> {\n const map = await this.readAllMaps();\n const out: SessionMetadata[] = [];\n for (const [key, e] of Object.entries(map)) {\n const m = this.metadataFromEntry(key, e);\n if (m.routing?.peerKind === peerKind && m.routing.peerId === peerId) {\n out.push(m);\n }\n }\n return out;\n }\n\n async getMainSession(channel: string, accountId: string): Promise<SessionMetadata | null> {\n const map = await this.readAllMaps();\n for (const [key, e] of Object.entries(map)) {\n const m = this.metadataFromEntry(key, e);\n if (\n m.routing?.source === channel &&\n m.routing.accountId === accountId &&\n m.routing.peerKind === 'dm' &&\n m.routing.peerId === 'main'\n ) {\n return m;\n }\n }\n return null;\n }\n\n async refreshIndex(): Promise<void> {\n /* no-op: sessions.json is authoritative */\n }\n\n async list(query: SessionListQuery = {}): Promise<PaginatedResult<SessionMetadata>> {\n const map = await this.readAllMaps();\n let sessions = Object.entries(map).map(([k, e]) => this.metadataFromEntry(k, e));\n\n if (query.status) {\n const statuses = Array.isArray(query.status) ? query.status : [query.status];\n sessions = sessions.filter((s) => statuses.includes(s.status));\n }\n if (query.channel) {\n const rawChannels = query.channel\n .split(',')\n .map((c) => c.trim().toLowerCase())\n .filter(Boolean);\n /**\n * `ui` is a legacy console source; treat as webchat when filtering web sessions.\n * `webui` matches slash-command normalization to `gateway` (see `chat-commands/session-key.ts`).\n */\n const channels = [\n ...new Set(\n rawChannels.flatMap((c) => {\n if (c === 'webchat') return ['webchat', 'ui'];\n if (c === 'gateway') return ['gateway', 'webui'];\n return [c];\n }),\n ),\n ];\n if (channels.length === 0) {\n sessions = [];\n } else if (channels.length === 1) {\n const ch = channels[0]!;\n sessions = sessions.filter((s) => (s.sourceChannel ?? '').toLowerCase() === ch);\n } else {\n sessions = sessions.filter((s) => channels.includes((s.sourceChannel ?? '').toLowerCase()));\n }\n }\n if (query.tags?.length) {\n sessions = sessions.filter((s) => query.tags!.some((t) => s.tags.includes(t)));\n }\n if (query.search) {\n const q = query.search.toLowerCase();\n const metadataMatches = sessions.filter((session) => this.sessionMetadataMatchesSearch(session, q));\n const metadataMatchedKeys = new Set(metadataMatches.map((session) => session.key));\n const contentMatches: SessionMetadata[] = [];\n const candidates = sessions.filter((session) => !metadataMatchedKeys.has(session.key));\n for (const candidate of candidates) {\n if (await this.sessionContentMatchesSearch(candidate.key, q)) {\n contentMatches.push(candidate);\n }\n }\n sessions = [...metadataMatches, ...contentMatches];\n }\n\n const sortBy = query.sortBy || 'updatedAt';\n const sortOrder = query.sortOrder || 'desc';\n sessions.sort((a, b) => {\n const av = a[sortBy];\n const bv = b[sortBy];\n const c = av < bv ? -1 : av > bv ? 1 : 0;\n return sortOrder === 'asc' ? c : -c;\n });\n\n const total = sessions.length;\n const limit = query.limit || 50;\n const offset = query.offset || 0;\n const items = sessions.slice(offset, offset + limit);\n return { items, total, limit, offset, hasMore: offset + limit < total };\n }\n\n private sessionMetadataMatchesSearch(session: SessionMetadata, query: string): boolean {\n return Boolean(\n session.key.toLowerCase().includes(query) ||\n session.name?.toLowerCase().includes(query) ||\n session.sourceChannel.toLowerCase().includes(query) ||\n session.sourceChatId.toLowerCase().includes(query) ||\n session.tags.some((tag) => tag.toLowerCase().includes(query)),\n );\n }\n\n private async sessionContentMatchesSearch(sessionKey: string, query: string): Promise<boolean> {\n const messages = await this.loadDisplayMessages(sessionKey);\n return messages.some((message) => {\n const content = this.extractTextContent(this.messageContent(message)).toLowerCase();\n if (content.includes(query)) {\n return true;\n }\n const attachments = (message as unknown as Record<string, unknown>).attachments;\n if (!Array.isArray(attachments)) {\n return false;\n }\n return attachments.some((attachment) => {\n if (!attachment || typeof attachment !== 'object') {\n return false;\n }\n const name = (attachment as { name?: string }).name;\n return typeof name === 'string' && name.toLowerCase().includes(query);\n });\n });\n }\n\n async get(\n key: string,\n options?: { includeTranscriptSummary?: boolean; includeTranscriptRows?: boolean },\n ): Promise<SessionDetail | null> {\n const metadata = await this.getMetadata(key);\n if (!metadata) {\n return null;\n }\n const messages = await this.loadDisplayMessages(key);\n const detail = await this.buildSessionDetail(key, metadata, messages, options);\n return detail;\n }\n\n async getMessagePage(\n key: string,\n options: {\n offset?: number;\n limit?: number;\n before?: string;\n includeTranscriptSummary?: boolean;\n includeTranscriptRows?: boolean;\n } = {},\n ): Promise<{\n session: SessionDetail;\n pagination: {\n total: number;\n limit: number;\n offset: number;\n hasMore: boolean;\n before?: string;\n nextBeforeCursor?: string;\n };\n } | null> {\n const metadata = await this.getMetadata(key);\n if (!metadata) {\n return null;\n }\n\n const limit = Math.min(200, Math.max(1, Math.trunc(options.limit ?? 50)));\n const offset = Math.max(0, Math.trunc(options.offset ?? 0));\n const parsedBefore = options.before ? Number.parseInt(options.before, 10) : undefined;\n const hasBeforeCursor = parsedBefore !== undefined && Number.isFinite(parsedBefore);\n\n const checkpoints = await this.listCompactionCheckpoints(key);\n if (checkpoints.length === 0) {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return null;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const primary = this.transcriptPathForEntry(entry, keySessionsDir);\n const page = await readDisplayMessagePageFromTranscriptFile(primary, {\n limit,\n offset: hasBeforeCursor ? undefined : offset,\n beforeIndex: hasBeforeCursor ? parsedBefore : undefined,\n });\n const session = await this.buildSessionDetail(key, metadata, page.messages, options);\n const nextBeforeCursor = page.startIndex > 0 ? String(page.startIndex) : undefined;\n\n return {\n session,\n pagination: {\n total: page.total,\n limit,\n offset,\n hasMore: page.startIndex > 0,\n ...(hasBeforeCursor ? { before: String(page.endIndex) } : {}),\n ...(nextBeforeCursor ? { nextBeforeCursor } : {}),\n },\n };\n }\n\n const messages = await this.loadDisplayMessages(key);\n const total = messages.length;\n const endExclusive = hasBeforeCursor\n ? Math.min(total, Math.max(0, Math.trunc(parsedBefore!)))\n : Math.max(0, total - offset);\n const startInclusive = Math.max(0, endExclusive - limit);\n const pageMessages = messages.slice(startInclusive, endExclusive);\n const session = await this.buildSessionDetail(key, metadata, pageMessages, options);\n const nextBeforeCursor = startInclusive > 0 ? String(startInclusive) : undefined;\n\n return {\n session,\n pagination: {\n total,\n limit,\n offset,\n hasMore: startInclusive > 0,\n ...(hasBeforeCursor ? { before: String(endExclusive) } : {}),\n ...(nextBeforeCursor ? { nextBeforeCursor } : {}),\n },\n };\n }\n\n private async buildSessionDetail(\n key: string,\n metadata: SessionMetadata,\n messages: AgentMessage[],\n options?: { includeTranscriptSummary?: boolean; includeTranscriptRows?: boolean },\n ): Promise<SessionDetail> {\n let transcriptSummary: SessionTranscriptSummary | undefined;\n if (options?.includeTranscriptSummary) {\n const env = await this.loadTranscriptDocument(key);\n if (env) {\n transcriptSummary = {\n id: env.id,\n version: env.version,\n createdAt: env.createdAt,\n updatedAt: env.updatedAt,\n compactionCount: env.compactions?.length ?? 0,\n };\n }\n }\n let transcriptRows: TranscriptStoredRow[] | undefined;\n if (options?.includeTranscriptRows) {\n transcriptRows = await this.loadTranscriptRows(key);\n }\n return {\n ...metadata,\n messages: this.convertMessages(messages),\n ...(transcriptSummary ? { transcriptSummary } : {}),\n ...(transcriptRows !== undefined ? { transcriptRows } : {}),\n };\n }\n\n async loadTranscriptRows(key: string): Promise<TranscriptStoredRow[]> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return [];\n }\n const path = this.transcriptPathForEntry(entry, this.resolveSessionsDirForKey(key));\n return readTranscriptRowsFromFile(path);\n }\n\n async getMetadata(key: string): Promise<SessionMetadata | null> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return null;\n }\n return this.metadataFromEntry(key, entry);\n }\n\n async updateMetadata(key: string, updates: Partial<SessionMetadata>): Promise<void> {\n return this.runStoreMutation(async () => {\n const keyStorePath = this.resolveStorePathForKey(key);\n await withSessionsJsonLock(keyStorePath, async (map) => {\n const entry = map[key] as XopcSessionDiskEntry | undefined;\n if (!entry?.pluginExtensions?.xopc?.metadata) {\n throw new Error(`Session not found: ${key}`);\n }\n const meta = { ...entry.pluginExtensions.xopc.metadata, ...updates, updatedAt: new Date().toISOString() };\n entry.pluginExtensions.xopc.metadata = meta;\n entry.updatedAt = Date.now();\n map[key] = entry as Record<string, unknown>;\n });\n this.invalidateAllSessionsMapCache();\n invalidateSessionSearchIndexCache();\n log.debug({ key, updates }, 'Session metadata updated');\n });\n }\n\n /**\n * Reset transcript for an existing session key: archive the current JSONL as\n * `*.reset.*`, assign a new `sessionId`, and preserve per-session overrides\n * on the disk entry (thinking/verbose) and in `sessions/config/*.json`.\n */\n async reset(key: string): Promise<{ sessionId: string; previousSessionId: string } | null> {\n return this.runStoreMutation(async () => {\n const existing = await this.getDiskEntry(key);\n if (!existing?.pluginExtensions?.xopc?.metadata) {\n return null;\n }\n\n const previousSessionId = existing.sessionId;\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const abs = this.transcriptPathForEntry(existing, keySessionsDir);\n if (existsSync(abs)) {\n try {\n archiveFileOnDisk(abs, 'reset');\n } catch (err) {\n log.warn({ err, key }, 'Transcript archive on reset failed');\n }\n }\n\n const sessionId = randomUUID();\n validateSessionId(sessionId);\n const now = Date.now();\n const nextAbs = resolveSessionTranscriptPathInDir(sessionId, keySessionsDir);\n await writeTranscriptJsonl({\n absPath: nextAbs,\n sessionId,\n cwd: process.cwd(),\n rows: [],\n });\n\n const keyStorePath = this.resolveStorePathForKey(key);\n await withSessionsJsonLock(keyStorePath, async (map) => {\n const e = map[key] as XopcSessionDiskEntry | undefined;\n if (!e?.pluginExtensions?.xopc?.metadata) {\n return;\n }\n e.sessionId = sessionId;\n e.sessionFile = `${sessionId}.jsonl`;\n e.updatedAt = now;\n e.sessionStartedAt = now;\n e.lastInteractionAt = undefined;\n patchSessionsJsonEntryStats(e, buildSessionsJsonStatsPatch(0, 0));\n const meta = e.pluginExtensions.xopc.metadata;\n meta.transcriptId = sessionId;\n meta.updatedAt = new Date(now).toISOString();\n map[key] = e as Record<string, unknown>;\n });\n\n this.invalidateAllSessionsMapCache();\n invalidateSessionSearchIndexCache();\n log.info({ key, previousSessionId, sessionId }, 'Session reset');\n return { sessionId, previousSessionId };\n });\n }\n\n async delete(key: string): Promise<boolean> {\n return this.runStoreMutation(async () => {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return false;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const abs = this.transcriptPathForEntry(entry, keySessionsDir);\n const keyStorePath = this.resolveStorePathForKey(key);\n await withSessionsJsonLock(keyStorePath, async (map) => {\n delete map[key];\n });\n try {\n if (existsSync(abs)) {\n archiveFileOnDisk(abs, 'deleted');\n }\n } catch (err) {\n log.warn({ err, key }, 'Transcript archive on delete failed');\n }\n this.invalidateAllSessionsMapCache();\n invalidateSessionSearchIndexCache();\n log.info({ key }, 'Session deleted');\n return true;\n });\n }\n\n async deleteMany(keys: string[]): Promise<{ success: string[]; failed: string[] }> {\n const success: string[] = [];\n const failed: string[] = [];\n for (const key of keys) {\n try {\n if (await this.delete(key)) {\n success.push(key);\n } else {\n failed.push(key);\n }\n } catch {\n failed.push(key);\n }\n }\n return { success, failed };\n }\n\n async setStatus(key: string, status: SessionStatus): Promise<void> {\n await this.updateMetadata(key, { status });\n if (status === SessionStatus.ARCHIVED) {\n await this.moveToArchive(key);\n } else {\n await this.moveFromArchive(key);\n }\n }\n\n async archive(key: string): Promise<void> {\n await this.setStatus(key, SessionStatus.ARCHIVED);\n }\n\n async unarchive(key: string): Promise<void> {\n await this.setStatus(key, SessionStatus.ACTIVE);\n }\n\n async pin(key: string): Promise<void> {\n await this.setStatus(key, SessionStatus.PINNED);\n }\n\n async unpin(key: string): Promise<void> {\n await this.setStatus(key, SessionStatus.ACTIVE);\n }\n\n private async moveToArchive(key: string): Promise<void> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const abs = this.transcriptPathForEntry(entry, keySessionsDir);\n if (existsSync(abs)) {\n try {\n archiveFileOnDisk(abs, 'deleted');\n } catch (err) {\n log.warn({ err, key }, 'Archive transcript rename failed');\n }\n }\n }\n\n private async findMostRecentDeletedTranscript(sessionId: string, sessionsDir: string): Promise<string | null> {\n let names: string[];\n try {\n names = await readdir(sessionsDir);\n } catch {\n return null;\n }\n const prefix = `${sessionId}${DELETED_MARKER}`;\n const hits = names.filter((n) => n.startsWith(prefix) && n.endsWith('Z'));\n hits.sort().reverse();\n const first = hits[0];\n return first ? join(sessionsDir, first) : null;\n }\n\n private async moveFromArchive(key: string): Promise<void> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const target = resolveSessionTranscriptPathInDir(entry.sessionId, keySessionsDir);\n if (existsSync(target)) {\n return;\n }\n const src = await this.findMostRecentDeletedTranscript(entry.sessionId, keySessionsDir);\n if (!src) {\n return;\n }\n try {\n const { rename } = await import('fs/promises');\n await rename(src, target);\n } catch (err) {\n log.warn({ err, key, src, target }, 'Unarchive transcript rename failed');\n }\n }\n\n async loadMessages(key: string, options?: { fromArchive?: boolean }): Promise<AgentMessage[]> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return [];\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const primary = this.transcriptPathForEntry(entry, keySessionsDir);\n if (existsSync(primary)) {\n const rows = await readTranscriptRowsFromFile(primary);\n return rowsToLlmMessages(rows);\n }\n if (options?.fromArchive) {\n const archived = await this.findMostRecentDeletedTranscript(entry.sessionId, keySessionsDir);\n if (!archived) {\n return [];\n }\n const rows = await readTranscriptRowsFromFile(archived);\n return rowsToLlmMessages(rows);\n }\n return [];\n }\n\n private async loadDisplayMessages(key: string): Promise<AgentMessage[]> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return [];\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const primary = this.transcriptPathForEntry(entry, keySessionsDir);\n const transcriptPaths: string[] = [];\n const checkpoints = await this.listCompactionCheckpoints(key);\n for (const checkpoint of [...checkpoints].reverse()) {\n transcriptPaths.push(join(keySessionsDir, `${this.checkpointBasename(entry.sessionId)}${checkpoint.id}.jsonl`));\n }\n transcriptPaths.push(primary);\n\n const messages: AgentMessage[] = [];\n const seenMessages = new Set<string>();\n for (const transcriptPath of transcriptPaths) {\n if (!existsSync(transcriptPath)) {\n continue;\n }\n const rows = await readTranscriptRowsFromFile(transcriptPath);\n for (const message of rowsToLlmMessages(rows)) {\n if (this.isCompactionSummaryMessage(message)) {\n continue;\n }\n const key = this.displayMessageIdentity(message);\n if (seenMessages.has(key)) {\n continue;\n }\n seenMessages.add(key);\n messages.push(message);\n }\n }\n return messages;\n }\n\n async loadTranscriptDocument(key: string): Promise<XopcSessionTranscriptV1 | null> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return null;\n }\n const path = this.transcriptPathForEntry(entry, this.resolveSessionsDirForKey(key));\n if (!existsSync(path)) {\n return null;\n }\n const entries = loadEntriesFromFile(path);\n const header = entries.find((e) => e.type === 'session');\n if (!header || typeof (header as { id?: unknown }).id !== 'string') {\n return null;\n }\n const sessionHeader = header as { type: 'session'; id: string; timestamp?: string };\n const rows = await readTranscriptRowsFromFile(path);\n const compactions = entries\n .filter((e): e is CompactionEntry => e.type === 'compaction')\n .map((c) => ({\n at: c.timestamp,\n summary: c.summary,\n firstKeptIndex: Number.parseInt(String(c.firstKeptEntryId), 10) || 0,\n tokensBefore: c.tokensBefore,\n tokensAfter:\n typeof c.details === 'object' &&\n c.details &&\n 'tokensAfter' in c.details &&\n typeof (c.details as { tokensAfter?: unknown }).tokensAfter === 'number'\n ? (c.details as { tokensAfter: number }).tokensAfter\n : 0,\n }));\n return {\n type: 'xopc_session_transcript',\n version: 1,\n id: sessionHeader.id,\n createdAt: sessionHeader.timestamp ?? new Date().toISOString(),\n updatedAt: new Date().toISOString(),\n messages: rows,\n ...(compactions.length > 0 ? { compactions } : {}),\n };\n }\n\n private async writeTranscriptAndSyncIndex(\n key: string,\n rows: TranscriptStoredRow[],\n opts?: { appendCompaction?: TranscriptCompactionRecord },\n ): Promise<void> {\n const entry = await this.ensureSession(key);\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const abs = this.transcriptPathForEntry(entry, keySessionsDir);\n const llm = rowsToLlmMessages(rows);\n await persistMergedTranscriptRows({\n absPath: abs,\n sessionId: entry.sessionId,\n cwd: process.cwd(),\n rows,\n appendCompaction: opts?.appendCompaction,\n });\n const keyStorePath = this.resolveStorePathForKey(key);\n await withSessionsJsonLock(keyStorePath, async (map) => {\n const e = map[key] as XopcSessionDiskEntry | undefined;\n if (!e?.pluginExtensions?.xopc?.metadata) {\n return;\n }\n patchSessionsJsonEntryStats(\n e,\n buildSessionsJsonStatsPatch(llm.length, this.estimateTokens(llm)),\n );\n map[key] = e as Record<string, unknown>;\n });\n this.invalidateAllSessionsMapCache();\n invalidateSessionSearchIndexCache();\n }\n\n /** Incremental sessions.json stats after guard append (OpenClaw transcript-events). */\n async syncSessionsJsonFromTranscriptUpdate(update: SessionTranscriptUpdate): Promise<void> {\n const sessionKey = update.sessionKey?.trim();\n if (!sessionKey || !existsSync(update.sessionFile)) {\n return;\n }\n return this.runStoreMutation(async () => {\n const keyStorePath = this.resolveStorePathForKey(sessionKey);\n await withSessionsJsonLock(keyStorePath, async (map) => {\n const e = map[sessionKey] as XopcSessionDiskEntry | undefined;\n if (!e?.pluginExtensions?.xopc?.metadata) {\n return;\n }\n\n if (update.message && isAppendOnlyLlmTranscriptMessage(update.message)) {\n incrementSessionsJsonStatsForAppend(e);\n map[sessionKey] = e as Record<string, unknown>;\n return;\n }\n\n const messageCount = await countTranscriptMessageRows(update.sessionFile);\n const rows = await readTranscriptRowsFromFile(update.sessionFile);\n const llm = rowsToLlmMessages(rows);\n patchSessionsJsonEntryStats(\n e,\n buildSessionsJsonStatsPatch(messageCount, this.estimateTokens(llm)),\n );\n map[sessionKey] = e as Record<string, unknown>;\n });\n this.invalidateAllSessionsMapCache();\n invalidateSessionSearchIndexCache();\n });\n }\n\n async appendTranscriptContextEntry(\n key: string,\n entry: Omit<XopcTranscriptContextEntry, 'kind'> & Partial<Pick<XopcTranscriptContextEntry, 'kind'>>,\n ): Promise<void> {\n return this.runStoreMutation(async () => {\n await this.ensureSession(key);\n const disk = await this.getDiskEntry(key);\n if (!disk) {\n return;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const absPath = this.transcriptPathForEntry(disk, keySessionsDir);\n const row: XopcTranscriptContextEntry = {\n kind: 'context',\n id: typeof entry.id === 'string' ? entry.id : undefined,\n text: typeof entry.text === 'string' ? entry.text : undefined,\n data: entry.data,\n createdAt: entry.createdAt ?? new Date().toISOString(),\n };\n await appendPiTranscriptContextEntry({\n absPath,\n cwd: process.cwd(),\n entry: row,\n sessionKey: key,\n });\n const rows = existsSync(absPath) ? await readTranscriptRowsFromFile(absPath) : [];\n const llm = rowsToLlmMessages(rows);\n const keyStorePath = this.resolveStorePathForKey(key);\n await withSessionsJsonLock(keyStorePath, async (map) => {\n const e = map[key] as XopcSessionDiskEntry | undefined;\n if (!e?.pluginExtensions?.xopc?.metadata) {\n return;\n }\n patchSessionsJsonEntryStats(\n e,\n buildSessionsJsonStatsPatch(llm.length, this.estimateTokens(llm)),\n );\n map[key] = e as Record<string, unknown>;\n });\n this.invalidateAllSessionsMapCache();\n invalidateSessionSearchIndexCache();\n });\n }\n\n /**\n * Bulk write entry point used by compaction, tests, and admin tools.\n * Runtime agent turns must persist via {@link guardSessionManager} + appendMessage.\n */\n async saveMessages(key: string, messages: AgentMessage[]): Promise<void> {\n return this.runStoreMutation(async () => {\n await this.ensureSession(key);\n const prev = await this.loadTranscriptRows(key);\n const merged = mergeLlmMessagesPreservingContextRows(prev, messages);\n await this.writeTranscriptAndSyncIndex(key, merged);\n });\n }\n\n getWindowStats(messages: AgentMessage[]) {\n return this.window.getStats(messages);\n }\n\n needsCompaction(key: string, messages: AgentMessage[], contextWindow: number) {\n return this.compactor.needsCompaction(messages, contextWindow);\n }\n\n prepareCompaction(\n key: string,\n messages: AgentMessage[],\n contextWindow: number,\n ): { needsCompaction: boolean; messages: AgentMessage[]; stats?: ReturnType<typeof this.compactor.needsCompaction> } {\n const result = this.compactor.needsCompaction(messages, contextWindow);\n return { needsCompaction: result.needed, messages, stats: result };\n }\n\n private checkpointBasename(sessionId: string): string {\n return `${sessionId}.checkpoint.`;\n }\n\n private async pruneCompactionCheckpoints(sessionId: string, sessionsDir: string): Promise<void> {\n const MAX = 15;\n const prefix = this.checkpointBasename(sessionId);\n let names: string[];\n try {\n names = await readdir(sessionsDir);\n } catch {\n return;\n }\n const candidates = names.filter((n) => n.startsWith(prefix) && n.endsWith('.jsonl'));\n if (candidates.length <= MAX) {\n return;\n }\n const stats = await Promise.all(\n candidates.map(async (name) => {\n const p = join(sessionsDir, name);\n try {\n const s = await stat(p);\n return { p, mtimeMs: s.mtimeMs };\n } catch {\n return { p: join(sessionsDir, name), mtimeMs: 0 };\n }\n }),\n );\n stats.sort((a, b) => a.mtimeMs - b.mtimeMs);\n for (let i = 0; i < stats.length - MAX; i++) {\n try {\n await unlink(stats[i]!.p);\n } catch {\n /* ignore */\n }\n }\n }\n\n private async captureCompactionCheckpoint(sessionId: string, transcriptAbs: string, sessionsDir: string): Promise<void> {\n if (!existsSync(transcriptAbs)) {\n return;\n }\n const id = randomUUID();\n const dest = join(sessionsDir, `${sessionId}.checkpoint.${id}.jsonl`);\n try {\n await copyFile(transcriptAbs, dest);\n await this.pruneCompactionCheckpoints(sessionId, sessionsDir);\n } catch (err) {\n log.warn({ err, sessionId }, 'Compaction checkpoint copy failed');\n }\n }\n\n async applyCompaction(\n key: string,\n messages: AgentMessage[],\n result: CompactionResult,\n ): Promise<AgentMessage[]> {\n const compacted = this.compactor.applyCompaction(messages, result);\n return this.runStoreMutation(async () => {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return compacted;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const abs = this.transcriptPathForEntry(entry, keySessionsDir);\n await this.captureCompactionCheckpoint(entry.sessionId, abs, keySessionsDir);\n const prev = await this.loadTranscriptRows(key);\n const merged = mergeLlmMessagesPreservingContextRows(prev, compacted);\n await this.writeTranscriptAndSyncIndex(key, merged, {\n appendCompaction: {\n at: new Date().toISOString(),\n summary: result.summary,\n firstKeptIndex: result.firstKeptIndex,\n tokensBefore: result.tokensBefore,\n tokensAfter: result.tokensAfter,\n },\n });\n const metadata = await this.getMetadata(key);\n if (metadata) {\n await this.updateMetadata(key, { compactedCount: metadata.compactedCount + 1 });\n }\n log.info(\n { key, tokensBefore: result.tokensBefore, tokensAfter: result.tokensAfter, keptMessages: compacted.length },\n 'Session compacted',\n );\n await this.injectPostCompactionContext(key);\n return compacted;\n });\n }\n\n async compact(\n key: string,\n messages: AgentMessage[],\n contextWindow: number,\n instructions?: string,\n force?: boolean,\n ): Promise<CompactionResult> {\n const result = await this.compactor.compact(messages, instructions, force);\n if (result.compacted) {\n await this.applyCompaction(key, messages, result);\n }\n return result;\n }\n\n async getCompactionStats(key: string) {\n const metadata = await this.getMetadata(key);\n if (!metadata) {\n return undefined;\n }\n return {\n compactionCount: metadata.compactedCount,\n totalTokensBefore: 0,\n totalTokensAfter: 0,\n lastCompactionAt: undefined,\n };\n }\n\n async listCompactionCheckpoints(key: string): Promise<CompactionCheckpointSummary[]> {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return [];\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const sessionId = entry.sessionId;\n const prefix = this.checkpointBasename(sessionId);\n let names: string[];\n try {\n names = await readdir(keySessionsDir);\n } catch {\n return [];\n }\n const files = names.filter((n) => n.startsWith(prefix) && n.endsWith('.jsonl'));\n const rows = await Promise.all(\n files.map(async (name) => {\n const p = join(keySessionsDir, name);\n const parsed = parseCompactionCheckpointTranscriptFileName(name);\n const id = parsed?.checkpointId;\n if (!id || !normalizeCompactionCheckpointId(id)) {\n return null;\n }\n try {\n const s = await stat(p);\n return {\n id: normalizeCompactionCheckpointId(id)!,\n sizeBytes: s.size,\n modifiedAt: new Date(s.mtimeMs).toISOString(),\n } satisfies CompactionCheckpointSummary;\n } catch {\n return null;\n }\n }),\n );\n const valid = rows.filter((r): r is CompactionCheckpointSummary => r !== null);\n valid.sort((a, b) => b.modifiedAt.localeCompare(a.modifiedAt));\n return valid;\n }\n\n async getCompactionCheckpointDetail(\n key: string,\n checkpointId: string,\n ): Promise<CompactionCheckpointDetail | null> {\n const id = normalizeCompactionCheckpointId(checkpointId);\n if (!id) {\n return null;\n }\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return null;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const fname = `${this.checkpointBasename(entry.sessionId)}${id}.jsonl`;\n const cpPath = join(keySessionsDir, fname);\n if (!existsSync(cpPath)) {\n return null;\n }\n try {\n const rows = await readTranscriptRowsFromFile(cpPath);\n const llm = rowsToLlmMessages(rows);\n const s = await stat(cpPath);\n return {\n id,\n sizeBytes: s.size,\n modifiedAt: new Date(s.mtimeMs).toISOString(),\n messageCount: llm.length,\n };\n } catch {\n return null;\n }\n }\n\n async restoreCompactionCheckpoint(key: string, checkpointId: string): Promise<void> {\n const id = normalizeCompactionCheckpointId(checkpointId);\n if (!id) {\n throw new Error('Invalid checkpoint id');\n }\n return this.runStoreMutation(async () => {\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n throw new Error(`Session not found: ${key}`);\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const cpPath = join(keySessionsDir, `${this.checkpointBasename(entry.sessionId)}${id}.jsonl`);\n if (!existsSync(cpPath)) {\n throw new Error(`Checkpoint not found: ${id}`);\n }\n const target = this.transcriptPathForEntry(entry, keySessionsDir);\n await copyFile(cpPath, target);\n const messages = await this.loadMessages(key);\n await this.saveMessages(key, messages);\n log.info({ key, checkpointId: id }, 'Session transcript restored from compaction checkpoint');\n });\n }\n\n async deleteSession(key: string): Promise<boolean> {\n return this.delete(key);\n }\n\n async load(key: string, options?: { fromArchive?: boolean }): Promise<AgentMessage[]> {\n return this.loadMessages(key, options);\n }\n\n async estimateTokenUsage(_key: string, messages: AgentMessage[]): Promise<number> {\n return this.estimateTokens(messages);\n }\n\n async searchInSession(key: string, keyword: string): Promise<Message[]> {\n const messages = await this.loadDisplayMessages(key);\n const keywordLower = keyword.toLowerCase();\n return this.convertMessages(\n messages.filter((m) => {\n const content = this.extractTextContent(this.messageContent(m));\n return content.toLowerCase().includes(keywordLower);\n }),\n );\n }\n\n async exportSession(key: string, format: ExportFormat): Promise<string> {\n const detail = await this.get(key);\n if (!detail) {\n throw new Error(`Session not found: ${key}`);\n }\n if (format === 'json') {\n const transcriptRows = await this.loadTranscriptRows(key);\n const exportData: SessionExport = {\n version: INDEX_VERSION,\n exportedAt: new Date().toISOString(),\n metadata: detail,\n messages: detail.messages,\n transcriptRows,\n };\n return JSON.stringify(exportData, null, 2);\n }\n const lines = [\n `# ${detail.name || detail.key}`,\n '',\n `- **Channel:** ${detail.sourceChannel}`,\n `- **Created:** ${detail.createdAt}`,\n `- **Messages:** ${detail.messageCount}`,\n `- **Tags:** ${detail.tags.join(', ') || 'none'}`,\n '',\n '---',\n '',\n ];\n for (const msg of detail.messages) {\n const role = msg.role === 'assistant' ? 'Assistant' : msg.role === 'user' ? 'User' : msg.role;\n lines.push(`## ${role}`, '');\n const body = typeof msg.content === 'string' ? msg.content : JSON.stringify(msg.content, null, 2);\n lines.push(body, '', '---', '');\n }\n return lines.join('\\n');\n }\n\n async getStats(): Promise<GlobalSessionStats> {\n const list = await this.list({ limit: 100000 });\n const sessions = list.items;\n const byChannel: Record<string, number> = {};\n for (const s of sessions) {\n byChannel[s.sourceChannel] = (byChannel[s.sourceChannel] || 0) + 1;\n }\n let oldestSession: string | undefined;\n let newestSession: string | undefined;\n if (sessions.length > 0) {\n const sorted = [...sessions].sort((a, b) => a.createdAt.localeCompare(b.createdAt));\n oldestSession = sorted[0]!.createdAt;\n newestSession = sorted[sorted.length - 1]!.createdAt;\n }\n return {\n totalSessions: sessions.length,\n activeSessions: sessions.filter((s) => s.status === SessionStatus.ACTIVE || s.status === SessionStatus.IDLE).length,\n archivedSessions: sessions.filter((s) => s.status === SessionStatus.ARCHIVED).length,\n pinnedSessions: sessions.filter((s) => s.status === SessionStatus.PINNED).length,\n totalMessages: sessions.reduce((sum, s) => sum + s.messageCount, 0),\n totalTokens: sessions.reduce((sum, s) => sum + s.estimatedTokens, 0),\n oldestSession,\n newestSession,\n byChannel,\n };\n }\n\n async archiveOld(olderThanDays: number): Promise<number> {\n const cutoff = new Date();\n cutoff.setDate(cutoff.getDate() - olderThanDays);\n const list = await this.list({ limit: 100000 });\n let archived = 0;\n for (const session of list.items) {\n if (session.status !== SessionStatus.ARCHIVED && session.status !== SessionStatus.PINNED) {\n const lastAccess = new Date(session.lastAccessedAt);\n if (lastAccess < cutoff) {\n await this.archive(session.key);\n archived++;\n }\n }\n }\n return archived;\n }\n\n estimateTokens(messages: AgentMessage[]): number {\n let total = 0;\n for (const msg of messages) {\n total += Math.ceil(this.extractTextContent(this.messageContent(msg)).length / 4);\n }\n return total;\n }\n\n private messageContent(msg: AgentMessage): unknown {\n return (msg as { content?: unknown }).content;\n }\n\n private isCompactionSummaryMessage(msg: AgentMessage): boolean {\n if (msg.role !== 'user') {\n return false;\n }\n const text = this.extractTextContent(this.messageContent(msg)).trim();\n return /^\\[Previous conversation summary\\]/i.test(text);\n }\n\n private displayMessageIdentity(message: AgentMessage): string {\n const record = message as unknown as Record<string, unknown>;\n return JSON.stringify({\n role: message.role,\n timestamp: record.timestamp,\n toolCallId: record.toolCallId ?? record.tool_call_id,\n toolName: record.toolName,\n content: this.messageContent(message),\n });\n }\n\n private extractTextContent(content: unknown): string {\n if (typeof content === 'string') {\n return content;\n }\n if (Array.isArray(content)) {\n const parts: string[] = [];\n for (const item of content) {\n if (typeof item !== 'object' || item === null || !('type' in item)) {\n continue;\n }\n const c = item as { type?: string; text?: string; name?: string };\n if (c.type === 'text' && typeof c.text === 'string') {\n parts.push(c.text);\n } else if (c.type === 'toolCall' || c.type === 'tool_use') {\n parts.push(c.name ? `[${c.name}]` : '');\n }\n }\n return parts.join('');\n }\n return '';\n }\n\n private async injectPostCompactionContext(key: string): Promise<void> {\n const contextText = readPostCompactionContext({\n cfg: this.options.config,\n sessionKey: key,\n });\n if (!contextText?.trim()) {\n return;\n }\n const entry = await this.getDiskEntry(key);\n if (!entry) {\n return;\n }\n const keySessionsDir = this.resolveSessionsDirForKey(key);\n const abs = this.transcriptPathForEntry(entry, keySessionsDir);\n const workspaceDir = resolveEffectiveAgentProfileForSession(this.options.config, key).resolvedWorkspacePath;\n try {\n await appendPiTranscriptContextEntry({\n absPath: abs,\n cwd: workspaceDir,\n sessionKey: key,\n entry: {\n kind: 'context',\n id: `post-compaction-${Date.now()}`,\n text: contextText,\n createdAt: new Date().toISOString(),\n },\n });\n } catch (err) {\n log.warn({ err, key }, 'Post-compaction context injection failed');\n }\n }\n\n private convertMessages(messages: AgentMessage[]): Message[] {\n return messages.map((m: AgentMessage & Record<string, unknown>) => {\n const c = this.messageContent(m);\n const content: string | unknown[] =\n typeof c === 'string' ? c : Array.isArray(c) ? c : this.extractTextContent(c);\n const row: Message = {\n role: m.role as Message['role'],\n content,\n timestamp: m.timestamp ? new Date(m.timestamp as string | number).toISOString() : undefined,\n tool_call_id: (m.tool_call_id as string | undefined) || (m.toolCallId as string | undefined),\n tool_calls: m.tool_calls as Message['tool_calls'],\n name: m.name as string | undefined,\n };\n if (Array.isArray(m.attachments) && m.attachments.length > 0) {\n row.attachments = m.attachments as Message['attachments'];\n }\n const rawUsage = m.usage as { input?: number; output?: number; totalTokens?: number; total?: number } | undefined;\n if (rawUsage && typeof rawUsage === 'object') {\n const inputTokens = typeof rawUsage.input === 'number' ? rawUsage.input : undefined;\n const outputTokens = typeof rawUsage.output === 'number' ? rawUsage.output : undefined;\n const totalTokens = typeof rawUsage.totalTokens === 'number'\n ? rawUsage.totalTokens\n : typeof rawUsage.total === 'number'\n ? rawUsage.total\n : undefined;\n if (inputTokens != null || outputTokens != null || totalTokens != null) {\n row.usage = { inputTokens, outputTokens, totalTokens };\n }\n }\n return row;\n });\n }\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;YAYoF;kBACF;kBAGI;aACpC;gBAyBkC;uBACwC;iBACjE;AAsB3D,MAAM,MAAM,aAAa,eAAe;AAExC,MAAM,gBAAgB;AACtB,MAAM,iBAAiB;AACvB,MAAM,gCAAgC;AAQtC,IAAa,eAAb,MAA0B;CACxB;CACA;CACA;CACA;CACA;CACA,qBAA6B;CAC7B,qBAA4C,QAAQ,SAAS;CAC7D;;CAKA,wCAAgC,IAAI,KAAqB;CAEzD,YACE,SACA,cACA,kBACA;AAHQ,OAAA,UAAA;EAIR,MAAM,UAAU,QAAQ,WAAW,sBAAsB,QAAQ,OAAO;AACxE,OAAK,cAAc,QAAQ,eAAe,mBAAmB,QAAQ,QAAQ,QAAQ;AACrF,OAAK,aAAa,KAAK,KAAK,aAAa,UAAU;AACnD,OAAK,YAAY,KAAK,KAAK,aAAa,UAAU,aAAa;AAC/D,OAAK,SAAS,IAAI,cAAc,aAAa;AAC7C,OAAK,YAAY,IAAI,iBAAiB,iBAAiB;;CAGzD,kBAA0B;AACxB,SAAO,KAAK;;;;;;;;CASd,yBAAiC,YAA4B;AAC3D,MAAI,KAAK,QAAQ,YACf,QAAO,KAAK;EAEd,MAAM,SAASA,gBAAuB,WAAW;AACjD,MAAI,CAAC,OACH,QAAO,KAAK;EAEd,MAAM,UAAU,OAAO;EACvB,MAAM,SAAS,KAAK,sBAAsB,IAAI,QAAQ;AACtD,MAAI,OACF,QAAO;EAET,MAAM,WAAW,mBAAmB,KAAK,QAAQ,QAAQ,QAAQ;AACjE,OAAK,sBAAsB,IAAI,SAAS,SAAS;AACjD,SAAO;;CAGT,uBAA+B,YAA4B;AACzD,SAAO,KAAK,KAAK,yBAAyB,WAAW,EAAE,UAAU,aAAa;;CAGhF,MAAc,iBAAoB,IAAkC;AAClE,MAAI,KAAK,qBAAqB,EAC5B,QAAO,IAAI;EAEb,MAAM,MAAM,KAAK,mBAAmB,KAAK,YAAY;AACnD,QAAK;AACL,OAAI;AACF,WAAO,MAAM,IAAI;aACT;AACR,SAAK;;IAEP;AACF,OAAK,qBAAqB,IAAI,WAAW,KAAA,EAAU,CAAC,YAAY,KAAA,EAAU;AAC1E,SAAO;;CAGT,MAAM,aAA4B;AAChC,QAAM,MAAM,KAAK,aAAa,EAAE,WAAW,MAAM,CAAC;AAClD,QAAM,MAAM,KAAK,YAAY,EAAE,WAAW,MAAM,CAAC;AACjD,MAAI,CAAC,WAAW,KAAK,UAAU,CAC7B,OAAM,qBAAqB,KAAK,WAAW,YAAY,KAAA,EAAU;AAEnE,MAAI,MAAM,oDAAoD;;CAGhE,uBAA+B,OAA6B,aAA8B;AACxF,SAAO,uBAAuB,MAAM,WAAW,OAAO,EAAE,aAAa,eAAe,KAAK,aAAa,CAAC;;CAGzG,MAAc,cAAc,YAAmE;AAE7F,SAAO,qBADW,KAAK,uBAAuB,WACa,CAAC;;CAG9D,MAAc,UAAyD;AACrE,SAAO,qBAA2C,KAAK,UAAU;;CAGnE,gCAA8C;AAC5C,OAAK,sBAAsB,KAAA;;CAG7B,MAAc,0BAAgF;EAC5F,MAAM,SAAS,iBAAiB,KAAK,QAAQ,OAAO;EACpD,MAAM,YAAY,sBAAsB,KAAK,QAAQ,OAAO;EAC5D,MAAM,WAAW,IAAI,IAAY,CAAC,WAAW,GAAG,OAAO,KAAK,UAAU,MAAM,GAAG,CAAC,CAAC;EAEjF,MAAM,aAAa,KAAK,gBAAgB,QAAQ,IAAI,EAAE,SAAS;AAC/D,MAAI,WAAW,WAAW,EAAE;GAC1B,MAAM,UAAU,MAAM,QAAQ,YAAY,EAAE,eAAe,MAAM,CAAC,CAAC,YAAY,EAAE,CAAC;AAClF,QAAK,MAAM,SAAS,QAClB,KAAI,MAAM,aAAa,CACrB,UAAS,IAAI,MAAM,KAAK;;AAK9B,SAAO,CAAC,GAAG,SAAS,CAAC,KAAK,aAAa;GACrC;GACA,SAAS,KAAK,mBAAmB,KAAK,QAAQ,QAAQ,QAAQ,EAAE,UAAU,aAAa;GACxF,EAAE;;;;;;CAOL,MAAc,cAA6D;AACzE,MAAI,KAAK,QAAQ,YACf,QAAO,KAAK,SAAS;EAGvB,MAAM,QAAQ,KAAK,KAAK;AACxB,MAAI,KAAK,uBAAuB,KAAK,oBAAoB,cAAc,OAAO;AAC5E,OAAI,MACF,EAAE,cAAc,OAAO,KAAK,KAAK,oBAAoB,IAAI,CAAC,QAAQ,EAClE,6BACD;AACD,UAAO,KAAK,oBAAoB;;EAGlC,MAAM,YAAY,YAAY,KAAK;EACnC,MAAM,QAAQ,MAAM,KAAK,yBAAyB;EAClD,MAAM,SAA+C,EAAE;EACvD,IAAI,kBAAkB;AACtB,OAAK,MAAM,EAAE,aAAa,OAAO;AAC/B,OAAI,CAAC,WAAW,QAAQ,CACtB;GAEF,MAAM,MAAM,MAAM,qBAA2C,QAAQ;AACrE,UAAO,OAAO,QAAQ,IAAI;AAC1B;;AAGF,OAAK,sBAAsB;GACzB,aAAa,QAAQ;GACrB,KAAK;GACN;AACD,MAAI,MACF;GACE,qBAAqB,MAAM;GAC3B;GACA,cAAc,OAAO,KAAK,OAAO,CAAC;GAClC,YAAY,KAAK,MAAM,YAAY,KAAK,GAAG,UAAU;GACtD,EACD,2BACD;AACD,SAAO;;CAGT,MAAc,aAAa,YAA+D;AAExF,UAAO,MADW,KAAK,cAAc,WAAW,EACrC;;CAGb,qBAA6B,KAA8B;EACzD,MAAM,EAAE,SAAS,WAAW,KAAK,gBAAgB,IAAI;EACrD,MAAM,UAAU,KAAK,sBAAsB,IAAI;EAC/C,MAAM,gBAAgB,YAAY;EAClC,MAAM,qBAAqB,YAAY;EACvC,MAAM,uBAAM,IAAI,MAAM,EAAC,aAAa;AACpC,SAAO;GACL;GACA,QAAA;GACA,MAAM,EAAE;GACR,WAAW;GACX,WAAW;GACX,gBAAgB;GAChB,cAAc;GACd,iBAAiB;GACjB,gBAAgB;GAChB,eAAe;GACf,cAAc;GACd;GACA,GAAI,gBACA;IAAE,aAAa;IAAQ,YAAY,EAAE,WAAW,QAAQ;IAAE,GAC1D,EAAE;GACN,GAAI,qBACA;IAAE,aAAa;IAAa,YAAY,EAAE,iBAAiB,QAAQ;IAAE,GACrE,EAAE;GACN,OAAO;IAAE,cAAc;IAAG,YAAY;IAAG;GAC1C;;CAGH,gBAAwB,KAAkD;EACxE,MAAM,QAAQ,IAAI,MAAM,IAAI;AAC5B,MAAI,MAAM,UAAU,KAAK,MAAM,OAAO,YACpC,QAAO;GAAE,SAAS;GAAa,QAAQ,MAAM,MAAM,EAAE,CAAC,KAAK,IAAI;GAAE;EAEnE,MAAM,SAASA,gBAAuB,IAAI;AAC1C,MAAI,QAAQ;AACV,OAAI,OAAO,WAAW,OACpB,QAAO;IAAE,SAAS;IAAQ,QAAQ,OAAO;IAAQ;AAEnD,UAAO;IACL,SAAS,OAAO;IAChB,QAAQ;KAAC,OAAO;KAAW,OAAO;KAAU,OAAO;KAAO,CAAC,KAAK,IAAI;IACrE;;AAEH,SAAO;GAAE,SAAS;GAAW,QAAQ;GAAK;;CAG5C,sBAA8B,KAAyC;EACrE,MAAM,SAASA,gBAAuB,IAAI;AAC1C,MAAI,CAAC,OACH;AAEF,SAAO;GACL,SAAS,OAAO,SAAS,aAAa,IAAI;GAC1C,QAAQ,OAAO,QAAQ,aAAa,IAAI;GACxC,WAAW,OAAO,WAAW,aAAa,IAAI;GAC9C,UAAU,OAAO,UAAU,aAAa,IAAI;GAC5C,QAAQ,OAAO,QAAQ,aAAa,IAAI;GACxC,UAAU,OAAO;GACjB,SAAS,OAAO;GACjB;;;CAIH,MAAM,sBACJ,YACsE;EACtE,MAAM,QAAQ,MAAM,KAAK,cAAc,WAAW;EAClD,MAAM,cAAc,KAAK,yBAAyB,WAAW;EAC7D,MAAM,UAAU,KAAK,uBAAuB,OAAO,YAAY;AAC/D,SAAO;GAAE,WAAW,MAAM;GAAW;GAAS;GAAa;;;CAI7D,MAAc,cAAc,YAAmD;EAC7E,MAAM,eAAe,KAAK,uBAAuB,WAAW;EAC5D,MAAM,iBAAiB,KAAK,yBAAyB,WAAW;AAChE,QAAM,MAAM,gBAAgB,EAAE,WAAW,MAAM,CAAC;EAChD,IAAI,UAAU;EACd,MAAM,QAAQ,MAAM,qBAAqB,cAAc,OAAO,QAAQ;GACpE,MAAM,WAAW,IAAI;AACrB,OAAI,UAAU,kBAAkB,MAAM,SACpC,QAAO;GAET,IAAI,YAAY;AAChB,OAAI,CAAC,WAAW;IACd,MAAM,YAAY,YAAY;AAC9B,sBAAkB,UAAU;IAC5B,MAAM,cAAc,GAAG,UAAU;IACjC,MAAM,MAAM,KAAK,KAAK;IACtB,MAAM,WAAW,KAAK,qBAAqB,WAAW;AACtD,aAAS,eAAe;AACxB,gBAAY;KACV;KACA,WAAW;KACX,kBAAkB;KAClB;KACA,kBAAkB,EAAE,MAAM,EAAE,UAAU,EAAE;KACzC;AACD,QAAI,cAAc;AAElB,UAAM,qBAAqB;KACzB,SAFU,kCAAkC,WAAW,eAE3C;KACZ;KACA,KAAK,QAAQ,KAAK;KAClB,MAAM,EAAE;KACT,CAAC;AACF,cAAU;cACD,CAAC,UAAU,kBAAkB,MAAM,UAAU;IACtD,MAAM,WAAW,KAAK,qBAAqB,WAAW;AACtD,aAAS,eAAe,UAAU;AAClC,cAAU,mBAAmB,EAAE,MAAM,EAAE,UAAU,EAAE;AACnD,QAAI,cAAc;AAClB,cAAU;;AAEZ,UAAO;IACP;AACF,MAAI,QACF,MAAK,+BAA+B;AAEtC,SAAO;;CAGT,kBAA0B,YAAoB,OAA8C;EAC1F,MAAM,OAAO,MAAM,kBAAkB,MAAM,YAAY,KAAK,qBAAqB,WAAW;EAC5F,MAAM,EAAE,SAAS,WAAW,QAAQ,cAAc,KAAK,gBAAgB,WAAW;EAClF,MAAM,SAAS,OAAO,KAAK,kBAAkB,WAAW,KAAK,cAAc,MAAM,GAAG;EACpF,MAAM,WAAW,OAAO,KAAK,iBAAiB,WAAW,KAAK,aAAa,MAAM,GAAG;AACpF,SAAO;GACL,GAAG;GACH,KAAK;GACL,cAAc,MAAM;GACpB,eAAe,UAAU;GACzB,cAAc,YAAY;GAC3B;;CAGH,MAAM,WAAW,SAA6C;EAC5D,MAAM,MAAM,MAAM,KAAK,aAAa;EACpC,MAAM,MAAyB,EAAE;AACjC,OAAK,MAAM,CAAC,KAAK,MAAM,OAAO,QAAQ,IAAI,EAAE;GAC1C,MAAM,IAAI,KAAK,kBAAkB,KAAK,EAAE;AACxC,OAAI,EAAE,SAAS,SAAS,aAAa,KAAK,QAAQ,aAAa,CAC7D,KAAI,KAAK,EAAE;;AAGf,SAAO;;CAGT,MAAM,aAAa,WAA+C;EAChE,MAAM,MAAM,MAAM,KAAK,aAAa;EACpC,MAAM,MAAyB,EAAE;AACjC,OAAK,MAAM,CAAC,KAAK,MAAM,OAAO,QAAQ,IAAI,EAAE;GAC1C,MAAM,IAAI,KAAK,kBAAkB,KAAK,EAAE;AACxC,OAAI,EAAE,SAAS,cAAc,UAC3B,KAAI,KAAK,EAAE;;AAGf,SAAO;;CAGT,MAAM,UAAU,UAAkB,QAA4C;EAC5E,MAAM,MAAM,MAAM,KAAK,aAAa;EACpC,MAAM,MAAyB,EAAE;AACjC,OAAK,MAAM,CAAC,KAAK,MAAM,OAAO,QAAQ,IAAI,EAAE;GAC1C,MAAM,IAAI,KAAK,kBAAkB,KAAK,EAAE;AACxC,OAAI,EAAE,SAAS,aAAa,YAAY,EAAE,QAAQ,WAAW,OAC3D,KAAI,KAAK,EAAE;;AAGf,SAAO;;CAGT,MAAM,eAAe,SAAiB,WAAoD;EACxF,MAAM,MAAM,MAAM,KAAK,aAAa;AACpC,OAAK,MAAM,CAAC,KAAK,MAAM,OAAO,QAAQ,IAAI,EAAE;GAC1C,MAAM,IAAI,KAAK,kBAAkB,KAAK,EAAE;AACxC,OACE,EAAE,SAAS,WAAW,WACtB,EAAE,QAAQ,cAAc,aACxB,EAAE,QAAQ,aAAa,QACvB,EAAE,QAAQ,WAAW,OAErB,QAAO;;AAGX,SAAO;;CAGT,MAAM,eAA8B;CAIpC,MAAM,KAAK,QAA0B,EAAE,EAA6C;EAClF,MAAM,MAAM,MAAM,KAAK,aAAa;EACpC,IAAI,WAAW,OAAO,QAAQ,IAAI,CAAC,KAAK,CAAC,GAAG,OAAO,KAAK,kBAAkB,GAAG,EAAE,CAAC;AAEhF,MAAI,MAAM,QAAQ;GAChB,MAAM,WAAW,MAAM,QAAQ,MAAM,OAAO,GAAG,MAAM,SAAS,CAAC,MAAM,OAAO;AAC5E,cAAW,SAAS,QAAQ,MAAM,SAAS,SAAS,EAAE,OAAO,CAAC;;AAEhE,MAAI,MAAM,SAAS;GACjB,MAAM,cAAc,MAAM,QACvB,MAAM,IAAI,CACV,KAAK,MAAM,EAAE,MAAM,CAAC,aAAa,CAAC,CAClC,OAAO,QAAQ;;;;;GAKlB,MAAM,WAAW,CACf,GAAG,IAAI,IACL,YAAY,SAAS,MAAM;AACzB,QAAI,MAAM,UAAW,QAAO,CAAC,WAAW,KAAK;AAC7C,QAAI,MAAM,UAAW,QAAO,CAAC,WAAW,QAAQ;AAChD,WAAO,CAAC,EAAE;KACV,CACH,CACF;AACD,OAAI,SAAS,WAAW,EACtB,YAAW,EAAE;YACJ,SAAS,WAAW,GAAG;IAChC,MAAM,KAAK,SAAS;AACpB,eAAW,SAAS,QAAQ,OAAO,EAAE,iBAAiB,IAAI,aAAa,KAAK,GAAG;SAE/E,YAAW,SAAS,QAAQ,MAAM,SAAS,UAAU,EAAE,iBAAiB,IAAI,aAAa,CAAC,CAAC;;AAG/F,MAAI,MAAM,MAAM,OACd,YAAW,SAAS,QAAQ,MAAM,MAAM,KAAM,MAAM,MAAM,EAAE,KAAK,SAAS,EAAE,CAAC,CAAC;AAEhF,MAAI,MAAM,QAAQ;GAChB,MAAM,IAAI,MAAM,OAAO,aAAa;GACpC,MAAM,kBAAkB,SAAS,QAAQ,YAAY,KAAK,6BAA6B,SAAS,EAAE,CAAC;GACnG,MAAM,sBAAsB,IAAI,IAAI,gBAAgB,KAAK,YAAY,QAAQ,IAAI,CAAC;GAClF,MAAM,iBAAoC,EAAE;GAC5C,MAAM,aAAa,SAAS,QAAQ,YAAY,CAAC,oBAAoB,IAAI,QAAQ,IAAI,CAAC;AACtF,QAAK,MAAM,aAAa,WACtB,KAAI,MAAM,KAAK,4BAA4B,UAAU,KAAK,EAAE,CAC1D,gBAAe,KAAK,UAAU;AAGlC,cAAW,CAAC,GAAG,iBAAiB,GAAG,eAAe;;EAGpD,MAAM,SAAS,MAAM,UAAU;EAC/B,MAAM,YAAY,MAAM,aAAa;AACrC,WAAS,MAAM,GAAG,MAAM;GACtB,MAAM,KAAK,EAAE;GACb,MAAM,KAAK,EAAE;GACb,MAAM,IAAI,KAAK,KAAK,KAAK,KAAK,KAAK,IAAI;AACvC,UAAO,cAAc,QAAQ,IAAI,CAAC;IAClC;EAEF,MAAM,QAAQ,SAAS;EACvB,MAAM,QAAQ,MAAM,SAAS;EAC7B,MAAM,SAAS,MAAM,UAAU;AAE/B,SAAO;GAAE,OADK,SAAS,MAAM,QAAQ,SAAS,MAChC;GAAE;GAAO;GAAO;GAAQ,SAAS,SAAS,QAAQ;GAAO;;CAGzE,6BAAqC,SAA0B,OAAwB;AACrF,SAAO,QACL,QAAQ,IAAI,aAAa,CAAC,SAAS,MAAM,IACzC,QAAQ,MAAM,aAAa,CAAC,SAAS,MAAM,IAC3C,QAAQ,cAAc,aAAa,CAAC,SAAS,MAAM,IACnD,QAAQ,aAAa,aAAa,CAAC,SAAS,MAAM,IAClD,QAAQ,KAAK,MAAM,QAAQ,IAAI,aAAa,CAAC,SAAS,MAAM,CAAC,CAC9D;;CAGH,MAAc,4BAA4B,YAAoB,OAAiC;AAE7F,UAAO,MADgB,KAAK,oBAAoB,WAAW,EAC3C,MAAM,YAAY;AAEhC,OADgB,KAAK,mBAAmB,KAAK,eAAe,QAAQ,CAAC,CAAC,aAC3D,CAAC,SAAS,MAAM,CACzB,QAAO;GAET,MAAM,cAAe,QAA+C;AACpE,OAAI,CAAC,MAAM,QAAQ,YAAY,CAC7B,QAAO;AAET,UAAO,YAAY,MAAM,eAAe;AACtC,QAAI,CAAC,cAAc,OAAO,eAAe,SACvC,QAAO;IAET,MAAM,OAAQ,WAAiC;AAC/C,WAAO,OAAO,SAAS,YAAY,KAAK,aAAa,CAAC,SAAS,MAAM;KACrE;IACF;;CAGJ,MAAM,IACJ,KACA,SAC+B;EAC/B,MAAM,WAAW,MAAM,KAAK,YAAY,IAAI;AAC5C,MAAI,CAAC,SACH,QAAO;EAET,MAAM,WAAW,MAAM,KAAK,oBAAoB,IAAI;AAEpD,SAAO,MADc,KAAK,mBAAmB,KAAK,UAAU,UAAU,QAAQ;;CAIhF,MAAM,eACJ,KACA,UAMI,EAAE,EAWE;EACR,MAAM,WAAW,MAAM,KAAK,YAAY,IAAI;AAC5C,MAAI,CAAC,SACH,QAAO;EAGT,MAAM,QAAQ,KAAK,IAAI,KAAK,KAAK,IAAI,GAAG,KAAK,MAAM,QAAQ,SAAS,GAAG,CAAC,CAAC;EACzE,MAAM,SAAS,KAAK,IAAI,GAAG,KAAK,MAAM,QAAQ,UAAU,EAAE,CAAC;EAC3D,MAAM,eAAe,QAAQ,SAAS,OAAO,SAAS,QAAQ,QAAQ,GAAG,GAAG,KAAA;EAC5E,MAAM,kBAAkB,iBAAiB,KAAA,KAAa,OAAO,SAAS,aAAa;AAGnF,OAAI,MADsB,KAAK,0BAA0B,IAAI,EAC7C,WAAW,GAAG;GAC5B,MAAM,QAAQ,MAAM,KAAK,aAAa,IAAI;AAC1C,OAAI,CAAC,MACH,QAAO;GAET,MAAM,iBAAiB,KAAK,yBAAyB,IAAI;GAEzD,MAAM,OAAO,MAAM,yCADH,KAAK,uBAAuB,OAAO,eACgB,EAAE;IACnE;IACA,QAAQ,kBAAkB,KAAA,IAAY;IACtC,aAAa,kBAAkB,eAAe,KAAA;IAC/C,CAAC;GACF,MAAM,UAAU,MAAM,KAAK,mBAAmB,KAAK,UAAU,KAAK,UAAU,QAAQ;GACpF,MAAM,mBAAmB,KAAK,aAAa,IAAI,OAAO,KAAK,WAAW,GAAG,KAAA;AAEzE,UAAO;IACL;IACA,YAAY;KACV,OAAO,KAAK;KACZ;KACA;KACA,SAAS,KAAK,aAAa;KAC3B,GAAI,kBAAkB,EAAE,QAAQ,OAAO,KAAK,SAAS,EAAE,GAAG,EAAE;KAC5D,GAAI,mBAAmB,EAAE,kBAAkB,GAAG,EAAE;KACjD;IACF;;EAGH,MAAM,WAAW,MAAM,KAAK,oBAAoB,IAAI;EACpD,MAAM,QAAQ,SAAS;EACvB,MAAM,eAAe,kBACjB,KAAK,IAAI,OAAO,KAAK,IAAI,GAAG,KAAK,MAAM,aAAc,CAAC,CAAC,GACvD,KAAK,IAAI,GAAG,QAAQ,OAAO;EAC/B,MAAM,iBAAiB,KAAK,IAAI,GAAG,eAAe,MAAM;EACxD,MAAM,eAAe,SAAS,MAAM,gBAAgB,aAAa;EACjE,MAAM,UAAU,MAAM,KAAK,mBAAmB,KAAK,UAAU,cAAc,QAAQ;EACnF,MAAM,mBAAmB,iBAAiB,IAAI,OAAO,eAAe,GAAG,KAAA;AAEvE,SAAO;GACL;GACA,YAAY;IACV;IACA;IACA;IACA,SAAS,iBAAiB;IAC1B,GAAI,kBAAkB,EAAE,QAAQ,OAAO,aAAa,EAAE,GAAG,EAAE;IAC3D,GAAI,mBAAmB,EAAE,kBAAkB,GAAG,EAAE;IACjD;GACF;;CAGH,MAAc,mBACZ,KACA,UACA,UACA,SACwB;EACxB,IAAI;AACJ,MAAI,SAAS,0BAA0B;GACrC,MAAM,MAAM,MAAM,KAAK,uBAAuB,IAAI;AAClD,OAAI,IACF,qBAAoB;IAClB,IAAI,IAAI;IACR,SAAS,IAAI;IACb,WAAW,IAAI;IACf,WAAW,IAAI;IACf,iBAAiB,IAAI,aAAa,UAAU;IAC7C;;EAGL,IAAI;AACJ,MAAI,SAAS,sBACX,kBAAiB,MAAM,KAAK,mBAAmB,IAAI;AAErD,SAAO;GACL,GAAG;GACH,UAAU,KAAK,gBAAgB,SAAS;GACxC,GAAI,oBAAoB,EAAE,mBAAmB,GAAG,EAAE;GAClD,GAAI,mBAAmB,KAAA,IAAY,EAAE,gBAAgB,GAAG,EAAE;GAC3D;;CAGH,MAAM,mBAAmB,KAA6C;EACpE,MAAM,QAAQ,MAAM,KAAK,aAAa,IAAI;AAC1C,MAAI,CAAC,MACH,QAAO,EAAE;AAGX,SAAO,2BADM,KAAK,uBAAuB,OAAO,KAAK,yBAAyB,IAAI,CAC5C,CAAC;;CAGzC,MAAM,YAAY,KAA8C;EAC9D,MAAM,QAAQ,MAAM,KAAK,aAAa,IAAI;AAC1C,MAAI,CAAC,MACH,QAAO;AAET,SAAO,KAAK,kBAAkB,KAAK,MAAM;;CAG3C,MAAM,eAAe,KAAa,SAAkD;AAClF,SAAO,KAAK,iBAAiB,YAAY;AAEvC,SAAM,qBADe,KAAK,uBAAuB,IACV,EAAE,OAAO,QAAQ;IACtD,MAAM,QAAQ,IAAI;AAClB,QAAI,CAAC,OAAO,kBAAkB,MAAM,SAClC,OAAM,IAAI,MAAM,sBAAsB,MAAM;IAE9C,MAAM,OAAO;KAAE,GAAG,MAAM,iBAAiB,KAAK;KAAU,GAAG;KAAS,4BAAW,IAAI,MAAM,EAAC,aAAa;KAAE;AACzG,UAAM,iBAAiB,KAAK,WAAW;AACvC,UAAM,YAAY,KAAK,KAAK;AAC5B,QAAI,OAAO;KACX;AACF,QAAK,+BAA+B;AACpC,sCAAmC;AACnC,OAAI,MAAM;IAAE;IAAK;IAAS,EAAE,2BAA2B;IACvD;;;;;;;CAQJ,MAAM,MAAM,KAA+E;AACzF,SAAO,KAAK,iBAAiB,YAAY;GACvC,MAAM,WAAW,MAAM,KAAK,aAAa,IAAI;AAC7C,OAAI,CAAC,UAAU,kBAAkB,MAAM,SACrC,QAAO;GAGT,MAAM,oBAAoB,SAAS;GACnC,MAAM,iBAAiB,KAAK,yBAAyB,IAAI;GACzD,MAAM,MAAM,KAAK,uBAAuB,UAAU,eAAe;AACjE,OAAI,WAAW,IAAI,CACjB,KAAI;AACF,sBAAkB,KAAK,QAAQ;YACxB,KAAK;AACZ,QAAI,KAAK;KAAE;KAAK;KAAK,EAAE,qCAAqC;;GAIhE,MAAM,YAAY,YAAY;AAC9B,qBAAkB,UAAU;GAC5B,MAAM,MAAM,KAAK,KAAK;AAEtB,SAAM,qBAAqB;IACzB,SAFc,kCAAkC,WAAW,eAE3C;IAChB;IACA,KAAK,QAAQ,KAAK;IAClB,MAAM,EAAE;IACT,CAAC;AAGF,SAAM,qBADe,KAAK,uBAAuB,IACV,EAAE,OAAO,QAAQ;IACtD,MAAM,IAAI,IAAI;AACd,QAAI,CAAC,GAAG,kBAAkB,MAAM,SAC9B;AAEF,MAAE,YAAY;AACd,MAAE,cAAc,GAAG,UAAU;AAC7B,MAAE,YAAY;AACd,MAAE,mBAAmB;AACrB,MAAE,oBAAoB,KAAA;AACtB,gCAA4B,GAAG,4BAA4B,GAAG,EAAE,CAAC;IACjE,MAAM,OAAO,EAAE,iBAAiB,KAAK;AACrC,SAAK,eAAe;AACpB,SAAK,YAAY,IAAI,KAAK,IAAI,CAAC,aAAa;AAC5C,QAAI,OAAO;KACX;AAEF,QAAK,+BAA+B;AACpC,sCAAmC;AACnC,OAAI,KAAK;IAAE;IAAK;IAAmB;IAAW,EAAE,gBAAgB;AAChE,UAAO;IAAE;IAAW;IAAmB;IACvC;;CAGJ,MAAM,OAAO,KAA+B;AAC1C,SAAO,KAAK,iBAAiB,YAAY;GACvC,MAAM,QAAQ,MAAM,KAAK,aAAa,IAAI;AAC1C,OAAI,CAAC,MACH,QAAO;GAET,MAAM,iBAAiB,KAAK,yBAAyB,IAAI;GACzD,MAAM,MAAM,KAAK,uBAAuB,OAAO,eAAe;AAE9D,SAAM,qBADe,KAAK,uBAAuB,IACV,EAAE,OAAO,QAAQ;AACtD,WAAO,IAAI;KACX;AACF,OAAI;AACF,QAAI,WAAW,IAAI,CACjB,mBAAkB,KAAK,UAAU;YAE5B,KAAK;AACZ,QAAI,KAAK;KAAE;KAAK;KAAK,EAAE,sCAAsC;;AAE/D,QAAK,+BAA+B;AACpC,sCAAmC;AACnC,OAAI,KAAK,EAAE,KAAK,EAAE,kBAAkB;AACpC,UAAO;IACP;;CAGJ,MAAM,WAAW,MAAkE;EACjF,MAAM,UAAoB,EAAE;EAC5B,MAAM,SAAmB,EAAE;AAC3B,OAAK,MAAM,OAAO,KAChB,KAAI;AACF,OAAI,MAAM,KAAK,OAAO,IAAI,CACxB,SAAQ,KAAK,IAAI;OAEjB,QAAO,KAAK,IAAI;UAEZ;AACN,UAAO,KAAK,IAAI;;AAGpB,SAAO;GAAE;GAAS;GAAQ;;CAG5B,MAAM,UAAU,KAAa,QAAsC;AACjE,QAAM,KAAK,eAAe,KAAK,EAAE,QAAQ,CAAC;AAC1C,MAAI,WAAA,WACF,OAAM,KAAK,cAAc,IAAI;MAE7B,OAAM,KAAK,gBAAgB,IAAI;;CAInC,MAAM,QAAQ,KAA4B;AACxC,QAAM,KAAK,UAAU,KAAA,WAA4B;;CAGnD,MAAM,UAAU,KAA4B;AAC1C,QAAM,KAAK,UAAU,KAAA,SAA0B;;CAGjD,MAAM,IAAI,KAA4B;AACpC,QAAM,KAAK,UAAU,KAAA,SAA0B;;CAGjD,MAAM,MAAM,KAA4B;AACtC,QAAM,KAAK,UAAU,KAAA,SAA0B;;CAGjD,MAAc,cAAc,KAA4B;EACtD,MAAM,QAAQ,MAAM,KAAK,aAAa,IAAI;AAC1C,MAAI,CAAC,MACH;EAEF,MAAM,iBAAiB,KAAK,yBAAyB,IAAI;EACzD,MAAM,MAAM,KAAK,uBAAuB,OAAO,eAAe;AAC9D,MAAI,WAAW,IAAI,CACjB,KAAI;AACF,qBAAkB,KAAK,UAAU;WAC1B,KAAK;AACZ,OAAI,KAAK;IAAE;IAAK;IAAK,EAAE,mCAAmC;;;CAKhE,MAAc,gCAAgC,WAAmB,aAA6C;EAC5G,IAAI;AACJ,MAAI;AACF,WAAQ,MAAM,QAAQ,YAAY;UAC5B;AACN,UAAO;;EAET,MAAM,SAAS,GAAG,YAAY;EAC9B,MAAM,OAAO,MAAM,QAAQ,MAAM,EAAE,WAAW,OAAO,IAAI,EAAE,SAAS,IAAI,CAAC;AACzE,OAAK,MAAM,CAAC,SAAS;EACrB,MAAM,QAAQ,KAAK;AACnB,SAAO,QAAQ,KAAK,aAAa,MAAM,GAAG;;CAG5C,MAAc,gBAAgB,KAA4B;EACxD,MAAM,QAAQ,MAAM,KAAK,aAAa,IAAI;AAC1C,MAAI,CAAC,MACH;EAEF,MAAM,iBAAiB,KAAK,yBAAyB,IAAI;EACzD,MAAM,SAAS,kCAAkC,MAAM,WAAW,eAAe;AACjF,MAAI,WAAW,OAAO,CACpB;EAEF,MAAM,MAAM,MAAM,KAAK,gCAAgC,MAAM,WAAW,eAAe;AACvF,MAAI,CAAC,IACH;AAEF,MAAI;GACF,MAAM,EAAE,WAAW,MAAM,OAAO;AAChC,SAAM,OAAO,KAAK,OAAO;WAClB,KAAK;AACZ,OAAI,KAAK;IAAE;IAAK;IAAK;IAAK;IAAQ,EAAE,qCAAqC;;;CAI7E,MAAM,aAAa,KAAa,SAA8D;EAC5F,MAAM,QAAQ,MAAM,KAAK,aAAa,IAAI;AAC1C,MAAI,CAAC,MACH,QAAO,EAAE;EAEX,MAAM,iBAAiB,KAAK,yBAAyB,IAAI;EACzD,MAAM,UAAU,KAAK,uBAAuB,OAAO,eAAe;AAClE,MAAI,WAAW,QAAQ,CAErB,QAAO,kBAAkB,MADN,2BAA2B,QAAQ,CACxB;AAEhC,MAAI,SAAS,aAAa;GACxB,MAAM,WAAW,MAAM,KAAK,gCAAgC,MAAM,WAAW,eAAe;AAC5F,OAAI,CAAC,SACH,QAAO,EAAE;AAGX,UAAO,kBAAkB,MADN,2BAA2B,SAAS,CACzB;;AAEhC,SAAO,EAAE;;CAGX,MAAc,oBAAoB,KAAsC;EACtE,MAAM,QAAQ,MAAM,KAAK,aAAa,IAAI;AAC1C,MAAI,CAAC,MACH,QAAO,EAAE;EAEX,MAAM,iBAAiB,KAAK,yBAAyB,IAAI;EACzD,MAAM,UAAU,KAAK,uBAAuB,OAAO,eAAe;EAClE,MAAM,kBAA4B,EAAE;EACpC,MAAM,cAAc,MAAM,KAAK,0BAA0B,IAAI;AAC7D,OAAK,MAAM,cAAc,CAAC,GAAG,YAAY,CAAC,SAAS,CACjD,iBAAgB,KAAK,KAAK,gBAAgB,GAAG,KAAK,mBAAmB,MAAM,UAAU,GAAG,WAAW,GAAG,QAAQ,CAAC;AAEjH,kBAAgB,KAAK,QAAQ;EAE7B,MAAM,WAA2B,EAAE;EACnC,MAAM,+BAAe,IAAI,KAAa;AACtC,OAAK,MAAM,kBAAkB,iBAAiB;AAC5C,OAAI,CAAC,WAAW,eAAe,CAC7B;GAEF,MAAM,OAAO,MAAM,2BAA2B,eAAe;AAC7D,QAAK,MAAM,WAAW,kBAAkB,KAAK,EAAE;AAC7C,QAAI,KAAK,2BAA2B,QAAQ,CAC1C;IAEF,MAAM,MAAM,KAAK,uBAAuB,QAAQ;AAChD,QAAI,aAAa,IAAI,IAAI,CACvB;AAEF,iBAAa,IAAI,IAAI;AACrB,aAAS,KAAK,QAAQ;;;AAG1B,SAAO;;CAGT,MAAM,uBAAuB,KAAsD;EACjF,MAAM,QAAQ,MAAM,KAAK,aAAa,IAAI;AAC1C,MAAI,CAAC,MACH,QAAO;EAET,MAAM,OAAO,KAAK,uBAAuB,OAAO,KAAK,yBAAyB,IAAI,CAAC;AACnF,MAAI,CAAC,WAAW,KAAK,CACnB,QAAO;EAET,MAAM,UAAU,oBAAoB,KAAK;EACzC,MAAM,SAAS,QAAQ,MAAM,MAAM,EAAE,SAAS,UAAU;AACxD,MAAI,CAAC,UAAU,OAAQ,OAA4B,OAAO,SACxD,QAAO;EAET,MAAM,gBAAgB;EACtB,MAAM,OAAO,MAAM,2BAA2B,KAAK;EACnD,MAAM,cAAc,QACjB,QAAQ,MAA4B,EAAE,SAAS,aAAa,CAC5D,KAAK,OAAO;GACX,IAAI,EAAE;GACN,SAAS,EAAE;GACX,gBAAgB,OAAO,SAAS,OAAO,EAAE,iBAAiB,EAAE,GAAG,IAAI;GACnE,cAAc,EAAE;GAChB,aACE,OAAO,EAAE,YAAY,YACrB,EAAE,WACF,iBAAiB,EAAE,WACnB,OAAQ,EAAE,QAAsC,gBAAgB,WAC3D,EAAE,QAAoC,cACvC;GACP,EAAE;AACL,SAAO;GACL,MAAM;GACN,SAAS;GACT,IAAI,cAAc;GAClB,WAAW,cAAc,8BAAa,IAAI,MAAM,EAAC,aAAa;GAC9D,4BAAW,IAAI,MAAM,EAAC,aAAa;GACnC,UAAU;GACV,GAAI,YAAY,SAAS,IAAI,EAAE,aAAa,GAAG,EAAE;GAClD;;CAGH,MAAc,4BACZ,KACA,MACA,MACe;EACf,MAAM,QAAQ,MAAM,KAAK,cAAc,IAAI;EAC3C,MAAM,iBAAiB,KAAK,yBAAyB,IAAI;EACzD,MAAM,MAAM,KAAK,uBAAuB,OAAO,eAAe;EAC9D,MAAM,MAAM,kBAAkB,KAAK;AACnC,QAAM,4BAA4B;GAChC,SAAS;GACT,WAAW,MAAM;GACjB,KAAK,QAAQ,KAAK;GAClB;GACA,kBAAkB,MAAM;GACzB,CAAC;AAEF,QAAM,qBADe,KAAK,uBAAuB,IACV,EAAE,OAAO,QAAQ;GACtD,MAAM,IAAI,IAAI;AACd,OAAI,CAAC,GAAG,kBAAkB,MAAM,SAC9B;AAEF,+BACE,GACA,4BAA4B,IAAI,QAAQ,KAAK,eAAe,IAAI,CAAC,CAClE;AACD,OAAI,OAAO;IACX;AACF,OAAK,+BAA+B;AACpC,qCAAmC;;;CAIrC,MAAM,qCAAqC,QAAgD;EACzF,MAAM,aAAa,OAAO,YAAY,MAAM;AAC5C,MAAI,CAAC,cAAc,CAAC,WAAW,OAAO,YAAY,CAChD;AAEF,SAAO,KAAK,iBAAiB,YAAY;AAEvC,SAAM,qBADe,KAAK,uBAAuB,WACV,EAAE,OAAO,QAAQ;IACtD,MAAM,IAAI,IAAI;AACd,QAAI,CAAC,GAAG,kBAAkB,MAAM,SAC9B;AAGF,QAAI,OAAO,WAAW,iCAAiC,OAAO,QAAQ,EAAE;AACtE,yCAAoC,EAAE;AACtC,SAAI,cAAc;AAClB;;IAGF,MAAM,eAAe,MAAM,2BAA2B,OAAO,YAAY;IAEzE,MAAM,MAAM,kBAAkB,MADX,2BAA2B,OAAO,YAAY,CAC9B;AACnC,gCACE,GACA,4BAA4B,cAAc,KAAK,eAAe,IAAI,CAAC,CACpE;AACD,QAAI,cAAc;KAClB;AACF,QAAK,+BAA+B;AACpC,sCAAmC;IACnC;;CAGJ,MAAM,6BACJ,KACA,OACe;AACf,SAAO,KAAK,iBAAiB,YAAY;AACvC,SAAM,KAAK,cAAc,IAAI;GAC7B,MAAM,OAAO,MAAM,KAAK,aAAa,IAAI;AACzC,OAAI,CAAC,KACH;GAEF,MAAM,iBAAiB,KAAK,yBAAyB,IAAI;GACzD,MAAM,UAAU,KAAK,uBAAuB,MAAM,eAAe;GACjE,MAAM,MAAkC;IACtC,MAAM;IACN,IAAI,OAAO,MAAM,OAAO,WAAW,MAAM,KAAK,KAAA;IAC9C,MAAM,OAAO,MAAM,SAAS,WAAW,MAAM,OAAO,KAAA;IACpD,MAAM,MAAM;IACZ,WAAW,MAAM,8BAAa,IAAI,MAAM,EAAC,aAAa;IACvD;AACD,SAAM,+BAA+B;IACnC;IACA,KAAK,QAAQ,KAAK;IAClB,OAAO;IACP,YAAY;IACb,CAAC;GAEF,MAAM,MAAM,kBADC,WAAW,QAAQ,GAAG,MAAM,2BAA2B,QAAQ,GAAG,EAAE,CAC9C;AAEnC,SAAM,qBADe,KAAK,uBAAuB,IACV,EAAE,OAAO,QAAQ;IACtD,MAAM,IAAI,IAAI;AACd,QAAI,CAAC,GAAG,kBAAkB,MAAM,SAC9B;AAEF,gCACE,GACA,4BAA4B,IAAI,QAAQ,KAAK,eAAe,IAAI,CAAC,CAClE;AACD,QAAI,OAAO;KACX;AACF,QAAK,+BAA+B;AACpC,sCAAmC;IACnC;;;;;;CAOJ,MAAM,aAAa,KAAa,UAAyC;AACvE,SAAO,KAAK,iBAAiB,YAAY;AACvC,SAAM,KAAK,cAAc,IAAI;GAE7B,MAAM,SAAS,sCAAsC,MADlC,KAAK,mBAAmB,IAAI,EACY,SAAS;AACpE,SAAM,KAAK,4BAA4B,KAAK,OAAO;IACnD;;CAGJ,eAAe,UAA0B;AACvC,SAAO,KAAK,OAAO,SAAS,SAAS;;CAGvC,gBAAgB,KAAa,UAA0B,eAAuB;AAC5E,SAAO,KAAK,UAAU,gBAAgB,UAAU,cAAc;;CAGhE,kBACE,KACA,UACA,eACmH;EACnH,MAAM,SAAS,KAAK,UAAU,gBAAgB,UAAU,cAAc;AACtE,SAAO;GAAE,iBAAiB,OAAO;GAAQ;GAAU,OAAO;GAAQ;;CAGpE,mBAA2B,WAA2B;AACpD,SAAO,GAAG,UAAU;;CAGtB,MAAc,2BAA2B,WAAmB,aAAoC;EAC9F,MAAM,MAAM;EACZ,MAAM,SAAS,KAAK,mBAAmB,UAAU;EACjD,IAAI;AACJ,MAAI;AACF,WAAQ,MAAM,QAAQ,YAAY;UAC5B;AACN;;EAEF,MAAM,aAAa,MAAM,QAAQ,MAAM,EAAE,WAAW,OAAO,IAAI,EAAE,SAAS,SAAS,CAAC;AACpF,MAAI,WAAW,UAAU,IACvB;EAEF,MAAM,QAAQ,MAAM,QAAQ,IAC1B,WAAW,IAAI,OAAO,SAAS;GAC7B,MAAM,IAAI,KAAK,aAAa,KAAK;AACjC,OAAI;AAEF,WAAO;KAAE;KAAG,UAAS,MADL,KAAK,EAAE,EACA;KAAS;WAC1B;AACN,WAAO;KAAE,GAAG,KAAK,aAAa,KAAK;KAAE,SAAS;KAAG;;IAEnD,CACH;AACD,QAAM,MAAM,GAAG,MAAM,EAAE,UAAU,EAAE,QAAQ;AAC3C,OAAK,IAAI,IAAI,GAAG,IAAI,MAAM,SAAS,KAAK,IACtC,KAAI;AACF,SAAM,OAAO,MAAM,GAAI,EAAE;UACnB;;CAMZ,MAAc,4BAA4B,WAAmB,eAAuB,aAAoC;AACtH,MAAI,CAAC,WAAW,cAAc,CAC5B;EAGF,MAAM,OAAO,KAAK,aAAa,GAAG,UAAU,cADjC,YACiD,CAAC,QAAQ;AACrE,MAAI;AACF,SAAM,SAAS,eAAe,KAAK;AACnC,SAAM,KAAK,2BAA2B,WAAW,YAAY;WACtD,KAAK;AACZ,OAAI,KAAK;IAAE;IAAK;IAAW,EAAE,oCAAoC;;;CAIrE,MAAM,gBACJ,KACA,UACA,QACyB;EACzB,MAAM,YAAY,KAAK,UAAU,gBAAgB,UAAU,OAAO;AAClE,SAAO,KAAK,iBAAiB,YAAY;GACvC,MAAM,QAAQ,MAAM,KAAK,aAAa,IAAI;AAC1C,OAAI,CAAC,MACH,QAAO;GAET,MAAM,iBAAiB,KAAK,yBAAyB,IAAI;GACzD,MAAM,MAAM,KAAK,uBAAuB,OAAO,eAAe;AAC9D,SAAM,KAAK,4BAA4B,MAAM,WAAW,KAAK,eAAe;GAE5E,MAAM,SAAS,sCAAsC,MADlC,KAAK,mBAAmB,IAAI,EACY,UAAU;AACrE,SAAM,KAAK,4BAA4B,KAAK,QAAQ,EAClD,kBAAkB;IAChB,qBAAI,IAAI,MAAM,EAAC,aAAa;IAC5B,SAAS,OAAO;IAChB,gBAAgB,OAAO;IACvB,cAAc,OAAO;IACrB,aAAa,OAAO;IACrB,EACF,CAAC;GACF,MAAM,WAAW,MAAM,KAAK,YAAY,IAAI;AAC5C,OAAI,SACF,OAAM,KAAK,eAAe,KAAK,EAAE,gBAAgB,SAAS,iBAAiB,GAAG,CAAC;AAEjF,OAAI,KACF;IAAE;IAAK,cAAc,OAAO;IAAc,aAAa,OAAO;IAAa,cAAc,UAAU;IAAQ,EAC3G,oBACD;AACD,SAAM,KAAK,4BAA4B,IAAI;AAC3C,UAAO;IACP;;CAGJ,MAAM,QACJ,KACA,UACA,eACA,cACA,OAC2B;EAC3B,MAAM,SAAS,MAAM,KAAK,UAAU,QAAQ,UAAU,cAAc,MAAM;AAC1E,MAAI,OAAO,UACT,OAAM,KAAK,gBAAgB,KAAK,UAAU,OAAO;AAEnD,SAAO;;CAGT,MAAM,mBAAmB,KAAa;EACpC,MAAM,WAAW,MAAM,KAAK,YAAY,IAAI;AAC5C,MAAI,CAAC,SACH;AAEF,SAAO;GACL,iBAAiB,SAAS;GAC1B,mBAAmB;GACnB,kBAAkB;GAClB,kBAAkB,KAAA;GACnB;;CAGH,MAAM,0BAA0B,KAAqD;EACnF,MAAM,QAAQ,MAAM,KAAK,aAAa,IAAI;AAC1C,MAAI,CAAC,MACH,QAAO,EAAE;EAEX,MAAM,iBAAiB,KAAK,yBAAyB,IAAI;EACzD,MAAM,YAAY,MAAM;EACxB,MAAM,SAAS,KAAK,mBAAmB,UAAU;EACjD,IAAI;AACJ,MAAI;AACF,WAAQ,MAAM,QAAQ,eAAe;UAC/B;AACN,UAAO,EAAE;;EAEX,MAAM,QAAQ,MAAM,QAAQ,MAAM,EAAE,WAAW,OAAO,IAAI,EAAE,SAAS,SAAS,CAAC;EAqB/E,MAAM,SAAQ,MApBK,QAAQ,IACzB,MAAM,IAAI,OAAO,SAAS;GACxB,MAAM,IAAI,KAAK,gBAAgB,KAAK;GAEpC,MAAM,KADS,4CAA4C,KAC1C,EAAE;AACnB,OAAI,CAAC,MAAM,CAAC,gCAAgC,GAAG,CAC7C,QAAO;AAET,OAAI;IACF,MAAM,IAAI,MAAM,KAAK,EAAE;AACvB,WAAO;KACL,IAAI,gCAAgC,GAAG;KACvC,WAAW,EAAE;KACb,YAAY,IAAI,KAAK,EAAE,QAAQ,CAAC,aAAa;KAC9C;WACK;AACN,WAAO;;IAET,CACH,EACkB,QAAQ,MAAwC,MAAM,KAAK;AAC9E,QAAM,MAAM,GAAG,MAAM,EAAE,WAAW,cAAc,EAAE,WAAW,CAAC;AAC9D,SAAO;;CAGT,MAAM,8BACJ,KACA,cAC4C;EAC5C,MAAM,KAAK,gCAAgC,aAAa;AACxD,MAAI,CAAC,GACH,QAAO;EAET,MAAM,QAAQ,MAAM,KAAK,aAAa,IAAI;AAC1C,MAAI,CAAC,MACH,QAAO;EAIT,MAAM,SAAS,KAFQ,KAAK,yBAAyB,IAEnB,EAAE,GADnB,KAAK,mBAAmB,MAAM,UAAU,GAAG,GAAG,QACrB;AAC1C,MAAI,CAAC,WAAW,OAAO,CACrB,QAAO;AAET,MAAI;GAEF,MAAM,MAAM,kBAAkB,MADX,2BAA2B,OAAO,CAClB;GACnC,MAAM,IAAI,MAAM,KAAK,OAAO;AAC5B,UAAO;IACL;IACA,WAAW,EAAE;IACb,YAAY,IAAI,KAAK,EAAE,QAAQ,CAAC,aAAa;IAC7C,cAAc,IAAI;IACnB;UACK;AACN,UAAO;;;CAIX,MAAM,4BAA4B,KAAa,cAAqC;EAClF,MAAM,KAAK,gCAAgC,aAAa;AACxD,MAAI,CAAC,GACH,OAAM,IAAI,MAAM,wBAAwB;AAE1C,SAAO,KAAK,iBAAiB,YAAY;GACvC,MAAM,QAAQ,MAAM,KAAK,aAAa,IAAI;AAC1C,OAAI,CAAC,MACH,OAAM,IAAI,MAAM,sBAAsB,MAAM;GAE9C,MAAM,iBAAiB,KAAK,yBAAyB,IAAI;GACzD,MAAM,SAAS,KAAK,gBAAgB,GAAG,KAAK,mBAAmB,MAAM,UAAU,GAAG,GAAG,QAAQ;AAC7F,OAAI,CAAC,WAAW,OAAO,CACrB,OAAM,IAAI,MAAM,yBAAyB,KAAK;AAGhD,SAAM,SAAS,QADA,KAAK,uBAAuB,OAAO,eACrB,CAAC;GAC9B,MAAM,WAAW,MAAM,KAAK,aAAa,IAAI;AAC7C,SAAM,KAAK,aAAa,KAAK,SAAS;AACtC,OAAI,KAAK;IAAE;IAAK,cAAc;IAAI,EAAE,yDAAyD;IAC7F;;CAGJ,MAAM,cAAc,KAA+B;AACjD,SAAO,KAAK,OAAO,IAAI;;CAGzB,MAAM,KAAK,KAAa,SAA8D;AACpF,SAAO,KAAK,aAAa,KAAK,QAAQ;;CAGxC,MAAM,mBAAmB,MAAc,UAA2C;AAChF,SAAO,KAAK,eAAe,SAAS;;CAGtC,MAAM,gBAAgB,KAAa,SAAqC;EACtE,MAAM,WAAW,MAAM,KAAK,oBAAoB,IAAI;EACpD,MAAM,eAAe,QAAQ,aAAa;AAC1C,SAAO,KAAK,gBACV,SAAS,QAAQ,MAAM;AAErB,UADgB,KAAK,mBAAmB,KAAK,eAAe,EAAE,CAChD,CAAC,aAAa,CAAC,SAAS,aAAa;IACnD,CACH;;CAGH,MAAM,cAAc,KAAa,QAAuC;EACtE,MAAM,SAAS,MAAM,KAAK,IAAI,IAAI;AAClC,MAAI,CAAC,OACH,OAAM,IAAI,MAAM,sBAAsB,MAAM;AAE9C,MAAI,WAAW,QAAQ;GACrB,MAAM,iBAAiB,MAAM,KAAK,mBAAmB,IAAI;GACzD,MAAM,aAA4B;IAChC,SAAS;IACT,6BAAY,IAAI,MAAM,EAAC,aAAa;IACpC,UAAU;IACV,UAAU,OAAO;IACjB;IACD;AACD,UAAO,KAAK,UAAU,YAAY,MAAM,EAAE;;EAE5C,MAAM,QAAQ;GACZ,KAAK,OAAO,QAAQ,OAAO;GAC3B;GACA,kBAAkB,OAAO;GACzB,kBAAkB,OAAO;GACzB,mBAAmB,OAAO;GAC1B,eAAe,OAAO,KAAK,KAAK,KAAK,IAAI;GACzC;GACA;GACA;GACD;AACD,OAAK,MAAM,OAAO,OAAO,UAAU;GACjC,MAAM,OAAO,IAAI,SAAS,cAAc,cAAc,IAAI,SAAS,SAAS,SAAS,IAAI;AACzF,SAAM,KAAK,MAAM,QAAQ,GAAG;GAC5B,MAAM,OAAO,OAAO,IAAI,YAAY,WAAW,IAAI,UAAU,KAAK,UAAU,IAAI,SAAS,MAAM,EAAE;AACjG,SAAM,KAAK,MAAM,IAAI,OAAO,GAAG;;AAEjC,SAAO,MAAM,KAAK,KAAK;;CAGzB,MAAM,WAAwC;EAE5C,MAAM,YAAW,MADE,KAAK,KAAK,EAAE,OAAO,KAAQ,CAAC,EACzB;EACtB,MAAM,YAAoC,EAAE;AAC5C,OAAK,MAAM,KAAK,SACd,WAAU,EAAE,kBAAkB,UAAU,EAAE,kBAAkB,KAAK;EAEnE,IAAI;EACJ,IAAI;AACJ,MAAI,SAAS,SAAS,GAAG;GACvB,MAAM,SAAS,CAAC,GAAG,SAAS,CAAC,MAAM,GAAG,MAAM,EAAE,UAAU,cAAc,EAAE,UAAU,CAAC;AACnF,mBAAgB,OAAO,GAAI;AAC3B,mBAAgB,OAAO,OAAO,SAAS,GAAI;;AAE7C,SAAO;GACL,eAAe,SAAS;GACxB,gBAAgB,SAAS,QAAQ,MAAM,EAAE,WAAA,YAAmC,EAAE,WAAA,OAA8B,CAAC;GAC7G,kBAAkB,SAAS,QAAQ,MAAM,EAAE,WAAA,WAAkC,CAAC;GAC9E,gBAAgB,SAAS,QAAQ,MAAM,EAAE,WAAA,SAAgC,CAAC;GAC1E,eAAe,SAAS,QAAQ,KAAK,MAAM,MAAM,EAAE,cAAc,EAAE;GACnE,aAAa,SAAS,QAAQ,KAAK,MAAM,MAAM,EAAE,iBAAiB,EAAE;GACpE;GACA;GACA;GACD;;CAGH,MAAM,WAAW,eAAwC;EACvD,MAAM,yBAAS,IAAI,MAAM;AACzB,SAAO,QAAQ,OAAO,SAAS,GAAG,cAAc;EAChD,MAAM,OAAO,MAAM,KAAK,KAAK,EAAE,OAAO,KAAQ,CAAC;EAC/C,IAAI,WAAW;AACf,OAAK,MAAM,WAAW,KAAK,MACzB,KAAI,QAAQ,WAAA,cAAqC,QAAQ,WAAA;OAEnD,IADmB,KAAK,QAAQ,eACtB,GAAG,QAAQ;AACvB,UAAM,KAAK,QAAQ,QAAQ,IAAI;AAC/B;;;AAIN,SAAO;;CAGT,eAAe,UAAkC;EAC/C,IAAI,QAAQ;AACZ,OAAK,MAAM,OAAO,SAChB,UAAS,KAAK,KAAK,KAAK,mBAAmB,KAAK,eAAe,IAAI,CAAC,CAAC,SAAS,EAAE;AAElF,SAAO;;CAGT,eAAuB,KAA4B;AACjD,SAAQ,IAA8B;;CAGxC,2BAAmC,KAA4B;AAC7D,MAAI,IAAI,SAAS,OACf,QAAO;EAET,MAAM,OAAO,KAAK,mBAAmB,KAAK,eAAe,IAAI,CAAC,CAAC,MAAM;AACrE,SAAO,sCAAsC,KAAK,KAAK;;CAGzD,uBAA+B,SAA+B;EAC5D,MAAM,SAAS;AACf,SAAO,KAAK,UAAU;GACpB,MAAM,QAAQ;GACd,WAAW,OAAO;GAClB,YAAY,OAAO,cAAc,OAAO;GACxC,UAAU,OAAO;GACjB,SAAS,KAAK,eAAe,QAAQ;GACtC,CAAC;;CAGJ,mBAA2B,SAA0B;AACnD,MAAI,OAAO,YAAY,SACrB,QAAO;AAET,MAAI,MAAM,QAAQ,QAAQ,EAAE;GAC1B,MAAM,QAAkB,EAAE;AAC1B,QAAK,MAAM,QAAQ,SAAS;AAC1B,QAAI,OAAO,SAAS,YAAY,SAAS,QAAQ,EAAE,UAAU,MAC3D;IAEF,MAAM,IAAI;AACV,QAAI,EAAE,SAAS,UAAU,OAAO,EAAE,SAAS,SACzC,OAAM,KAAK,EAAE,KAAK;aACT,EAAE,SAAS,cAAc,EAAE,SAAS,WAC7C,OAAM,KAAK,EAAE,OAAO,IAAI,EAAE,KAAK,KAAK,GAAG;;AAG3C,UAAO,MAAM,KAAK,GAAG;;AAEvB,SAAO;;CAGT,MAAc,4BAA4B,KAA4B;EACpE,MAAM,cAAc,0BAA0B;GAC5C,KAAK,KAAK,QAAQ;GAClB,YAAY;GACb,CAAC;AACF,MAAI,CAAC,aAAa,MAAM,CACtB;EAEF,MAAM,QAAQ,MAAM,KAAK,aAAa,IAAI;AAC1C,MAAI,CAAC,MACH;EAEF,MAAM,iBAAiB,KAAK,yBAAyB,IAAI;EACzD,MAAM,MAAM,KAAK,uBAAuB,OAAO,eAAe;EAC9D,MAAM,eAAe,uCAAuC,KAAK,QAAQ,QAAQ,IAAI,CAAC;AACtF,MAAI;AACF,SAAM,+BAA+B;IACnC,SAAS;IACT,KAAK;IACL,YAAY;IACZ,OAAO;KACL,MAAM;KACN,IAAI,mBAAmB,KAAK,KAAK;KACjC,MAAM;KACN,4BAAW,IAAI,MAAM,EAAC,aAAa;KACpC;IACF,CAAC;WACK,KAAK;AACZ,OAAI,KAAK;IAAE;IAAK;IAAK,EAAE,2CAA2C;;;CAItE,gBAAwB,UAAqC;AAC3D,SAAO,SAAS,KAAK,MAA8C;GACjE,MAAM,IAAI,KAAK,eAAe,EAAE;GAChC,MAAM,UACJ,OAAO,MAAM,WAAW,IAAI,MAAM,QAAQ,EAAE,GAAG,IAAI,KAAK,mBAAmB,EAAE;GAC/E,MAAM,MAAe;IACnB,MAAM,EAAE;IACR;IACA,WAAW,EAAE,YAAY,IAAI,KAAK,EAAE,UAA6B,CAAC,aAAa,GAAG,KAAA;IAClF,cAAe,EAAE,gBAAwC,EAAE;IAC3D,YAAY,EAAE;IACd,MAAM,EAAE;IACT;AACD,OAAI,MAAM,QAAQ,EAAE,YAAY,IAAI,EAAE,YAAY,SAAS,EACzD,KAAI,cAAc,EAAE;GAEtB,MAAM,WAAW,EAAE;AACnB,OAAI,YAAY,OAAO,aAAa,UAAU;IAC5C,MAAM,cAAc,OAAO,SAAS,UAAU,WAAW,SAAS,QAAQ,KAAA;IAC1E,MAAM,eAAe,OAAO,SAAS,WAAW,WAAW,SAAS,SAAS,KAAA;IAC7E,MAAM,cAAc,OAAO,SAAS,gBAAgB,WAChD,SAAS,cACT,OAAO,SAAS,UAAU,WACxB,SAAS,QACT,KAAA;AACN,QAAI,eAAe,QAAQ,gBAAgB,QAAQ,eAAe,KAChE,KAAI,QAAQ;KAAE;KAAa;KAAc;KAAa;;AAG1D,UAAO;IACP"}
|
|
1
|
+
{"version":3,"file":"store.js","names":[],"sources":["../../../src/session/store.ts"],"sourcesContent":["import type { AgentMessage } from '@earendil-works/pi-agent-core';\n\nimport type { Config } from '../config/schema.js';\nimport { resolveStateDir } from '../config/paths-state.js';\nimport { resolveEffectiveAgentProfileForSession } from '../config/agent-profile.js';\nimport { readPostCompactionContext } from '../agent/reply/post-compaction-context.js';\nimport { createLogger } from '../utils/logger.js';\nimport { SessionCompactor, type CompactionConfig, type CompactionResult } from '../agent/memory/compaction.js';\nimport { SlidingWindow, type WindowConfig } from '../agent/memory/window.js';\nimport {\n requireXopcDatabase,\n appendTranscriptEntry,\n captureCompactionCheckpoint,\n deleteSessionRecord,\n ensureSessionRecord,\n estimateTokensFromMessages,\n getCompactionCheckpointDetail,\n getGlobalSessionStats,\n getSessionMetadata,\n listCompactionCheckpoints,\n listSessionMetadata,\n listSessionsByAgent,\n loadCheckpointRows,\n loadLlmMessagesForSession,\n loadTranscriptRowsForSession,\n patchSessionMetadata,\n replaceTranscriptRows,\n resetSessionRecord,\n restoreCompactionCheckpoint,\n} from '../storage/sqlite/index.js';\nimport type { TranscriptCompactionRecord, XopcSessionTranscriptV1 } from './transcript-format.js';\nimport {\n buildSessionContextForLlm,\n mergeLlmMessagesPreservingContextRows,\n type TranscriptStoredRow,\n type XopcTranscriptContextEntry,\n} from './session-context-for-llm.js';\nimport { normalizeCompactionCheckpointId } from './compaction-checkpoints.js';\nimport type {\n SessionMetadata,\n SessionDetail,\n SessionListQuery,\n PaginatedResult,\n GlobalSessionStats,\n ExportFormat,\n SessionExport,\n SessionTranscriptSummary,\n CompactionCheckpointSummary,\n CompactionCheckpointDetail,\n} from './types.js';\nimport { SessionStatus } from './types.js';\nimport type { Message } from './types.js';\nimport type { SessionTranscriptUpdate } from './transcript-events.js';\nimport { isAppendOnlyLlmTranscriptMessage } from './transcript-stats.js';\n\nconst log = createLogger('SessionStore');\n\nconst INDEX_VERSION = '1.0';\n\nexport interface SessionStoreOptions {\n config: Config;\n}\n\nexport class SessionStore {\n private window: SlidingWindow;\n private compactor: SessionCompactor;\n\n constructor(\n private options: SessionStoreOptions,\n windowConfig?: Partial<WindowConfig>,\n compactionConfig?: Partial<CompactionConfig>,\n ) {\n this.window = new SlidingWindow(windowConfig);\n this.compactor = new SessionCompactor(compactionConfig);\n }\n\n private resolveWorkspaceCwd(sessionKey: string): string {\n return resolveEffectiveAgentProfileForSession(this.options.config, sessionKey).resolvedWorkspacePath;\n }\n\n private async runStoreMutation<T>(fn: () => Promise<T>): Promise<T> {\n return fn();\n }\n\n async initialize(): Promise<void> {\n requireXopcDatabase();\n log.debug('Session store initialized (SQLite)');\n }\n\n getSessionsRoot(): string {\n return resolveStateDir();\n }\n\n async resolveTranscriptPath(\n sessionKey: string,\n ): Promise<{ sessionId: string; sessionKey: string }> {\n requireXopcDatabase();\n const cwd = this.resolveWorkspaceCwd(sessionKey);\n const meta = ensureSessionRecord(sessionKey, cwd);\n return { sessionId: meta.transcriptId!, sessionKey };\n }\n\n async appendTranscriptMessage(sessionKey: string, message: AgentMessage): Promise<void> {\n return this.runStoreMutation(async () => {\n requireXopcDatabase();\n const cwd = this.resolveWorkspaceCwd(sessionKey);\n ensureSessionRecord(sessionKey, cwd);\n appendTranscriptEntry(sessionKey, message);\n });\n }\n\n async getByAgent(agentId: string): Promise<SessionMetadata[]> {\n requireXopcDatabase();\n return listSessionsByAgent(agentId);\n }\n\n async getByAccount(accountId: string): Promise<SessionMetadata[]> {\n const { items } = await this.list({ limit: 100_000 });\n return items.filter((m) => m.routing?.accountId === accountId);\n }\n\n async getByPeer(peerKind: string, peerId: string): Promise<SessionMetadata[]> {\n const { items } = await this.list({ limit: 100_000 });\n return items.filter((m) => m.routing?.peerKind === peerKind && m.routing.peerId === peerId);\n }\n\n async getMainSession(channel: string, accountId: string): Promise<SessionMetadata | null> {\n const { items } = await this.list({ limit: 100_000 });\n return (\n items.find(\n (m) =>\n m.routing?.source === channel &&\n m.routing.accountId === accountId &&\n m.routing.peerKind === 'dm' &&\n m.routing.peerId === 'main',\n ) ?? null\n );\n }\n\n async refreshIndex(): Promise<void> {\n /* no-op: SQLite is authoritative */\n }\n\n async list(query: SessionListQuery = {}): Promise<PaginatedResult<SessionMetadata>> {\n requireXopcDatabase();\n return listSessionMetadata(query);\n }\n\n async get(\n key: string,\n options?: { includeTranscriptSummary?: boolean; includeTranscriptRows?: boolean },\n ): Promise<SessionDetail | null> {\n const metadata = await this.getMetadata(key);\n if (!metadata) {\n return null;\n }\n const messages = await this.loadDisplayMessages(key);\n return this.buildSessionDetail(key, metadata, messages, options);\n }\n\n async getMessagePage(\n key: string,\n options: {\n offset?: number;\n limit?: number;\n before?: string;\n includeTranscriptSummary?: boolean;\n includeTranscriptRows?: boolean;\n } = {},\n ): Promise<{\n session: SessionDetail;\n pagination: {\n total: number;\n limit: number;\n offset: number;\n hasMore: boolean;\n before?: string;\n nextBeforeCursor?: string;\n };\n } | null> {\n const metadata = await this.getMetadata(key);\n if (!metadata) {\n return null;\n }\n\n const limit = Math.min(200, Math.max(1, Math.trunc(options.limit ?? 50)));\n const offset = Math.max(0, Math.trunc(options.offset ?? 0));\n const parsedBefore = options.before ? Number.parseInt(options.before, 10) : undefined;\n const hasBeforeCursor = parsedBefore !== undefined && Number.isFinite(parsedBefore);\n\n const displayMessages = await this.loadDisplayMessages(key);\n const page = this.paginateDisplayMessages(displayMessages, {\n limit,\n offset: hasBeforeCursor ? undefined : offset,\n beforeEndIndex: hasBeforeCursor ? parsedBefore : undefined,\n });\n\n const session = await this.buildSessionDetail(key, metadata, page.messages, options);\n const nextBeforeCursor = page.startIndex > 0 ? String(page.startIndex) : undefined;\n\n return {\n session,\n pagination: {\n total: page.total,\n limit,\n offset,\n hasMore: hasBeforeCursor ? page.startIndex > 0 : offset + limit < page.total,\n ...(hasBeforeCursor ? { before: String(page.endIndex) } : {}),\n ...(nextBeforeCursor ? { nextBeforeCursor } : {}),\n },\n };\n }\n\n private async buildSessionDetail(\n key: string,\n metadata: SessionMetadata,\n messages: AgentMessage[],\n options?: { includeTranscriptSummary?: boolean; includeTranscriptRows?: boolean },\n ): Promise<SessionDetail> {\n let transcriptSummary: SessionTranscriptSummary | undefined;\n if (options?.includeTranscriptSummary) {\n const env = await this.loadTranscriptDocument(key);\n if (env) {\n transcriptSummary = {\n id: env.id,\n version: env.version,\n createdAt: env.createdAt,\n updatedAt: env.updatedAt,\n compactionCount: env.compactions?.length ?? 0,\n };\n }\n }\n let transcriptRows: TranscriptStoredRow[] | undefined;\n if (options?.includeTranscriptRows) {\n transcriptRows = await this.loadTranscriptRows(key);\n }\n return {\n ...metadata,\n messages: this.convertMessages(messages),\n ...(transcriptSummary ? { transcriptSummary } : {}),\n ...(transcriptRows !== undefined ? { transcriptRows } : {}),\n };\n }\n\n async loadTranscriptRows(key: string): Promise<TranscriptStoredRow[]> {\n requireXopcDatabase();\n return loadTranscriptRowsForSession(key);\n }\n\n async getMetadata(key: string): Promise<SessionMetadata | null> {\n requireXopcDatabase();\n return getSessionMetadata(key);\n }\n\n async updateMetadata(key: string, updates: Partial<SessionMetadata>): Promise<void> {\n return this.runStoreMutation(async () => {\n requireXopcDatabase();\n patchSessionMetadata(key, updates);\n log.debug({ key, updates }, 'Session metadata updated');\n });\n }\n\n async reset(key: string): Promise<{ sessionId: string; previousSessionId: string } | null> {\n return this.runStoreMutation(async () => {\n requireXopcDatabase();\n const cwd = this.resolveWorkspaceCwd(key);\n const outcome = resetSessionRecord(key, cwd);\n if (outcome) {\n log.info({ key, ...outcome }, 'Session reset');\n }\n return outcome;\n });\n }\n\n async delete(key: string): Promise<boolean> {\n return this.runStoreMutation(async () => {\n requireXopcDatabase();\n const ok = deleteSessionRecord(key);\n if (ok) {\n log.info({ key }, 'Session deleted');\n }\n return ok;\n });\n }\n\n async deleteMany(keys: string[]): Promise<{ success: string[]; failed: string[] }> {\n const success: string[] = [];\n const failed: string[] = [];\n for (const key of keys) {\n try {\n if (await this.delete(key)) {\n success.push(key);\n } else {\n failed.push(key);\n }\n } catch {\n failed.push(key);\n }\n }\n return { success, failed };\n }\n\n async setStatus(key: string, status: SessionStatus): Promise<void> {\n await this.updateMetadata(key, { status });\n }\n\n async archive(key: string): Promise<void> {\n await this.setStatus(key, SessionStatus.ARCHIVED);\n }\n\n async unarchive(key: string): Promise<void> {\n await this.setStatus(key, SessionStatus.ACTIVE);\n }\n\n async pin(key: string): Promise<void> {\n await this.setStatus(key, SessionStatus.PINNED);\n }\n\n async unpin(key: string): Promise<void> {\n await this.setStatus(key, SessionStatus.ACTIVE);\n }\n\n async loadMessages(_key: string, _options?: { fromArchive?: boolean }): Promise<AgentMessage[]> {\n requireXopcDatabase();\n return loadLlmMessagesForSession(_key);\n }\n\n async loadTranscriptDocument(key: string): Promise<XopcSessionTranscriptV1 | null> {\n const metadata = await this.getMetadata(key);\n if (!metadata?.transcriptId) {\n return null;\n }\n const rows = await this.loadTranscriptRows(key);\n const compactions: TranscriptCompactionRecord[] = [];\n for (const row of rows) {\n if ((row as { type?: string }).type === 'compaction') {\n const c = row as unknown as TranscriptCompactionRecord & { type: 'compaction' };\n compactions.push({\n at: c.at ?? new Date().toISOString(),\n summary: c.summary ?? '',\n firstKeptIndex: Number(c.firstKeptIndex ?? 0),\n tokensBefore: c.tokensBefore ?? 0,\n tokensAfter: (c as { tokensAfter?: number }).tokensAfter ?? 0,\n });\n }\n }\n return {\n type: 'xopc_session_transcript',\n version: 1,\n id: metadata.transcriptId,\n createdAt: metadata.createdAt,\n updatedAt: metadata.updatedAt,\n messages: rows,\n ...(compactions.length > 0 ? { compactions } : {}),\n };\n }\n\n async syncEmbeddedTranscriptUpdate(update: SessionTranscriptUpdate): Promise<void> {\n const sessionKey = update.sessionKey?.trim();\n if (!sessionKey) {\n return;\n }\n return this.runStoreMutation(async () => {\n requireXopcDatabase();\n const cwd = this.resolveWorkspaceCwd(sessionKey);\n ensureSessionRecord(sessionKey, cwd);\n\n if (update.message && isAppendOnlyLlmTranscriptMessage(update.message)) {\n appendTranscriptEntry(sessionKey, update.message as AgentMessage);\n }\n });\n }\n\n async appendTranscriptContextEntry(\n key: string,\n entry: Omit<XopcTranscriptContextEntry, 'kind'> & Partial<Pick<XopcTranscriptContextEntry, 'kind'>>,\n ): Promise<void> {\n return this.runStoreMutation(async () => {\n requireXopcDatabase();\n const cwd = this.resolveWorkspaceCwd(key);\n ensureSessionRecord(key, cwd);\n const row: XopcTranscriptContextEntry = {\n kind: 'context',\n id: typeof entry.id === 'string' ? entry.id : undefined,\n text: typeof entry.text === 'string' ? entry.text : undefined,\n data: entry.data,\n createdAt: entry.createdAt ?? new Date().toISOString(),\n };\n appendTranscriptEntry(key, row);\n });\n }\n\n async saveMessages(key: string, messages: AgentMessage[]): Promise<void> {\n return this.runStoreMutation(async () => {\n requireXopcDatabase();\n const cwd = this.resolveWorkspaceCwd(key);\n ensureSessionRecord(key, cwd);\n const prev = await this.loadTranscriptRows(key);\n const merged = mergeLlmMessagesPreservingContextRows(prev, messages);\n replaceTranscriptRows(key, merged);\n });\n }\n\n getWindowStats(messages: AgentMessage[]) {\n return this.window.getStats(messages);\n }\n\n needsCompaction(key: string, messages: AgentMessage[], contextWindow: number) {\n return this.compactor.needsCompaction(messages, contextWindow);\n }\n\n prepareCompaction(key: string, messages: AgentMessage[], contextWindow: number) {\n const result = this.compactor.needsCompaction(messages, contextWindow);\n return { needsCompaction: result.needed, messages, stats: result };\n }\n\n async applyCompaction(\n key: string,\n messages: AgentMessage[],\n result: CompactionResult,\n ): Promise<AgentMessage[]> {\n const compacted = this.compactor.applyCompaction(messages, result);\n return this.runStoreMutation(async () => {\n const prev = await this.loadTranscriptRows(key);\n const merged = mergeLlmMessagesPreservingContextRows(prev, compacted);\n captureCompactionCheckpoint(key);\n replaceTranscriptRows(key, merged, {\n appendCompaction: {\n at: new Date().toISOString(),\n summary: result.summary,\n firstKeptIndex: result.firstKeptIndex,\n tokensBefore: result.tokensBefore,\n tokensAfter: result.tokensAfter,\n },\n });\n const metadata = await this.getMetadata(key);\n if (metadata) {\n await this.updateMetadata(key, { compactedCount: metadata.compactedCount + 1 });\n }\n log.info(\n { key, tokensBefore: result.tokensBefore, tokensAfter: result.tokensAfter, keptMessages: compacted.length },\n 'Session compacted',\n );\n await this.injectPostCompactionContext(key);\n return compacted;\n });\n }\n\n async compact(\n key: string,\n messages: AgentMessage[],\n contextWindow: number,\n instructions?: string,\n force?: boolean,\n ): Promise<CompactionResult> {\n void contextWindow;\n const result = await this.compactor.compact(messages, instructions, force);\n if (result.compacted) {\n await this.applyCompaction(key, messages, result);\n }\n return result;\n }\n\n async getCompactionStats(key: string) {\n const metadata = await this.getMetadata(key);\n if (!metadata) {\n return undefined;\n }\n return {\n compactionCount: metadata.compactedCount,\n totalTokensBefore: 0,\n totalTokensAfter: 0,\n lastCompactionAt: undefined,\n };\n }\n\n async listCompactionCheckpoints(key: string): Promise<CompactionCheckpointSummary[]> {\n requireXopcDatabase();\n return listCompactionCheckpoints(key);\n }\n\n async getCompactionCheckpointDetail(\n key: string,\n checkpointId: string,\n ): Promise<CompactionCheckpointDetail | null> {\n requireXopcDatabase();\n const id = normalizeCompactionCheckpointId(checkpointId);\n if (!id) {\n return null;\n }\n return getCompactionCheckpointDetail(key, id);\n }\n\n async restoreCompactionCheckpoint(key: string, checkpointId: string): Promise<void> {\n return this.runStoreMutation(async () => {\n requireXopcDatabase();\n const id = normalizeCompactionCheckpointId(checkpointId);\n if (!id) {\n throw new Error(`Invalid checkpoint id: ${checkpointId}`);\n }\n restoreCompactionCheckpoint(key, id);\n });\n }\n\n async deleteSession(key: string): Promise<boolean> {\n return this.delete(key);\n }\n\n async load(key: string, options?: { fromArchive?: boolean }): Promise<AgentMessage[]> {\n return this.loadMessages(key, options);\n }\n\n async estimateTokenUsage(_key: string, messages: AgentMessage[]): Promise<number> {\n return this.estimateTokens(messages);\n }\n\n async searchInSession(key: string, keyword: string): Promise<Message[]> {\n const messages = await this.loadDisplayMessages(key);\n const q = keyword.toLowerCase();\n return this.convertMessages(\n messages.filter((m) => this.extractTextContent(this.messageContent(m)).toLowerCase().includes(q)),\n );\n }\n\n async exportSession(key: string, format: ExportFormat): Promise<string> {\n const metadata = await this.getMetadata(key);\n if (!metadata) {\n throw new Error(`Session not found: ${key}`);\n }\n const rows = await this.loadTranscriptRows(key);\n const messages = this.convertMessages(buildSessionContextForLlm(rows));\n const payload: SessionExport = {\n version: INDEX_VERSION,\n exportedAt: new Date().toISOString(),\n metadata,\n messages,\n transcriptRows: rows,\n };\n if (format === 'json') {\n return JSON.stringify(payload, null, 2);\n }\n const lines = [`# ${metadata.name ?? metadata.key}`, '', `Exported: ${payload.exportedAt}`, ''];\n for (const msg of messages) {\n lines.push(`## ${msg.role}`, '', typeof msg.content === 'string' ? msg.content : JSON.stringify(msg.content), '');\n }\n return lines.join('\\n');\n }\n\n async getStats(): Promise<GlobalSessionStats> {\n requireXopcDatabase();\n return getGlobalSessionStats();\n }\n\n async archiveOld(olderThanDays: number): Promise<number> {\n const cutoff = new Date();\n cutoff.setDate(cutoff.getDate() - olderThanDays);\n const list = await this.list({ limit: 100_000 });\n let archived = 0;\n for (const session of list.items) {\n if (session.status !== SessionStatus.ARCHIVED && session.status !== SessionStatus.PINNED) {\n const lastAccess = new Date(session.lastAccessedAt);\n if (lastAccess < cutoff) {\n await this.archive(session.key);\n archived++;\n }\n }\n }\n return archived;\n }\n\n estimateTokens(messages: AgentMessage[]): number {\n return estimateTokensFromMessages(messages);\n }\n\n private async loadDisplayMessages(key: string): Promise<AgentMessage[]> {\n requireXopcDatabase();\n const checkpoints = listCompactionCheckpoints(key);\n const rowSets: TranscriptStoredRow[][] = [];\n for (const checkpoint of [...checkpoints].reverse()) {\n rowSets.push(loadCheckpointRows(key, checkpoint.id));\n }\n rowSets.push(await this.loadTranscriptRows(key));\n\n const messages: AgentMessage[] = [];\n const seen = new Set<string>();\n for (const rows of rowSets) {\n for (const message of buildSessionContextForLlm(rows)) {\n if (this.isCompactionSummaryMessage(message)) {\n continue;\n }\n const identity = this.displayMessageIdentity(message);\n if (seen.has(identity)) {\n continue;\n }\n seen.add(identity);\n messages.push(message);\n }\n }\n return messages;\n }\n\n private paginateDisplayMessages(\n messages: AgentMessage[],\n options: {\n offset?: number;\n limit?: number;\n beforeEndIndex?: number;\n } = {},\n ): { messages: AgentMessage[]; total: number; startIndex: number; endIndex: number } {\n const total = messages.length;\n const limit = Math.min(200, Math.max(1, Math.trunc(options.limit ?? 50)));\n const offset = Math.max(0, Math.trunc(options.offset ?? 0));\n\n let startIndex: number;\n let endIndex: number;\n if (options.beforeEndIndex !== undefined && Number.isFinite(options.beforeEndIndex)) {\n endIndex = Math.min(total, Math.max(0, Math.trunc(options.beforeEndIndex)));\n startIndex = Math.max(0, endIndex - limit);\n } else {\n endIndex = Math.max(0, total - offset);\n startIndex = Math.max(0, endIndex - limit);\n }\n\n return {\n messages: messages.slice(startIndex, endIndex),\n total,\n startIndex,\n endIndex,\n };\n }\n\n private displayMessageIdentity(message: AgentMessage): string {\n const record = message as unknown as Record<string, unknown>;\n return JSON.stringify({\n role: message.role,\n timestamp: record.timestamp,\n toolCallId: record.toolCallId ?? record.tool_call_id,\n toolName: record.toolName,\n content: this.messageContent(message),\n });\n }\n\n private messageContent(msg: AgentMessage): unknown {\n return (msg as { content?: unknown }).content;\n }\n\n private isCompactionSummaryMessage(msg: AgentMessage): boolean {\n if (msg.role !== 'user') {\n return false;\n }\n const text = this.extractTextContent(this.messageContent(msg)).trim();\n return /^\\[Previous conversation summary\\]/i.test(text);\n }\n\n private extractTextContent(content: unknown): string {\n if (typeof content === 'string') {\n return content;\n }\n if (Array.isArray(content)) {\n const parts: string[] = [];\n for (const item of content) {\n if (typeof item !== 'object' || item === null || !('type' in item)) {\n continue;\n }\n const c = item as { type?: string; text?: string; name?: string };\n if (c.type === 'text' && typeof c.text === 'string') {\n parts.push(c.text);\n } else if (c.type === 'toolCall' || c.type === 'tool_use') {\n parts.push(c.name ? `[${c.name}]` : '');\n }\n }\n return parts.join('');\n }\n return '';\n }\n\n private async injectPostCompactionContext(key: string): Promise<void> {\n const contextText = readPostCompactionContext({\n cfg: this.options.config,\n sessionKey: key,\n });\n if (!contextText?.trim()) {\n return;\n }\n try {\n await this.appendTranscriptContextEntry(key, {\n id: `post-compaction-${Date.now()}`,\n text: contextText,\n createdAt: new Date().toISOString(),\n });\n } catch (err) {\n log.warn({ err, key }, 'Post-compaction context injection failed');\n }\n }\n\n private convertMessages(messages: AgentMessage[]): Message[] {\n return messages.map((m: AgentMessage & Record<string, unknown>) => {\n const c = this.messageContent(m);\n const content: string | unknown[] =\n typeof c === 'string' ? c : Array.isArray(c) ? c : this.extractTextContent(c);\n const row: Message = {\n role: m.role as Message['role'],\n content,\n timestamp: m.timestamp ? new Date(m.timestamp as string | number).toISOString() : undefined,\n tool_call_id: (m.tool_call_id as string | undefined) || (m.toolCallId as string | undefined),\n tool_calls: m.tool_calls as Message['tool_calls'],\n name: m.name as string | undefined,\n };\n if (Array.isArray(m.attachments) && m.attachments.length > 0) {\n row.attachments = m.attachments as Message['attachments'];\n }\n const rawUsage = m.usage as { input?: number; output?: number; totalTokens?: number; total?: number } | undefined;\n if (rawUsage && typeof rawUsage === 'object') {\n const inputTokens = typeof rawUsage.input === 'number' ? rawUsage.input : undefined;\n const outputTokens = typeof rawUsage.output === 'number' ? rawUsage.output : undefined;\n const totalTokens = typeof rawUsage.totalTokens === 'number'\n ? rawUsage.totalTokens\n : typeof rawUsage.total === 'number'\n ? rawUsage.total\n : undefined;\n if (inputTokens !== undefined || outputTokens !== undefined || totalTokens !== undefined) {\n row.usage = { inputTokens, outputTokens, totalTokens };\n }\n }\n return row;\n });\n }\n}\n"],"mappings":"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"}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import { type FileEntry } from '@earendil-works/pi-coding-agent';
|
|
2
|
+
import type { TranscriptCompactionRecord } from './transcript-format.js';
|
|
3
|
+
import { type TranscriptStoredRow } from './session-context-for-llm.js';
|
|
4
|
+
export declare const XOPC_CONTEXT_CUSTOM_TYPE = "xopc:transcript-row";
|
|
5
|
+
/** Convert SQLite transcript rows into pi-coding-agent JSONL file entries (header + tree). */
|
|
6
|
+
export declare function storedRowsToFileEntries(params: {
|
|
7
|
+
sessionId: string;
|
|
8
|
+
cwd: string;
|
|
9
|
+
rows: TranscriptStoredRow[];
|
|
10
|
+
appendCompaction?: TranscriptCompactionRecord;
|
|
11
|
+
}): FileEntry[];
|
|
@@ -0,0 +1,95 @@
|
|
|
1
|
+
import { isTranscriptContextEntry } from "./session-context-for-llm.js";
|
|
2
|
+
import { randomUUID } from "node:crypto";
|
|
3
|
+
import { CURRENT_SESSION_VERSION } from "@earendil-works/pi-coding-agent";
|
|
4
|
+
//#region src/session/stored-rows-to-file-entries.ts
|
|
5
|
+
const XOPC_CONTEXT_CUSTOM_TYPE = "xopc:transcript-row";
|
|
6
|
+
function generateShortId(byId) {
|
|
7
|
+
for (let i = 0; i < 100; i++) {
|
|
8
|
+
const id = randomUUID().slice(0, 8);
|
|
9
|
+
if (!byId.has(id)) return id;
|
|
10
|
+
}
|
|
11
|
+
return randomUUID();
|
|
12
|
+
}
|
|
13
|
+
function contextRowToCustomEntry(row, parentId, byId) {
|
|
14
|
+
return {
|
|
15
|
+
type: "custom",
|
|
16
|
+
customType: XOPC_CONTEXT_CUSTOM_TYPE,
|
|
17
|
+
id: generateShortId(byId),
|
|
18
|
+
parentId,
|
|
19
|
+
timestamp: row.createdAt ?? (/* @__PURE__ */ new Date()).toISOString(),
|
|
20
|
+
data: {
|
|
21
|
+
kind: "context",
|
|
22
|
+
id: row.id,
|
|
23
|
+
text: row.text,
|
|
24
|
+
data: row.data,
|
|
25
|
+
createdAt: row.createdAt
|
|
26
|
+
}
|
|
27
|
+
};
|
|
28
|
+
}
|
|
29
|
+
function compactionRecordToEntry(rec, parentId, byId) {
|
|
30
|
+
return {
|
|
31
|
+
type: "compaction",
|
|
32
|
+
id: generateShortId(byId),
|
|
33
|
+
parentId,
|
|
34
|
+
timestamp: rec.at,
|
|
35
|
+
summary: rec.summary,
|
|
36
|
+
firstKeptEntryId: String(rec.firstKeptIndex),
|
|
37
|
+
tokensBefore: rec.tokensBefore,
|
|
38
|
+
details: {
|
|
39
|
+
tokensAfter: rec.tokensAfter,
|
|
40
|
+
xopc: true
|
|
41
|
+
},
|
|
42
|
+
fromHook: true
|
|
43
|
+
};
|
|
44
|
+
}
|
|
45
|
+
function agentMessageToEntry(msg, parentId, byId) {
|
|
46
|
+
return {
|
|
47
|
+
type: "message",
|
|
48
|
+
id: generateShortId(byId),
|
|
49
|
+
parentId,
|
|
50
|
+
timestamp: (/* @__PURE__ */ new Date()).toISOString(),
|
|
51
|
+
message: msg
|
|
52
|
+
};
|
|
53
|
+
}
|
|
54
|
+
/** Convert SQLite transcript rows into pi-coding-agent JSONL file entries (header + tree). */
|
|
55
|
+
function storedRowsToFileEntries(params) {
|
|
56
|
+
const byId = /* @__PURE__ */ new Set();
|
|
57
|
+
const entries = [{
|
|
58
|
+
type: "session",
|
|
59
|
+
version: CURRENT_SESSION_VERSION,
|
|
60
|
+
id: params.sessionId,
|
|
61
|
+
timestamp: (/* @__PURE__ */ new Date()).toISOString(),
|
|
62
|
+
cwd: params.cwd
|
|
63
|
+
}];
|
|
64
|
+
let parentId = null;
|
|
65
|
+
for (const row of params.rows) {
|
|
66
|
+
if (isTranscriptContextEntry(row)) {
|
|
67
|
+
const entry = contextRowToCustomEntry(row, parentId, byId);
|
|
68
|
+
byId.add(entry.id);
|
|
69
|
+
entries.push(entry);
|
|
70
|
+
parentId = entry.id;
|
|
71
|
+
continue;
|
|
72
|
+
}
|
|
73
|
+
if (row.type === "compaction") {
|
|
74
|
+
const entry = compactionRecordToEntry(row, parentId, byId);
|
|
75
|
+
byId.add(entry.id);
|
|
76
|
+
entries.push(entry);
|
|
77
|
+
parentId = entry.id;
|
|
78
|
+
continue;
|
|
79
|
+
}
|
|
80
|
+
const entry = agentMessageToEntry(row, parentId, byId);
|
|
81
|
+
byId.add(entry.id);
|
|
82
|
+
entries.push(entry);
|
|
83
|
+
parentId = entry.id;
|
|
84
|
+
}
|
|
85
|
+
if (params.appendCompaction) {
|
|
86
|
+
const entry = compactionRecordToEntry(params.appendCompaction, parentId, byId);
|
|
87
|
+
byId.add(entry.id);
|
|
88
|
+
entries.push(entry);
|
|
89
|
+
}
|
|
90
|
+
return entries;
|
|
91
|
+
}
|
|
92
|
+
//#endregion
|
|
93
|
+
export { XOPC_CONTEXT_CUSTOM_TYPE, storedRowsToFileEntries };
|
|
94
|
+
|
|
95
|
+
//# sourceMappingURL=stored-rows-to-file-entries.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"stored-rows-to-file-entries.js","names":[],"sources":["../../../src/session/stored-rows-to-file-entries.ts"],"sourcesContent":["import { randomUUID } from 'node:crypto';\n\nimport type { AgentMessage } from '@earendil-works/pi-agent-core';\nimport {\n CURRENT_SESSION_VERSION,\n type CompactionEntry,\n type CustomEntry,\n type FileEntry,\n type SessionMessageEntry,\n} from '@earendil-works/pi-coding-agent';\n\nimport type { TranscriptCompactionRecord } from './transcript-format.js';\nimport {\n isTranscriptContextEntry,\n type TranscriptStoredRow,\n type XopcTranscriptContextEntry,\n} from './session-context-for-llm.js';\n\nexport const XOPC_CONTEXT_CUSTOM_TYPE = 'xopc:transcript-row';\n\nfunction generateShortId(byId: Set<string>): string {\n for (let i = 0; i < 100; i++) {\n const id = randomUUID().slice(0, 8);\n if (!byId.has(id)) {\n return id;\n }\n }\n return randomUUID();\n}\n\nfunction contextRowToCustomEntry(\n row: XopcTranscriptContextEntry,\n parentId: string | null,\n byId: Set<string>,\n): CustomEntry {\n return {\n type: 'custom',\n customType: XOPC_CONTEXT_CUSTOM_TYPE,\n id: generateShortId(byId),\n parentId,\n timestamp: row.createdAt ?? new Date().toISOString(),\n data: {\n kind: 'context',\n id: row.id,\n text: row.text,\n data: row.data,\n createdAt: row.createdAt,\n },\n };\n}\n\nfunction compactionRecordToEntry(\n rec: TranscriptCompactionRecord,\n parentId: string | null,\n byId: Set<string>,\n): CompactionEntry {\n return {\n type: 'compaction',\n id: generateShortId(byId),\n parentId,\n timestamp: rec.at,\n summary: rec.summary,\n firstKeptEntryId: String(rec.firstKeptIndex),\n tokensBefore: rec.tokensBefore,\n details: { tokensAfter: rec.tokensAfter, xopc: true },\n fromHook: true,\n };\n}\n\nfunction agentMessageToEntry(\n msg: AgentMessage,\n parentId: string | null,\n byId: Set<string>,\n): SessionMessageEntry {\n return {\n type: 'message',\n id: generateShortId(byId),\n parentId,\n timestamp: new Date().toISOString(),\n message: msg,\n };\n}\n\n/** Convert SQLite transcript rows into pi-coding-agent JSONL file entries (header + tree). */\nexport function storedRowsToFileEntries(params: {\n sessionId: string;\n cwd: string;\n rows: TranscriptStoredRow[];\n appendCompaction?: TranscriptCompactionRecord;\n}): FileEntry[] {\n const byId = new Set<string>();\n const entries: FileEntry[] = [\n {\n type: 'session',\n version: CURRENT_SESSION_VERSION,\n id: params.sessionId,\n timestamp: new Date().toISOString(),\n cwd: params.cwd,\n },\n ];\n\n let parentId: string | null = null;\n for (const row of params.rows) {\n if (isTranscriptContextEntry(row)) {\n const entry = contextRowToCustomEntry(row, parentId, byId);\n byId.add(entry.id);\n entries.push(entry);\n parentId = entry.id;\n continue;\n }\n if ((row as { type?: string }).type === 'compaction') {\n const entry = compactionRecordToEntry(row as unknown as TranscriptCompactionRecord, parentId, byId);\n byId.add(entry.id);\n entries.push(entry);\n parentId = entry.id;\n continue;\n }\n const entry = agentMessageToEntry(row as AgentMessage, parentId, byId);\n byId.add(entry.id);\n entries.push(entry);\n parentId = entry.id;\n }\n\n if (params.appendCompaction) {\n const entry = compactionRecordToEntry(params.appendCompaction, parentId, byId);\n byId.add(entry.id);\n entries.push(entry);\n }\n\n return entries;\n}\n"],"mappings":";;;;AAkBA,MAAa,2BAA2B;AAExC,SAAS,gBAAgB,MAA2B;AAClD,MAAK,IAAI,IAAI,GAAG,IAAI,KAAK,KAAK;EAC5B,MAAM,KAAK,YAAY,CAAC,MAAM,GAAG,EAAE;AACnC,MAAI,CAAC,KAAK,IAAI,GAAG,CACf,QAAO;;AAGX,QAAO,YAAY;;AAGrB,SAAS,wBACP,KACA,UACA,MACa;AACb,QAAO;EACL,MAAM;EACN,YAAY;EACZ,IAAI,gBAAgB,KAAK;EACzB;EACA,WAAW,IAAI,8BAAa,IAAI,MAAM,EAAC,aAAa;EACpD,MAAM;GACJ,MAAM;GACN,IAAI,IAAI;GACR,MAAM,IAAI;GACV,MAAM,IAAI;GACV,WAAW,IAAI;GAChB;EACF;;AAGH,SAAS,wBACP,KACA,UACA,MACiB;AACjB,QAAO;EACL,MAAM;EACN,IAAI,gBAAgB,KAAK;EACzB;EACA,WAAW,IAAI;EACf,SAAS,IAAI;EACb,kBAAkB,OAAO,IAAI,eAAe;EAC5C,cAAc,IAAI;EAClB,SAAS;GAAE,aAAa,IAAI;GAAa,MAAM;GAAM;EACrD,UAAU;EACX;;AAGH,SAAS,oBACP,KACA,UACA,MACqB;AACrB,QAAO;EACL,MAAM;EACN,IAAI,gBAAgB,KAAK;EACzB;EACA,4BAAW,IAAI,MAAM,EAAC,aAAa;EACnC,SAAS;EACV;;;AAIH,SAAgB,wBAAwB,QAKxB;CACd,MAAM,uBAAO,IAAI,KAAa;CAC9B,MAAM,UAAuB,CAC3B;EACE,MAAM;EACN,SAAS;EACT,IAAI,OAAO;EACX,4BAAW,IAAI,MAAM,EAAC,aAAa;EACnC,KAAK,OAAO;EACb,CACF;CAED,IAAI,WAA0B;AAC9B,MAAK,MAAM,OAAO,OAAO,MAAM;AAC7B,MAAI,yBAAyB,IAAI,EAAE;GACjC,MAAM,QAAQ,wBAAwB,KAAK,UAAU,KAAK;AAC1D,QAAK,IAAI,MAAM,GAAG;AAClB,WAAQ,KAAK,MAAM;AACnB,cAAW,MAAM;AACjB;;AAEF,MAAK,IAA0B,SAAS,cAAc;GACpD,MAAM,QAAQ,wBAAwB,KAA8C,UAAU,KAAK;AACnG,QAAK,IAAI,MAAM,GAAG;AAClB,WAAQ,KAAK,MAAM;AACnB,cAAW,MAAM;AACjB;;EAEF,MAAM,QAAQ,oBAAoB,KAAqB,UAAU,KAAK;AACtE,OAAK,IAAI,MAAM,GAAG;AAClB,UAAQ,KAAK,MAAM;AACnB,aAAW,MAAM;;AAGnB,KAAI,OAAO,kBAAkB;EAC3B,MAAM,QAAQ,wBAAwB,OAAO,kBAAkB,UAAU,KAAK;AAC9E,OAAK,IAAI,MAAM,GAAG;AAClB,UAAQ,KAAK,MAAM;;AAGrB,QAAO"}
|
|
@@ -1,10 +1,9 @@
|
|
|
1
1
|
export type SessionTranscriptUpdate = {
|
|
2
|
-
sessionFile: string;
|
|
3
2
|
sessionKey?: string;
|
|
4
3
|
message?: unknown;
|
|
5
4
|
messageId?: string;
|
|
6
5
|
};
|
|
7
6
|
type SessionTranscriptListener = (update: SessionTranscriptUpdate) => void;
|
|
8
7
|
export declare function onSessionTranscriptUpdate(listener: SessionTranscriptListener): () => void;
|
|
9
|
-
export declare function emitSessionTranscriptUpdate(update:
|
|
8
|
+
export declare function emitSessionTranscriptUpdate(update: SessionTranscriptUpdate): void;
|
|
10
9
|
export {};
|
|
@@ -9,19 +9,12 @@ function onSessionTranscriptUpdate(listener) {
|
|
|
9
9
|
};
|
|
10
10
|
}
|
|
11
11
|
function emitSessionTranscriptUpdate(update) {
|
|
12
|
-
const
|
|
13
|
-
|
|
14
|
-
sessionKey: update.sessionKey,
|
|
15
|
-
message: update.message,
|
|
16
|
-
messageId: update.messageId
|
|
17
|
-
};
|
|
18
|
-
const trimmed = normalizeOptionalString(normalized.sessionFile);
|
|
19
|
-
if (!trimmed) return;
|
|
12
|
+
const sessionKey = normalizeOptionalString(update.sessionKey);
|
|
13
|
+
if (!sessionKey) return;
|
|
20
14
|
const nextUpdate = {
|
|
21
|
-
|
|
22
|
-
...
|
|
23
|
-
...
|
|
24
|
-
...normalizeOptionalString(normalized.messageId) ? { messageId: normalizeOptionalString(normalized.messageId) } : {}
|
|
15
|
+
...sessionKey ? { sessionKey } : {},
|
|
16
|
+
...update.message !== void 0 ? { message: update.message } : {},
|
|
17
|
+
...normalizeOptionalString(update.messageId) ? { messageId: normalizeOptionalString(update.messageId) } : {}
|
|
25
18
|
};
|
|
26
19
|
for (const listener of SESSION_TRANSCRIPT_LISTENERS) try {
|
|
27
20
|
listener(nextUpdate);
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"transcript-events.js","names":[],"sources":["../../../src/session/transcript-events.ts"],"sourcesContent":["import { normalizeOptionalString } from '../utils/string-coerce.js';\n\nexport type SessionTranscriptUpdate = {\n
|
|
1
|
+
{"version":3,"file":"transcript-events.js","names":[],"sources":["../../../src/session/transcript-events.ts"],"sourcesContent":["import { normalizeOptionalString } from '../utils/string-coerce.js';\n\nexport type SessionTranscriptUpdate = {\n sessionKey?: string;\n message?: unknown;\n messageId?: string;\n};\n\ntype SessionTranscriptListener = (update: SessionTranscriptUpdate) => void;\n\nconst SESSION_TRANSCRIPT_LISTENERS = new Set<SessionTranscriptListener>();\n\nexport function onSessionTranscriptUpdate(listener: SessionTranscriptListener): () => void {\n SESSION_TRANSCRIPT_LISTENERS.add(listener);\n return () => {\n SESSION_TRANSCRIPT_LISTENERS.delete(listener);\n };\n}\n\nexport function emitSessionTranscriptUpdate(update: SessionTranscriptUpdate): void {\n const sessionKey = normalizeOptionalString(update.sessionKey);\n if (!sessionKey) {\n return;\n }\n const nextUpdate: SessionTranscriptUpdate = {\n ...(sessionKey ? { sessionKey } : {}),\n ...(update.message !== undefined ? { message: update.message } : {}),\n ...(normalizeOptionalString(update.messageId)\n ? { messageId: normalizeOptionalString(update.messageId) }\n : {}),\n };\n for (const listener of SESSION_TRANSCRIPT_LISTENERS) {\n try {\n listener(nextUpdate);\n } catch {\n /* ignore */\n }\n }\n}\n"],"mappings":";;oBAAoE;AAUpE,MAAM,+CAA+B,IAAI,KAAgC;AAEzE,SAAgB,0BAA0B,UAAiD;AACzF,8BAA6B,IAAI,SAAS;AAC1C,cAAa;AACX,+BAA6B,OAAO,SAAS;;;AAIjD,SAAgB,4BAA4B,QAAuC;CACjF,MAAM,aAAa,wBAAwB,OAAO,WAAW;AAC7D,KAAI,CAAC,WACH;CAEF,MAAM,aAAsC;EAC1C,GAAI,aAAa,EAAE,YAAY,GAAG,EAAE;EACpC,GAAI,OAAO,YAAY,KAAA,IAAY,EAAE,SAAS,OAAO,SAAS,GAAG,EAAE;EACnE,GAAI,wBAAwB,OAAO,UAAU,GACzC,EAAE,WAAW,wBAAwB,OAAO,UAAU,EAAE,GACxD,EAAE;EACP;AACD,MAAK,MAAM,YAAY,6BACrB,KAAI;AACF,WAAS,WAAW;SACd"}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* API-level transcript summary types (
|
|
2
|
+
* API-level transcript summary types (authoritative storage is SQLite `transcript_entries`).
|
|
3
3
|
*/
|
|
4
4
|
import type { TranscriptStoredRow } from './session-context-for-llm.js';
|
|
5
5
|
export declare const XOPC_SESSION_TRANSCRIPT_TYPE: "xopc_session_transcript";
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"transcript-format.js","names":[],"sources":["../../../src/session/transcript-format.ts"],"sourcesContent":["/**\n * API-level transcript summary types (
|
|
1
|
+
{"version":3,"file":"transcript-format.js","names":[],"sources":["../../../src/session/transcript-format.ts"],"sourcesContent":["/**\n * API-level transcript summary types (authoritative storage is SQLite `transcript_entries`).\n */\n\nimport type { TranscriptStoredRow } from './session-context-for-llm.js';\n\nexport const XOPC_SESSION_TRANSCRIPT_TYPE = 'xopc_session_transcript' as const;\n\nexport const CURRENT_SESSION_TRANSCRIPT_VERSION = 1;\n\n/** Record appended when context compaction runs (mirrors pi `CompactionEntry` audit fields). */\nexport interface TranscriptCompactionRecord {\n at: string;\n summary: string;\n firstKeptIndex: number;\n tokensBefore: number;\n tokensAfter: number;\n}\n\n/** Synthetic document shape returned by {@link SessionStore.loadTranscriptDocument} for API parity. */\nexport interface XopcSessionTranscriptV1 {\n type: typeof XOPC_SESSION_TRANSCRIPT_TYPE;\n version: number;\n id: string;\n createdAt: string;\n updatedAt: string;\n messages: TranscriptStoredRow[];\n compactions?: TranscriptCompactionRecord[];\n}\n"],"mappings":";AAMA,MAAa,+BAA+B;AAE5C,MAAa,qCAAqC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export declare function isAppendOnlyLlmTranscriptMessage(message: unknown): boolean;
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
//#region src/session/transcript-stats.ts
|
|
2
|
+
function isAppendOnlyLlmTranscriptMessage(message) {
|
|
3
|
+
if (!message || typeof message !== "object") return false;
|
|
4
|
+
const role = message.role;
|
|
5
|
+
return role === "user" || role === "assistant" || role === "tool" || role === "toolResult";
|
|
6
|
+
}
|
|
7
|
+
//#endregion
|
|
8
|
+
export { isAppendOnlyLlmTranscriptMessage };
|
|
9
|
+
|
|
10
|
+
//# sourceMappingURL=transcript-stats.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"transcript-stats.js","names":[],"sources":["../../../src/session/transcript-stats.ts"],"sourcesContent":["export function isAppendOnlyLlmTranscriptMessage(message: unknown): boolean {\n if (!message || typeof message !== 'object') {\n return false;\n }\n const role = (message as { role?: string }).role;\n return role === 'user' || role === 'assistant' || role === 'tool' || role === 'toolResult';\n}\n"],"mappings":";AAAA,SAAgB,iCAAiC,SAA2B;AAC1E,KAAI,CAAC,WAAW,OAAO,YAAY,SACjC,QAAO;CAET,MAAM,OAAQ,QAA8B;AAC5C,QAAO,SAAS,UAAU,SAAS,eAAe,SAAS,UAAU,SAAS"}
|
|
@@ -1,8 +1,8 @@
|
|
|
1
1
|
import { resolveMimeType } from "./share-store.js";
|
|
2
|
-
import { basename, join, resolve } from "node:path";
|
|
3
|
-
import { readdirSync } from "node:fs";
|
|
4
2
|
import { randomUUID } from "node:crypto";
|
|
3
|
+
import { readdirSync } from "node:fs";
|
|
5
4
|
import { copyFile, mkdir, rm, stat } from "node:fs/promises";
|
|
5
|
+
import { basename, join, resolve } from "node:path";
|
|
6
6
|
//#region src/share/share-auto.ts
|
|
7
7
|
/**
|
|
8
8
|
* Smart-share decision layer.
|
|
@@ -1,14 +1,14 @@
|
|
|
1
|
-
import { resolveStateDir } from "../config/paths-state.js";
|
|
2
1
|
import { createLogger } from "../utils/logger/index.js";
|
|
3
2
|
import { init_logger } from "../utils/logger.js";
|
|
3
|
+
import { resolveStateDir } from "../config/paths-state.js";
|
|
4
4
|
import { init_paths } from "../config/paths.js";
|
|
5
5
|
import { SHARE_CONFIG_DEFAULTS } from "./share-types.js";
|
|
6
6
|
import { isPathUnderWorkspace } from "../gateway/workspace-editor-path.js";
|
|
7
7
|
import { logShareAudit } from "./share-audit.js";
|
|
8
|
-
import { join, relative, resolve } from "node:path";
|
|
9
|
-
import { existsSync, mkdirSync, readFileSync, writeFileSync } from "node:fs";
|
|
10
8
|
import { randomBytes, randomUUID } from "node:crypto";
|
|
9
|
+
import { existsSync, mkdirSync, readFileSync, writeFileSync } from "node:fs";
|
|
11
10
|
import { lstat, readdir, realpath, stat } from "node:fs/promises";
|
|
11
|
+
import { join, relative, resolve } from "node:path";
|
|
12
12
|
//#region src/share/share-store.ts
|
|
13
13
|
init_paths();
|
|
14
14
|
init_logger();
|
|
@@ -1,12 +1,12 @@
|
|
|
1
|
-
import { resolveStateDir } from "../config/paths-state.js";
|
|
2
1
|
import { createLogger } from "../utils/logger/index.js";
|
|
3
2
|
import { init_logger } from "../utils/logger.js";
|
|
3
|
+
import { resolveStateDir } from "../config/paths-state.js";
|
|
4
4
|
import { init_paths } from "../config/paths.js";
|
|
5
5
|
import { getShareStore } from "./share-store.js";
|
|
6
6
|
import { getSiteShareStore } from "./site-share-store.js";
|
|
7
7
|
import { loadPlaywrightCoreModule } from "../browser/providers/playwright-doctor.js";
|
|
8
|
-
import { join } from "node:path";
|
|
9
8
|
import { mkdir, readFile, stat, unlink, writeFile } from "node:fs/promises";
|
|
9
|
+
import { join } from "node:path";
|
|
10
10
|
//#region src/share/share-thumbnail.ts
|
|
11
11
|
/**
|
|
12
12
|
* Thumbnail generator for shares.
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { isPathUnderWorkspace } from "../gateway/workspace-editor-path.js";
|
|
2
|
-
import { relative, resolve } from "node:path";
|
|
3
2
|
import { createReadStream } from "node:fs";
|
|
4
3
|
import { lstat, readdir, realpath, stat } from "node:fs/promises";
|
|
4
|
+
import { relative, resolve } from "node:path";
|
|
5
5
|
import { Readable, Transform } from "node:stream";
|
|
6
6
|
import { pipeline } from "node:stream/promises";
|
|
7
7
|
import { crc32, createDeflateRaw } from "node:zlib";
|
|
@@ -1,13 +1,13 @@
|
|
|
1
|
-
import { resolveStateDir } from "../config/paths-state.js";
|
|
2
1
|
import { createLogger } from "../utils/logger/index.js";
|
|
3
2
|
import { init_logger } from "../utils/logger.js";
|
|
3
|
+
import { resolveStateDir } from "../config/paths-state.js";
|
|
4
4
|
import { init_paths } from "../config/paths.js";
|
|
5
5
|
import { SITE_SHARE_CONFIG_DEFAULTS } from "./site-share-types.js";
|
|
6
6
|
import { isPathUnderWorkspace } from "../gateway/workspace-editor-path.js";
|
|
7
|
-
import { join, relative, resolve } from "node:path";
|
|
8
|
-
import { existsSync, mkdirSync, readFileSync, writeFileSync } from "node:fs";
|
|
9
7
|
import { randomBytes, randomUUID } from "node:crypto";
|
|
8
|
+
import { existsSync, mkdirSync, readFileSync, writeFileSync } from "node:fs";
|
|
10
9
|
import { lstat, readdir, realpath, stat } from "node:fs/promises";
|
|
10
|
+
import { join, relative, resolve } from "node:path";
|
|
11
11
|
//#region src/share/site-share-store.ts
|
|
12
12
|
init_paths();
|
|
13
13
|
init_logger();
|
|
@@ -1,8 +1,8 @@
|
|
|
1
1
|
import { isPathUnderWorkspace } from "../gateway/workspace-editor-path.js";
|
|
2
2
|
import { resolveMimeType, shareResponseContentType } from "./share-store.js";
|
|
3
|
-
import { join, relative, resolve } from "node:path";
|
|
4
3
|
import { createReadStream } from "node:fs";
|
|
5
4
|
import { readFile, stat } from "node:fs/promises";
|
|
5
|
+
import { join, relative, resolve } from "node:path";
|
|
6
6
|
import { Readable } from "node:stream";
|
|
7
7
|
//#region src/share/site-static-serve.ts
|
|
8
8
|
const HASHED_ASSET_RE = /\.[a-f0-9]{8,}\.(?:js|mjs|css|woff2?|ttf|otf|svg|png|jpg|jpeg|gif|webp|avif|ico)$/i;
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
import type { SessionAgentConfig } from '../../session/config-types.js';
|
|
2
|
+
export declare function getSessionConfig(sessionKey: string): SessionAgentConfig | null;
|
|
3
|
+
export declare function setSessionConfig(sessionKey: string, config: SessionAgentConfig, cwd: string): SessionAgentConfig;
|
|
4
|
+
export declare function updateSessionConfig(sessionKey: string, partial: Partial<SessionAgentConfig>, cwd: string): SessionAgentConfig;
|
|
5
|
+
export declare function deleteSessionConfig(sessionKey: string): void;
|
|
6
|
+
export declare function hasSessionConfig(sessionKey: string): boolean;
|