@xopcai/xopc 0.0.8 → 0.0.11
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/extensions/telegram/src/plugin.js +1 -1
- package/dist/extensions/telegram/src/routing-integration.js +2 -2
- package/dist/extensions/weixin/src/plugin.js +1 -1
- package/dist/gateway/static/root/assets/{agents-BSNzJWbQ.js → agents-BdC4Y-HX.js} +2 -2
- package/dist/gateway/static/root/assets/agents-BdC4Y-HX.js.map +1 -0
- package/dist/gateway/static/root/assets/{apps-page-BKk9SB4D.js → apps-page-C-oaSHkm.js} +2 -2
- package/dist/gateway/static/root/assets/{apps-page-BKk9SB4D.js.map → apps-page-C-oaSHkm.js.map} +1 -1
- package/dist/gateway/static/root/assets/attachment-load-BDDlItdE.js +1 -0
- package/dist/gateway/static/root/assets/{channels-settings-_J6cQN6G.js → channels-settings-BqEUppPO.js} +2 -2
- package/dist/gateway/static/root/assets/{channels-settings-_J6cQN6G.js.map → channels-settings-BqEUppPO.js.map} +1 -1
- package/dist/gateway/static/root/assets/{chat-agents-api-DPb_0O8M.js → chat-agents-api-BhqjQ7iL.js} +2 -2
- package/dist/gateway/static/root/assets/{chat-agents-api-DPb_0O8M.js.map → chat-agents-api-BhqjQ7iL.js.map} +1 -1
- package/dist/gateway/static/root/assets/{cron-page-BUJOuuKX.js → cron-page-Cli49RKR.js} +2 -2
- package/dist/gateway/static/root/assets/{cron-page-BUJOuuKX.js.map → cron-page-Cli49RKR.js.map} +1 -1
- package/dist/gateway/static/root/assets/{cron-utils-Cn0YVg8x.js → cron-utils-Dkj-Ldpf.js} +2 -2
- package/dist/gateway/static/root/assets/{cron-utils-Cn0YVg8x.js.map → cron-utils-Dkj-Ldpf.js.map} +1 -1
- package/dist/gateway/static/root/assets/{electron-env-D9bm1FIu.js → electron-env-BDtJw9AY.js} +2 -2
- package/dist/gateway/static/root/assets/{electron-env-D9bm1FIu.js.map → electron-env-BDtJw9AY.js.map} +1 -1
- package/dist/gateway/static/root/assets/{extension-debug-page-DTz4O5Ua.js → extension-debug-page-BMcZlaxF.js} +2 -2
- package/dist/gateway/static/root/assets/{extension-debug-page-DTz4O5Ua.js.map → extension-debug-page-BMcZlaxF.js.map} +1 -1
- package/dist/gateway/static/root/assets/{extension-iframe-host-Cs1Kde9o.js → extension-iframe-host-D5HEF0KR.js} +2 -2
- package/dist/gateway/static/root/assets/{extension-iframe-host-Cs1Kde9o.js.map → extension-iframe-host-D5HEF0KR.js.map} +1 -1
- package/dist/gateway/static/root/assets/{extension-page-G52iX0Bo.js → extension-page-CXdCSSPl.js} +2 -2
- package/dist/gateway/static/root/assets/{extension-page-G52iX0Bo.js.map → extension-page-CXdCSSPl.js.map} +1 -1
- package/dist/gateway/static/root/assets/{extension-provider-CO2jxBA9.js → extension-provider-DZCZgQE2.js} +2 -2
- package/dist/gateway/static/root/assets/{extension-provider-CO2jxBA9.js.map → extension-provider-DZCZgQE2.js.map} +1 -1
- package/dist/gateway/static/root/assets/{extension-settings-page-D9Ul8uSt.js → extension-settings-page-CX6STpx3.js} +2 -2
- package/dist/gateway/static/root/assets/{extension-settings-page-D9Ul8uSt.js.map → extension-settings-page-CX6STpx3.js.map} +1 -1
- package/dist/gateway/static/root/assets/{gateway-config-swr-Bc8SVD15.js → gateway-config-swr-Cph02QZn.js} +2 -2
- package/dist/gateway/static/root/assets/{gateway-config-swr-Bc8SVD15.js.map → gateway-config-swr-Cph02QZn.js.map} +1 -1
- package/dist/gateway/static/root/assets/index-Bty3m0mS.css +2 -0
- package/dist/gateway/static/root/assets/{index-BXUJbteW.js → index-iTUyfzNr.js} +10 -10
- package/dist/gateway/static/root/assets/index-iTUyfzNr.js.map +1 -0
- package/dist/gateway/static/root/assets/{logs-page-5V25JkQY.js → logs-page-B9O5l3I8.js} +2 -2
- package/dist/gateway/static/root/assets/{logs-page-5V25JkQY.js.map → logs-page-B9O5l3I8.js.map} +1 -1
- package/dist/gateway/static/root/assets/{model-selector-he3aQfme.js → model-selector-BLiY_O25.js} +2 -2
- package/dist/gateway/static/root/assets/{model-selector-he3aQfme.js.map → model-selector-BLiY_O25.js.map} +1 -1
- package/dist/gateway/static/root/assets/page-header-store-BFpnFTed.js +2 -0
- package/dist/gateway/static/root/assets/{page-header-store-DJHD9Ean.js.map → page-header-store-BFpnFTed.js.map} +1 -1
- package/dist/gateway/static/root/assets/{session-api-n-4O5d9U.js → session-api-DEhQXWJg.js} +2 -2
- package/dist/gateway/static/root/assets/{session-api-n-4O5d9U.js.map → session-api-DEhQXWJg.js.map} +1 -1
- package/dist/gateway/static/root/assets/{session-working-directory-control-B6dHLvbr.js → session-working-directory-control-DKOtWs3-.js} +2 -2
- package/dist/gateway/static/root/assets/{session-working-directory-control-B6dHLvbr.js.map → session-working-directory-control-DKOtWs3-.js.map} +1 -1
- package/dist/gateway/static/root/assets/{sessions-page-rBUfTdm3.js → sessions-page-BYlWP1ep.js} +2 -2
- package/dist/gateway/static/root/assets/{sessions-page-rBUfTdm3.js.map → sessions-page-BYlWP1ep.js.map} +1 -1
- package/dist/gateway/static/root/assets/{settings-page-B3QrJm-E.js → settings-page-oCnIavdg.js} +2 -2
- package/dist/gateway/static/root/assets/settings-page-oCnIavdg.js.map +1 -0
- package/dist/gateway/static/root/assets/{skill-api-vxtE8kI6.js → skill-api-DWrn8Az0.js} +2 -2
- package/dist/gateway/static/root/assets/{skill-api-vxtE8kI6.js.map → skill-api-DWrn8Az0.js.map} +1 -1
- package/dist/gateway/static/root/assets/{skills-page-D36_O2Ub.js → skills-page-C59WQpM1.js} +2 -2
- package/dist/gateway/static/root/assets/{skills-page-D36_O2Ub.js.map → skills-page-C59WQpM1.js.map} +1 -1
- package/dist/gateway/static/root/assets/{theme-store-CmiSsYBd.js → theme-store-CywXkKml.js} +2 -2
- package/dist/gateway/static/root/assets/{theme-store-CmiSsYBd.js.map → theme-store-CywXkKml.js.map} +1 -1
- package/dist/gateway/static/root/assets/url-D7yWllI8.js +2 -0
- package/dist/gateway/static/root/assets/url-D7yWllI8.js.map +1 -0
- package/dist/gateway/static/root/assets/{useTranslation-DYORQ7x6.js → useTranslation-CACj0DBJ.js} +2 -2
- package/dist/gateway/static/root/assets/{useTranslation-DYORQ7x6.js.map → useTranslation-CACj0DBJ.js.map} +1 -1
- package/dist/gateway/static/root/index.html +15 -15
- package/dist/package.js +1 -1
- package/dist/src/agent/agent-manager.d.ts +1 -0
- package/dist/src/agent/agent-manager.js +20 -12
- package/dist/src/agent/agent-manager.js.map +1 -1
- package/dist/src/agent/background-review/run-background-review.js +2 -0
- package/dist/src/agent/background-review/run-background-review.js.map +1 -1
- package/dist/src/agent/child-agent-factory.js +2 -0
- package/dist/src/agent/child-agent-factory.js.map +1 -1
- package/dist/src/agent/context/expand-at-file-mentions.d.ts +4 -0
- package/dist/src/agent/context/expand-at-file-mentions.js +69 -0
- package/dist/src/agent/context/expand-at-file-mentions.js.map +1 -0
- package/dist/src/agent/context/workspace-seed.js +1 -1
- package/dist/src/agent/image/understanding/pi-ai-provider.js.map +1 -1
- package/dist/src/agent/ipc/bus.js +1 -1
- package/dist/src/agent/ipc/inbox.js +1 -1
- package/dist/src/agent/ipc/socket.js +1 -1
- package/dist/src/agent/memory/compaction.d.ts +1 -1
- package/dist/src/agent/memory/compaction.js +38 -11
- package/dist/src/agent/memory/compaction.js.map +1 -1
- package/dist/src/agent/messaging/command-handler.d.ts +13 -0
- package/dist/src/agent/messaging/command-handler.js +14 -2
- package/dist/src/agent/messaging/command-handler.js.map +1 -1
- package/dist/src/agent/models/manager.js +1 -1
- package/dist/src/agent/orchestration/agent-orchestrator.js +6 -1
- package/dist/src/agent/orchestration/agent-orchestrator.js.map +1 -1
- package/dist/src/agent/prompt/service-prompt-builder.js +1 -1
- package/dist/src/agent/service.d.ts +16 -1
- package/dist/src/agent/service.js +178 -20
- package/dist/src/agent/service.js.map +1 -1
- package/dist/src/agent/skills/format-skills-prompt.js.map +1 -1
- package/dist/src/agent/skills/index.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-manage-ops.js.map +1 -1
- package/dist/src/agent/skills/skill-manager.js +1 -1
- package/dist/src/agent/tools/browser/tools.js.map +1 -1
- package/dist/src/agent/tools/factory.js +1 -1
- package/dist/src/agent/tools/image-tool.js.map +1 -1
- package/dist/src/agent/tools/send-media.js +1 -1
- package/dist/src/agent/tools/skill-manage-tool.js +1 -1
- package/dist/src/agent/tools/write.js +1 -1
- package/dist/src/auth/credentials.js +2 -2
- package/dist/src/auth/sync-provider-auth.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/index.d.ts +1 -1
- package/dist/src/channels/index.js +2 -2
- package/dist/src/channels/pipeline.d.ts +8 -1
- package/dist/src/channels/pipeline.js +49 -4
- package/dist/src/channels/pipeline.js.map +1 -1
- package/dist/src/chat-commands/builtins/config.d.ts +4 -0
- package/dist/src/chat-commands/builtins/config.js +197 -0
- package/dist/src/chat-commands/builtins/config.js.map +1 -0
- package/dist/src/chat-commands/builtins/context.d.ts +4 -0
- package/dist/src/chat-commands/builtins/context.js +44 -0
- package/dist/src/chat-commands/builtins/context.js.map +1 -0
- package/dist/src/chat-commands/builtins/session.js +111 -0
- package/dist/src/chat-commands/builtins/session.js.map +1 -1
- package/dist/src/chat-commands/builtins/thinking.js +49 -21
- package/dist/src/chat-commands/builtins/thinking.js.map +1 -1
- package/dist/src/chat-commands/config-paths.d.ts +10 -0
- package/dist/src/chat-commands/config-paths.js +45 -0
- package/dist/src/chat-commands/config-paths.js.map +1 -0
- package/dist/src/chat-commands/config-value.d.ts +12 -0
- package/dist/src/chat-commands/config-value.js +53 -0
- package/dist/src/chat-commands/config-value.js.map +1 -0
- package/dist/src/chat-commands/context.d.ts +24 -1
- package/dist/src/chat-commands/context.js +41 -0
- package/dist/src/chat-commands/context.js.map +1 -1
- package/dist/src/chat-commands/index.d.ts +2 -0
- package/dist/src/chat-commands/index.js +5 -1
- package/dist/src/chat-commands/index.js.map +1 -1
- package/dist/src/chat-commands/types.d.ts +33 -1
- package/dist/src/cli/commands/agent/interactive.js +1 -1
- package/dist/src/cli/commands/agent/interactive.js.map +1 -1
- package/dist/src/cli/commands/agent.js +22 -10
- package/dist/src/cli/commands/agent.js.map +1 -1
- package/dist/src/cli/commands/auth.js.map +1 -1
- package/dist/src/cli/commands/doctor/checks/provider-auth.js +1 -1
- package/dist/src/cli/commands/extension.js +10 -0
- package/dist/src/cli/commands/extension.js.map +1 -1
- package/dist/src/cli/commands/init.js +3 -4
- package/dist/src/cli/commands/init.js.map +1 -1
- package/dist/src/cli/commands/session/utils.js.map +1 -1
- package/dist/src/cli/commands/update.d.ts +1 -0
- package/dist/src/cli/commands/update.js +171 -0
- package/dist/src/cli/commands/update.js.map +1 -0
- package/dist/src/cli/index.d.ts +1 -1
- package/dist/src/cli/index.js +3 -1
- package/dist/src/cli/index.js.map +1 -1
- package/dist/src/config/index.d.ts +1 -0
- package/dist/src/config/index.js +5 -4
- package/dist/src/config/index.js.map +1 -1
- package/dist/src/config/loader.js +1 -1
- package/dist/src/config/models-json.js +1 -1
- package/dist/src/config/paths.js.map +1 -1
- package/dist/src/config/profile.js +2 -2
- package/dist/src/config/runtime-overrides.d.ts +8 -0
- package/dist/src/config/runtime-overrides.js +40 -0
- package/dist/src/config/runtime-overrides.js.map +1 -0
- package/dist/src/config/schema.d.ts +35 -0
- package/dist/src/config/schema.js +19 -3
- package/dist/src/config/schema.js.map +1 -1
- package/dist/src/cron/executor.js +2 -2
- package/dist/src/cron/persistence.js +1 -1
- package/dist/src/cron/run-log-store.js +1 -1
- package/dist/src/extensions/health.js +1 -1
- package/dist/src/extensions/loader.d.ts +1 -1
- package/dist/src/extensions/loader.js +6 -9
- package/dist/src/extensions/loader.js.map +1 -1
- package/dist/src/extensions/lockfile.js +1 -1
- package/dist/src/extensions/sdk/index.js +6 -1
- package/dist/src/extensions/sdk/index.js.map +1 -0
- package/dist/src/gateway/agents-admin.js +2 -2
- package/dist/src/gateway/agents-admin.js.map +1 -1
- package/dist/src/gateway/hono/oauth.js +1 -1
- package/dist/src/gateway/hono/routes/config.js +1 -1
- package/dist/src/gateway/hono/routes/index.js +2 -0
- package/dist/src/gateway/hono/routes/index.js.map +1 -1
- package/dist/src/gateway/hono/routes/models.js +64 -11
- package/dist/src/gateway/hono/routes/models.js.map +1 -1
- package/dist/src/gateway/hono/routes/public-gateway.js +10 -0
- package/dist/src/gateway/hono/routes/public-gateway.js.map +1 -1
- package/dist/src/gateway/hono/routes/update.d.ts +3 -0
- package/dist/src/gateway/hono/routes/update.js +141 -0
- package/dist/src/gateway/hono/routes/update.js.map +1 -0
- package/dist/src/gateway/hono/routes/workspace.js +84 -4
- package/dist/src/gateway/hono/routes/workspace.js.map +1 -1
- package/dist/src/gateway/hono/sse.js +2 -2
- package/dist/src/gateway/service.d.ts +1 -0
- package/dist/src/gateway/service.js +16 -4
- package/dist/src/gateway/service.js.map +1 -1
- package/dist/src/gateway/workspace-fs-file-list.d.ts +5 -0
- package/dist/src/gateway/workspace-fs-file-list.js +56 -0
- package/dist/src/gateway/workspace-fs-file-list.js.map +1 -0
- package/dist/src/gateway/workspace-heartbeat-path.js +1 -1
- package/dist/src/gateway/workspace-ripgrep.d.ts +5 -0
- package/dist/src/gateway/workspace-ripgrep.js +88 -4
- package/dist/src/gateway/workspace-ripgrep.js.map +1 -1
- package/dist/src/infra/update-channels.d.ts +14 -0
- package/dist/src/infra/update-channels.js +30 -0
- package/dist/src/infra/update-channels.js.map +1 -0
- package/dist/src/infra/update-check.d.ts +53 -0
- package/dist/src/infra/update-check.js +155 -0
- package/dist/src/infra/update-check.js.map +1 -0
- package/dist/src/infra/update-runner.d.ts +18 -0
- package/dist/src/infra/update-runner.js +112 -0
- package/dist/src/infra/update-runner.js.map +1 -0
- package/dist/src/infra/update-startup.d.ts +20 -0
- package/dist/src/infra/update-startup.js +246 -0
- package/dist/src/infra/update-startup.js.map +1 -0
- package/dist/src/providers/extension-stream-bridge.d.ts +3 -0
- package/dist/src/providers/extension-stream-bridge.js +239 -0
- package/dist/src/providers/extension-stream-bridge.js.map +1 -0
- package/dist/src/providers/index.d.ts +7 -2
- package/dist/src/providers/index.js +77 -14
- package/dist/src/providers/index.js.map +1 -1
- package/dist/src/providers/model-registry.js +1 -1
- package/dist/src/providers/plugin-registry.js +92 -87
- package/dist/src/providers/plugin-registry.js.map +1 -1
- package/dist/src/session/chat-export.d.ts +5 -0
- package/dist/src/session/chat-export.js +35 -0
- package/dist/src/session/chat-export.js.map +1 -0
- package/dist/src/session/config-store.js +1 -1
- package/dist/src/session/manager.d.ts +1 -1
- package/dist/src/session/manager.js +2 -2
- package/dist/src/session/manager.js.map +1 -1
- package/dist/src/session/session-title.js +1 -1
- package/dist/src/session/store.d.ts +1 -1
- package/dist/src/session/store.js +5 -5
- package/dist/src/session/store.js.map +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/package.json +2 -1
- package/dist/gateway/static/root/assets/agents-BSNzJWbQ.js.map +0 -1
- package/dist/gateway/static/root/assets/attachment-load-DXcJLSWT.js +0 -1
- package/dist/gateway/static/root/assets/index-BXUJbteW.js.map +0 -1
- package/dist/gateway/static/root/assets/index-CQLMxWSA.css +0 -2
- package/dist/gateway/static/root/assets/page-header-store-DJHD9Ean.js +0 -2
- package/dist/gateway/static/root/assets/settings-page-B3QrJm-E.js.map +0 -1
- package/dist/gateway/static/root/assets/url-CtSqjF9J.js +0 -2
- package/dist/gateway/static/root/assets/url-CtSqjF9J.js.map +0 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"thinking.js","names":[],"sources":["../../../../src/chat-commands/builtins/thinking.ts"],"sourcesContent":["/**\n * Thinking Commands\n * \n * Built-in commands for thinking ability management:\n * - /think - Set thinking level\n * - /reasoning - Set reasoning visibility\n * - /verbose - Set verbose mode\n * - /status - Show current status (including thinking/reasoning/verbose)\n */\n\nimport type { CommandDefinition, CommandContext } from '../types.js';\nimport { commandRegistry } from '../registry.js';\nimport { \n normalizeThinkLevel, \n normalizeReasoningLevel, \n normalizeVerboseLevel,\n formatThinkingLevels,\n type ThinkLevel,\n type ReasoningLevel,\n type VerboseLevel,\n} from '../../agent/transcript/thinking-types.js';\n\n// Think command\nconst thinkCommand: CommandDefinition = {\n id: 'agent.think',\n name: 'think',\n description: 'Set thinking level (usage: /think <level>)',\n category: 'model',\n scope: ['global', 'private', 'group'],\n acceptsArgs: true,\n examples: ['/think', '/think high', '/think off', '/think adaptive'],\n handler: async (ctx: CommandContext, args: string) => {\n await ctx.setTyping(true);\n \n const extendedCtx = ctx as CommandContext & {\n getThinkingLevel?: () => Promise<ThinkLevel | undefined>;\n getSessionConfigStore?: () => unknown;\n };\n \n if (!args.trim()) {\n // Show current thinking level\n const currentLevel = await extendedCtx.getThinkingLevel?.() || ctx.getConfig()?.agents?.defaults?.thinkingDefault;\n \n return {\n content: `🧠 *Thinking Level*\\n\\n` +\n `Current: *${currentLevel || 'default'}*\\n\\n` +\n `Available levels:\\n` +\n `• off - No thinking\\n` +\n `• minimal - Minimal thinking\\n` +\n `• low - Low thinking\\n` +\n `• medium - Medium thinking (default)\\n` +\n `• high - High thinking\\n` +\n `• xhigh - Extra high (certain models)\\n` +\n `• adaptive - Auto-adjust based on task\\n\\n` +\n `Usage: /think <level>\\n` +\n `Example: /think high`,\n success: true,\n };\n }\n \n // Parse and validate the level\n const level = normalizeThinkLevel(args.trim());\n if (!level) {\n const levels = formatThinkingLevels();\n return {\n content: `❌ Invalid thinking level: ${args}\\n\\n` +\n `Available levels: ${levels}\\n\\n` +\n `Usage: /think <level>\\n` +\n `Example: /think high`,\n success: false,\n };\n }\n \n if (ctx.setThinkingLevel) {\n await ctx.setThinkingLevel(level);\n }\n \n const levelDescriptions: Record<ThinkLevel, string> = {\n off: 'No thinking',\n minimal: 'Minimal thinking effort',\n low: 'Low thinking effort',\n medium: 'Medium thinking effort',\n high: 'High thinking effort',\n xhigh: 'Extra high thinking effort',\n adaptive: 'Auto-adjust based on task complexity',\n };\n \n return {\n content: `🧠 *Thinking Level Set*\\n\\n` +\n `Level: *${level}*\\n` +\n `${levelDescriptions[level]}\\n\\n` +\n `This setting will apply to your next message.`,\n success: true,\n };\n },\n};\n\n// Reasoning command\nconst reasoningCommand: CommandDefinition = {\n id: 'agent.reasoning',\n name: 'reasoning',\n description: 'Set reasoning visibility (usage: /reasoning <mode>)',\n category: 'model',\n scope: ['global', 'private', 'group'],\n acceptsArgs: true,\n examples: ['/reasoning', '/reasoning on', '/reasoning stream', '/reasoning off'],\n handler: async (ctx: CommandContext, args: string) => {\n await ctx.setTyping(true);\n \n // Get the extended context\n const extendedCtx = ctx as any;\n const configStore = extendedCtx.getSessionConfigStore?.();\n \n if (!args.trim()) {\n // Show current reasoning level\n const currentLevel = await extendedCtx.getReasoningLevel?.() || ctx.getConfig()?.agents?.defaults?.reasoningDefault;\n \n return {\n content: `💭 *Reasoning Visibility*\\n\\n` +\n `Current: *${currentLevel || 'default'}*\\n\\n` +\n `Available modes:\\n` +\n `• off - Hide reasoning\\n` +\n `• on - Show reasoning after completion\\n` +\n `• stream - Stream reasoning in real-time\\n\\n` +\n `Usage: /reasoning <mode>\\n` +\n `Example: /reasoning stream`,\n success: true,\n };\n }\n \n // Parse and validate the level\n const level = normalizeReasoningLevel(args.trim());\n if (!level) {\n return {\n content: `❌ Invalid reasoning mode: ${args}\\n\\n` +\n `Available modes: off, on, stream\\n\\n` +\n `Usage: /reasoning <mode>\\n` +\n `Example: /reasoning stream`,\n success: false,\n };\n }\n \n // Set the reasoning level\n if (configStore) {\n await configStore.update(ctx.sessionKey, { reasoningLevel: level });\n }\n \n const modeDescriptions: Record<ReasoningLevel, string> = {\n off: 'Hide reasoning from user',\n on: 'Show reasoning after completion',\n stream: 'Stream reasoning in real-time',\n };\n \n return {\n content: `💭 *Reasoning Visibility Set*\\n\\n` +\n `Mode: *${level}*\\n` +\n `${modeDescriptions[level]}\\n\\n` +\n `This setting will apply to your next message.`,\n success: true,\n };\n },\n};\n\n// Verbose command\nconst verboseCommand: CommandDefinition = {\n id: 'agent.verbose',\n name: 'verbose',\n description: 'Toggle verbose mode (usage: /verbose [on|off|full])',\n category: 'model',\n scope: ['global', 'private', 'group'],\n acceptsArgs: true,\n examples: ['/verbose', '/verbose on', '/verbose off', '/verbose full'],\n handler: async (ctx: CommandContext, args: string) => {\n await ctx.setTyping(true);\n \n // Get the extended context\n const extendedCtx = ctx as any;\n const configStore = extendedCtx.getSessionConfigStore?.();\n \n if (!args.trim()) {\n // Toggle verbose mode\n const currentLevel = await extendedCtx.getVerboseLevel?.() || ctx.getConfig()?.agents?.defaults?.verboseDefault;\n const newLevel: VerboseLevel = currentLevel === 'on' ? 'off' : 'on';\n \n if (configStore) {\n await configStore.update(ctx.sessionKey, { verboseLevel: newLevel });\n }\n \n return {\n content: `📝 *Verbose Mode*\\n\\n` +\n `Mode: *${newLevel}*\\n\\n` +\n `${newLevel === 'on' ? 'Verbose output enabled for next message.' : 'Verbose output disabled.'}`,\n success: true,\n };\n }\n \n // Parse and validate the level\n const level = normalizeVerboseLevel(args.trim());\n if (!level) {\n return {\n content: `❌ Invalid verbose mode: ${args}\\n\\n` +\n `Available modes: off, on, full\\n\\n` +\n `Usage: /verbose [mode]\\n` +\n `Example: /verbose on`,\n success: false,\n };\n }\n \n // Set the verbose level\n if (configStore) {\n await configStore.update(ctx.sessionKey, { verboseLevel: level });\n }\n \n const modeDescriptions: Record<VerboseLevel, string> = {\n off: 'Minimal output',\n on: 'Normal verbose output',\n full: 'Full verbose output with all details',\n };\n \n return {\n content: `📝 *Verbose Mode Set*\\n\\n` +\n `Mode: *${level}*\\n` +\n `${modeDescriptions[level]}`,\n success: true,\n };\n },\n};\n\n// Status command - enhanced to show thinking/reasoning/verbose\nconst enhancedStatusCommand: CommandDefinition = {\n id: 'agent.status',\n name: 'status',\n description: 'Show current agent status including thinking settings',\n category: 'system',\n scope: ['global', 'private', 'group'],\n handler: async (ctx: CommandContext) => {\n await ctx.setTyping(true);\n \n // Get the extended context\n const extendedCtx = ctx as any;\n \n // Get current settings\n const config = ctx.getConfig();\n const currentModel = ctx.getCurrentModel();\n const thinkingLevel = await extendedCtx.getThinkingLevel?.() || config.agents?.defaults?.thinkingDefault;\n const reasoningLevel = await extendedCtx.getReasoningLevel?.() || config.agents?.defaults?.reasoningDefault;\n const verboseLevel = await extendedCtx.getVerboseLevel?.() || config.agents?.defaults?.verboseDefault;\n const usage = await ctx.getUsage();\n \n // Build status message\n const lines = [\n '📊 *Agent Status*',\n '',\n `🤖 *Model:* \\`${currentModel}\\``,\n `🧠 *Thinking:* \\`${thinkingLevel || 'default'}\\``,\n `💭 *Reasoning:* \\`${reasoningLevel || 'default'}\\``,\n `📝 *Verbose:* \\`${verboseLevel || 'default'}\\``,\n '',\n '---',\n '',\n '📈 *Session Usage*',\n `💬 Messages: ${usage.messageCount}`,\n `📥 Prompt: ${usage.promptTokens.toLocaleString()} tokens`,\n `📤 Completion: ${usage.completionTokens.toLocaleString()} tokens`,\n `📊 Total: ${usage.totalTokens.toLocaleString()} tokens`,\n ];\n \n return {\n content: lines.join('\\n'),\n success: true,\n };\n },\n};\n\n// Register all thinking commands\nexport function registerThinkingCommands(): void {\n commandRegistry.register(thinkCommand);\n commandRegistry.register(reasoningCommand);\n commandRegistry.register(verboseCommand);\n commandRegistry.register(enhancedStatusCommand);\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"thinking.js","names":[],"sources":["../../../../src/chat-commands/builtins/thinking.ts"],"sourcesContent":["/**\n * Thinking Commands\n * \n * Built-in commands for thinking ability management:\n * - /think - Set thinking level\n * - /reasoning - Set reasoning visibility\n * - /verbose - Set verbose mode\n * - /status - Show current status (including thinking/reasoning/verbose)\n */\n\nimport type { CommandDefinition, CommandContext } from '../types.js';\nimport { commandRegistry } from '../registry.js';\nimport {\n normalizeThinkLevel,\n normalizeReasoningLevel,\n normalizeVerboseLevel,\n formatThinkingLevels,\n type ThinkLevel,\n type ReasoningLevel,\n type VerboseLevel,\n} from '../../agent/transcript/thinking-types.js';\nimport { applyConfigOverrides, setConfigOverride } from '../../config/runtime-overrides.js';\nimport type { Config } from '../../config/schema.js';\nimport type { SessionConfigStore } from '../../session/index.js';\n\nfunction effectiveAgentDefaults(ctx: CommandContext) {\n const base = (ctx.getConfig?.() ?? ctx.config) as Config;\n return applyConfigOverrides(base).agents?.defaults;\n}\n\nasync function stripSessionAgentFields(\n store: SessionConfigStore | undefined,\n sessionKey: string,\n fields: Array<'thinkingLevel' | 'reasoningLevel' | 'verboseLevel'>,\n): Promise<void> {\n if (!store) return;\n const existing = await store.get(sessionKey);\n if (!existing) return;\n const next = { ...existing };\n for (const f of fields) {\n delete next[f];\n }\n const restKeys = Object.keys(next).filter((k) => k !== 'updatedAt');\n if (restKeys.length === 0) {\n await store.delete(sessionKey);\n } else {\n await store.set(sessionKey, next);\n }\n}\n\n// Think command\nconst thinkCommand: CommandDefinition = {\n id: 'agent.think',\n name: 'think',\n description: 'Set thinking level (usage: /think <level>)',\n category: 'model',\n scope: ['global', 'private', 'group'],\n acceptsArgs: true,\n examples: ['/think', '/think high', '/think off', '/think adaptive'],\n handler: async (ctx: CommandContext, args: string) => {\n await ctx.setTyping(true);\n\n if (!args.trim()) {\n // Show current thinking level\n const currentLevel =\n (await ctx.getThinkingLevel?.()) ?? effectiveAgentDefaults(ctx)?.thinkingDefault;\n \n return {\n content: `🧠 *Thinking Level*\\n\\n` +\n `Current: *${currentLevel || 'default'}*\\n\\n` +\n `Available levels:\\n` +\n `• off - No thinking\\n` +\n `• minimal - Minimal thinking\\n` +\n `• low - Low thinking\\n` +\n `• medium - Medium thinking (default)\\n` +\n `• high - High thinking\\n` +\n `• xhigh - Extra high (certain models)\\n` +\n `• adaptive - Auto-adjust based on task\\n\\n` +\n `Usage: /think <level>\\n` +\n `Example: /think high`,\n success: true,\n };\n }\n \n // Parse and validate the level\n const level = normalizeThinkLevel(args.trim());\n if (!level) {\n const levels = formatThinkingLevels();\n return {\n content: `❌ Invalid thinking level: ${args}\\n\\n` +\n `Available levels: ${levels}\\n\\n` +\n `Usage: /think <level>\\n` +\n `Example: /think high`,\n success: false,\n };\n }\n \n setConfigOverride(['agents', 'defaults', 'thinkingDefault'], level);\n await stripSessionAgentFields(ctx.getSessionConfigStore?.(), ctx.sessionKey, ['thinkingLevel']);\n ctx.syncAgentThinkingLevel?.(level);\n\n const levelDescriptions: Record<ThinkLevel, string> = {\n off: 'No thinking',\n minimal: 'Minimal thinking effort',\n low: 'Low thinking effort',\n medium: 'Medium thinking effort',\n high: 'High thinking effort',\n xhigh: 'Extra high thinking effort',\n adaptive: 'Auto-adjust based on task complexity',\n };\n \n return {\n content: `🧠 *Thinking Level Set*\\n\\n` +\n `Level: *${level}*\\n` +\n `${levelDescriptions[level]}\\n\\n` +\n `Applies for this gateway process (until restart). Persists only if you use /config set.`,\n success: true,\n };\n },\n};\n\n// Reasoning command\nconst reasoningCommand: CommandDefinition = {\n id: 'agent.reasoning',\n name: 'reasoning',\n description: 'Set reasoning visibility (usage: /reasoning <mode>)',\n category: 'model',\n scope: ['global', 'private', 'group'],\n acceptsArgs: true,\n examples: ['/reasoning', '/reasoning on', '/reasoning stream', '/reasoning off'],\n handler: async (ctx: CommandContext, args: string) => {\n await ctx.setTyping(true);\n\n if (!args.trim()) {\n // Show current reasoning level\n const currentLevel =\n (await ctx.getReasoningLevel?.()) ?? effectiveAgentDefaults(ctx)?.reasoningDefault;\n \n return {\n content: `💭 *Reasoning Visibility*\\n\\n` +\n `Current: *${currentLevel || 'default'}*\\n\\n` +\n `Available modes:\\n` +\n `• off - Hide reasoning\\n` +\n `• on - Show reasoning after completion\\n` +\n `• stream - Stream reasoning in real-time\\n\\n` +\n `Usage: /reasoning <mode>\\n` +\n `Example: /reasoning stream`,\n success: true,\n };\n }\n \n // Parse and validate the level\n const level = normalizeReasoningLevel(args.trim());\n if (!level) {\n return {\n content: `❌ Invalid reasoning mode: ${args}\\n\\n` +\n `Available modes: off, on, stream\\n\\n` +\n `Usage: /reasoning <mode>\\n` +\n `Example: /reasoning stream`,\n success: false,\n };\n }\n \n setConfigOverride(['agents', 'defaults', 'reasoningDefault'], level);\n await stripSessionAgentFields(ctx.getSessionConfigStore?.(), ctx.sessionKey, ['reasoningLevel']);\n\n const modeDescriptions: Record<ReasoningLevel, string> = {\n off: 'Hide reasoning from user',\n on: 'Show reasoning after completion',\n stream: 'Stream reasoning in real-time',\n };\n \n return {\n content: `💭 *Reasoning Visibility Set*\\n\\n` +\n `Mode: *${level}*\\n` +\n `${modeDescriptions[level]}\\n\\n` +\n `Applies for this gateway process (until restart). Persists only if you use /config set.`,\n success: true,\n };\n },\n};\n\n// Verbose command\nconst verboseCommand: CommandDefinition = {\n id: 'agent.verbose',\n name: 'verbose',\n description: 'Toggle verbose mode (usage: /verbose [on|off|full])',\n category: 'model',\n scope: ['global', 'private', 'group'],\n acceptsArgs: true,\n examples: ['/verbose', '/verbose on', '/verbose off', '/verbose full'],\n handler: async (ctx: CommandContext, args: string) => {\n await ctx.setTyping(true);\n\n if (!args.trim()) {\n // Toggle verbose mode\n const currentLevel =\n (await ctx.getVerboseLevel?.()) ?? effectiveAgentDefaults(ctx)?.verboseDefault;\n const newLevel: VerboseLevel = currentLevel === 'on' ? 'off' : 'on';\n\n setConfigOverride(['agents', 'defaults', 'verboseDefault'], newLevel);\n await stripSessionAgentFields(ctx.getSessionConfigStore?.(), ctx.sessionKey, ['verboseLevel']);\n\n return {\n content: `📝 *Verbose Mode*\\n\\n` +\n `Mode: *${newLevel}*\\n\\n` +\n `${newLevel === 'on' ? 'Verbose output enabled (process lifetime).' : 'Verbose output disabled (process lifetime).'}`,\n success: true,\n };\n }\n \n // Parse and validate the level\n const level = normalizeVerboseLevel(args.trim());\n if (!level) {\n return {\n content: `❌ Invalid verbose mode: ${args}\\n\\n` +\n `Available modes: off, on, full\\n\\n` +\n `Usage: /verbose [mode]\\n` +\n `Example: /verbose on`,\n success: false,\n };\n }\n \n setConfigOverride(['agents', 'defaults', 'verboseDefault'], level);\n await stripSessionAgentFields(ctx.getSessionConfigStore?.(), ctx.sessionKey, ['verboseLevel']);\n\n const modeDescriptions: Record<VerboseLevel, string> = {\n off: 'Minimal output',\n on: 'Normal verbose output',\n full: 'Full verbose output with all details',\n };\n\n return {\n content: `📝 *Verbose Mode Set*\\n\\n` +\n `Mode: *${level}*\\n` +\n `${modeDescriptions[level]}\\n\\n` +\n `Applies for this gateway process (until restart). Persists only if you use /config set.`,\n success: true,\n };\n },\n};\n\n// Status command - enhanced to show thinking/reasoning/verbose\nconst enhancedStatusCommand: CommandDefinition = {\n id: 'agent.status',\n name: 'status',\n description: 'Show current agent status including thinking settings',\n category: 'system',\n scope: ['global', 'private', 'group'],\n handler: async (ctx: CommandContext) => {\n await ctx.setTyping(true);\n\n const defaults = effectiveAgentDefaults(ctx);\n const currentModel = ctx.getCurrentModel();\n const thinkingLevel = (await ctx.getThinkingLevel?.()) ?? defaults?.thinkingDefault;\n const reasoningLevel = (await ctx.getReasoningLevel?.()) ?? defaults?.reasoningDefault;\n const verboseLevel = (await ctx.getVerboseLevel?.()) ?? defaults?.verboseDefault;\n const usage = await ctx.getUsage();\n \n // Build status message\n const lines = [\n '📊 *Agent Status*',\n '',\n `🤖 *Model:* \\`${currentModel}\\``,\n `🧠 *Thinking:* \\`${thinkingLevel || 'default'}\\``,\n `💭 *Reasoning:* \\`${reasoningLevel || 'default'}\\``,\n `📝 *Verbose:* \\`${verboseLevel || 'default'}\\``,\n '',\n '---',\n '',\n '📈 *Session Usage*',\n `💬 Messages: ${usage.messageCount}`,\n `📥 Prompt: ${usage.promptTokens.toLocaleString()} tokens`,\n `📤 Completion: ${usage.completionTokens.toLocaleString()} tokens`,\n `📊 Total: ${usage.totalTokens.toLocaleString()} tokens`,\n ];\n \n return {\n content: lines.join('\\n'),\n success: true,\n };\n },\n};\n\n// Register all thinking commands\nexport function registerThinkingCommands(): void {\n commandRegistry.register(thinkCommand);\n commandRegistry.register(reasoningCommand);\n commandRegistry.register(verboseCommand);\n commandRegistry.register(enhancedStatusCommand);\n}\n"],"mappings":"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"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
type PathNode = Record<string, unknown>;
|
|
2
|
+
export declare function parseConfigPath(raw: string): {
|
|
3
|
+
ok: boolean;
|
|
4
|
+
path?: string[];
|
|
5
|
+
error?: string;
|
|
6
|
+
};
|
|
7
|
+
export declare function getConfigValueAtPath(root: PathNode, path: string[]): unknown;
|
|
8
|
+
export declare function setConfigValueAtPath(root: PathNode, path: string[], value: unknown): void;
|
|
9
|
+
export declare function unsetConfigValueAtPath(root: PathNode, path: string[]): boolean;
|
|
10
|
+
export {};
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
//#region src/chat-commands/config-paths.ts
|
|
2
|
+
function parseConfigPath(raw) {
|
|
3
|
+
const parts = raw.trim().split(".").map((p) => p.trim());
|
|
4
|
+
if (parts.some((p) => !p)) return {
|
|
5
|
+
ok: false,
|
|
6
|
+
error: "Invalid path. Use dot notation (e.g. agents.defaults.model)."
|
|
7
|
+
};
|
|
8
|
+
return {
|
|
9
|
+
ok: true,
|
|
10
|
+
path: parts
|
|
11
|
+
};
|
|
12
|
+
}
|
|
13
|
+
function getConfigValueAtPath(root, path) {
|
|
14
|
+
let cursor = root;
|
|
15
|
+
for (const key of path) {
|
|
16
|
+
if (typeof cursor !== "object" || cursor === null) return void 0;
|
|
17
|
+
cursor = cursor[key];
|
|
18
|
+
}
|
|
19
|
+
return cursor;
|
|
20
|
+
}
|
|
21
|
+
function setConfigValueAtPath(root, path, value) {
|
|
22
|
+
let cursor = root;
|
|
23
|
+
for (let i = 0; i < path.length - 1; i++) {
|
|
24
|
+
const key = path[i];
|
|
25
|
+
if (typeof cursor[key] !== "object" || cursor[key] === null) cursor[key] = {};
|
|
26
|
+
cursor = cursor[key];
|
|
27
|
+
}
|
|
28
|
+
cursor[path[path.length - 1]] = value;
|
|
29
|
+
}
|
|
30
|
+
function unsetConfigValueAtPath(root, path) {
|
|
31
|
+
let cursor = root;
|
|
32
|
+
for (let i = 0; i < path.length - 1; i++) {
|
|
33
|
+
const next = cursor[path[i]];
|
|
34
|
+
if (typeof next !== "object" || next === null) return false;
|
|
35
|
+
cursor = next;
|
|
36
|
+
}
|
|
37
|
+
const leaf = path[path.length - 1];
|
|
38
|
+
if (!(leaf in cursor)) return false;
|
|
39
|
+
delete cursor[leaf];
|
|
40
|
+
return true;
|
|
41
|
+
}
|
|
42
|
+
//#endregion
|
|
43
|
+
export { getConfigValueAtPath, parseConfigPath, setConfigValueAtPath, unsetConfigValueAtPath };
|
|
44
|
+
|
|
45
|
+
//# sourceMappingURL=config-paths.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config-paths.js","names":[],"sources":["../../../src/chat-commands/config-paths.ts"],"sourcesContent":["type PathNode = Record<string, unknown>;\n\nexport function parseConfigPath(raw: string): { ok: boolean; path?: string[]; error?: string } {\n const parts = raw.trim().split('.').map((p) => p.trim());\n if (parts.some((p) => !p)) {\n return { ok: false, error: 'Invalid path. Use dot notation (e.g. agents.defaults.model).' };\n }\n return { ok: true, path: parts };\n}\n\nexport function getConfigValueAtPath(root: PathNode, path: string[]): unknown {\n let cursor: unknown = root;\n for (const key of path) {\n if (typeof cursor !== 'object' || cursor === null) return undefined;\n cursor = (cursor as PathNode)[key];\n }\n return cursor;\n}\n\nexport function setConfigValueAtPath(root: PathNode, path: string[], value: unknown): void {\n let cursor: PathNode = root;\n for (let i = 0; i < path.length - 1; i++) {\n const key = path[i];\n if (typeof cursor[key] !== 'object' || cursor[key] === null) cursor[key] = {};\n cursor = cursor[key] as PathNode;\n }\n cursor[path[path.length - 1]] = value;\n}\n\nexport function unsetConfigValueAtPath(root: PathNode, path: string[]): boolean {\n let cursor: PathNode = root;\n for (let i = 0; i < path.length - 1; i++) {\n const next = cursor[path[i]];\n if (typeof next !== 'object' || next === null) return false;\n cursor = next as PathNode;\n }\n const leaf = path[path.length - 1];\n if (!(leaf in cursor)) return false;\n delete cursor[leaf];\n return true;\n}\n"],"mappings":";AAEA,SAAgB,gBAAgB,KAA+D;CAC7F,MAAM,QAAQ,IAAI,MAAM,CAAC,MAAM,IAAI,CAAC,KAAK,MAAM,EAAE,MAAM,CAAC;AACxD,KAAI,MAAM,MAAM,MAAM,CAAC,EAAE,CACvB,QAAO;EAAE,IAAI;EAAO,OAAO;EAAgE;AAE7F,QAAO;EAAE,IAAI;EAAM,MAAM;EAAO;;AAGlC,SAAgB,qBAAqB,MAAgB,MAAyB;CAC5E,IAAI,SAAkB;AACtB,MAAK,MAAM,OAAO,MAAM;AACtB,MAAI,OAAO,WAAW,YAAY,WAAW,KAAM,QAAO,KAAA;AAC1D,WAAU,OAAoB;;AAEhC,QAAO;;AAGT,SAAgB,qBAAqB,MAAgB,MAAgB,OAAsB;CACzF,IAAI,SAAmB;AACvB,MAAK,IAAI,IAAI,GAAG,IAAI,KAAK,SAAS,GAAG,KAAK;EACxC,MAAM,MAAM,KAAK;AACjB,MAAI,OAAO,OAAO,SAAS,YAAY,OAAO,SAAS,KAAM,QAAO,OAAO,EAAE;AAC7E,WAAS,OAAO;;AAElB,QAAO,KAAK,KAAK,SAAS,MAAM;;AAGlC,SAAgB,uBAAuB,MAAgB,MAAyB;CAC9E,IAAI,SAAmB;AACvB,MAAK,IAAI,IAAI,GAAG,IAAI,KAAK,SAAS,GAAG,KAAK;EACxC,MAAM,OAAO,OAAO,KAAK;AACzB,MAAI,OAAO,SAAS,YAAY,SAAS,KAAM,QAAO;AACtD,WAAS;;CAEX,MAAM,OAAO,KAAK,KAAK,SAAS;AAChC,KAAI,EAAE,QAAQ,QAAS,QAAO;AAC9B,QAAO,OAAO;AACd,QAAO"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Infer the intended type of a config value from its raw string representation.
|
|
3
|
+
* Mirrors openclaw-style parseConfigValue behavior.
|
|
4
|
+
*/
|
|
5
|
+
export type ParsedConfigValue = {
|
|
6
|
+
ok: true;
|
|
7
|
+
value: unknown;
|
|
8
|
+
} | {
|
|
9
|
+
ok: false;
|
|
10
|
+
error: string;
|
|
11
|
+
};
|
|
12
|
+
export declare function parseConfigValue(raw: string): ParsedConfigValue;
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
//#region src/chat-commands/config-value.ts
|
|
2
|
+
function parseConfigValue(raw) {
|
|
3
|
+
const trimmed = raw.trim();
|
|
4
|
+
if (trimmed === "true") return {
|
|
5
|
+
ok: true,
|
|
6
|
+
value: true
|
|
7
|
+
};
|
|
8
|
+
if (trimmed === "false") return {
|
|
9
|
+
ok: true,
|
|
10
|
+
value: false
|
|
11
|
+
};
|
|
12
|
+
if (trimmed === "null") return {
|
|
13
|
+
ok: true,
|
|
14
|
+
value: null
|
|
15
|
+
};
|
|
16
|
+
if (/^-?\d+(\.\d+)?$/.test(trimmed)) {
|
|
17
|
+
const num = Number(trimmed);
|
|
18
|
+
if (!Number.isNaN(num)) return {
|
|
19
|
+
ok: true,
|
|
20
|
+
value: num
|
|
21
|
+
};
|
|
22
|
+
}
|
|
23
|
+
if (trimmed.startsWith("{") && trimmed.endsWith("}") || trimmed.startsWith("[") && trimmed.endsWith("]")) try {
|
|
24
|
+
return {
|
|
25
|
+
ok: true,
|
|
26
|
+
value: JSON.parse(trimmed)
|
|
27
|
+
};
|
|
28
|
+
} catch {
|
|
29
|
+
return {
|
|
30
|
+
ok: false,
|
|
31
|
+
error: "Invalid JSON value."
|
|
32
|
+
};
|
|
33
|
+
}
|
|
34
|
+
if (trimmed.startsWith("\"") && trimmed.endsWith("\"")) try {
|
|
35
|
+
return {
|
|
36
|
+
ok: true,
|
|
37
|
+
value: JSON.parse(trimmed)
|
|
38
|
+
};
|
|
39
|
+
} catch {
|
|
40
|
+
return {
|
|
41
|
+
ok: false,
|
|
42
|
+
error: "Invalid quoted string."
|
|
43
|
+
};
|
|
44
|
+
}
|
|
45
|
+
return {
|
|
46
|
+
ok: true,
|
|
47
|
+
value: trimmed
|
|
48
|
+
};
|
|
49
|
+
}
|
|
50
|
+
//#endregion
|
|
51
|
+
export { parseConfigValue };
|
|
52
|
+
|
|
53
|
+
//# sourceMappingURL=config-value.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config-value.js","names":[],"sources":["../../../src/chat-commands/config-value.ts"],"sourcesContent":["/**\n * Infer the intended type of a config value from its raw string representation.\n * Mirrors openclaw-style parseConfigValue behavior.\n */\n\nexport type ParsedConfigValue =\n | { ok: true; value: unknown }\n | { ok: false; error: string };\n\nexport function parseConfigValue(raw: string): ParsedConfigValue {\n const trimmed = raw.trim();\n\n if (trimmed === 'true') return { ok: true, value: true };\n if (trimmed === 'false') return { ok: true, value: false };\n if (trimmed === 'null') return { ok: true, value: null };\n\n if (/^-?\\d+(\\.\\d+)?$/.test(trimmed)) {\n const num = Number(trimmed);\n if (!Number.isNaN(num)) return { ok: true, value: num };\n }\n\n if (\n (trimmed.startsWith('{') && trimmed.endsWith('}')) ||\n (trimmed.startsWith('[') && trimmed.endsWith(']'))\n ) {\n try {\n return { ok: true, value: JSON.parse(trimmed) };\n } catch {\n return { ok: false, error: 'Invalid JSON value.' };\n }\n }\n\n if (trimmed.startsWith('\"') && trimmed.endsWith('\"')) {\n try {\n return { ok: true, value: JSON.parse(trimmed) };\n } catch {\n return { ok: false, error: 'Invalid quoted string.' };\n }\n }\n\n return { ok: true, value: trimmed };\n}\n"],"mappings":";AASA,SAAgB,iBAAiB,KAAgC;CAC/D,MAAM,UAAU,IAAI,MAAM;AAE1B,KAAI,YAAY,OAAQ,QAAO;EAAE,IAAI;EAAM,OAAO;EAAM;AACxD,KAAI,YAAY,QAAS,QAAO;EAAE,IAAI;EAAM,OAAO;EAAO;AAC1D,KAAI,YAAY,OAAQ,QAAO;EAAE,IAAI;EAAM,OAAO;EAAM;AAExD,KAAI,kBAAkB,KAAK,QAAQ,EAAE;EACnC,MAAM,MAAM,OAAO,QAAQ;AAC3B,MAAI,CAAC,OAAO,MAAM,IAAI,CAAE,QAAO;GAAE,IAAI;GAAM,OAAO;GAAK;;AAGzD,KACG,QAAQ,WAAW,IAAI,IAAI,QAAQ,SAAS,IAAI,IAChD,QAAQ,WAAW,IAAI,IAAI,QAAQ,SAAS,IAAI,CAEjD,KAAI;AACF,SAAO;GAAE,IAAI;GAAM,OAAO,KAAK,MAAM,QAAQ;GAAE;SACzC;AACN,SAAO;GAAE,IAAI;GAAO,OAAO;GAAuB;;AAItD,KAAI,QAAQ,WAAW,KAAI,IAAI,QAAQ,SAAS,KAAI,CAClD,KAAI;AACF,SAAO;GAAE,IAAI;GAAM,OAAO,KAAK,MAAM,QAAQ;GAAE;SACzC;AACN,SAAO;GAAE,IAAI;GAAO,OAAO;GAA0B;;AAIzD,QAAO;EAAE,IAAI;EAAM,OAAO;EAAS"}
|
|
@@ -4,12 +4,13 @@
|
|
|
4
4
|
* Provides the concrete implementation of CommandContext interface,
|
|
5
5
|
* bridging commands to core services (AgentService, SessionStore, etc.)
|
|
6
6
|
*/
|
|
7
|
-
import type { CommandContext, ReplyOptions, UIComponent, SessionInfo, ModelInfo, UsageStats, PlatformFeature, MessageSource } from './types.js';
|
|
7
|
+
import type { CommandContext, ReplyOptions, UIComponent, SessionInfo, ModelInfo, UsageStats, PlatformFeature, MessageSource, CompactSessionResult } from './types.js';
|
|
8
8
|
import type { Config } from '../config/schema.js';
|
|
9
9
|
import type { AgentMessage } from '@mariozechner/pi-agent-core';
|
|
10
10
|
import type { MessageBus } from '../infra/bus/index.js';
|
|
11
11
|
import type { SessionStore, SessionConfigStore } from '../session/index.js';
|
|
12
12
|
import type { ThinkLevel, ReasoningLevel, VerboseLevel } from '../agent/transcript/thinking-types.js';
|
|
13
|
+
import type { CompactionResult } from '../agent/memory/compaction.js';
|
|
13
14
|
export interface CommandContextDeps {
|
|
14
15
|
sessionKey: string;
|
|
15
16
|
source: MessageSource;
|
|
@@ -35,6 +36,15 @@ export interface CommandContextDeps {
|
|
|
35
36
|
getUsage?: () => Promise<UsageStats>;
|
|
36
37
|
/** Stop current LLM turn and clear channel preview stream (Telegram draft, etc.) */
|
|
37
38
|
abortCurrentTurn?: () => Promise<void>;
|
|
39
|
+
compactSession?: (sessionKey: string, options?: {
|
|
40
|
+
instructions?: string;
|
|
41
|
+
force?: boolean;
|
|
42
|
+
}) => Promise<CompactionResult>;
|
|
43
|
+
btwQuery?: (sessionKey: string, question: string) => Promise<{
|
|
44
|
+
text: string;
|
|
45
|
+
error?: string;
|
|
46
|
+
}>;
|
|
47
|
+
getSessionContextReport?: (sessionKey: string, mode: 'list' | 'detail' | 'json') => Promise<string>;
|
|
38
48
|
}
|
|
39
49
|
export declare class CommandContextImpl implements CommandContext {
|
|
40
50
|
readonly sessionKey: string;
|
|
@@ -74,6 +84,19 @@ export declare class CommandContextImpl implements CommandContext {
|
|
|
74
84
|
* Set thinking level for this session
|
|
75
85
|
*/
|
|
76
86
|
setThinkingLevel(level: ThinkLevel): Promise<void>;
|
|
87
|
+
syncAgentThinkingLevel(level: ThinkLevel): void;
|
|
88
|
+
compactSession(options?: {
|
|
89
|
+
instructions?: string;
|
|
90
|
+
force?: boolean;
|
|
91
|
+
}): Promise<CompactSessionResult | null>;
|
|
92
|
+
btwQuery(question: string): Promise<{
|
|
93
|
+
text: string;
|
|
94
|
+
error?: string;
|
|
95
|
+
}>;
|
|
96
|
+
exportSessionToWorkspace(format: 'markdown' | 'html' | 'json'): Promise<{
|
|
97
|
+
path: string;
|
|
98
|
+
}>;
|
|
99
|
+
agentContextReport(mode?: 'list' | 'detail' | 'json'): Promise<string>;
|
|
77
100
|
/**
|
|
78
101
|
* Get current reasoning level (session override or default)
|
|
79
102
|
*/
|
|
@@ -1,7 +1,11 @@
|
|
|
1
1
|
import { createLogger } from "../utils/logger/index.js";
|
|
2
2
|
import { init_logger } from "../utils/logger.js";
|
|
3
3
|
import { init_loader, saveConfig } from "../config/loader.js";
|
|
4
|
+
import { effectiveWorkspacePathForSession } from "../session/session-workspace.js";
|
|
4
5
|
import { getRoutingInfo, getSessionDisplayName } from "./session-key.js";
|
|
6
|
+
import { wrapMarkdownExportAsHtml } from "../session/chat-export.js";
|
|
7
|
+
import { mkdir, writeFile } from "fs/promises";
|
|
8
|
+
import { join } from "path";
|
|
5
9
|
//#region src/chat-commands/context.ts
|
|
6
10
|
init_logger();
|
|
7
11
|
init_loader();
|
|
@@ -173,6 +177,43 @@ var CommandContextImpl = class {
|
|
|
173
177
|
if (configStore) await configStore.update(this.sessionKey, { thinkingLevel: level });
|
|
174
178
|
this.deps.applySessionThinkingLevel?.(this.sessionKey, level);
|
|
175
179
|
}
|
|
180
|
+
syncAgentThinkingLevel(level) {
|
|
181
|
+
this.deps.applySessionThinkingLevel?.(this.sessionKey, level);
|
|
182
|
+
}
|
|
183
|
+
async compactSession(options) {
|
|
184
|
+
if (!this.deps.compactSession) return null;
|
|
185
|
+
const r = await this.deps.compactSession(this.sessionKey, options);
|
|
186
|
+
return {
|
|
187
|
+
compacted: r.compacted,
|
|
188
|
+
tokensBefore: r.tokensBefore,
|
|
189
|
+
tokensAfter: r.tokensAfter,
|
|
190
|
+
summary: r.summary
|
|
191
|
+
};
|
|
192
|
+
}
|
|
193
|
+
async btwQuery(question) {
|
|
194
|
+
if (!this.deps.btwQuery) return {
|
|
195
|
+
text: "",
|
|
196
|
+
error: "Side questions are not available in this environment."
|
|
197
|
+
};
|
|
198
|
+
return this.deps.btwQuery(this.sessionKey, question);
|
|
199
|
+
}
|
|
200
|
+
async exportSessionToWorkspace(format) {
|
|
201
|
+
const exportFmt = format === "json" ? "json" : "markdown";
|
|
202
|
+
let body = await this.deps.sessionStore.exportSession(this.sessionKey, exportFmt);
|
|
203
|
+
if (format === "html") body = wrapMarkdownExportAsHtml(`Session ${this.sessionKey}`, body);
|
|
204
|
+
const sc = this.deps.sessionConfigStore ? await this.deps.sessionConfigStore.get(this.sessionKey) : null;
|
|
205
|
+
const dir = join(effectiveWorkspacePathForSession(this.config, this.sessionKey, sc), "exports");
|
|
206
|
+
await mkdir(dir, { recursive: true });
|
|
207
|
+
const safe = this.sessionKey.replace(/[^a-zA-Z0-9._-]+/g, "_").slice(0, 96);
|
|
208
|
+
const ext = format === "json" ? "json" : format === "html" ? "html" : "md";
|
|
209
|
+
const outPath = join(dir, `session-${safe}-${Date.now()}.${ext}`);
|
|
210
|
+
await writeFile(outPath, body, "utf-8");
|
|
211
|
+
return { path: outPath };
|
|
212
|
+
}
|
|
213
|
+
async agentContextReport(mode = "list") {
|
|
214
|
+
if (!this.deps.getSessionContextReport) return "Context report is not available in this environment.";
|
|
215
|
+
return this.deps.getSessionContextReport(this.sessionKey, mode);
|
|
216
|
+
}
|
|
176
217
|
/**
|
|
177
218
|
* Get current reasoning level (session override or default)
|
|
178
219
|
*/
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"context.js","names":[],"sources":["../../../src/chat-commands/context.ts"],"sourcesContent":["/**\n * Command Context Implementation\n * \n * Provides the concrete implementation of CommandContext interface,\n * bridging commands to core services (AgentService, SessionStore, etc.)\n */\n\nimport type {\n CommandContext,\n ReplyOptions,\n UIComponent,\n SessionInfo,\n ModelInfo,\n UsageStats,\n PlatformFeature,\n MessageSource,\n} from './types.js';\nimport { getSessionDisplayName } from './session-key.js';\nimport type { Config } from '../config/schema.js';\nimport type { AgentMessage } from '@mariozechner/pi-agent-core';\nimport type { MessageBus } from '../infra/bus/index.js';\nimport type { SessionStore, SessionConfigStore } from '../session/index.js';\nimport { createLogger } from '../utils/logger.js';\nimport { getRoutingInfo } from './session-key.js';\nimport { saveConfig } from '../config/loader.js';\nimport type { ThinkLevel, ReasoningLevel, VerboseLevel } from '../agent/transcript/thinking-types.js';\n\nconst log = createLogger('CommandContext');\n\nexport interface CommandContextDeps {\n sessionKey: string;\n source: MessageSource;\n channelId: string;\n chatId: string;\n senderId: string;\n isGroup: boolean;\n config: Config;\n bus: MessageBus;\n sessionStore: SessionStore;\n sessionConfigStore?: SessionConfigStore;\n /** After persisting session thinking, sync pi-agent in-memory state */\n applySessionThinkingLevel?: (sessionKey: string, level: ThinkLevel) => void;\n // Callbacks for platform-specific operations\n replyHandler: (text: string, options?: ReplyOptions) => Promise<void>;\n componentHandler?: (component: UIComponent) => Promise<void>;\n typingHandler?: (typing: boolean) => Promise<void>;\n supportedFeatures: PlatformFeature[];\n /** Called after session files are removed so in-memory agents match disk */\n invalidateAgentSession?: (sessionKey: string) => void;\n // Model management (optional, will be injected)\n getCurrentModel?: () => string;\n switchModel?: (modelId: string) => Promise<boolean>;\n listModels?: () => Promise<ModelInfo[]>;\n getUsage?: () => Promise<UsageStats>;\n /** Stop current LLM turn and clear channel preview stream (Telegram draft, etc.) */\n abortCurrentTurn?: () => Promise<void>;\n}\n\nexport class CommandContextImpl implements CommandContext {\n readonly sessionKey: string;\n readonly source: MessageSource;\n readonly channelId: string;\n readonly chatId: string;\n readonly senderId: string;\n readonly isGroup: boolean;\n readonly config: Config;\n readonly abortCurrentTurn?: () => Promise<void>;\n\n private deps: CommandContextDeps;\n\n constructor(deps: CommandContextDeps) {\n this.sessionKey = deps.sessionKey;\n this.source = deps.source;\n this.channelId = deps.channelId;\n this.chatId = deps.chatId;\n this.senderId = deps.senderId;\n this.isGroup = deps.isGroup;\n this.config = deps.config;\n this.deps = deps;\n\n if (deps.abortCurrentTurn) {\n const run = deps.abortCurrentTurn;\n this.abortCurrentTurn = async () => {\n await run();\n };\n }\n }\n\n // === Reply API ===\n\n async reply(text: string, options?: ReplyOptions): Promise<void> {\n await this.deps.replyHandler(text, options);\n }\n\n async replyComponent(component: UIComponent): Promise<void> {\n if (this.deps.componentHandler) {\n await this.deps.componentHandler(component);\n } else {\n // Fallback to text representation\n await this.reply(this.renderComponentAsText(component));\n }\n }\n\n async setTyping(typing: boolean): Promise<void> {\n if (this.deps.typingHandler) {\n await this.deps.typingHandler(typing);\n }\n }\n\n // === Session Management ===\n\n async getSession(): Promise<AgentMessage[]> {\n return this.deps.sessionStore.load(this.sessionKey);\n }\n\n async clearSession(): Promise<void> {\n // Archive first if has messages\n const messages = await this.getSession();\n if (messages.length > 0) {\n await this.deps.sessionStore.archive(this.sessionKey);\n log.info({ sessionKey: this.sessionKey, messageCount: messages.length }, 'Session archived');\n }\n\n // Delete session\n await this.deps.sessionStore.deleteSession(this.sessionKey);\n this.deps.invalidateAgentSession?.(this.sessionKey);\n\n // Publish outbound message to confirm\n const routing = getRoutingInfo(this.sessionKey);\n await this.deps.bus.publishOutbound({\n channel: routing.channel,\n chat_id: routing.chatId,\n content: '✅ New session started. Previous session has been archived.',\n type: 'message',\n metadata: {\n threadId: routing.threadId,\n },\n });\n\n log.info({ sessionKey: this.sessionKey }, 'Session cleared');\n }\n\n async archiveSession(): Promise<void> {\n await this.deps.sessionStore.archive(this.sessionKey);\n log.info({ sessionKey: this.sessionKey }, 'Session archived');\n }\n\n async listSessions(): Promise<SessionInfo[]> {\n // TODO: Implement listSessions in SessionStore\n // For now, return current session only\n const messages = await this.getSession();\n return [{\n key: this.sessionKey,\n name: getSessionDisplayName(this.sessionKey),\n messageCount: messages.length,\n createdAt: new Date(),\n updatedAt: new Date(),\n isActive: true,\n }];\n }\n\n async switchSession(sessionKey: string): Promise<void> {\n // This is mainly for CLI/Web UI where you can switch between sessions\n // For Telegram, each chat has its own session\n log.info({ from: this.sessionKey, to: sessionKey }, 'Session switch requested');\n \n // Note: In the current architecture, switching session means\n // the next message will use a different sessionKey\n // The actual switch happens at the adapter level\n }\n\n // === Model Management ===\n\n getCurrentModel(): string {\n if (this.deps.getCurrentModel) {\n return this.deps.getCurrentModel();\n }\n \n // Fallback to config default\n const modelConfig = this.config.agents?.defaults?.model;\n return typeof modelConfig === 'string' ? modelConfig : modelConfig?.primary || 'minimax/minimax-m2.1';\n }\n\n async listModels(): Promise<ModelInfo[]> {\n if (this.deps.listModels) {\n return this.deps.listModels();\n }\n \n // Fallback to empty list\n return [];\n }\n\n async switchModel(modelId: string): Promise<boolean> {\n if (this.deps.switchModel) {\n return this.deps.switchModel(modelId);\n }\n \n // No model manager available\n await this.reply('❌ Model switching not available in this context.');\n return false;\n }\n\n async getUsage(): Promise<UsageStats> {\n if (this.deps.getUsage) {\n return this.deps.getUsage();\n }\n \n // Fallback: calculate from session\n const messages = await this.getSession();\n let promptTokens = 0;\n let completionTokens = 0;\n \n for (const msg of messages) {\n if ('usage' in msg && msg.usage) {\n promptTokens += msg.usage.input || 0;\n completionTokens += msg.usage.output || 0;\n }\n }\n \n return {\n promptTokens,\n completionTokens,\n totalTokens: promptTokens + completionTokens,\n messageCount: messages.length,\n };\n }\n\n // === Platform Features ===\n\n supports(feature: PlatformFeature): boolean {\n return this.deps.supportedFeatures.includes(feature);\n }\n\n // === Configuration ===\n\n getConfig(): Config {\n return this.config;\n }\n\n async updateConfig(path: string, value: unknown): Promise<boolean> {\n try {\n // Update config object using path\n const keys = path.split('.');\n let target: Record<string, unknown> = this.config as Record<string, unknown>;\n \n for (let i = 0; i < keys.length - 1; i++) {\n const key = keys[i];\n if (!(key in target) || typeof target[key] !== 'object' || target[key] === null) {\n target[key] = {};\n }\n target = target[key] as Record<string, unknown>;\n }\n \n target[keys[keys.length - 1]] = value;\n \n // Save to disk\n await saveConfig(this.config);\n \n log.info({ path, value }, 'Config updated via command');\n return true;\n } catch (error) {\n log.error({ err: error, path, value }, 'Failed to update config');\n return false;\n }\n }\n\n // === Private Helpers ===\n\n // === Thinking Configuration ===\n\n /**\n * Get the session config store (if available)\n */\n getSessionConfigStore(): SessionConfigStore | undefined {\n return this.deps.sessionConfigStore;\n }\n\n /**\n * Get current thinking level (session override or default)\n */\n async getThinkingLevel(): Promise<ThinkLevel | undefined> {\n const configStore = this.deps.sessionConfigStore;\n if (configStore) {\n const sessionConfig = await configStore.get(this.sessionKey);\n if (sessionConfig?.thinkingLevel) {\n return sessionConfig.thinkingLevel;\n }\n }\n // Fallback to agent default\n return this.config.agents?.defaults?.thinkingDefault;\n }\n\n /**\n * Set thinking level for this session\n */\n async setThinkingLevel(level: ThinkLevel): Promise<void> {\n const configStore = this.deps.sessionConfigStore;\n if (configStore) {\n await configStore.update(this.sessionKey, { thinkingLevel: level });\n }\n this.deps.applySessionThinkingLevel?.(this.sessionKey, level);\n }\n\n /**\n * Get current reasoning level (session override or default)\n */\n async getReasoningLevel(): Promise<ReasoningLevel | undefined> {\n const configStore = this.deps.sessionConfigStore;\n if (configStore) {\n const sessionConfig = await configStore.get(this.sessionKey);\n if (sessionConfig?.reasoningLevel) {\n return sessionConfig.reasoningLevel;\n }\n }\n // Fallback to agent default\n return this.config.agents?.defaults?.reasoningDefault;\n }\n\n /**\n * Set reasoning level for this session\n */\n async setReasoningLevel(level: ReasoningLevel): Promise<void> {\n const configStore = this.deps.sessionConfigStore;\n if (configStore) {\n await configStore.update(this.sessionKey, { reasoningLevel: level });\n }\n }\n\n /**\n * Get current verbose level (session override or default)\n */\n async getVerboseLevel(): Promise<VerboseLevel | undefined> {\n const configStore = this.deps.sessionConfigStore;\n if (configStore) {\n const sessionConfig = await configStore.get(this.sessionKey);\n if (sessionConfig?.verboseLevel) {\n return sessionConfig.verboseLevel;\n }\n }\n // Fallback to agent default\n return this.config.agents?.defaults?.verboseDefault;\n }\n\n /**\n * Set verbose level for this session\n */\n async setVerboseLevel(level: VerboseLevel): Promise<void> {\n const configStore = this.deps.sessionConfigStore;\n if (configStore) {\n await configStore.update(this.sessionKey, { verboseLevel: level });\n }\n }\n\n private renderComponentAsText(component: UIComponent): string {\n switch (component.type) {\n case 'buttons':\n return component.buttons.map(b => `[${b.text}]`).join(' ');\n \n case 'select':\n return component.options.map(o => `- ${o.label}`).join('\\n');\n \n case 'model-picker':\n return component.providers.map(p => \n `**${p.name}**\\n${p.models.map(m => ` - ${m.name}`).join('\\n')}`\n ).join('\\n\\n');\n \n case 'usage-display':\n return `📊 Usage Stats:\\n` +\n `📥 Prompt: ${component.stats.promptTokens.toLocaleString()} tokens\\n` +\n `📤 Completion: ${component.stats.completionTokens.toLocaleString()} tokens\\n` +\n `📊 Total: ${component.stats.totalTokens.toLocaleString()} tokens`;\n \n case 'session-list':\n return component.sessions.map(s => \n `${s.isActive ? '▶️' : ' '} ${s.key} (${s.messageCount} messages)`\n ).join('\\n');\n \n case 'text-input':\n return component.placeholder || 'Enter text...';\n \n default:\n return '[UI Component]';\n }\n }\n}\n\n/**\n * Create a command context from dependencies\n */\nexport function createCommandContext(deps: CommandContextDeps): CommandContext {\n return new CommandContextImpl(deps);\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"context.js","names":[],"sources":["../../../src/chat-commands/context.ts"],"sourcesContent":["/**\n * Command Context Implementation\n * \n * Provides the concrete implementation of CommandContext interface,\n * bridging commands to core services (AgentService, SessionStore, etc.)\n */\n\nimport type {\n CommandContext,\n ReplyOptions,\n UIComponent,\n SessionInfo,\n ModelInfo,\n UsageStats,\n PlatformFeature,\n MessageSource,\n CompactSessionResult,\n} from './types.js';\nimport { getSessionDisplayName } from './session-key.js';\nimport type { Config } from '../config/schema.js';\nimport type { AgentMessage } from '@mariozechner/pi-agent-core';\nimport type { MessageBus } from '../infra/bus/index.js';\nimport type { SessionStore, SessionConfigStore } from '../session/index.js';\nimport { createLogger } from '../utils/logger.js';\nimport { getRoutingInfo } from './session-key.js';\nimport { saveConfig } from '../config/loader.js';\nimport type { ThinkLevel, ReasoningLevel, VerboseLevel } from '../agent/transcript/thinking-types.js';\nimport { mkdir, writeFile } from 'fs/promises';\nimport { join } from 'path';\nimport { effectiveWorkspacePathForSession } from '../session/session-workspace.js';\nimport { wrapMarkdownExportAsHtml } from '../session/chat-export.js';\nimport type { CompactionResult } from '../agent/memory/compaction.js';\n\nconst log = createLogger('CommandContext');\n\nexport interface CommandContextDeps {\n sessionKey: string;\n source: MessageSource;\n channelId: string;\n chatId: string;\n senderId: string;\n isGroup: boolean;\n config: Config;\n bus: MessageBus;\n sessionStore: SessionStore;\n sessionConfigStore?: SessionConfigStore;\n /** After persisting session thinking, sync pi-agent in-memory state */\n applySessionThinkingLevel?: (sessionKey: string, level: ThinkLevel) => void;\n // Callbacks for platform-specific operations\n replyHandler: (text: string, options?: ReplyOptions) => Promise<void>;\n componentHandler?: (component: UIComponent) => Promise<void>;\n typingHandler?: (typing: boolean) => Promise<void>;\n supportedFeatures: PlatformFeature[];\n /** Called after session files are removed so in-memory agents match disk */\n invalidateAgentSession?: (sessionKey: string) => void;\n // Model management (optional, will be injected)\n getCurrentModel?: () => string;\n switchModel?: (modelId: string) => Promise<boolean>;\n listModels?: () => Promise<ModelInfo[]>;\n getUsage?: () => Promise<UsageStats>;\n /** Stop current LLM turn and clear channel preview stream (Telegram draft, etc.) */\n abortCurrentTurn?: () => Promise<void>;\n\n compactSession?: (\n sessionKey: string,\n options?: { instructions?: string; force?: boolean },\n ) => Promise<CompactionResult>;\n\n btwQuery?: (sessionKey: string, question: string) => Promise<{ text: string; error?: string }>;\n\n getSessionContextReport?: (\n sessionKey: string,\n mode: 'list' | 'detail' | 'json',\n ) => Promise<string>;\n}\n\nexport class CommandContextImpl implements CommandContext {\n readonly sessionKey: string;\n readonly source: MessageSource;\n readonly channelId: string;\n readonly chatId: string;\n readonly senderId: string;\n readonly isGroup: boolean;\n readonly config: Config;\n readonly abortCurrentTurn?: () => Promise<void>;\n\n private deps: CommandContextDeps;\n\n constructor(deps: CommandContextDeps) {\n this.sessionKey = deps.sessionKey;\n this.source = deps.source;\n this.channelId = deps.channelId;\n this.chatId = deps.chatId;\n this.senderId = deps.senderId;\n this.isGroup = deps.isGroup;\n this.config = deps.config;\n this.deps = deps;\n\n if (deps.abortCurrentTurn) {\n const run = deps.abortCurrentTurn;\n this.abortCurrentTurn = async () => {\n await run();\n };\n }\n }\n\n // === Reply API ===\n\n async reply(text: string, options?: ReplyOptions): Promise<void> {\n await this.deps.replyHandler(text, options);\n }\n\n async replyComponent(component: UIComponent): Promise<void> {\n if (this.deps.componentHandler) {\n await this.deps.componentHandler(component);\n } else {\n // Fallback to text representation\n await this.reply(this.renderComponentAsText(component));\n }\n }\n\n async setTyping(typing: boolean): Promise<void> {\n if (this.deps.typingHandler) {\n await this.deps.typingHandler(typing);\n }\n }\n\n // === Session Management ===\n\n async getSession(): Promise<AgentMessage[]> {\n return this.deps.sessionStore.load(this.sessionKey);\n }\n\n async clearSession(): Promise<void> {\n // Archive first if has messages\n const messages = await this.getSession();\n if (messages.length > 0) {\n await this.deps.sessionStore.archive(this.sessionKey);\n log.info({ sessionKey: this.sessionKey, messageCount: messages.length }, 'Session archived');\n }\n\n // Delete session\n await this.deps.sessionStore.deleteSession(this.sessionKey);\n this.deps.invalidateAgentSession?.(this.sessionKey);\n\n // Publish outbound message to confirm\n const routing = getRoutingInfo(this.sessionKey);\n await this.deps.bus.publishOutbound({\n channel: routing.channel,\n chat_id: routing.chatId,\n content: '✅ New session started. Previous session has been archived.',\n type: 'message',\n metadata: {\n threadId: routing.threadId,\n },\n });\n\n log.info({ sessionKey: this.sessionKey }, 'Session cleared');\n }\n\n async archiveSession(): Promise<void> {\n await this.deps.sessionStore.archive(this.sessionKey);\n log.info({ sessionKey: this.sessionKey }, 'Session archived');\n }\n\n async listSessions(): Promise<SessionInfo[]> {\n // TODO: Implement listSessions in SessionStore\n // For now, return current session only\n const messages = await this.getSession();\n return [{\n key: this.sessionKey,\n name: getSessionDisplayName(this.sessionKey),\n messageCount: messages.length,\n createdAt: new Date(),\n updatedAt: new Date(),\n isActive: true,\n }];\n }\n\n async switchSession(sessionKey: string): Promise<void> {\n // This is mainly for CLI/Web UI where you can switch between sessions\n // For Telegram, each chat has its own session\n log.info({ from: this.sessionKey, to: sessionKey }, 'Session switch requested');\n \n // Note: In the current architecture, switching session means\n // the next message will use a different sessionKey\n // The actual switch happens at the adapter level\n }\n\n // === Model Management ===\n\n getCurrentModel(): string {\n if (this.deps.getCurrentModel) {\n return this.deps.getCurrentModel();\n }\n \n // Fallback to config default\n const modelConfig = this.config.agents?.defaults?.model;\n return typeof modelConfig === 'string' ? modelConfig : modelConfig?.primary || 'minimax/minimax-m2.1';\n }\n\n async listModels(): Promise<ModelInfo[]> {\n if (this.deps.listModels) {\n return this.deps.listModels();\n }\n \n // Fallback to empty list\n return [];\n }\n\n async switchModel(modelId: string): Promise<boolean> {\n if (this.deps.switchModel) {\n return this.deps.switchModel(modelId);\n }\n \n // No model manager available\n await this.reply('❌ Model switching not available in this context.');\n return false;\n }\n\n async getUsage(): Promise<UsageStats> {\n if (this.deps.getUsage) {\n return this.deps.getUsage();\n }\n \n // Fallback: calculate from session\n const messages = await this.getSession();\n let promptTokens = 0;\n let completionTokens = 0;\n \n for (const msg of messages) {\n if ('usage' in msg && msg.usage) {\n promptTokens += msg.usage.input || 0;\n completionTokens += msg.usage.output || 0;\n }\n }\n \n return {\n promptTokens,\n completionTokens,\n totalTokens: promptTokens + completionTokens,\n messageCount: messages.length,\n };\n }\n\n // === Platform Features ===\n\n supports(feature: PlatformFeature): boolean {\n return this.deps.supportedFeatures.includes(feature);\n }\n\n // === Configuration ===\n\n getConfig(): Config {\n return this.config;\n }\n\n async updateConfig(path: string, value: unknown): Promise<boolean> {\n try {\n // Update config object using path\n const keys = path.split('.');\n let target: Record<string, unknown> = this.config as Record<string, unknown>;\n \n for (let i = 0; i < keys.length - 1; i++) {\n const key = keys[i];\n if (!(key in target) || typeof target[key] !== 'object' || target[key] === null) {\n target[key] = {};\n }\n target = target[key] as Record<string, unknown>;\n }\n \n target[keys[keys.length - 1]] = value;\n \n // Save to disk\n await saveConfig(this.config);\n \n log.info({ path, value }, 'Config updated via command');\n return true;\n } catch (error) {\n log.error({ err: error, path, value }, 'Failed to update config');\n return false;\n }\n }\n\n // === Private Helpers ===\n\n // === Thinking Configuration ===\n\n /**\n * Get the session config store (if available)\n */\n getSessionConfigStore(): SessionConfigStore | undefined {\n return this.deps.sessionConfigStore;\n }\n\n /**\n * Get current thinking level (session override or default)\n */\n async getThinkingLevel(): Promise<ThinkLevel | undefined> {\n const configStore = this.deps.sessionConfigStore;\n if (configStore) {\n const sessionConfig = await configStore.get(this.sessionKey);\n if (sessionConfig?.thinkingLevel) {\n return sessionConfig.thinkingLevel;\n }\n }\n // Fallback to agent default\n return this.config.agents?.defaults?.thinkingDefault;\n }\n\n /**\n * Set thinking level for this session\n */\n async setThinkingLevel(level: ThinkLevel): Promise<void> {\n const configStore = this.deps.sessionConfigStore;\n if (configStore) {\n await configStore.update(this.sessionKey, { thinkingLevel: level });\n }\n this.deps.applySessionThinkingLevel?.(this.sessionKey, level);\n }\n\n syncAgentThinkingLevel(level: ThinkLevel): void {\n this.deps.applySessionThinkingLevel?.(this.sessionKey, level);\n }\n\n async compactSession(options?: { instructions?: string; force?: boolean }): Promise<CompactSessionResult | null> {\n if (!this.deps.compactSession) {\n return null;\n }\n const r = await this.deps.compactSession(this.sessionKey, options);\n return {\n compacted: r.compacted,\n tokensBefore: r.tokensBefore,\n tokensAfter: r.tokensAfter,\n summary: r.summary,\n };\n }\n\n async btwQuery(question: string): Promise<{ text: string; error?: string }> {\n if (!this.deps.btwQuery) {\n return { text: '', error: 'Side questions are not available in this environment.' };\n }\n return this.deps.btwQuery(this.sessionKey, question);\n }\n\n async exportSessionToWorkspace(format: 'markdown' | 'html' | 'json'): Promise<{ path: string }> {\n const exportFmt = format === 'json' ? 'json' : 'markdown';\n let body = await this.deps.sessionStore.exportSession(this.sessionKey, exportFmt);\n if (format === 'html') {\n body = wrapMarkdownExportAsHtml(`Session ${this.sessionKey}`, body);\n }\n const sc = this.deps.sessionConfigStore\n ? await this.deps.sessionConfigStore.get(this.sessionKey)\n : null;\n const root = effectiveWorkspacePathForSession(this.config, this.sessionKey, sc);\n const dir = join(root, 'exports');\n await mkdir(dir, { recursive: true });\n const safe = this.sessionKey.replace(/[^a-zA-Z0-9._-]+/g, '_').slice(0, 96);\n const ext = format === 'json' ? 'json' : format === 'html' ? 'html' : 'md';\n const name = `session-${safe}-${Date.now()}.${ext}`;\n const outPath = join(dir, name);\n await writeFile(outPath, body, 'utf-8');\n return { path: outPath };\n }\n\n async agentContextReport(mode: 'list' | 'detail' | 'json' = 'list'): Promise<string> {\n if (!this.deps.getSessionContextReport) {\n return 'Context report is not available in this environment.';\n }\n return this.deps.getSessionContextReport(this.sessionKey, mode);\n }\n\n /**\n * Get current reasoning level (session override or default)\n */\n async getReasoningLevel(): Promise<ReasoningLevel | undefined> {\n const configStore = this.deps.sessionConfigStore;\n if (configStore) {\n const sessionConfig = await configStore.get(this.sessionKey);\n if (sessionConfig?.reasoningLevel) {\n return sessionConfig.reasoningLevel;\n }\n }\n // Fallback to agent default\n return this.config.agents?.defaults?.reasoningDefault;\n }\n\n /**\n * Set reasoning level for this session\n */\n async setReasoningLevel(level: ReasoningLevel): Promise<void> {\n const configStore = this.deps.sessionConfigStore;\n if (configStore) {\n await configStore.update(this.sessionKey, { reasoningLevel: level });\n }\n }\n\n /**\n * Get current verbose level (session override or default)\n */\n async getVerboseLevel(): Promise<VerboseLevel | undefined> {\n const configStore = this.deps.sessionConfigStore;\n if (configStore) {\n const sessionConfig = await configStore.get(this.sessionKey);\n if (sessionConfig?.verboseLevel) {\n return sessionConfig.verboseLevel;\n }\n }\n // Fallback to agent default\n return this.config.agents?.defaults?.verboseDefault;\n }\n\n /**\n * Set verbose level for this session\n */\n async setVerboseLevel(level: VerboseLevel): Promise<void> {\n const configStore = this.deps.sessionConfigStore;\n if (configStore) {\n await configStore.update(this.sessionKey, { verboseLevel: level });\n }\n }\n\n private renderComponentAsText(component: UIComponent): string {\n switch (component.type) {\n case 'buttons':\n return component.buttons.map(b => `[${b.text}]`).join(' ');\n \n case 'select':\n return component.options.map(o => `- ${o.label}`).join('\\n');\n \n case 'model-picker':\n return component.providers.map(p => \n `**${p.name}**\\n${p.models.map(m => ` - ${m.name}`).join('\\n')}`\n ).join('\\n\\n');\n \n case 'usage-display':\n return `📊 Usage Stats:\\n` +\n `📥 Prompt: ${component.stats.promptTokens.toLocaleString()} tokens\\n` +\n `📤 Completion: ${component.stats.completionTokens.toLocaleString()} tokens\\n` +\n `📊 Total: ${component.stats.totalTokens.toLocaleString()} tokens`;\n \n case 'session-list':\n return component.sessions.map(s => \n `${s.isActive ? '▶️' : ' '} ${s.key} (${s.messageCount} messages)`\n ).join('\\n');\n \n case 'text-input':\n return component.placeholder || 'Enter text...';\n \n default:\n return '[UI Component]';\n }\n }\n}\n\n/**\n * Create a command context from dependencies\n */\nexport function createCommandContext(deps: CommandContextDeps): CommandContext {\n return new CommandContextImpl(deps);\n}\n"],"mappings":"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"}
|
|
@@ -15,6 +15,8 @@ export { registerModelCommands } from './builtins/model.js';
|
|
|
15
15
|
export { registerSystemCommands } from './builtins/system.js';
|
|
16
16
|
export { registerTTSCommands } from './builtins/tts.js';
|
|
17
17
|
export { registerThinkingCommands } from './builtins/thinking.js';
|
|
18
|
+
export { registerConfigCommand } from './builtins/config.js';
|
|
19
|
+
export { registerContextCommands } from './builtins/context.js';
|
|
18
20
|
/**
|
|
19
21
|
* Initialize the command system with all built-in commands
|
|
20
22
|
*/
|
|
@@ -6,6 +6,8 @@ import { registerModelCommands } from "./builtins/model.js";
|
|
|
6
6
|
import { registerSystemCommands } from "./builtins/system.js";
|
|
7
7
|
import { registerTTSCommands } from "./builtins/tts.js";
|
|
8
8
|
import { registerThinkingCommands } from "./builtins/thinking.js";
|
|
9
|
+
import { registerConfigCommand } from "./builtins/config.js";
|
|
10
|
+
import { registerContextCommands } from "./builtins/context.js";
|
|
9
11
|
import { CommandContextImpl, createCommandContext } from "./context.js";
|
|
10
12
|
//#region src/chat-commands/index.ts
|
|
11
13
|
/**
|
|
@@ -21,10 +23,12 @@ function initializeCommands() {
|
|
|
21
23
|
registerSessionCommands();
|
|
22
24
|
registerModelCommands();
|
|
23
25
|
registerSystemCommands();
|
|
26
|
+
registerConfigCommand();
|
|
27
|
+
registerContextCommands();
|
|
24
28
|
registerTTSCommands();
|
|
25
29
|
registerThinkingCommands();
|
|
26
30
|
}
|
|
27
31
|
//#endregion
|
|
28
|
-
export { CommandContextImpl, CommandRegistry, commandRegistry, createCommandContext, generateSessionKey, getRoutingInfo, getSessionDisplayName, initializeCommands, isValidSessionKey, normalizeTelegramCommandName, parseSessionKey, parseSlashCommand, registerModelCommands, registerSessionCommands, registerSystemCommands, registerTTSCommands, registerThinkingCommands };
|
|
32
|
+
export { CommandContextImpl, CommandRegistry, commandRegistry, createCommandContext, generateSessionKey, getRoutingInfo, getSessionDisplayName, initializeCommands, isValidSessionKey, normalizeTelegramCommandName, parseSessionKey, parseSlashCommand, registerConfigCommand, registerContextCommands, registerModelCommands, registerSessionCommands, registerSystemCommands, registerTTSCommands, registerThinkingCommands };
|
|
29
33
|
|
|
30
34
|
//# sourceMappingURL=index.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","names":[],"sources":["../../../src/chat-commands/index.ts"],"sourcesContent":["/**\n * Unified Command System\n *\n * Provides a platform-agnostic command system that works across\n * Telegram, Feishu, Web UI, CLI, and other channel surfaces.\n */\n\n// Built-in Commands (import first to avoid circular deps)\nimport { registerSessionCommands } from './builtins/session.js';\nimport { registerModelCommands } from './builtins/model.js';\nimport { registerSystemCommands } from './builtins/system.js';\nimport { registerTTSCommands } from './builtins/tts.js';\nimport { registerThinkingCommands } from './builtins/thinking.js';\n\n// Types\nexport type {\n MessageSource,\n UnifiedMessage,\n PlatformMetadata,\n MessageAttachment,\n CommandDefinition,\n CommandCategory,\n CommandScope,\n CommandHandler,\n CommandResult,\n CommandContext,\n ReplyOptions,\n UIComponent,\n ButtonGroup,\n SelectMenu,\n ModelPicker,\n UsageDisplay,\n SessionList,\n TextInput,\n ProviderInfo,\n ModelInfo,\n UsageStats,\n SessionInfo,\n PlatformFeature,\n ChannelAdapter,\n ReplyPayload,\n} from './types.js';\n\n// Session Key\nexport {\n generateSessionKey,\n parseSessionKey,\n isValidSessionKey,\n getSessionDisplayName,\n getRoutingInfo,\n type SessionKeyContext,\n} from './session-key.js';\n\n// Command parsing helpers\nexport { normalizeTelegramCommandName, parseSlashCommand } from './command-parse.js';\n\n// Registry\nexport { CommandRegistry, commandRegistry } from './registry.js';\nexport type { CommandRegistry as CommandRegistryType } from './types.js';\n\n// Context\nexport { CommandContextImpl, createCommandContext } from './context.js';\n\n// Built-in Commands\nexport { registerSessionCommands } from './builtins/session.js';\nexport { registerModelCommands } from './builtins/model.js';\nexport { registerSystemCommands } from './builtins/system.js';\nexport { registerTTSCommands } from './builtins/tts.js';\nexport { registerThinkingCommands } from './builtins/thinking.js';\n\n/**\n * Initialize the command system with all built-in commands\n */\nexport function initializeCommands(): void {\n registerSessionCommands();\n registerModelCommands();\n registerSystemCommands();\n registerTTSCommands();\n registerThinkingCommands();\n}\n"],"mappings":"
|
|
1
|
+
{"version":3,"file":"index.js","names":[],"sources":["../../../src/chat-commands/index.ts"],"sourcesContent":["/**\n * Unified Command System\n *\n * Provides a platform-agnostic command system that works across\n * Telegram, Feishu, Web UI, CLI, and other channel surfaces.\n */\n\n// Built-in Commands (import first to avoid circular deps)\nimport { registerSessionCommands } from './builtins/session.js';\nimport { registerModelCommands } from './builtins/model.js';\nimport { registerSystemCommands } from './builtins/system.js';\nimport { registerTTSCommands } from './builtins/tts.js';\nimport { registerThinkingCommands } from './builtins/thinking.js';\nimport { registerConfigCommand } from './builtins/config.js';\nimport { registerContextCommands } from './builtins/context.js';\n\n// Types\nexport type {\n MessageSource,\n UnifiedMessage,\n PlatformMetadata,\n MessageAttachment,\n CommandDefinition,\n CommandCategory,\n CommandScope,\n CommandHandler,\n CommandResult,\n CommandContext,\n ReplyOptions,\n UIComponent,\n ButtonGroup,\n SelectMenu,\n ModelPicker,\n UsageDisplay,\n SessionList,\n TextInput,\n ProviderInfo,\n ModelInfo,\n UsageStats,\n SessionInfo,\n PlatformFeature,\n ChannelAdapter,\n ReplyPayload,\n} from './types.js';\n\n// Session Key\nexport {\n generateSessionKey,\n parseSessionKey,\n isValidSessionKey,\n getSessionDisplayName,\n getRoutingInfo,\n type SessionKeyContext,\n} from './session-key.js';\n\n// Command parsing helpers\nexport { normalizeTelegramCommandName, parseSlashCommand } from './command-parse.js';\n\n// Registry\nexport { CommandRegistry, commandRegistry } from './registry.js';\nexport type { CommandRegistry as CommandRegistryType } from './types.js';\n\n// Context\nexport { CommandContextImpl, createCommandContext } from './context.js';\n\n// Built-in Commands\nexport { registerSessionCommands } from './builtins/session.js';\nexport { registerModelCommands } from './builtins/model.js';\nexport { registerSystemCommands } from './builtins/system.js';\nexport { registerTTSCommands } from './builtins/tts.js';\nexport { registerThinkingCommands } from './builtins/thinking.js';\nexport { registerConfigCommand } from './builtins/config.js';\nexport { registerContextCommands } from './builtins/context.js';\n\n/**\n * Initialize the command system with all built-in commands\n */\nexport function initializeCommands(): void {\n registerSessionCommands();\n registerModelCommands();\n registerSystemCommands();\n registerConfigCommand();\n registerContextCommands();\n registerTTSCommands();\n registerThinkingCommands();\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;AA6EA,SAAgB,qBAA2B;AACzC,0BAAyB;AACzB,wBAAuB;AACvB,yBAAwB;AACxB,wBAAuB;AACvB,0BAAyB;AACzB,sBAAqB;AACrB,2BAA0B"}
|
|
@@ -6,7 +6,8 @@
|
|
|
6
6
|
*/
|
|
7
7
|
import type { Config } from '../config/schema.js';
|
|
8
8
|
import type { AgentMessage } from '@mariozechner/pi-agent-core';
|
|
9
|
-
import type { ThinkLevel } from '../agent/transcript/thinking-types.js';
|
|
9
|
+
import type { ThinkLevel, ReasoningLevel, VerboseLevel } from '../agent/transcript/thinking-types.js';
|
|
10
|
+
import type { SessionConfigStore } from '../session/index.js';
|
|
10
11
|
export type MessageSource = 'telegram' | 'weixin' | 'webui' | 'cli' | 'api' | 'system' | 'gateway';
|
|
11
12
|
export interface UnifiedMessage {
|
|
12
13
|
/** Message source platform */
|
|
@@ -85,6 +86,12 @@ export interface CommandDefinition {
|
|
|
85
86
|
handler: CommandHandler;
|
|
86
87
|
}
|
|
87
88
|
export type CommandHandler = (context: CommandContext, args: string) => Promise<CommandResult | void>;
|
|
89
|
+
export interface CompactSessionResult {
|
|
90
|
+
compacted: boolean;
|
|
91
|
+
tokensBefore: number;
|
|
92
|
+
tokensAfter: number;
|
|
93
|
+
summary?: string;
|
|
94
|
+
}
|
|
88
95
|
export interface CommandResult {
|
|
89
96
|
/** Response text */
|
|
90
97
|
content: string;
|
|
@@ -142,6 +149,31 @@ export interface CommandContext {
|
|
|
142
149
|
updateConfig?(path: string, value: unknown): Promise<boolean>;
|
|
143
150
|
/** Persist session thinking level and sync in-memory agent (when wired) */
|
|
144
151
|
setThinkingLevel?(level: ThinkLevel): Promise<void>;
|
|
152
|
+
/**
|
|
153
|
+
* Update pi-agent thinking level in memory only (used with runtime config overrides so session
|
|
154
|
+
* store does not mask merged defaults).
|
|
155
|
+
*/
|
|
156
|
+
syncAgentThinkingLevel?(level: ThinkLevel): void;
|
|
157
|
+
getSessionConfigStore?(): SessionConfigStore | undefined;
|
|
158
|
+
getThinkingLevel?(): Promise<ThinkLevel | undefined>;
|
|
159
|
+
getReasoningLevel?(): Promise<ReasoningLevel | undefined>;
|
|
160
|
+
getVerboseLevel?(): Promise<VerboseLevel | undefined>;
|
|
161
|
+
/** Summarize older turns into a memory line; optional focus text for the summarizer. */
|
|
162
|
+
compactSession?(options?: {
|
|
163
|
+
instructions?: string;
|
|
164
|
+
force?: boolean;
|
|
165
|
+
}): Promise<CompactSessionResult | null>;
|
|
166
|
+
/** Side question: LLM answer without appending to the session transcript. */
|
|
167
|
+
btwQuery?(question: string): Promise<{
|
|
168
|
+
text: string;
|
|
169
|
+
error?: string;
|
|
170
|
+
}>;
|
|
171
|
+
/** Write session transcript to workspace `exports/` (markdown, html, or json). */
|
|
172
|
+
exportSessionToWorkspace?(format: 'markdown' | 'html' | 'json'): Promise<{
|
|
173
|
+
path: string;
|
|
174
|
+
}>;
|
|
175
|
+
/** Debug view: config + transcript stats (`list` | `detail` | `json`). */
|
|
176
|
+
agentContextReport?(mode?: 'list' | 'detail' | 'json'): Promise<string>;
|
|
145
177
|
/** Abort in-flight assistant generation and channel streaming for this session (e.g. /abort) */
|
|
146
178
|
abortCurrentTurn?(): Promise<void>;
|
|
147
179
|
}
|
|
@@ -5,7 +5,7 @@ import { listSessions } from "./sessions.js";
|
|
|
5
5
|
* Start interactive chat mode
|
|
6
6
|
*/
|
|
7
7
|
async function startInteractiveChat(agent, options) {
|
|
8
|
-
const {
|
|
8
|
+
const { sessionKey: initialSessionKey, continuingSession } = options;
|
|
9
9
|
let sessionKey = initialSessionKey;
|
|
10
10
|
if (continuingSession) console.log("🧠 Interactive chat mode - Continuing session\n");
|
|
11
11
|
else console.log("🧠 Interactive chat mode (Ctrl+C to exit)\n");
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"interactive.js","names":[],"sources":["../../../../../src/cli/commands/agent/interactive.ts"],"sourcesContent":["/**\n * Interactive chat mode for agent command\n */\n\nimport type { Interface as _Interface } from 'readline';\nimport type { AgentService } from '../../../agent/index.js';\nimport { getSessionManager } from '../../utils/session.js';\nimport { listSessions } from './sessions.js';\n\nexport interface InteractiveOptions {\n workspace: string;\n sessionKey: string;\n continuingSession: boolean;\n}\n\n/**\n * Start interactive chat mode\n */\nexport async function startInteractiveChat(\n agent: AgentService,\n options: InteractiveOptions\n): Promise<void> {\n const {
|
|
1
|
+
{"version":3,"file":"interactive.js","names":[],"sources":["../../../../../src/cli/commands/agent/interactive.ts"],"sourcesContent":["/**\n * Interactive chat mode for agent command\n */\n\nimport type { Interface as _Interface } from 'readline';\nimport type { AgentService } from '../../../agent/index.js';\nimport { getSessionManager } from '../../utils/session.js';\nimport { listSessions } from './sessions.js';\n\nexport interface InteractiveOptions {\n workspace: string;\n sessionKey: string;\n continuingSession: boolean;\n}\n\n/**\n * Start interactive chat mode\n */\nexport async function startInteractiveChat(\n agent: AgentService,\n options: InteractiveOptions\n): Promise<void> {\n const { sessionKey: initialSessionKey, continuingSession } = options;\n \n let sessionKey = initialSessionKey;\n\n if (continuingSession) {\n console.log('🧠 Interactive chat mode - Continuing session\\n');\n } else {\n console.log('🧠 Interactive chat mode (Ctrl+C to exit)\\n');\n }\n\n const readline = await import('readline');\n const rl = readline.createInterface({\n input: process.stdin,\n output: process.stdout,\n terminal: true,\n });\n\n rl.on('line', async (input) => {\n const trimmed = input.trim();\n \n // Handle special commands\n if (trimmed === ':sessions' || trimmed === ':list') {\n rl.pause();\n await listSessions();\n rl.resume();\n rl.prompt();\n return;\n }\n \n if (trimmed.startsWith(':session ')) {\n const newSessionKey = trimmed.slice(9).trim();\n const manager = await getSessionManager();\n const session = await manager.getSessionMetadata(newSessionKey);\n if (session) {\n sessionKey = newSessionKey;\n console.log(`🔄 Switched to session: ${sessionKey}\\n`);\n } else {\n console.log(`❌ Session not found: ${newSessionKey}\\n`);\n }\n rl.prompt();\n return;\n }\n\n if (trimmed === ':help') {\n printHelp();\n rl.prompt();\n return;\n }\n\n if (trimmed === ':quit' || trimmed === ':exit') {\n rl.close();\n return;\n }\n\n const response = await agent.processDirect(input, sessionKey);\n console.log('\\n🤖:', response);\n rl.prompt();\n });\n\n rl.on('close', async () => {\n console.log('\\n👋 Goodbye!');\n process.exit(0);\n });\n\n rl.setPrompt('You: ');\n rl.prompt();\n}\n\nfunction printHelp(): void {\n console.log(`\n📖 Available commands:\n :sessions, :list - List available sessions\n :session <key> - Switch to another session\n :quit, :exit - Exit interactive mode\n :help - Show this help\n`);\n}\n"],"mappings":";;;;;;AAkBA,eAAsB,qBACpB,OACA,SACe;CACf,MAAM,EAAE,YAAY,mBAAmB,sBAAsB;CAE7D,IAAI,aAAa;AAEjB,KAAI,kBACF,SAAQ,IAAI,kDAAkD;KAE9D,SAAQ,IAAI,8CAA8C;CAI5D,MAAM,MADW,MAAM,OAAO,aACV,gBAAgB;EAClC,OAAO,QAAQ;EACf,QAAQ,QAAQ;EAChB,UAAU;EACX,CAAC;AAEF,IAAG,GAAG,QAAQ,OAAO,UAAU;EAC7B,MAAM,UAAU,MAAM,MAAM;AAG5B,MAAI,YAAY,eAAe,YAAY,SAAS;AAClD,MAAG,OAAO;AACV,SAAM,cAAc;AACpB,MAAG,QAAQ;AACX,MAAG,QAAQ;AACX;;AAGF,MAAI,QAAQ,WAAW,YAAY,EAAE;GACnC,MAAM,gBAAgB,QAAQ,MAAM,EAAE,CAAC,MAAM;AAG7C,OADgB,OADA,MAAM,mBAAmB,EACX,mBAAmB,cAAc,EAClD;AACX,iBAAa;AACb,YAAQ,IAAI,2BAA2B,WAAW,IAAI;SAEtD,SAAQ,IAAI,wBAAwB,cAAc,IAAI;AAExD,MAAG,QAAQ;AACX;;AAGF,MAAI,YAAY,SAAS;AACvB,cAAW;AACX,MAAG,QAAQ;AACX;;AAGF,MAAI,YAAY,WAAW,YAAY,SAAS;AAC9C,MAAG,OAAO;AACV;;EAGF,MAAM,WAAW,MAAM,MAAM,cAAc,OAAO,WAAW;AAC7D,UAAQ,IAAI,SAAS,SAAS;AAC9B,KAAG,QAAQ;GACX;AAEF,IAAG,GAAG,SAAS,YAAY;AACzB,UAAQ,IAAI,gBAAgB;AAC5B,UAAQ,KAAK,EAAE;GACf;AAEF,IAAG,UAAU,QAAQ;AACrB,IAAG,QAAQ;;AAGb,SAAS,YAAkB;AACzB,SAAQ,IAAI;;;;;;EAMZ"}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
|
-
import { getWorkspacePath } from "../../config/schema.js";
|
|
2
1
|
import { createLogger } from "../../utils/logger/index.js";
|
|
3
2
|
import { init_logger } from "../../utils/logger.js";
|
|
3
|
+
import { getWorkspacePath } from "../../config/schema.js";
|
|
4
4
|
import { loadConfig } from "../../config/loader.js";
|
|
5
5
|
import { MessageBus, MessageBusShutdownError } from "../../infra/bus/queue.js";
|
|
6
6
|
import "../../infra/bus/index.js";
|
|
@@ -21,10 +21,11 @@ const log = createLogger("AgentCommand");
|
|
|
21
21
|
function createAgentCommand(_ctx) {
|
|
22
22
|
const cmd = new Command("agent").description("Chat with the AI agent").addHelpText("after", formatExamples([
|
|
23
23
|
"xopc agent -m \"Hello\" # Single message",
|
|
24
|
+
"xopc agent --model demo/demo-chat-7b -m \"Hi\" # Override model for one shot",
|
|
24
25
|
"xopc agent -i # Interactive chat mode",
|
|
25
26
|
"xopc agent -i --session telegram:dm:123456 # Continue existing session",
|
|
26
27
|
"xopc agent --list # List available sessions"
|
|
27
|
-
])).option("-m, --message <text>", "Single message to send").option("-i, --interactive", "Interactive chat mode").option("-s, --session <key>", "Continue an existing session (use --list to see available sessions)").option("-l, --list", "List available sessions and exit").action(async (options) => {
|
|
28
|
+
])).option("--model <id>", "Model ref for this run (e.g. demo/demo-chat-7b, anthropic/claude-sonnet-4-5)").option("-m, --message <text>", "Single message to send").option("-i, --interactive", "Interactive chat mode").option("-s, --session <key>", "Continue an existing session (use --list to see available sessions)").option("-l, --list", "List available sessions and exit").action(async (options) => {
|
|
28
29
|
const ctx = getContextWithOpts();
|
|
29
30
|
const config = loadConfig(ctx.configPath);
|
|
30
31
|
const workspace = getWorkspacePath(config) || ctx.workspacePath;
|
|
@@ -33,7 +34,8 @@ function createAgentCommand(_ctx) {
|
|
|
33
34
|
return;
|
|
34
35
|
}
|
|
35
36
|
const modelConfig = config.agents?.defaults?.model;
|
|
36
|
-
const
|
|
37
|
+
const modelFromConfig = typeof modelConfig === "string" ? modelConfig : modelConfig?.primary;
|
|
38
|
+
const modelId = options.model?.trim() || modelFromConfig;
|
|
37
39
|
const bus = new MessageBus();
|
|
38
40
|
if (ctx.isVerbose) log.info({
|
|
39
41
|
model: modelId,
|
|
@@ -107,15 +109,21 @@ function createAgentCommand(_ctx) {
|
|
|
107
109
|
process.on("SIGINT", shutdown);
|
|
108
110
|
process.on("SIGTERM", shutdown);
|
|
109
111
|
if (options.message) {
|
|
112
|
+
const oneShotModel = options.model?.trim();
|
|
113
|
+
if (oneShotModel) await agent.switchModelForSession(sessionKey, oneShotModel);
|
|
110
114
|
const response = await agent.processDirect(options.message, sessionKey);
|
|
111
115
|
console.log("\n🤖:", response);
|
|
116
|
+
if (oneShotModel) await agent.resetSessionModelToAgentDefault(sessionKey);
|
|
112
117
|
await shutdown();
|
|
113
|
-
} else if (options.interactive)
|
|
114
|
-
|
|
115
|
-
sessionKey,
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
118
|
+
} else if (options.interactive) {
|
|
119
|
+
const interactiveModel = options.model?.trim();
|
|
120
|
+
if (interactiveModel) await agent.switchModelForSession(sessionKey, interactiveModel);
|
|
121
|
+
await startInteractiveChat(agent, {
|
|
122
|
+
workspace,
|
|
123
|
+
sessionKey,
|
|
124
|
+
continuingSession: !!options.session
|
|
125
|
+
});
|
|
126
|
+
} else {
|
|
119
127
|
await shutdown();
|
|
120
128
|
cmd.help();
|
|
121
129
|
}
|
|
@@ -129,7 +137,11 @@ register({
|
|
|
129
137
|
factory: createAgentCommand,
|
|
130
138
|
metadata: {
|
|
131
139
|
category: "runtime",
|
|
132
|
-
examples: [
|
|
140
|
+
examples: [
|
|
141
|
+
"xopc agent -m \"Hello\"",
|
|
142
|
+
"xopc agent --model demo/demo-chat-7b -m \"Hello demo!\"",
|
|
143
|
+
"xopc agent -i"
|
|
144
|
+
]
|
|
133
145
|
}
|
|
134
146
|
});
|
|
135
147
|
//#endregion
|