whale-code 6.4.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +95 -0
- package/bin/swag-agent.js +9 -0
- package/bin/swagmanager-mcp.js +321 -0
- package/dist/cli/app.d.ts +26 -0
- package/dist/cli/app.js +64 -0
- package/dist/cli/chat/AgentSelector.d.ts +14 -0
- package/dist/cli/chat/AgentSelector.js +14 -0
- package/dist/cli/chat/ChatApp.d.ts +9 -0
- package/dist/cli/chat/ChatApp.js +267 -0
- package/dist/cli/chat/ChatInput.d.ts +39 -0
- package/dist/cli/chat/ChatInput.js +509 -0
- package/dist/cli/chat/MarkdownText.d.ts +10 -0
- package/dist/cli/chat/MarkdownText.js +20 -0
- package/dist/cli/chat/MessageList.d.ts +37 -0
- package/dist/cli/chat/MessageList.js +80 -0
- package/dist/cli/chat/ModelSelector.d.ts +20 -0
- package/dist/cli/chat/ModelSelector.js +73 -0
- package/dist/cli/chat/RewindViewer.d.ts +26 -0
- package/dist/cli/chat/RewindViewer.js +185 -0
- package/dist/cli/chat/StoreSelector.d.ts +14 -0
- package/dist/cli/chat/StoreSelector.js +24 -0
- package/dist/cli/chat/StreamingText.d.ts +12 -0
- package/dist/cli/chat/StreamingText.js +12 -0
- package/dist/cli/chat/SubagentPanel.d.ts +45 -0
- package/dist/cli/chat/SubagentPanel.js +110 -0
- package/dist/cli/chat/TeamPanel.d.ts +21 -0
- package/dist/cli/chat/TeamPanel.js +42 -0
- package/dist/cli/chat/ToolIndicator.d.ts +25 -0
- package/dist/cli/chat/ToolIndicator.js +436 -0
- package/dist/cli/chat/hooks/useAgentLoop.d.ts +39 -0
- package/dist/cli/chat/hooks/useAgentLoop.js +382 -0
- package/dist/cli/chat/hooks/useSlashCommands.d.ts +37 -0
- package/dist/cli/chat/hooks/useSlashCommands.js +387 -0
- package/dist/cli/commands/config-cmd.d.ts +10 -0
- package/dist/cli/commands/config-cmd.js +99 -0
- package/dist/cli/commands/doctor.d.ts +14 -0
- package/dist/cli/commands/doctor.js +172 -0
- package/dist/cli/commands/init.d.ts +16 -0
- package/dist/cli/commands/init.js +278 -0
- package/dist/cli/commands/mcp.d.ts +12 -0
- package/dist/cli/commands/mcp.js +162 -0
- package/dist/cli/login/LoginApp.d.ts +7 -0
- package/dist/cli/login/LoginApp.js +157 -0
- package/dist/cli/print-mode.d.ts +31 -0
- package/dist/cli/print-mode.js +202 -0
- package/dist/cli/serve-mode.d.ts +37 -0
- package/dist/cli/serve-mode.js +636 -0
- package/dist/cli/services/agent-definitions.d.ts +25 -0
- package/dist/cli/services/agent-definitions.js +91 -0
- package/dist/cli/services/agent-events.d.ts +178 -0
- package/dist/cli/services/agent-events.js +175 -0
- package/dist/cli/services/agent-loop.d.ts +90 -0
- package/dist/cli/services/agent-loop.js +762 -0
- package/dist/cli/services/agent-worker-base.d.ts +97 -0
- package/dist/cli/services/agent-worker-base.js +220 -0
- package/dist/cli/services/auth-service.d.ts +30 -0
- package/dist/cli/services/auth-service.js +160 -0
- package/dist/cli/services/background-processes.d.ts +126 -0
- package/dist/cli/services/background-processes.js +318 -0
- package/dist/cli/services/browser-auth.d.ts +24 -0
- package/dist/cli/services/browser-auth.js +180 -0
- package/dist/cli/services/claude-md-loader.d.ts +16 -0
- package/dist/cli/services/claude-md-loader.js +58 -0
- package/dist/cli/services/config-store.d.ts +47 -0
- package/dist/cli/services/config-store.js +79 -0
- package/dist/cli/services/debug-log.d.ts +10 -0
- package/dist/cli/services/debug-log.js +52 -0
- package/dist/cli/services/error-logger.d.ts +58 -0
- package/dist/cli/services/error-logger.js +269 -0
- package/dist/cli/services/file-history.d.ts +21 -0
- package/dist/cli/services/file-history.js +83 -0
- package/dist/cli/services/format-server-response.d.ts +16 -0
- package/dist/cli/services/format-server-response.js +440 -0
- package/dist/cli/services/git-context.d.ts +11 -0
- package/dist/cli/services/git-context.js +66 -0
- package/dist/cli/services/hooks.d.ts +85 -0
- package/dist/cli/services/hooks.js +258 -0
- package/dist/cli/services/interactive-tools.d.ts +125 -0
- package/dist/cli/services/interactive-tools.js +260 -0
- package/dist/cli/services/keybinding-manager.d.ts +52 -0
- package/dist/cli/services/keybinding-manager.js +115 -0
- package/dist/cli/services/local-tools.d.ts +22 -0
- package/dist/cli/services/local-tools.js +697 -0
- package/dist/cli/services/lsp-manager.d.ts +18 -0
- package/dist/cli/services/lsp-manager.js +717 -0
- package/dist/cli/services/mcp-client.d.ts +48 -0
- package/dist/cli/services/mcp-client.js +157 -0
- package/dist/cli/services/memory-manager.d.ts +16 -0
- package/dist/cli/services/memory-manager.js +57 -0
- package/dist/cli/services/model-manager.d.ts +18 -0
- package/dist/cli/services/model-manager.js +71 -0
- package/dist/cli/services/model-router.d.ts +26 -0
- package/dist/cli/services/model-router.js +149 -0
- package/dist/cli/services/permission-modes.d.ts +13 -0
- package/dist/cli/services/permission-modes.js +43 -0
- package/dist/cli/services/rewind.d.ts +84 -0
- package/dist/cli/services/rewind.js +194 -0
- package/dist/cli/services/ripgrep.d.ts +28 -0
- package/dist/cli/services/ripgrep.js +138 -0
- package/dist/cli/services/sandbox.d.ts +29 -0
- package/dist/cli/services/sandbox.js +97 -0
- package/dist/cli/services/server-tools.d.ts +61 -0
- package/dist/cli/services/server-tools.js +543 -0
- package/dist/cli/services/session-persistence.d.ts +23 -0
- package/dist/cli/services/session-persistence.js +99 -0
- package/dist/cli/services/subagent-worker.d.ts +19 -0
- package/dist/cli/services/subagent-worker.js +41 -0
- package/dist/cli/services/subagent.d.ts +47 -0
- package/dist/cli/services/subagent.js +647 -0
- package/dist/cli/services/system-prompt.d.ts +7 -0
- package/dist/cli/services/system-prompt.js +198 -0
- package/dist/cli/services/team-lead.d.ts +73 -0
- package/dist/cli/services/team-lead.js +512 -0
- package/dist/cli/services/team-state.d.ts +77 -0
- package/dist/cli/services/team-state.js +398 -0
- package/dist/cli/services/teammate.d.ts +31 -0
- package/dist/cli/services/teammate.js +689 -0
- package/dist/cli/services/telemetry.d.ts +61 -0
- package/dist/cli/services/telemetry.js +209 -0
- package/dist/cli/services/tools/agent-tools.d.ts +14 -0
- package/dist/cli/services/tools/agent-tools.js +347 -0
- package/dist/cli/services/tools/file-ops.d.ts +15 -0
- package/dist/cli/services/tools/file-ops.js +487 -0
- package/dist/cli/services/tools/search-tools.d.ts +8 -0
- package/dist/cli/services/tools/search-tools.js +186 -0
- package/dist/cli/services/tools/shell-exec.d.ts +10 -0
- package/dist/cli/services/tools/shell-exec.js +168 -0
- package/dist/cli/services/tools/task-manager.d.ts +28 -0
- package/dist/cli/services/tools/task-manager.js +209 -0
- package/dist/cli/services/tools/web-tools.d.ts +11 -0
- package/dist/cli/services/tools/web-tools.js +395 -0
- package/dist/cli/setup/SetupApp.d.ts +9 -0
- package/dist/cli/setup/SetupApp.js +191 -0
- package/dist/cli/shared/MatrixIntro.d.ts +4 -0
- package/dist/cli/shared/MatrixIntro.js +83 -0
- package/dist/cli/shared/Theme.d.ts +74 -0
- package/dist/cli/shared/Theme.js +127 -0
- package/dist/cli/shared/WhaleBanner.d.ts +10 -0
- package/dist/cli/shared/WhaleBanner.js +12 -0
- package/dist/cli/shared/markdown.d.ts +21 -0
- package/dist/cli/shared/markdown.js +756 -0
- package/dist/cli/status/StatusApp.d.ts +4 -0
- package/dist/cli/status/StatusApp.js +105 -0
- package/dist/cli/stores/StoreApp.d.ts +7 -0
- package/dist/cli/stores/StoreApp.js +81 -0
- package/dist/index.d.ts +15 -0
- package/dist/index.js +538 -0
- package/dist/local-agent/connection.d.ts +48 -0
- package/dist/local-agent/connection.js +332 -0
- package/dist/local-agent/discovery.d.ts +18 -0
- package/dist/local-agent/discovery.js +146 -0
- package/dist/local-agent/executor.d.ts +34 -0
- package/dist/local-agent/executor.js +241 -0
- package/dist/local-agent/index.d.ts +14 -0
- package/dist/local-agent/index.js +198 -0
- package/dist/node/adapters/base.d.ts +35 -0
- package/dist/node/adapters/base.js +10 -0
- package/dist/node/adapters/discord.d.ts +29 -0
- package/dist/node/adapters/discord.js +299 -0
- package/dist/node/adapters/email.d.ts +23 -0
- package/dist/node/adapters/email.js +218 -0
- package/dist/node/adapters/imessage.d.ts +17 -0
- package/dist/node/adapters/imessage.js +118 -0
- package/dist/node/adapters/slack.d.ts +26 -0
- package/dist/node/adapters/slack.js +259 -0
- package/dist/node/adapters/sms.d.ts +23 -0
- package/dist/node/adapters/sms.js +161 -0
- package/dist/node/adapters/telegram.d.ts +17 -0
- package/dist/node/adapters/telegram.js +101 -0
- package/dist/node/adapters/webchat.d.ts +27 -0
- package/dist/node/adapters/webchat.js +160 -0
- package/dist/node/adapters/whatsapp.d.ts +28 -0
- package/dist/node/adapters/whatsapp.js +230 -0
- package/dist/node/cli.d.ts +2 -0
- package/dist/node/cli.js +325 -0
- package/dist/node/config.d.ts +17 -0
- package/dist/node/config.js +31 -0
- package/dist/node/runtime.d.ts +50 -0
- package/dist/node/runtime.js +351 -0
- package/dist/server/handlers/__test-utils__/mock-supabase.d.ts +11 -0
- package/dist/server/handlers/__test-utils__/mock-supabase.js +393 -0
- package/dist/server/handlers/analytics.d.ts +17 -0
- package/dist/server/handlers/analytics.js +266 -0
- package/dist/server/handlers/api-keys.d.ts +6 -0
- package/dist/server/handlers/api-keys.js +221 -0
- package/dist/server/handlers/billing.d.ts +33 -0
- package/dist/server/handlers/billing.js +272 -0
- package/dist/server/handlers/browser.d.ts +10 -0
- package/dist/server/handlers/browser.js +517 -0
- package/dist/server/handlers/catalog.d.ts +99 -0
- package/dist/server/handlers/catalog.js +976 -0
- package/dist/server/handlers/comms.d.ts +254 -0
- package/dist/server/handlers/comms.js +588 -0
- package/dist/server/handlers/creations.d.ts +6 -0
- package/dist/server/handlers/creations.js +479 -0
- package/dist/server/handlers/crm.d.ts +89 -0
- package/dist/server/handlers/crm.js +538 -0
- package/dist/server/handlers/discovery.d.ts +6 -0
- package/dist/server/handlers/discovery.js +288 -0
- package/dist/server/handlers/embeddings.d.ts +92 -0
- package/dist/server/handlers/embeddings.js +197 -0
- package/dist/server/handlers/enrichment.d.ts +8 -0
- package/dist/server/handlers/enrichment.js +768 -0
- package/dist/server/handlers/image-gen.d.ts +6 -0
- package/dist/server/handlers/image-gen.js +409 -0
- package/dist/server/handlers/inventory.d.ts +319 -0
- package/dist/server/handlers/inventory.js +447 -0
- package/dist/server/handlers/kali.d.ts +10 -0
- package/dist/server/handlers/kali.js +210 -0
- package/dist/server/handlers/llm-providers.d.ts +6 -0
- package/dist/server/handlers/llm-providers.js +673 -0
- package/dist/server/handlers/local-agent.d.ts +6 -0
- package/dist/server/handlers/local-agent.js +118 -0
- package/dist/server/handlers/meta-ads.d.ts +111 -0
- package/dist/server/handlers/meta-ads.js +2279 -0
- package/dist/server/handlers/nodes.d.ts +33 -0
- package/dist/server/handlers/nodes.js +699 -0
- package/dist/server/handlers/operations.d.ts +138 -0
- package/dist/server/handlers/operations.js +131 -0
- package/dist/server/handlers/platform.d.ts +23 -0
- package/dist/server/handlers/platform.js +227 -0
- package/dist/server/handlers/supply-chain.d.ts +19 -0
- package/dist/server/handlers/supply-chain.js +327 -0
- package/dist/server/handlers/transcription.d.ts +17 -0
- package/dist/server/handlers/transcription.js +121 -0
- package/dist/server/handlers/video-gen.d.ts +6 -0
- package/dist/server/handlers/video-gen.js +466 -0
- package/dist/server/handlers/voice.d.ts +8 -0
- package/dist/server/handlers/voice.js +1146 -0
- package/dist/server/handlers/workflow-steps.d.ts +86 -0
- package/dist/server/handlers/workflow-steps.js +2349 -0
- package/dist/server/handlers/workflows.d.ts +7 -0
- package/dist/server/handlers/workflows.js +989 -0
- package/dist/server/index.d.ts +1 -0
- package/dist/server/index.js +2427 -0
- package/dist/server/lib/batch-client.d.ts +80 -0
- package/dist/server/lib/batch-client.js +467 -0
- package/dist/server/lib/code-worker-pool.d.ts +31 -0
- package/dist/server/lib/code-worker-pool.js +224 -0
- package/dist/server/lib/code-worker.d.ts +1 -0
- package/dist/server/lib/code-worker.js +188 -0
- package/dist/server/lib/compaction-service.d.ts +32 -0
- package/dist/server/lib/compaction-service.js +162 -0
- package/dist/server/lib/logger.d.ts +19 -0
- package/dist/server/lib/logger.js +46 -0
- package/dist/server/lib/otel.d.ts +38 -0
- package/dist/server/lib/otel.js +126 -0
- package/dist/server/lib/pg-rate-limiter.d.ts +21 -0
- package/dist/server/lib/pg-rate-limiter.js +86 -0
- package/dist/server/lib/prompt-sanitizer.d.ts +37 -0
- package/dist/server/lib/prompt-sanitizer.js +177 -0
- package/dist/server/lib/provider-capabilities.d.ts +85 -0
- package/dist/server/lib/provider-capabilities.js +190 -0
- package/dist/server/lib/provider-failover.d.ts +74 -0
- package/dist/server/lib/provider-failover.js +210 -0
- package/dist/server/lib/rate-limiter.d.ts +39 -0
- package/dist/server/lib/rate-limiter.js +147 -0
- package/dist/server/lib/server-agent-loop.d.ts +107 -0
- package/dist/server/lib/server-agent-loop.js +667 -0
- package/dist/server/lib/server-subagent.d.ts +78 -0
- package/dist/server/lib/server-subagent.js +203 -0
- package/dist/server/lib/session-checkpoint.d.ts +51 -0
- package/dist/server/lib/session-checkpoint.js +145 -0
- package/dist/server/lib/ssrf-guard.d.ts +13 -0
- package/dist/server/lib/ssrf-guard.js +240 -0
- package/dist/server/lib/supabase-client.d.ts +7 -0
- package/dist/server/lib/supabase-client.js +78 -0
- package/dist/server/lib/template-resolver.d.ts +31 -0
- package/dist/server/lib/template-resolver.js +215 -0
- package/dist/server/lib/utils.d.ts +16 -0
- package/dist/server/lib/utils.js +147 -0
- package/dist/server/local-agent-gateway.d.ts +82 -0
- package/dist/server/local-agent-gateway.js +426 -0
- package/dist/server/providers/anthropic.d.ts +20 -0
- package/dist/server/providers/anthropic.js +199 -0
- package/dist/server/providers/bedrock.d.ts +20 -0
- package/dist/server/providers/bedrock.js +194 -0
- package/dist/server/providers/gemini.d.ts +24 -0
- package/dist/server/providers/gemini.js +486 -0
- package/dist/server/providers/openai.d.ts +24 -0
- package/dist/server/providers/openai.js +522 -0
- package/dist/server/providers/registry.d.ts +32 -0
- package/dist/server/providers/registry.js +58 -0
- package/dist/server/providers/shared.d.ts +32 -0
- package/dist/server/providers/shared.js +124 -0
- package/dist/server/providers/types.d.ts +92 -0
- package/dist/server/providers/types.js +12 -0
- package/dist/server/proxy-handlers.d.ts +6 -0
- package/dist/server/proxy-handlers.js +89 -0
- package/dist/server/tool-router.d.ts +149 -0
- package/dist/server/tool-router.js +803 -0
- package/dist/server/validation.d.ts +24 -0
- package/dist/server/validation.js +301 -0
- package/dist/server/worker.d.ts +19 -0
- package/dist/server/worker.js +201 -0
- package/dist/setup.d.ts +8 -0
- package/dist/setup.js +181 -0
- package/dist/shared/agent-core.d.ts +157 -0
- package/dist/shared/agent-core.js +534 -0
- package/dist/shared/anthropic-types.d.ts +105 -0
- package/dist/shared/anthropic-types.js +7 -0
- package/dist/shared/api-client.d.ts +90 -0
- package/dist/shared/api-client.js +379 -0
- package/dist/shared/constants.d.ts +33 -0
- package/dist/shared/constants.js +80 -0
- package/dist/shared/sse-parser.d.ts +26 -0
- package/dist/shared/sse-parser.js +259 -0
- package/dist/shared/tool-dispatch.d.ts +52 -0
- package/dist/shared/tool-dispatch.js +191 -0
- package/dist/shared/types.d.ts +72 -0
- package/dist/shared/types.js +7 -0
- package/dist/updater.d.ts +25 -0
- package/dist/updater.js +140 -0
- package/dist/webchat/widget.d.ts +0 -0
- package/dist/webchat/widget.js +397 -0
- package/package.json +95 -0
- package/src/cli/services/builtin-skills/commit.md +19 -0
- package/src/cli/services/builtin-skills/review-pr.md +21 -0
- package/src/cli/services/builtin-skills/review.md +18 -0
|
@@ -0,0 +1,267 @@
|
|
|
1
|
+
import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
|
|
2
|
+
/**
|
|
3
|
+
* ChatApp — whale code CLI
|
|
4
|
+
*
|
|
5
|
+
* Uses Ink's <Static> for completed messages — written to stdout once,
|
|
6
|
+
* never re-rendered. Only the active area (streaming, tools, input)
|
|
7
|
+
* is managed by Ink's render loop. This prevents scroll bounce when
|
|
8
|
+
* content exceeds the terminal height.
|
|
9
|
+
*/
|
|
10
|
+
import { useState, useEffect, useRef, useMemo, useCallback } from "react";
|
|
11
|
+
import { Box, Text, Static, useApp, useInput } from "ink";
|
|
12
|
+
import Spinner from "ink-spinner";
|
|
13
|
+
import { canUseAgent, getServerToolCount, getModelShortName, setModel, getPermissionMode, mcpClientManager, } from "../services/agent-loop.js";
|
|
14
|
+
import { CompletedMessage } from "./MessageList.js";
|
|
15
|
+
import { ToolIndicator } from "./ToolIndicator.js";
|
|
16
|
+
import { SubagentPanel } from "./SubagentPanel.js";
|
|
17
|
+
import { TeamPanel } from "./TeamPanel.js";
|
|
18
|
+
import { StreamingText } from "./StreamingText.js";
|
|
19
|
+
import { ChatInput } from "./ChatInput.js";
|
|
20
|
+
import { StoreSelector } from "./StoreSelector.js";
|
|
21
|
+
import { ModelSelector } from "./ModelSelector.js";
|
|
22
|
+
import { RewindViewer, RewindOutcome } from "./RewindViewer.js";
|
|
23
|
+
import { RewindManager } from "../services/rewind.js";
|
|
24
|
+
import { colors, symbols } from "../shared/Theme.js";
|
|
25
|
+
import { loadKeybindings, matchesBinding } from "../services/keybinding-manager.js";
|
|
26
|
+
import { loadConfig, updateConfig } from "../services/config-store.js";
|
|
27
|
+
import { createRequire } from "module";
|
|
28
|
+
import { fileURLToPath } from "url";
|
|
29
|
+
import { dirname, join } from "path";
|
|
30
|
+
// Extracted hooks
|
|
31
|
+
import { useSlashCommands } from "./hooks/useSlashCommands.js";
|
|
32
|
+
import { useAgentLoop } from "./hooks/useAgentLoop.js";
|
|
33
|
+
// Thinking verbs — rotate randomly each render (Claude Code parity)
|
|
34
|
+
const THINKING_VERBS = [
|
|
35
|
+
"thinking…",
|
|
36
|
+
"reasoning…",
|
|
37
|
+
"considering…",
|
|
38
|
+
"analyzing…",
|
|
39
|
+
"evaluating…",
|
|
40
|
+
"pondering…",
|
|
41
|
+
"processing…",
|
|
42
|
+
"reflecting…",
|
|
43
|
+
"examining…",
|
|
44
|
+
"working…",
|
|
45
|
+
];
|
|
46
|
+
function randomVerb() {
|
|
47
|
+
return THINKING_VERBS[Math.floor(Math.random() * THINKING_VERBS.length)];
|
|
48
|
+
}
|
|
49
|
+
const PKG_NAME = "whale-code";
|
|
50
|
+
const __filename = fileURLToPath(import.meta.url);
|
|
51
|
+
const __dirname = dirname(__filename);
|
|
52
|
+
const PKG_VERSION = createRequire(import.meta.url)(join(__dirname, "..", "..", "..", "package.json")).version;
|
|
53
|
+
// ── Component ──
|
|
54
|
+
export function ChatApp() {
|
|
55
|
+
const { exit } = useApp();
|
|
56
|
+
// Core state
|
|
57
|
+
const [ready, setReady] = useState(false);
|
|
58
|
+
const [error, setError] = useState("");
|
|
59
|
+
const [messages, setMessages] = useState([]);
|
|
60
|
+
const [streamingText, setStreamingText] = useState("");
|
|
61
|
+
const [isStreaming, setIsStreaming] = useState(false);
|
|
62
|
+
const [activeTools, setActiveTools] = useState([]);
|
|
63
|
+
const [subagentActivity, setSubagentActivity] = useState(new Map());
|
|
64
|
+
const [completedSubagents, setCompletedSubagents] = useState([]);
|
|
65
|
+
const [teamState, setTeamState] = useState(null);
|
|
66
|
+
// UI state
|
|
67
|
+
const [userLabel, setUserLabel] = useState("");
|
|
68
|
+
const [toolsExpanded, setToolsExpanded] = useState(false);
|
|
69
|
+
const [serverToolsAvailable, setServerToolsAvailable] = useState(0);
|
|
70
|
+
const [storeSelectMode, setStoreSelectMode] = useState(false);
|
|
71
|
+
const [storeList, setStoreList] = useState([]);
|
|
72
|
+
const [modelSelectMode, setModelSelectMode] = useState(false);
|
|
73
|
+
const [currentModel, setCurrentModel] = useState(getModelShortName());
|
|
74
|
+
const [sessionId, setSessionId] = useState(null);
|
|
75
|
+
const [thinkingEnabled, setThinkingEnabled] = useState(() => {
|
|
76
|
+
try {
|
|
77
|
+
return loadConfig().thinking_enabled ?? true;
|
|
78
|
+
}
|
|
79
|
+
catch {
|
|
80
|
+
return true;
|
|
81
|
+
}
|
|
82
|
+
});
|
|
83
|
+
// Stream activity indicator — show spinner when no new content for 800ms+
|
|
84
|
+
const [streamStale, setStreamStale] = useState(false);
|
|
85
|
+
const lastContentUpdateRef = useRef(Date.now());
|
|
86
|
+
// Poll-based staleness check — avoids re-renders on every content change
|
|
87
|
+
useEffect(() => {
|
|
88
|
+
if (!isStreaming) {
|
|
89
|
+
setStreamStale(false);
|
|
90
|
+
return;
|
|
91
|
+
}
|
|
92
|
+
lastContentUpdateRef.current = Date.now();
|
|
93
|
+
const interval = setInterval(() => {
|
|
94
|
+
setStreamStale(Date.now() - lastContentUpdateRef.current > 800);
|
|
95
|
+
}, 400);
|
|
96
|
+
return () => clearInterval(interval);
|
|
97
|
+
}, [isStreaming]);
|
|
98
|
+
// Rewind state
|
|
99
|
+
const [showRewind, setShowRewind] = useState(false);
|
|
100
|
+
const rewindManagerRef = useRef(new RewindManager());
|
|
101
|
+
const turnIndexRef = useRef(0);
|
|
102
|
+
// Refs
|
|
103
|
+
const conversationRef = useRef([]);
|
|
104
|
+
const abortRef = useRef(null);
|
|
105
|
+
const accTextRef = useRef("");
|
|
106
|
+
const thinkingVerbRef = useRef(randomVerb());
|
|
107
|
+
// No timer cleanup needed — unified flush timer is managed inside useAgentLoop,
|
|
108
|
+
// stale polling interval is cleaned up by its own effect.
|
|
109
|
+
// ── Init ──
|
|
110
|
+
useEffect(() => {
|
|
111
|
+
const check = canUseAgent();
|
|
112
|
+
if (!check.ready) {
|
|
113
|
+
setError(check.reason || "Run 'whale login' to authenticate.");
|
|
114
|
+
}
|
|
115
|
+
else {
|
|
116
|
+
const config = loadConfig();
|
|
117
|
+
if (config.email)
|
|
118
|
+
setUserLabel(config.store_name || config.email);
|
|
119
|
+
setReady(true);
|
|
120
|
+
getServerToolCount().then((count) => setServerToolsAvailable(count));
|
|
121
|
+
mcpClientManager.connectAll().catch(() => { });
|
|
122
|
+
}
|
|
123
|
+
return () => {
|
|
124
|
+
mcpClientManager.disconnectAll().catch(() => { });
|
|
125
|
+
};
|
|
126
|
+
}, []);
|
|
127
|
+
// ── Keys (configurable via ~/.swagmanager/keybindings.json) ──
|
|
128
|
+
const kb = useMemo(() => loadKeybindings(), []);
|
|
129
|
+
useInput((input, key) => {
|
|
130
|
+
if (matchesBinding(kb.exit, input, key)) {
|
|
131
|
+
if (abortRef.current)
|
|
132
|
+
abortRef.current.abort();
|
|
133
|
+
exit();
|
|
134
|
+
}
|
|
135
|
+
if (matchesBinding(kb.cancel_stream, input, key) && isStreaming && abortRef.current) {
|
|
136
|
+
abortRef.current.abort();
|
|
137
|
+
}
|
|
138
|
+
if (matchesBinding(kb.toggle_expand, input, key)) {
|
|
139
|
+
setToolsExpanded((prev) => !prev);
|
|
140
|
+
}
|
|
141
|
+
if (matchesBinding(kb.toggle_thinking, input, key)) {
|
|
142
|
+
setThinkingEnabled((prev) => {
|
|
143
|
+
const next = !prev;
|
|
144
|
+
try {
|
|
145
|
+
updateConfig({ thinking_enabled: next });
|
|
146
|
+
}
|
|
147
|
+
catch { /* best effort */ }
|
|
148
|
+
setMessages((msgs) => [...msgs, {
|
|
149
|
+
role: "assistant",
|
|
150
|
+
text: ` Thinking: ${next ? "on" : "off"}`,
|
|
151
|
+
}]);
|
|
152
|
+
return next;
|
|
153
|
+
});
|
|
154
|
+
}
|
|
155
|
+
});
|
|
156
|
+
// ── Extracted hooks ──
|
|
157
|
+
const { handleCommand, handleStoreSelect, handleStoreCancel } = useSlashCommands({
|
|
158
|
+
exit, toolsExpanded, serverToolsAvailable, sessionId, thinkingEnabled,
|
|
159
|
+
conversationRef,
|
|
160
|
+
setMessages, setStreamingText, setActiveTools, setTeamState,
|
|
161
|
+
setStoreList, setStoreSelectMode, setModelSelectMode, setCurrentModel,
|
|
162
|
+
setSessionId, setThinkingEnabled, setUserLabel, setServerToolsAvailable,
|
|
163
|
+
setShowRewind,
|
|
164
|
+
rewindCheckpointCount: rewindManagerRef.current.getCheckpointCount(),
|
|
165
|
+
PKG_NAME, PKG_VERSION,
|
|
166
|
+
});
|
|
167
|
+
const { handleSend } = useAgentLoop({
|
|
168
|
+
isStreaming, thinkingEnabled, conversationRef, abortRef,
|
|
169
|
+
accTextRef, lastContentUpdateRef, thinkingVerbRef,
|
|
170
|
+
rewindManagerRef, turnIndexRef,
|
|
171
|
+
setMessages, setStreamingText, setIsStreaming, setActiveTools,
|
|
172
|
+
setSubagentActivity, setCompletedSubagents, setTeamState,
|
|
173
|
+
});
|
|
174
|
+
// ── Rewind handler ──
|
|
175
|
+
const handleRewind = useCallback((checkpointIndex, outcome) => {
|
|
176
|
+
const rm = rewindManagerRef.current;
|
|
177
|
+
if (outcome === RewindOutcome.Cancel) {
|
|
178
|
+
setShowRewind(false);
|
|
179
|
+
return;
|
|
180
|
+
}
|
|
181
|
+
if (outcome === RewindOutcome.RevertOnly) {
|
|
182
|
+
// Revert files only, keep conversation
|
|
183
|
+
const result = rm.revertFilesFrom(checkpointIndex);
|
|
184
|
+
setShowRewind(false);
|
|
185
|
+
const total = result.filesReverted.length + result.filesDeleted.length;
|
|
186
|
+
const parts = [];
|
|
187
|
+
if (result.filesReverted.length > 0)
|
|
188
|
+
parts.push(`${result.filesReverted.length} reverted`);
|
|
189
|
+
if (result.filesDeleted.length > 0)
|
|
190
|
+
parts.push(`${result.filesDeleted.length} deleted`);
|
|
191
|
+
if (result.errors.length > 0)
|
|
192
|
+
parts.push(`${result.errors.length} errors`);
|
|
193
|
+
setMessages(prev => [...prev, {
|
|
194
|
+
role: "assistant",
|
|
195
|
+
text: total > 0
|
|
196
|
+
? ` ${symbols.check} File changes reverted (${parts.join(", ")})`
|
|
197
|
+
: ` ${symbols.check} No file changes to revert.`,
|
|
198
|
+
}]);
|
|
199
|
+
return;
|
|
200
|
+
}
|
|
201
|
+
// RewindOnly or RewindAndRevert
|
|
202
|
+
const shouldRevertFiles = outcome === RewindOutcome.RewindAndRevert;
|
|
203
|
+
const result = rm.rewindTo(checkpointIndex);
|
|
204
|
+
// Truncate messages
|
|
205
|
+
setMessages(prev => prev.slice(0, result.messageCount));
|
|
206
|
+
// Truncate conversation history to match
|
|
207
|
+
// We need to figure out how many Anthropic messages correspond to the checkpoint.
|
|
208
|
+
// The checkpoint stores messageCount which is the UI messages count at that point.
|
|
209
|
+
// Rebuild conversation from remaining messages by keeping only the conversation
|
|
210
|
+
// entries up to the corresponding turn.
|
|
211
|
+
const targetTurnIndex = rm.getCheckpoints()[checkpointIndex]?.turnIndex ?? 0;
|
|
212
|
+
// Each user/assistant exchange is roughly 2 conversation entries (user + assistant).
|
|
213
|
+
// But the actual conversation is maintained independently.
|
|
214
|
+
// We slice it to 2*(turnIndex+1) entries (user+assistant pairs)
|
|
215
|
+
const conversationSlicePoint = Math.min((targetTurnIndex + 1) * 2, conversationRef.current.length);
|
|
216
|
+
conversationRef.current = conversationRef.current.slice(0, conversationSlicePoint);
|
|
217
|
+
// Reset turn index to match
|
|
218
|
+
turnIndexRef.current = targetTurnIndex + 1;
|
|
219
|
+
setShowRewind(false);
|
|
220
|
+
const parts = [];
|
|
221
|
+
if (shouldRevertFiles) {
|
|
222
|
+
if (result.filesReverted.length > 0)
|
|
223
|
+
parts.push(`${result.filesReverted.length} files reverted`);
|
|
224
|
+
if (result.filesDeleted.length > 0)
|
|
225
|
+
parts.push(`${result.filesDeleted.length} files deleted`);
|
|
226
|
+
}
|
|
227
|
+
if (result.errors.length > 0)
|
|
228
|
+
parts.push(`${result.errors.length} revert errors`);
|
|
229
|
+
const detail = parts.length > 0 ? ` (${parts.join(", ")})` : "";
|
|
230
|
+
setMessages(prev => [...prev, {
|
|
231
|
+
role: "assistant",
|
|
232
|
+
text: ` ${symbols.check} Rewound to turn ${targetTurnIndex + 1}${detail}`,
|
|
233
|
+
}]);
|
|
234
|
+
}, []);
|
|
235
|
+
// ── Render ──
|
|
236
|
+
const termWidth = process.stdout.columns || 80;
|
|
237
|
+
const contentWidth = Math.max(20, termWidth - 2);
|
|
238
|
+
const { staticItems, dynamicMessages } = useMemo(() => {
|
|
239
|
+
const items = [{ id: "header", type: "header" }];
|
|
240
|
+
const cutoff = Math.max(0, messages.length - 1);
|
|
241
|
+
for (let i = 0; i < cutoff; i++) {
|
|
242
|
+
items.push({ id: `msg-${i}`, type: "message", msg: messages[i], index: i });
|
|
243
|
+
}
|
|
244
|
+
const tail = messages.length > 0 ? [messages[messages.length - 1]] : [];
|
|
245
|
+
return { staticItems: items, dynamicMessages: tail };
|
|
246
|
+
}, [messages]);
|
|
247
|
+
if (error) {
|
|
248
|
+
return (_jsxs(Box, { flexDirection: "column", paddingLeft: 1, paddingRight: 1, children: [_jsx(Text, { children: " " }), _jsx(Text, { color: colors.brand, bold: true, children: "\u25C6 whale code" }), _jsx(Text, { children: " " }), _jsx(Text, { color: colors.error, children: error })] }));
|
|
249
|
+
}
|
|
250
|
+
if (!ready) {
|
|
251
|
+
return (_jsx(Box, { flexDirection: "column", paddingLeft: 1, paddingRight: 1, children: _jsx(Text, { color: colors.tertiary, children: "loading..." }) }));
|
|
252
|
+
}
|
|
253
|
+
return (_jsxs(Box, { flexDirection: "column", paddingLeft: 1, paddingRight: 1, children: [_jsx(Static, { items: staticItems, children: (item) => {
|
|
254
|
+
if (item.type === "header") {
|
|
255
|
+
return (_jsxs(Box, { flexDirection: "column", children: [_jsx(Text, { children: " " }), _jsxs(Text, { children: [_jsx(Text, { color: colors.brand, bold: true, children: "\u25C6 whale code" }), userLabel ? _jsxs(Text, { color: colors.dim, children: [" ", userLabel] }) : null, _jsxs(Text, { color: colors.dim, children: [" ", currentModel] }), thinkingEnabled ? _jsx(Text, { color: colors.warning, children: " thinking" }) : null, getPermissionMode() !== "default" && (_jsxs(Text, { color: getPermissionMode() === "yolo" ? colors.error : colors.info, children: [" ", getPermissionMode()] })), serverToolsAvailable > 0 ? (_jsxs(Text, { color: colors.tertiary, children: [" ", symbols.dot, " ", serverToolsAvailable, " server tools"] })) : null] }), _jsx(Text, { color: colors.separator, children: "─".repeat(contentWidth) })] }, item.id));
|
|
256
|
+
}
|
|
257
|
+
return _jsx(CompletedMessage, { msg: item.msg, index: item.index, toolsExpanded: toolsExpanded }, item.id);
|
|
258
|
+
} }), dynamicMessages.map((msg) => (_jsx(CompletedMessage, { msg: msg, index: messages.length - 1, toolsExpanded: toolsExpanded }, `dynamic-${messages.length - 1}`))), teamState ? (_jsx(TeamPanel, { team: teamState })) : (_jsxs(_Fragment, { children: [isStreaming && !streamingText && activeTools.length === 0 && (_jsxs(Box, { marginLeft: 2, marginY: 1, children: [_jsx(Text, { color: colors.brand, children: _jsx(Spinner, { type: "dots" }) }), _jsxs(Text, { color: colors.dim, children: [" ", thinkingVerbRef.current] })] })), activeTools.length > 0 && (_jsx(Box, { flexDirection: "column", marginLeft: 2, children: activeTools.map((tc, i) => (_jsxs(Box, { flexDirection: "column", children: [_jsx(ToolIndicator, { id: `live-${tc.name}-${i}`, name: tc.name, status: tc.status, input: tc.input, expanded: toolsExpanded }), tc.name === "task" && tc.status === "running" && (subagentActivity.size > 0 || completedSubagents.length > 0) && (_jsx(SubagentPanel, { running: subagentActivity, completed: completedSubagents }))] }, `live-${tc.name}-${i}`))) })), streamingText && (_jsxs(Box, { marginLeft: 2, flexDirection: "column", children: [_jsx(StreamingText, { text: streamingText }), streamStale && (_jsx(Box, { marginTop: 0, children: _jsx(Text, { color: colors.brand, children: _jsx(Spinner, { type: "dots" }) }) }))] }))] })), showRewind ? (_jsx(RewindViewer, { checkpoints: rewindManagerRef.current.getCheckpoints(), onRewind: handleRewind, onCancel: () => setShowRewind(false) })) : storeSelectMode ? (_jsx(StoreSelector, { stores: storeList, currentStoreId: loadConfig().store_id || "", onSelect: handleStoreSelect, onCancel: handleStoreCancel })) : modelSelectMode ? (_jsx(ModelSelector, { currentModel: currentModel, onSelect: (model) => {
|
|
259
|
+
setModelSelectMode(false);
|
|
260
|
+
setModel(model.value);
|
|
261
|
+
setCurrentModel(model.value);
|
|
262
|
+
setMessages((prev) => [...prev, {
|
|
263
|
+
role: "assistant",
|
|
264
|
+
text: ` ${symbols.check} Model: ${model.label} (${model.modelId})`,
|
|
265
|
+
}]);
|
|
266
|
+
}, onCancel: () => setModelSelectMode(false) })) : (_jsx(ChatInput, { onSubmit: handleSend, onCommand: handleCommand, disabled: isStreaming, agentName: "whale code" }))] }));
|
|
267
|
+
}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* ChatInput — Custom input with bracketed paste + image attachments
|
|
3
|
+
*
|
|
4
|
+
* Replaces ink-text-input with raw stdin handling to fix:
|
|
5
|
+
* - Paste losing content on enter (newlines in paste triggered submit)
|
|
6
|
+
* - Multi-line paste mangled/unformatted
|
|
7
|
+
* - No drag-drop image support
|
|
8
|
+
*
|
|
9
|
+
* Features:
|
|
10
|
+
* - Bracketed paste mode — clean multi-line paste
|
|
11
|
+
* - Drag-drop images — detects image file paths, attaches as chips
|
|
12
|
+
* - Image chips above input like Claude Code: [image1.png] [image2.jpg]
|
|
13
|
+
* - Backspace on empty input removes last image
|
|
14
|
+
* - Slash command menu preserved
|
|
15
|
+
* - Multi-line input with ⎸ continuation markers
|
|
16
|
+
*/
|
|
17
|
+
export interface ImageAttachment {
|
|
18
|
+
path: string;
|
|
19
|
+
name: string;
|
|
20
|
+
base64: string;
|
|
21
|
+
mediaType: string;
|
|
22
|
+
}
|
|
23
|
+
export interface FileAttachment {
|
|
24
|
+
path: string;
|
|
25
|
+
name: string;
|
|
26
|
+
}
|
|
27
|
+
export interface SlashCommand {
|
|
28
|
+
name: string;
|
|
29
|
+
description: string;
|
|
30
|
+
}
|
|
31
|
+
export declare const SLASH_COMMANDS: SlashCommand[];
|
|
32
|
+
interface ChatInputProps {
|
|
33
|
+
onSubmit: (message: string, images?: ImageAttachment[]) => void;
|
|
34
|
+
onCommand: (command: string) => void;
|
|
35
|
+
disabled: boolean;
|
|
36
|
+
agentName?: string;
|
|
37
|
+
}
|
|
38
|
+
export declare function ChatInput({ onSubmit, onCommand, disabled, agentName }: ChatInputProps): import("react/jsx-runtime").JSX.Element;
|
|
39
|
+
export {};
|