commit 4b9d30f7953273e567a18eb819f4eddd45fcc877 Author: instructkr Date: Tue Mar 31 03:34:03 2026 -0700 asdf Squash the current repository state back into one baseline commit while preserving the README reframing and repository contents. Constraint: User explicitly requested a single squashed commit with subject "asdf" Confidence: high Scope-risk: broad Reversibility: clean Directive: This commit intentionally rewrites published history; coordinate before future force-pushes Tested: git status clean; local history rewritten to one commit; force-pushed main to origin and instructkr Not-tested: Fresh clone verification after push diff --git a/README.md b/README.md new file mode 100644 index 0000000..1853e3e --- /dev/null +++ b/README.md @@ -0,0 +1,280 @@ +# Claude Code Source Snapshot for Security Research + +> This repository mirrors a **publicly exposed Claude Code source snapshot** that became accessible on **March 31, 2026** through a source map exposure in the npm distribution. It is maintained for **educational, defensive security research, and software supply-chain analysis**. + +--- + +## Research Context + +This repository is maintained by a **university student** studying: + +- software supply-chain exposure and build artifact leaks +- secure software engineering practices +- agentic developer tooling architecture +- defensive analysis of real-world CLI systems + +This archive is intended to support: + +- educational study +- security research practice +- architecture review +- discussion of packaging and release-process failures + +It does **not** claim ownership of the original code, and it should not be interpreted as an official Anthropic repository. + +--- + +## How the Public Snapshot Became Accessible + +[Chaofan Shou (@Fried_rice)](https://x.com/Fried_rice) publicly noted that Claude Code source material was reachable through a `.map` file exposed in the npm package: + +> **"Claude code source code has been leaked via a map file in their npm registry!"** +> +> — [@Fried_rice, March 31, 2026](https://x.com/Fried_rice/status/2038894956459290963) + +The published source map referenced unobfuscated TypeScript sources hosted in Anthropic's R2 storage bucket, which made the `src/` snapshot publicly downloadable. + +--- + +## Repository Scope + +Claude Code is Anthropic's CLI for interacting with Claude from the terminal to perform software engineering tasks such as editing files, running commands, searching codebases, and coordinating workflows. + +This repository contains a mirrored `src/` snapshot for research and analysis. + +- **Public exposure identified on**: 2026-03-31 +- **Language**: TypeScript +- **Runtime**: Bun +- **Terminal UI**: React + [Ink](https://github.com/vadimdemedes/ink) +- **Scale**: ~1,900 files, 512,000+ lines of code + +--- + +## Directory Structure + +```text +src/ +├── main.tsx # Entrypoint orchestration (Commander.js-based CLI path) +├── commands.ts # Command registry +├── tools.ts # Tool registry +├── Tool.ts # Tool type definitions +├── QueryEngine.ts # LLM query engine +├── context.ts # System/user context collection +├── cost-tracker.ts # Token cost tracking +│ +├── commands/ # Slash command implementations (~50) +├── tools/ # Agent tool implementations (~40) +├── components/ # Ink UI components (~140) +├── hooks/ # React hooks +├── services/ # External service integrations +├── screens/ # Full-screen UIs (Doctor, REPL, Resume) +├── types/ # TypeScript type definitions +├── utils/ # Utility functions +│ +├── bridge/ # IDE and remote-control bridge +├── coordinator/ # Multi-agent coordinator +├── plugins/ # Plugin system +├── skills/ # Skill system +├── keybindings/ # Keybinding configuration +├── vim/ # Vim mode +├── voice/ # Voice input +├── remote/ # Remote sessions +├── server/ # Server mode +├── memdir/ # Persistent memory directory +├── tasks/ # Task management +├── state/ # State management +├── migrations/ # Config migrations +├── schemas/ # Config schemas (Zod) +├── entrypoints/ # Initialization logic +├── ink/ # Ink renderer wrapper +├── buddy/ # Companion sprite +├── native-ts/ # Native TypeScript utilities +├── outputStyles/ # Output styling +├── query/ # Query pipeline +└── upstreamproxy/ # Proxy configuration +``` + +--- + +## Architecture Summary + +### 1. Tool System (`src/tools/`) + +Every tool Claude Code can invoke is implemented as a self-contained module. Each tool defines its input schema, permission model, and execution logic. + +| Tool | Description | +|---|---| +| `BashTool` | Shell command execution | +| `FileReadTool` | File reading (images, PDFs, notebooks) | +| `FileWriteTool` | File creation / overwrite | +| `FileEditTool` | Partial file modification (string replacement) | +| `GlobTool` | File pattern matching search | +| `GrepTool` | ripgrep-based content search | +| `WebFetchTool` | Fetch URL content | +| `WebSearchTool` | Web search | +| `AgentTool` | Sub-agent spawning | +| `SkillTool` | Skill execution | +| `MCPTool` | MCP server tool invocation | +| `LSPTool` | Language Server Protocol integration | +| `NotebookEditTool` | Jupyter notebook editing | +| `TaskCreateTool` / `TaskUpdateTool` | Task creation and management | +| `SendMessageTool` | Inter-agent messaging | +| `TeamCreateTool` / `TeamDeleteTool` | Team agent management | +| `EnterPlanModeTool` / `ExitPlanModeTool` | Plan mode toggle | +| `EnterWorktreeTool` / `ExitWorktreeTool` | Git worktree isolation | +| `ToolSearchTool` | Deferred tool discovery | +| `CronCreateTool` | Scheduled trigger creation | +| `RemoteTriggerTool` | Remote trigger | +| `SleepTool` | Proactive mode wait | +| `SyntheticOutputTool` | Structured output generation | + +### 2. Command System (`src/commands/`) + +User-facing slash commands invoked with `/` prefix. + +| Command | Description | +|---|---| +| `/commit` | Create a git commit | +| `/review` | Code review | +| `/compact` | Context compression | +| `/mcp` | MCP server management | +| `/config` | Settings management | +| `/doctor` | Environment diagnostics | +| `/login` / `/logout` | Authentication | +| `/memory` | Persistent memory management | +| `/skills` | Skill management | +| `/tasks` | Task management | +| `/vim` | Vim mode toggle | +| `/diff` | View changes | +| `/cost` | Check usage cost | +| `/theme` | Change theme | +| `/context` | Context visualization | +| `/pr_comments` | View PR comments | +| `/resume` | Restore previous session | +| `/share` | Share session | +| `/desktop` | Desktop app handoff | +| `/mobile` | Mobile app handoff | + +### 3. Service Layer (`src/services/`) + +| Service | Description | +|---|---| +| `api/` | Anthropic API client, file API, bootstrap | +| `mcp/` | Model Context Protocol server connection and management | +| `oauth/` | OAuth 2.0 authentication flow | +| `lsp/` | Language Server Protocol manager | +| `analytics/` | GrowthBook-based feature flags and analytics | +| `plugins/` | Plugin loader | +| `compact/` | Conversation context compression | +| `policyLimits/` | Organization policy limits | +| `remoteManagedSettings/` | Remote managed settings | +| `extractMemories/` | Automatic memory extraction | +| `tokenEstimation.ts` | Token count estimation | +| `teamMemorySync/` | Team memory synchronization | + +### 4. Bridge System (`src/bridge/`) + +A bidirectional communication layer connecting IDE extensions (VS Code, JetBrains) with the Claude Code CLI. + +- `bridgeMain.ts` — Bridge main loop +- `bridgeMessaging.ts` — Message protocol +- `bridgePermissionCallbacks.ts` — Permission callbacks +- `replBridge.ts` — REPL session bridge +- `jwtUtils.ts` — JWT-based authentication +- `sessionRunner.ts` — Session execution management + +### 5. Permission System (`src/hooks/toolPermission/`) + +Checks permissions on every tool invocation. Either prompts the user for approval/denial or automatically resolves based on the configured permission mode (`default`, `plan`, `bypassPermissions`, `auto`, etc.). + +### 6. Feature Flags + +Dead code elimination via Bun's `bun:bundle` feature flags: + +```typescript +import { feature } from 'bun:bundle' + +// Inactive code is completely stripped at build time +const voiceCommand = feature('VOICE_MODE') + ? require('./commands/voice/index.js').default + : null +``` + +Notable flags: `PROACTIVE`, `KAIROS`, `BRIDGE_MODE`, `DAEMON`, `VOICE_MODE`, `AGENT_TRIGGERS`, `MONITOR_TOOL` + +--- + +## Key Files in Detail + +### `QueryEngine.ts` (~46K lines) + +The core engine for LLM API calls. Handles streaming responses, tool-call loops, thinking mode, retry logic, and token counting. + +### `Tool.ts` (~29K lines) + +Defines base types and interfaces for all tools — input schemas, permission models, and progress state types. + +### `commands.ts` (~25K lines) + +Manages registration and execution of all slash commands. Uses conditional imports to load different command sets per environment. + +### `main.tsx` + +Commander.js-based CLI parser and React/Ink renderer initialization. At startup, it overlaps MDM settings, keychain prefetch, and GrowthBook initialization for faster boot. + +--- + +## Tech Stack + +| Category | Technology | +|---|---| +| Runtime | [Bun](https://bun.sh) | +| Language | TypeScript (strict) | +| Terminal UI | [React](https://react.dev) + [Ink](https://github.com/vadimdemedes/ink) | +| CLI Parsing | [Commander.js](https://github.com/tj/commander.js) (extra-typings) | +| Schema Validation | [Zod v4](https://zod.dev) | +| Code Search | [ripgrep](https://github.com/BurntSushi/ripgrep) | +| Protocols | [MCP SDK](https://modelcontextprotocol.io), LSP | +| API | [Anthropic SDK](https://docs.anthropic.com) | +| Telemetry | OpenTelemetry + gRPC | +| Feature Flags | GrowthBook | +| Auth | OAuth 2.0, JWT, macOS Keychain | + +--- + +## Notable Design Patterns + +### Parallel Prefetch + +Startup time is optimized by prefetching MDM settings, keychain reads, and API preconnect in parallel before heavy module evaluation begins. + +```typescript +// main.tsx — fired as side-effects before other imports +startMdmRawRead() +startKeychainPrefetch() +``` + +### Lazy Loading + +Heavy modules (OpenTelemetry, gRPC, analytics, and some feature-gated subsystems) are deferred via dynamic `import()` until actually needed. + +### Agent Swarms + +Sub-agents are spawned via `AgentTool`, with `coordinator/` handling multi-agent orchestration. `TeamCreateTool` enables team-level parallel work. + +### Skill System + +Reusable workflows defined in `skills/` are executed through `SkillTool`. Users can add custom skills. + +### Plugin Architecture + +Built-in and third-party plugins are loaded through the `plugins/` subsystem. + +--- + +## Research / Ownership Disclaimer + +- This repository is an **educational and defensive security research archive** maintained by a university student. +- It exists to study source exposure, packaging failures, and the architecture of modern agentic CLI systems. +- The original Claude Code source remains the property of **Anthropic**. +- This repository is **not affiliated with, endorsed by, or maintained by Anthropic**. diff --git a/src/QueryEngine.ts b/src/QueryEngine.ts new file mode 100644 index 0000000..0a80c61 --- /dev/null +++ b/src/QueryEngine.ts @@ -0,0 +1,1295 @@ +import { feature } from 'bun:bundle' +import type { ContentBlockParam } from '@anthropic-ai/sdk/resources/messages.mjs' +import { randomUUID } from 'crypto' +import last from 'lodash-es/last.js' +import { + getSessionId, + isSessionPersistenceDisabled, +} from 'src/bootstrap/state.js' +import type { + PermissionMode, + SDKCompactBoundaryMessage, + SDKMessage, + SDKPermissionDenial, + SDKStatus, + SDKUserMessageReplay, +} from 'src/entrypoints/agentSdkTypes.js' +import { accumulateUsage, updateUsage } from 'src/services/api/claude.js' +import type { NonNullableUsage } from 'src/services/api/logging.js' +import { EMPTY_USAGE } from 'src/services/api/logging.js' +import stripAnsi from 'strip-ansi' +import type { Command } from './commands.js' +import { getSlashCommandToolSkills } from './commands.js' +import { + LOCAL_COMMAND_STDERR_TAG, + LOCAL_COMMAND_STDOUT_TAG, +} from './constants/xml.js' +import { + getModelUsage, + getTotalAPIDuration, + getTotalCost, +} from './cost-tracker.js' +import type { CanUseToolFn } from './hooks/useCanUseTool.js' +import { loadMemoryPrompt } from './memdir/memdir.js' +import { hasAutoMemPathOverride } from './memdir/paths.js' +import { query } from './query.js' +import { categorizeRetryableAPIError } from './services/api/errors.js' +import type { MCPServerConnection } from './services/mcp/types.js' +import type { AppState } from './state/AppState.js' +import { type Tools, type ToolUseContext, toolMatchesName } from './Tool.js' +import type { AgentDefinition } from './tools/AgentTool/loadAgentsDir.js' +import { SYNTHETIC_OUTPUT_TOOL_NAME } from './tools/SyntheticOutputTool/SyntheticOutputTool.js' +import type { Message } from './types/message.js' +import type { OrphanedPermission } from './types/textInputTypes.js' +import { createAbortController } from './utils/abortController.js' +import type { AttributionState } from './utils/commitAttribution.js' +import { getGlobalConfig } from './utils/config.js' +import { getCwd } from './utils/cwd.js' +import { isBareMode, isEnvTruthy } from './utils/envUtils.js' +import { getFastModeState } from './utils/fastMode.js' +import { + type FileHistoryState, + fileHistoryEnabled, + fileHistoryMakeSnapshot, +} from './utils/fileHistory.js' +import { + cloneFileStateCache, + type FileStateCache, +} from './utils/fileStateCache.js' +import { headlessProfilerCheckpoint } from './utils/headlessProfiler.js' +import { registerStructuredOutputEnforcement } from './utils/hooks/hookHelpers.js' +import { getInMemoryErrors } from './utils/log.js' +import { countToolCalls, SYNTHETIC_MESSAGES } from './utils/messages.js' +import { + getMainLoopModel, + parseUserSpecifiedModel, +} from './utils/model/model.js' +import { loadAllPluginsCacheOnly } from './utils/plugins/pluginLoader.js' +import { + type ProcessUserInputContext, + processUserInput, +} from './utils/processUserInput/processUserInput.js' +import { fetchSystemPromptParts } from './utils/queryContext.js' +import { setCwd } from './utils/Shell.js' +import { + flushSessionStorage, + recordTranscript, +} from './utils/sessionStorage.js' +import { asSystemPrompt } from './utils/systemPromptType.js' +import { resolveThemeSetting } from './utils/systemTheme.js' +import { + shouldEnableThinkingByDefault, + type ThinkingConfig, +} from './utils/thinking.js' + +// Lazy: MessageSelector.tsx pulls React/ink; only needed for message filtering at query time +/* eslint-disable @typescript-eslint/no-require-imports */ +const messageSelector = + (): typeof import('src/components/MessageSelector.js') => + require('src/components/MessageSelector.js') + +import { + localCommandOutputToSDKAssistantMessage, + toSDKCompactMetadata, +} from './utils/messages/mappers.js' +import { + buildSystemInitMessage, + sdkCompatToolName, +} from './utils/messages/systemInit.js' +import { + getScratchpadDir, + isScratchpadEnabled, +} from './utils/permissions/filesystem.js' +/* eslint-enable @typescript-eslint/no-require-imports */ +import { + handleOrphanedPermission, + isResultSuccessful, + normalizeMessage, +} from './utils/queryHelpers.js' + +// Dead code elimination: conditional import for coordinator mode +/* eslint-disable @typescript-eslint/no-require-imports */ +const getCoordinatorUserContext: ( + mcpClients: ReadonlyArray<{ name: string }>, + scratchpadDir?: string, +) => { [k: string]: string } = feature('COORDINATOR_MODE') + ? require('./coordinator/coordinatorMode.js').getCoordinatorUserContext + : () => ({}) +/* eslint-enable @typescript-eslint/no-require-imports */ + +// Dead code elimination: conditional import for snip compaction +/* eslint-disable @typescript-eslint/no-require-imports */ +const snipModule = feature('HISTORY_SNIP') + ? (require('./services/compact/snipCompact.js') as typeof import('./services/compact/snipCompact.js')) + : null +const snipProjection = feature('HISTORY_SNIP') + ? (require('./services/compact/snipProjection.js') as typeof import('./services/compact/snipProjection.js')) + : null +/* eslint-enable @typescript-eslint/no-require-imports */ + +export type QueryEngineConfig = { + cwd: string + tools: Tools + commands: Command[] + mcpClients: MCPServerConnection[] + agents: AgentDefinition[] + canUseTool: CanUseToolFn + getAppState: () => AppState + setAppState: (f: (prev: AppState) => AppState) => void + initialMessages?: Message[] + readFileCache: FileStateCache + customSystemPrompt?: string + appendSystemPrompt?: string + userSpecifiedModel?: string + fallbackModel?: string + thinkingConfig?: ThinkingConfig + maxTurns?: number + maxBudgetUsd?: number + taskBudget?: { total: number } + jsonSchema?: Record + verbose?: boolean + replayUserMessages?: boolean + /** Handler for URL elicitations triggered by MCP tool -32042 errors. */ + handleElicitation?: ToolUseContext['handleElicitation'] + includePartialMessages?: boolean + setSDKStatus?: (status: SDKStatus) => void + abortController?: AbortController + orphanedPermission?: OrphanedPermission + /** + * Snip-boundary handler: receives each yielded system message plus the + * current mutableMessages store. Returns undefined if the message is not a + * snip boundary; otherwise returns the replayed snip result. Injected by + * ask() when HISTORY_SNIP is enabled so feature-gated strings stay inside + * the gated module (keeps QueryEngine free of excluded strings and testable + * despite feature() returning false under bun test). SDK-only: the REPL + * keeps full history for UI scrollback and projects on demand via + * projectSnippedView; QueryEngine truncates here to bound memory in long + * headless sessions (no UI to preserve). + */ + snipReplay?: ( + yieldedSystemMsg: Message, + store: Message[], + ) => { messages: Message[]; executed: boolean } | undefined +} + +/** + * QueryEngine owns the query lifecycle and session state for a conversation. + * It extracts the core logic from ask() into a standalone class that can be + * used by both the headless/SDK path and (in a future phase) the REPL. + * + * One QueryEngine per conversation. Each submitMessage() call starts a new + * turn within the same conversation. State (messages, file cache, usage, etc.) + * persists across turns. + */ +export class QueryEngine { + private config: QueryEngineConfig + private mutableMessages: Message[] + private abortController: AbortController + private permissionDenials: SDKPermissionDenial[] + private totalUsage: NonNullableUsage + private hasHandledOrphanedPermission = false + private readFileState: FileStateCache + // Turn-scoped skill discovery tracking (feeds was_discovered on + // tengu_skill_tool_invocation). Must persist across the two + // processUserInputContext rebuilds inside submitMessage, but is cleared + // at the start of each submitMessage to avoid unbounded growth across + // many turns in SDK mode. + private discoveredSkillNames = new Set() + private loadedNestedMemoryPaths = new Set() + + constructor(config: QueryEngineConfig) { + this.config = config + this.mutableMessages = config.initialMessages ?? [] + this.abortController = config.abortController ?? createAbortController() + this.permissionDenials = [] + this.readFileState = config.readFileCache + this.totalUsage = EMPTY_USAGE + } + + async *submitMessage( + prompt: string | ContentBlockParam[], + options?: { uuid?: string; isMeta?: boolean }, + ): AsyncGenerator { + const { + cwd, + commands, + tools, + mcpClients, + verbose = false, + thinkingConfig, + maxTurns, + maxBudgetUsd, + taskBudget, + canUseTool, + customSystemPrompt, + appendSystemPrompt, + userSpecifiedModel, + fallbackModel, + jsonSchema, + getAppState, + setAppState, + replayUserMessages = false, + includePartialMessages = false, + agents = [], + setSDKStatus, + orphanedPermission, + } = this.config + + this.discoveredSkillNames.clear() + setCwd(cwd) + const persistSession = !isSessionPersistenceDisabled() + const startTime = Date.now() + + // Wrap canUseTool to track permission denials + const wrappedCanUseTool: CanUseToolFn = async ( + tool, + input, + toolUseContext, + assistantMessage, + toolUseID, + forceDecision, + ) => { + const result = await canUseTool( + tool, + input, + toolUseContext, + assistantMessage, + toolUseID, + forceDecision, + ) + + // Track denials for SDK reporting + if (result.behavior !== 'allow') { + this.permissionDenials.push({ + tool_name: sdkCompatToolName(tool.name), + tool_use_id: toolUseID, + tool_input: input, + }) + } + + return result + } + + const initialAppState = getAppState() + const initialMainLoopModel = userSpecifiedModel + ? parseUserSpecifiedModel(userSpecifiedModel) + : getMainLoopModel() + + const initialThinkingConfig: ThinkingConfig = thinkingConfig + ? thinkingConfig + : shouldEnableThinkingByDefault() !== false + ? { type: 'adaptive' } + : { type: 'disabled' } + + headlessProfilerCheckpoint('before_getSystemPrompt') + // Narrow once so TS tracks the type through the conditionals below. + const customPrompt = + typeof customSystemPrompt === 'string' ? customSystemPrompt : undefined + const { + defaultSystemPrompt, + userContext: baseUserContext, + systemContext, + } = await fetchSystemPromptParts({ + tools, + mainLoopModel: initialMainLoopModel, + additionalWorkingDirectories: Array.from( + initialAppState.toolPermissionContext.additionalWorkingDirectories.keys(), + ), + mcpClients, + customSystemPrompt: customPrompt, + }) + headlessProfilerCheckpoint('after_getSystemPrompt') + const userContext = { + ...baseUserContext, + ...getCoordinatorUserContext( + mcpClients, + isScratchpadEnabled() ? getScratchpadDir() : undefined, + ), + } + + // When an SDK caller provides a custom system prompt AND has set + // CLAUDE_COWORK_MEMORY_PATH_OVERRIDE, inject the memory-mechanics prompt. + // The env var is an explicit opt-in signal — the caller has wired up + // a memory directory and needs Claude to know how to use it (which + // Write/Edit tools to call, MEMORY.md filename, loading semantics). + // The caller can layer their own policy text via appendSystemPrompt. + const memoryMechanicsPrompt = + customPrompt !== undefined && hasAutoMemPathOverride() + ? await loadMemoryPrompt() + : null + + const systemPrompt = asSystemPrompt([ + ...(customPrompt !== undefined ? [customPrompt] : defaultSystemPrompt), + ...(memoryMechanicsPrompt ? [memoryMechanicsPrompt] : []), + ...(appendSystemPrompt ? [appendSystemPrompt] : []), + ]) + + // Register function hook for structured output enforcement + const hasStructuredOutputTool = tools.some(t => + toolMatchesName(t, SYNTHETIC_OUTPUT_TOOL_NAME), + ) + if (jsonSchema && hasStructuredOutputTool) { + registerStructuredOutputEnforcement(setAppState, getSessionId()) + } + + let processUserInputContext: ProcessUserInputContext = { + messages: this.mutableMessages, + // Slash commands that mutate the message array (e.g. /force-snip) + // call setMessages(fn). In interactive mode this writes back to + // AppState; in print mode we write back to mutableMessages so the + // rest of the query loop (push at :389, snapshot at :392) sees + // the result. The second processUserInputContext below (after + // slash-command processing) keeps the no-op — nothing else calls + // setMessages past that point. + setMessages: fn => { + this.mutableMessages = fn(this.mutableMessages) + }, + onChangeAPIKey: () => {}, + handleElicitation: this.config.handleElicitation, + options: { + commands, + debug: false, // we use stdout, so don't want to clobber it + tools, + verbose, + mainLoopModel: initialMainLoopModel, + thinkingConfig: initialThinkingConfig, + mcpClients, + mcpResources: {}, + ideInstallationStatus: null, + isNonInteractiveSession: true, + customSystemPrompt, + appendSystemPrompt, + agentDefinitions: { activeAgents: agents, allAgents: [] }, + theme: resolveThemeSetting(getGlobalConfig().theme), + maxBudgetUsd, + }, + getAppState, + setAppState, + abortController: this.abortController, + readFileState: this.readFileState, + nestedMemoryAttachmentTriggers: new Set(), + loadedNestedMemoryPaths: this.loadedNestedMemoryPaths, + dynamicSkillDirTriggers: new Set(), + discoveredSkillNames: this.discoveredSkillNames, + setInProgressToolUseIDs: () => {}, + setResponseLength: () => {}, + updateFileHistoryState: ( + updater: (prev: FileHistoryState) => FileHistoryState, + ) => { + setAppState(prev => { + const updated = updater(prev.fileHistory) + if (updated === prev.fileHistory) return prev + return { ...prev, fileHistory: updated } + }) + }, + updateAttributionState: ( + updater: (prev: AttributionState) => AttributionState, + ) => { + setAppState(prev => { + const updated = updater(prev.attribution) + if (updated === prev.attribution) return prev + return { ...prev, attribution: updated } + }) + }, + setSDKStatus, + } + + // Handle orphaned permission (only once per engine lifetime) + if (orphanedPermission && !this.hasHandledOrphanedPermission) { + this.hasHandledOrphanedPermission = true + for await (const message of handleOrphanedPermission( + orphanedPermission, + tools, + this.mutableMessages, + processUserInputContext, + )) { + yield message + } + } + + const { + messages: messagesFromUserInput, + shouldQuery, + allowedTools, + model: modelFromUserInput, + resultText, + } = await processUserInput({ + input: prompt, + mode: 'prompt', + setToolJSX: () => {}, + context: { + ...processUserInputContext, + messages: this.mutableMessages, + }, + messages: this.mutableMessages, + uuid: options?.uuid, + isMeta: options?.isMeta, + querySource: 'sdk', + }) + + // Push new messages, including user input and any attachments + this.mutableMessages.push(...messagesFromUserInput) + + // Update params to reflect updates from processing /slash commands + const messages = [...this.mutableMessages] + + // Persist the user's message(s) to transcript BEFORE entering the query + // loop. The for-await below only calls recordTranscript when ask() yields + // an assistant/user/compact_boundary message — which doesn't happen until + // the API responds. If the process is killed before that (e.g. user clicks + // Stop in cowork seconds after send), the transcript is left with only + // queue-operation entries; getLastSessionLog filters those out, returns + // null, and --resume fails with "No conversation found". Writing now makes + // the transcript resumable from the point the user message was accepted, + // even if no API response ever arrives. + // + // --bare / SIMPLE: fire-and-forget. Scripted calls don't --resume after + // kill-mid-request. The await is ~4ms on SSD, ~30ms under disk contention + // — the single largest controllable critical-path cost after module eval. + // Transcript is still written (for post-hoc debugging); just not blocking. + if (persistSession && messagesFromUserInput.length > 0) { + const transcriptPromise = recordTranscript(messages) + if (isBareMode()) { + void transcriptPromise + } else { + await transcriptPromise + if ( + isEnvTruthy(process.env.CLAUDE_CODE_EAGER_FLUSH) || + isEnvTruthy(process.env.CLAUDE_CODE_IS_COWORK) + ) { + await flushSessionStorage() + } + } + } + + // Filter messages that should be acknowledged after transcript + const replayableMessages = messagesFromUserInput.filter( + msg => + (msg.type === 'user' && + !msg.isMeta && // Skip synthetic caveat messages + !msg.toolUseResult && // Skip tool results (they'll be acked from query) + messageSelector().selectableUserMessagesFilter(msg)) || // Skip non-user-authored messages (task notifications, etc.) + (msg.type === 'system' && msg.subtype === 'compact_boundary'), // Always ack compact boundaries + ) + const messagesToAck = replayUserMessages ? replayableMessages : [] + + // Update the ToolPermissionContext based on user input processing (as necessary) + setAppState(prev => ({ + ...prev, + toolPermissionContext: { + ...prev.toolPermissionContext, + alwaysAllowRules: { + ...prev.toolPermissionContext.alwaysAllowRules, + command: allowedTools, + }, + }, + })) + + const mainLoopModel = modelFromUserInput ?? initialMainLoopModel + + // Recreate after processing the prompt to pick up updated messages and + // model (from slash commands). + processUserInputContext = { + messages, + setMessages: () => {}, + onChangeAPIKey: () => {}, + handleElicitation: this.config.handleElicitation, + options: { + commands, + debug: false, + tools, + verbose, + mainLoopModel, + thinkingConfig: initialThinkingConfig, + mcpClients, + mcpResources: {}, + ideInstallationStatus: null, + isNonInteractiveSession: true, + customSystemPrompt, + appendSystemPrompt, + theme: resolveThemeSetting(getGlobalConfig().theme), + agentDefinitions: { activeAgents: agents, allAgents: [] }, + maxBudgetUsd, + }, + getAppState, + setAppState, + abortController: this.abortController, + readFileState: this.readFileState, + nestedMemoryAttachmentTriggers: new Set(), + loadedNestedMemoryPaths: this.loadedNestedMemoryPaths, + dynamicSkillDirTriggers: new Set(), + discoveredSkillNames: this.discoveredSkillNames, + setInProgressToolUseIDs: () => {}, + setResponseLength: () => {}, + updateFileHistoryState: processUserInputContext.updateFileHistoryState, + updateAttributionState: processUserInputContext.updateAttributionState, + setSDKStatus, + } + + headlessProfilerCheckpoint('before_skills_plugins') + // Cache-only: headless/SDK/CCR startup must not block on network for + // ref-tracked plugins. CCR populates the cache via CLAUDE_CODE_SYNC_PLUGIN_INSTALL + // (headlessPluginInstall) or CLAUDE_CODE_PLUGIN_SEED_DIR before this runs; + // SDK callers that need fresh source can call /reload-plugins. + const [skills, { enabled: enabledPlugins }] = await Promise.all([ + getSlashCommandToolSkills(getCwd()), + loadAllPluginsCacheOnly(), + ]) + headlessProfilerCheckpoint('after_skills_plugins') + + yield buildSystemInitMessage({ + tools, + mcpClients, + model: mainLoopModel, + permissionMode: initialAppState.toolPermissionContext + .mode as PermissionMode, // TODO: avoid the cast + commands, + agents, + skills, + plugins: enabledPlugins, + fastMode: initialAppState.fastMode, + }) + + // Record when system message is yielded for headless latency tracking + headlessProfilerCheckpoint('system_message_yielded') + + if (!shouldQuery) { + // Return the results of local slash commands. + // Use messagesFromUserInput (not replayableMessages) for command output + // because selectableUserMessagesFilter excludes local-command-stdout tags. + for (const msg of messagesFromUserInput) { + if ( + msg.type === 'user' && + typeof msg.message.content === 'string' && + (msg.message.content.includes(`<${LOCAL_COMMAND_STDOUT_TAG}>`) || + msg.message.content.includes(`<${LOCAL_COMMAND_STDERR_TAG}>`) || + msg.isCompactSummary) + ) { + yield { + type: 'user', + message: { + ...msg.message, + content: stripAnsi(msg.message.content), + }, + session_id: getSessionId(), + parent_tool_use_id: null, + uuid: msg.uuid, + timestamp: msg.timestamp, + isReplay: !msg.isCompactSummary, + isSynthetic: msg.isMeta || msg.isVisibleInTranscriptOnly, + } as SDKUserMessageReplay + } + + // Local command output — yield as a synthetic assistant message so + // RC renders it as assistant-style text rather than a user bubble. + // Emitted as assistant (not the dedicated SDKLocalCommandOutputMessage + // system subtype) so mobile clients + session-ingress can parse it. + if ( + msg.type === 'system' && + msg.subtype === 'local_command' && + typeof msg.content === 'string' && + (msg.content.includes(`<${LOCAL_COMMAND_STDOUT_TAG}>`) || + msg.content.includes(`<${LOCAL_COMMAND_STDERR_TAG}>`)) + ) { + yield localCommandOutputToSDKAssistantMessage(msg.content, msg.uuid) + } + + if (msg.type === 'system' && msg.subtype === 'compact_boundary') { + yield { + type: 'system', + subtype: 'compact_boundary' as const, + session_id: getSessionId(), + uuid: msg.uuid, + compact_metadata: toSDKCompactMetadata(msg.compactMetadata), + } as SDKCompactBoundaryMessage + } + } + + if (persistSession) { + await recordTranscript(messages) + if ( + isEnvTruthy(process.env.CLAUDE_CODE_EAGER_FLUSH) || + isEnvTruthy(process.env.CLAUDE_CODE_IS_COWORK) + ) { + await flushSessionStorage() + } + } + + yield { + type: 'result', + subtype: 'success', + is_error: false, + duration_ms: Date.now() - startTime, + duration_api_ms: getTotalAPIDuration(), + num_turns: messages.length - 1, + result: resultText ?? '', + stop_reason: null, + session_id: getSessionId(), + total_cost_usd: getTotalCost(), + usage: this.totalUsage, + modelUsage: getModelUsage(), + permission_denials: this.permissionDenials, + fast_mode_state: getFastModeState( + mainLoopModel, + initialAppState.fastMode, + ), + uuid: randomUUID(), + } + return + } + + if (fileHistoryEnabled() && persistSession) { + messagesFromUserInput + .filter(messageSelector().selectableUserMessagesFilter) + .forEach(message => { + void fileHistoryMakeSnapshot( + (updater: (prev: FileHistoryState) => FileHistoryState) => { + setAppState(prev => ({ + ...prev, + fileHistory: updater(prev.fileHistory), + })) + }, + message.uuid, + ) + }) + } + + // Track current message usage (reset on each message_start) + let currentMessageUsage: NonNullableUsage = EMPTY_USAGE + let turnCount = 1 + let hasAcknowledgedInitialMessages = false + // Track structured output from StructuredOutput tool calls + let structuredOutputFromTool: unknown + // Track the last stop_reason from assistant messages + let lastStopReason: string | null = null + // Reference-based watermark so error_during_execution's errors[] is + // turn-scoped. A length-based index breaks when the 100-entry ring buffer + // shift()s during the turn — the index slides. If this entry is rotated + // out, lastIndexOf returns -1 and we include everything (safe fallback). + const errorLogWatermark = getInMemoryErrors().at(-1) + // Snapshot count before this query for delta-based retry limiting + const initialStructuredOutputCalls = jsonSchema + ? countToolCalls(this.mutableMessages, SYNTHETIC_OUTPUT_TOOL_NAME) + : 0 + + for await (const message of query({ + messages, + systemPrompt, + userContext, + systemContext, + canUseTool: wrappedCanUseTool, + toolUseContext: processUserInputContext, + fallbackModel, + querySource: 'sdk', + maxTurns, + taskBudget, + })) { + // Record assistant, user, and compact boundary messages + if ( + message.type === 'assistant' || + message.type === 'user' || + (message.type === 'system' && message.subtype === 'compact_boundary') + ) { + // Before writing a compact boundary, flush any in-memory-only + // messages up through the preservedSegment tail. Attachments and + // progress are now recorded inline (their switch cases below), but + // this flush still matters for the preservedSegment tail walk. + // If the SDK subprocess restarts before then (claude-desktop kills + // between turns), tailUuid points to a never-written message → + // applyPreservedSegmentRelinks fails its tail→head walk → returns + // without pruning → resume loads full pre-compact history. + if ( + persistSession && + message.type === 'system' && + message.subtype === 'compact_boundary' + ) { + const tailUuid = message.compactMetadata?.preservedSegment?.tailUuid + if (tailUuid) { + const tailIdx = this.mutableMessages.findLastIndex( + m => m.uuid === tailUuid, + ) + if (tailIdx !== -1) { + await recordTranscript(this.mutableMessages.slice(0, tailIdx + 1)) + } + } + } + messages.push(message) + if (persistSession) { + // Fire-and-forget for assistant messages. claude.ts yields one + // assistant message per content block, then mutates the last + // one's message.usage/stop_reason on message_delta — relying on + // the write queue's 100ms lazy jsonStringify. Awaiting here + // blocks ask()'s generator, so message_delta can't run until + // every block is consumed; the drain timer (started at block 1) + // elapses first. Interactive CC doesn't hit this because + // useLogMessages.ts fire-and-forgets. enqueueWrite is + // order-preserving so fire-and-forget here is safe. + if (message.type === 'assistant') { + void recordTranscript(messages) + } else { + await recordTranscript(messages) + } + } + + // Acknowledge initial user messages after first transcript recording + if (!hasAcknowledgedInitialMessages && messagesToAck.length > 0) { + hasAcknowledgedInitialMessages = true + for (const msgToAck of messagesToAck) { + if (msgToAck.type === 'user') { + yield { + type: 'user', + message: msgToAck.message, + session_id: getSessionId(), + parent_tool_use_id: null, + uuid: msgToAck.uuid, + timestamp: msgToAck.timestamp, + isReplay: true, + } as SDKUserMessageReplay + } + } + } + } + + if (message.type === 'user') { + turnCount++ + } + + switch (message.type) { + case 'tombstone': + // Tombstone messages are control signals for removing messages, skip them + break + case 'assistant': + // Capture stop_reason if already set (synthetic messages). For + // streamed responses, this is null at content_block_stop time; + // the real value arrives via message_delta (handled below). + if (message.message.stop_reason != null) { + lastStopReason = message.message.stop_reason + } + this.mutableMessages.push(message) + yield* normalizeMessage(message) + break + case 'progress': + this.mutableMessages.push(message) + // Record inline so the dedup loop in the next ask() call sees it + // as already-recorded. Without this, deferred progress interleaves + // with already-recorded tool_results in mutableMessages, and the + // dedup walk freezes startingParentUuid at the wrong message — + // forking the chain and orphaning the conversation on resume. + if (persistSession) { + messages.push(message) + void recordTranscript(messages) + } + yield* normalizeMessage(message) + break + case 'user': + this.mutableMessages.push(message) + yield* normalizeMessage(message) + break + case 'stream_event': + if (message.event.type === 'message_start') { + // Reset current message usage for new message + currentMessageUsage = EMPTY_USAGE + currentMessageUsage = updateUsage( + currentMessageUsage, + message.event.message.usage, + ) + } + if (message.event.type === 'message_delta') { + currentMessageUsage = updateUsage( + currentMessageUsage, + message.event.usage, + ) + // Capture stop_reason from message_delta. The assistant message + // is yielded at content_block_stop with stop_reason=null; the + // real value only arrives here (see claude.ts message_delta + // handler). Without this, result.stop_reason is always null. + if (message.event.delta.stop_reason != null) { + lastStopReason = message.event.delta.stop_reason + } + } + if (message.event.type === 'message_stop') { + // Accumulate current message usage into total + this.totalUsage = accumulateUsage( + this.totalUsage, + currentMessageUsage, + ) + } + + if (includePartialMessages) { + yield { + type: 'stream_event' as const, + event: message.event, + session_id: getSessionId(), + parent_tool_use_id: null, + uuid: randomUUID(), + } + } + + break + case 'attachment': + this.mutableMessages.push(message) + // Record inline (same reason as progress above). + if (persistSession) { + messages.push(message) + void recordTranscript(messages) + } + + // Extract structured output from StructuredOutput tool calls + if (message.attachment.type === 'structured_output') { + structuredOutputFromTool = message.attachment.data + } + // Handle max turns reached signal from query.ts + else if (message.attachment.type === 'max_turns_reached') { + if (persistSession) { + if ( + isEnvTruthy(process.env.CLAUDE_CODE_EAGER_FLUSH) || + isEnvTruthy(process.env.CLAUDE_CODE_IS_COWORK) + ) { + await flushSessionStorage() + } + } + yield { + type: 'result', + subtype: 'error_max_turns', + duration_ms: Date.now() - startTime, + duration_api_ms: getTotalAPIDuration(), + is_error: true, + num_turns: message.attachment.turnCount, + stop_reason: lastStopReason, + session_id: getSessionId(), + total_cost_usd: getTotalCost(), + usage: this.totalUsage, + modelUsage: getModelUsage(), + permission_denials: this.permissionDenials, + fast_mode_state: getFastModeState( + mainLoopModel, + initialAppState.fastMode, + ), + uuid: randomUUID(), + errors: [ + `Reached maximum number of turns (${message.attachment.maxTurns})`, + ], + } + return + } + // Yield queued_command attachments as SDK user message replays + else if ( + replayUserMessages && + message.attachment.type === 'queued_command' + ) { + yield { + type: 'user', + message: { + role: 'user' as const, + content: message.attachment.prompt, + }, + session_id: getSessionId(), + parent_tool_use_id: null, + uuid: message.attachment.source_uuid || message.uuid, + timestamp: message.timestamp, + isReplay: true, + } as SDKUserMessageReplay + } + break + case 'stream_request_start': + // Don't yield stream request start messages + break + case 'system': { + // Snip boundary: replay on our store to remove zombie messages and + // stale markers. The yielded boundary is a signal, not data to push — + // the replay produces its own equivalent boundary. Without this, + // markers persist and re-trigger on every turn, and mutableMessages + // never shrinks (memory leak in long SDK sessions). The subtype + // check lives inside the injected callback so feature-gated strings + // stay out of this file (excluded-strings check). + const snipResult = this.config.snipReplay?.( + message, + this.mutableMessages, + ) + if (snipResult !== undefined) { + if (snipResult.executed) { + this.mutableMessages.length = 0 + this.mutableMessages.push(...snipResult.messages) + } + break + } + this.mutableMessages.push(message) + // Yield compact boundary messages to SDK + if ( + message.subtype === 'compact_boundary' && + message.compactMetadata + ) { + // Release pre-compaction messages for GC. The boundary was just + // pushed so it's the last element. query.ts already uses + // getMessagesAfterCompactBoundary() internally, so only + // post-boundary messages are needed going forward. + const mutableBoundaryIdx = this.mutableMessages.length - 1 + if (mutableBoundaryIdx > 0) { + this.mutableMessages.splice(0, mutableBoundaryIdx) + } + const localBoundaryIdx = messages.length - 1 + if (localBoundaryIdx > 0) { + messages.splice(0, localBoundaryIdx) + } + + yield { + type: 'system', + subtype: 'compact_boundary' as const, + session_id: getSessionId(), + uuid: message.uuid, + compact_metadata: toSDKCompactMetadata(message.compactMetadata), + } + } + if (message.subtype === 'api_error') { + yield { + type: 'system', + subtype: 'api_retry' as const, + attempt: message.retryAttempt, + max_retries: message.maxRetries, + retry_delay_ms: message.retryInMs, + error_status: message.error.status ?? null, + error: categorizeRetryableAPIError(message.error), + session_id: getSessionId(), + uuid: message.uuid, + } + } + // Don't yield other system messages in headless mode + break + } + case 'tool_use_summary': + // Yield tool use summary messages to SDK + yield { + type: 'tool_use_summary' as const, + summary: message.summary, + preceding_tool_use_ids: message.precedingToolUseIds, + session_id: getSessionId(), + uuid: message.uuid, + } + break + } + + // Check if USD budget has been exceeded + if (maxBudgetUsd !== undefined && getTotalCost() >= maxBudgetUsd) { + if (persistSession) { + if ( + isEnvTruthy(process.env.CLAUDE_CODE_EAGER_FLUSH) || + isEnvTruthy(process.env.CLAUDE_CODE_IS_COWORK) + ) { + await flushSessionStorage() + } + } + yield { + type: 'result', + subtype: 'error_max_budget_usd', + duration_ms: Date.now() - startTime, + duration_api_ms: getTotalAPIDuration(), + is_error: true, + num_turns: turnCount, + stop_reason: lastStopReason, + session_id: getSessionId(), + total_cost_usd: getTotalCost(), + usage: this.totalUsage, + modelUsage: getModelUsage(), + permission_denials: this.permissionDenials, + fast_mode_state: getFastModeState( + mainLoopModel, + initialAppState.fastMode, + ), + uuid: randomUUID(), + errors: [`Reached maximum budget ($${maxBudgetUsd})`], + } + return + } + + // Check if structured output retry limit exceeded (only on user messages) + if (message.type === 'user' && jsonSchema) { + const currentCalls = countToolCalls( + this.mutableMessages, + SYNTHETIC_OUTPUT_TOOL_NAME, + ) + const callsThisQuery = currentCalls - initialStructuredOutputCalls + const maxRetries = parseInt( + process.env.MAX_STRUCTURED_OUTPUT_RETRIES || '5', + 10, + ) + if (callsThisQuery >= maxRetries) { + if (persistSession) { + if ( + isEnvTruthy(process.env.CLAUDE_CODE_EAGER_FLUSH) || + isEnvTruthy(process.env.CLAUDE_CODE_IS_COWORK) + ) { + await flushSessionStorage() + } + } + yield { + type: 'result', + subtype: 'error_max_structured_output_retries', + duration_ms: Date.now() - startTime, + duration_api_ms: getTotalAPIDuration(), + is_error: true, + num_turns: turnCount, + stop_reason: lastStopReason, + session_id: getSessionId(), + total_cost_usd: getTotalCost(), + usage: this.totalUsage, + modelUsage: getModelUsage(), + permission_denials: this.permissionDenials, + fast_mode_state: getFastModeState( + mainLoopModel, + initialAppState.fastMode, + ), + uuid: randomUUID(), + errors: [ + `Failed to provide valid structured output after ${maxRetries} attempts`, + ], + } + return + } + } + } + + // Stop hooks yield progress/attachment messages AFTER the assistant + // response (via yield* handleStopHooks in query.ts). Since #23537 pushes + // those to `messages` inline, last(messages) can be a progress/attachment + // instead of the assistant — which makes textResult extraction below + // return '' and -p mode emit a blank line. Allowlist to assistant|user: + // isResultSuccessful handles both (user with all tool_result blocks is a + // valid successful terminal state). + const result = messages.findLast( + m => m.type === 'assistant' || m.type === 'user', + ) + // Capture for the error_during_execution diagnostic — isResultSuccessful + // is a type predicate (message is Message), so inside the false branch + // `result` narrows to never and these accesses don't typecheck. + const edeResultType = result?.type ?? 'undefined' + const edeLastContentType = + result?.type === 'assistant' + ? (last(result.message.content)?.type ?? 'none') + : 'n/a' + + // Flush buffered transcript writes before yielding result. + // The desktop app kills the CLI process immediately after receiving the + // result message, so any unflushed writes would be lost. + if (persistSession) { + if ( + isEnvTruthy(process.env.CLAUDE_CODE_EAGER_FLUSH) || + isEnvTruthy(process.env.CLAUDE_CODE_IS_COWORK) + ) { + await flushSessionStorage() + } + } + + if (!isResultSuccessful(result, lastStopReason)) { + yield { + type: 'result', + subtype: 'error_during_execution', + duration_ms: Date.now() - startTime, + duration_api_ms: getTotalAPIDuration(), + is_error: true, + num_turns: turnCount, + stop_reason: lastStopReason, + session_id: getSessionId(), + total_cost_usd: getTotalCost(), + usage: this.totalUsage, + modelUsage: getModelUsage(), + permission_denials: this.permissionDenials, + fast_mode_state: getFastModeState( + mainLoopModel, + initialAppState.fastMode, + ), + uuid: randomUUID(), + // Diagnostic prefix: these are what isResultSuccessful() checks — if + // the result type isn't assistant-with-text/thinking or user-with- + // tool_result, and stop_reason isn't end_turn, that's why this fired. + // errors[] is turn-scoped via the watermark; previously it dumped the + // entire process's logError buffer (ripgrep timeouts, ENOENT, etc). + errors: (() => { + const all = getInMemoryErrors() + const start = errorLogWatermark + ? all.lastIndexOf(errorLogWatermark) + 1 + : 0 + return [ + `[ede_diagnostic] result_type=${edeResultType} last_content_type=${edeLastContentType} stop_reason=${lastStopReason}`, + ...all.slice(start).map(_ => _.error), + ] + })(), + } + return + } + + // Extract the text result based on message type + let textResult = '' + let isApiError = false + + if (result.type === 'assistant') { + const lastContent = last(result.message.content) + if ( + lastContent?.type === 'text' && + !SYNTHETIC_MESSAGES.has(lastContent.text) + ) { + textResult = lastContent.text + } + isApiError = Boolean(result.isApiErrorMessage) + } + + yield { + type: 'result', + subtype: 'success', + is_error: isApiError, + duration_ms: Date.now() - startTime, + duration_api_ms: getTotalAPIDuration(), + num_turns: turnCount, + result: textResult, + stop_reason: lastStopReason, + session_id: getSessionId(), + total_cost_usd: getTotalCost(), + usage: this.totalUsage, + modelUsage: getModelUsage(), + permission_denials: this.permissionDenials, + structured_output: structuredOutputFromTool, + fast_mode_state: getFastModeState( + mainLoopModel, + initialAppState.fastMode, + ), + uuid: randomUUID(), + } + } + + interrupt(): void { + this.abortController.abort() + } + + getMessages(): readonly Message[] { + return this.mutableMessages + } + + getReadFileState(): FileStateCache { + return this.readFileState + } + + getSessionId(): string { + return getSessionId() + } + + setModel(model: string): void { + this.config.userSpecifiedModel = model + } +} + +/** + * Sends a single prompt to the Claude API and returns the response. + * Assumes that claude is being used non-interactively -- will not + * ask the user for permissions or further input. + * + * Convenience wrapper around QueryEngine for one-shot usage. + */ +export async function* ask({ + commands, + prompt, + promptUuid, + isMeta, + cwd, + tools, + mcpClients, + verbose = false, + thinkingConfig, + maxTurns, + maxBudgetUsd, + taskBudget, + canUseTool, + mutableMessages = [], + getReadFileCache, + setReadFileCache, + customSystemPrompt, + appendSystemPrompt, + userSpecifiedModel, + fallbackModel, + jsonSchema, + getAppState, + setAppState, + abortController, + replayUserMessages = false, + includePartialMessages = false, + handleElicitation, + agents = [], + setSDKStatus, + orphanedPermission, +}: { + commands: Command[] + prompt: string | Array + promptUuid?: string + isMeta?: boolean + cwd: string + tools: Tools + verbose?: boolean + mcpClients: MCPServerConnection[] + thinkingConfig?: ThinkingConfig + maxTurns?: number + maxBudgetUsd?: number + taskBudget?: { total: number } + canUseTool: CanUseToolFn + mutableMessages?: Message[] + customSystemPrompt?: string + appendSystemPrompt?: string + userSpecifiedModel?: string + fallbackModel?: string + jsonSchema?: Record + getAppState: () => AppState + setAppState: (f: (prev: AppState) => AppState) => void + getReadFileCache: () => FileStateCache + setReadFileCache: (cache: FileStateCache) => void + abortController?: AbortController + replayUserMessages?: boolean + includePartialMessages?: boolean + handleElicitation?: ToolUseContext['handleElicitation'] + agents?: AgentDefinition[] + setSDKStatus?: (status: SDKStatus) => void + orphanedPermission?: OrphanedPermission +}): AsyncGenerator { + const engine = new QueryEngine({ + cwd, + tools, + commands, + mcpClients, + agents, + canUseTool, + getAppState, + setAppState, + initialMessages: mutableMessages, + readFileCache: cloneFileStateCache(getReadFileCache()), + customSystemPrompt, + appendSystemPrompt, + userSpecifiedModel, + fallbackModel, + thinkingConfig, + maxTurns, + maxBudgetUsd, + taskBudget, + jsonSchema, + verbose, + handleElicitation, + replayUserMessages, + includePartialMessages, + setSDKStatus, + abortController, + orphanedPermission, + ...(feature('HISTORY_SNIP') + ? { + snipReplay: (yielded: Message, store: Message[]) => { + if (!snipProjection!.isSnipBoundaryMessage(yielded)) + return undefined + return snipModule!.snipCompactIfNeeded(store, { force: true }) + }, + } + : {}), + }) + + try { + yield* engine.submitMessage(prompt, { + uuid: promptUuid, + isMeta, + }) + } finally { + setReadFileCache(engine.getReadFileState()) + } +} diff --git a/src/Task.ts b/src/Task.ts new file mode 100644 index 0000000..196caf3 --- /dev/null +++ b/src/Task.ts @@ -0,0 +1,125 @@ +import { randomBytes } from 'crypto' +import type { AppState } from './state/AppState.js' +import type { AgentId } from './types/ids.js' +import { getTaskOutputPath } from './utils/task/diskOutput.js' + +export type TaskType = + | 'local_bash' + | 'local_agent' + | 'remote_agent' + | 'in_process_teammate' + | 'local_workflow' + | 'monitor_mcp' + | 'dream' + +export type TaskStatus = + | 'pending' + | 'running' + | 'completed' + | 'failed' + | 'killed' + +/** + * True when a task is in a terminal state and will not transition further. + * Used to guard against injecting messages into dead teammates, evicting + * finished tasks from AppState, and orphan-cleanup paths. + */ +export function isTerminalTaskStatus(status: TaskStatus): boolean { + return status === 'completed' || status === 'failed' || status === 'killed' +} + +export type TaskHandle = { + taskId: string + cleanup?: () => void +} + +export type SetAppState = (f: (prev: AppState) => AppState) => void + +export type TaskContext = { + abortController: AbortController + getAppState: () => AppState + setAppState: SetAppState +} + +// Base fields shared by all task states +export type TaskStateBase = { + id: string + type: TaskType + status: TaskStatus + description: string + toolUseId?: string + startTime: number + endTime?: number + totalPausedMs?: number + outputFile: string + outputOffset: number + notified: boolean +} + +export type LocalShellSpawnInput = { + command: string + description: string + timeout?: number + toolUseId?: string + agentId?: AgentId + /** UI display variant: description-as-label, dialog title, status bar pill. */ + kind?: 'bash' | 'monitor' +} + +// What getTaskByType dispatches for: kill. spawn/render were never +// called polymorphically (removed in #22546). All six kill implementations +// use only setAppState — getAppState/abortController were dead weight. +export type Task = { + name: string + type: TaskType + kill(taskId: string, setAppState: SetAppState): Promise +} + +// Task ID prefixes +const TASK_ID_PREFIXES: Record = { + local_bash: 'b', // Keep as 'b' for backward compatibility + local_agent: 'a', + remote_agent: 'r', + in_process_teammate: 't', + local_workflow: 'w', + monitor_mcp: 'm', + dream: 'd', +} + +// Get task ID prefix +function getTaskIdPrefix(type: TaskType): string { + return TASK_ID_PREFIXES[type] ?? 'x' +} + +// Case-insensitive-safe alphabet (digits + lowercase) for task IDs. +// 36^8 ≈ 2.8 trillion combinations, sufficient to resist brute-force symlink attacks. +const TASK_ID_ALPHABET = '0123456789abcdefghijklmnopqrstuvwxyz' + +export function generateTaskId(type: TaskType): string { + const prefix = getTaskIdPrefix(type) + const bytes = randomBytes(8) + let id = prefix + for (let i = 0; i < 8; i++) { + id += TASK_ID_ALPHABET[bytes[i]! % TASK_ID_ALPHABET.length] + } + return id +} + +export function createTaskStateBase( + id: string, + type: TaskType, + description: string, + toolUseId?: string, +): TaskStateBase { + return { + id, + type, + status: 'pending', + description, + toolUseId, + startTime: Date.now(), + outputFile: getTaskOutputPath(id), + outputOffset: 0, + notified: false, + } +} diff --git a/src/Tool.ts b/src/Tool.ts new file mode 100644 index 0000000..205cac0 --- /dev/null +++ b/src/Tool.ts @@ -0,0 +1,792 @@ +import type { + ToolResultBlockParam, + ToolUseBlockParam, +} from '@anthropic-ai/sdk/resources/index.mjs' +import type { + ElicitRequestURLParams, + ElicitResult, +} from '@modelcontextprotocol/sdk/types.js' +import type { UUID } from 'crypto' +import type { z } from 'zod/v4' +import type { Command } from './commands.js' +import type { CanUseToolFn } from './hooks/useCanUseTool.js' +import type { ThinkingConfig } from './utils/thinking.js' + +export type ToolInputJSONSchema = { + [x: string]: unknown + type: 'object' + properties?: { + [x: string]: unknown + } +} + +import type { Notification } from './context/notifications.js' +import type { + MCPServerConnection, + ServerResource, +} from './services/mcp/types.js' +import type { + AgentDefinition, + AgentDefinitionsResult, +} from './tools/AgentTool/loadAgentsDir.js' +import type { + AssistantMessage, + AttachmentMessage, + Message, + ProgressMessage, + SystemLocalCommandMessage, + SystemMessage, + UserMessage, +} from './types/message.js' +// Import permission types from centralized location to break import cycles +// Import PermissionResult from centralized location to break import cycles +import type { + AdditionalWorkingDirectory, + PermissionMode, + PermissionResult, +} from './types/permissions.js' +// Import tool progress types from centralized location to break import cycles +import type { + AgentToolProgress, + BashProgress, + MCPProgress, + REPLToolProgress, + SkillToolProgress, + TaskOutputProgress, + ToolProgressData, + WebSearchProgress, +} from './types/tools.js' +import type { FileStateCache } from './utils/fileStateCache.js' +import type { DenialTrackingState } from './utils/permissions/denialTracking.js' +import type { SystemPrompt } from './utils/systemPromptType.js' +import type { ContentReplacementState } from './utils/toolResultStorage.js' + +// Re-export progress types for backwards compatibility +export type { + AgentToolProgress, + BashProgress, + MCPProgress, + REPLToolProgress, + SkillToolProgress, + TaskOutputProgress, + WebSearchProgress, +} + +import type { SpinnerMode } from './components/Spinner.js' +import type { QuerySource } from './constants/querySource.js' +import type { SDKStatus } from './entrypoints/agentSdkTypes.js' +import type { AppState } from './state/AppState.js' +import type { + HookProgress, + PromptRequest, + PromptResponse, +} from './types/hooks.js' +import type { AgentId } from './types/ids.js' +import type { DeepImmutable } from './types/utils.js' +import type { AttributionState } from './utils/commitAttribution.js' +import type { FileHistoryState } from './utils/fileHistory.js' +import type { Theme, ThemeName } from './utils/theme.js' + +export type QueryChainTracking = { + chainId: string + depth: number +} + +export type ValidationResult = + | { result: true } + | { + result: false + message: string + errorCode: number + } + +export type SetToolJSXFn = ( + args: { + jsx: React.ReactNode | null + shouldHidePromptInput: boolean + shouldContinueAnimation?: true + showSpinner?: boolean + isLocalJSXCommand?: boolean + isImmediate?: boolean + /** Set to true to clear a local JSX command (e.g., from its onDone callback) */ + clearLocalJSX?: boolean + } | null, +) => void + +// Import tool permission types from centralized location to break import cycles +import type { ToolPermissionRulesBySource } from './types/permissions.js' + +// Re-export for backwards compatibility +export type { ToolPermissionRulesBySource } + +// Apply DeepImmutable to the imported type +export type ToolPermissionContext = DeepImmutable<{ + mode: PermissionMode + additionalWorkingDirectories: Map + alwaysAllowRules: ToolPermissionRulesBySource + alwaysDenyRules: ToolPermissionRulesBySource + alwaysAskRules: ToolPermissionRulesBySource + isBypassPermissionsModeAvailable: boolean + isAutoModeAvailable?: boolean + strippedDangerousRules?: ToolPermissionRulesBySource + /** When true, permission prompts are auto-denied (e.g., background agents that can't show UI) */ + shouldAvoidPermissionPrompts?: boolean + /** When true, automated checks (classifier, hooks) are awaited before showing the permission dialog (coordinator workers) */ + awaitAutomatedChecksBeforeDialog?: boolean + /** Stores the permission mode before model-initiated plan mode entry, so it can be restored on exit */ + prePlanMode?: PermissionMode +}> + +export const getEmptyToolPermissionContext: () => ToolPermissionContext = + () => ({ + mode: 'default', + additionalWorkingDirectories: new Map(), + alwaysAllowRules: {}, + alwaysDenyRules: {}, + alwaysAskRules: {}, + isBypassPermissionsModeAvailable: false, + }) + +export type CompactProgressEvent = + | { + type: 'hooks_start' + hookType: 'pre_compact' | 'post_compact' | 'session_start' + } + | { type: 'compact_start' } + | { type: 'compact_end' } + +export type ToolUseContext = { + options: { + commands: Command[] + debug: boolean + mainLoopModel: string + tools: Tools + verbose: boolean + thinkingConfig: ThinkingConfig + mcpClients: MCPServerConnection[] + mcpResources: Record + isNonInteractiveSession: boolean + agentDefinitions: AgentDefinitionsResult + maxBudgetUsd?: number + /** Custom system prompt that replaces the default system prompt */ + customSystemPrompt?: string + /** Additional system prompt appended after the main system prompt */ + appendSystemPrompt?: string + /** Override querySource for analytics tracking */ + querySource?: QuerySource + /** Optional callback to get the latest tools (e.g., after MCP servers connect mid-query) */ + refreshTools?: () => Tools + } + abortController: AbortController + readFileState: FileStateCache + getAppState(): AppState + setAppState(f: (prev: AppState) => AppState): void + /** + * Always-shared setAppState for session-scoped infrastructure (background + * tasks, session hooks). Unlike setAppState, which is no-op for async agents + * (see createSubagentContext), this always reaches the root store so agents + * at any nesting depth can register/clean up infrastructure that outlives + * a single turn. Only set by createSubagentContext; main-thread contexts + * fall back to setAppState. + */ + setAppStateForTasks?: (f: (prev: AppState) => AppState) => void + /** + * Optional handler for URL elicitations triggered by tool call errors (-32042). + * In print/SDK mode, this delegates to structuredIO.handleElicitation. + * In REPL mode, this is undefined and the queue-based UI path is used. + */ + handleElicitation?: ( + serverName: string, + params: ElicitRequestURLParams, + signal: AbortSignal, + ) => Promise + setToolJSX?: SetToolJSXFn + addNotification?: (notif: Notification) => void + /** Append a UI-only system message to the REPL message list. Stripped at the + * normalizeMessagesForAPI boundary — the Exclude<> makes that type-enforced. */ + appendSystemMessage?: ( + msg: Exclude, + ) => void + /** Send an OS-level notification (iTerm2, Kitty, Ghostty, bell, etc.) */ + sendOSNotification?: (opts: { + message: string + notificationType: string + }) => void + nestedMemoryAttachmentTriggers?: Set + /** + * CLAUDE.md paths already injected as nested_memory attachments this + * session. Dedup for memoryFilesToAttachments — readFileState is an LRU + * that evicts entries in busy sessions, so its .has() check alone can + * re-inject the same CLAUDE.md dozens of times. + */ + loadedNestedMemoryPaths?: Set + dynamicSkillDirTriggers?: Set + /** Skill names surfaced via skill_discovery this session. Telemetry only (feeds was_discovered). */ + discoveredSkillNames?: Set + userModified?: boolean + setInProgressToolUseIDs: (f: (prev: Set) => Set) => void + /** Only wired in interactive (REPL) contexts; SDK/QueryEngine don't set this. */ + setHasInterruptibleToolInProgress?: (v: boolean) => void + setResponseLength: (f: (prev: number) => number) => void + /** Ant-only: push a new API metrics entry for OTPS tracking. + * Called by subagent streaming when a new API request starts. */ + pushApiMetricsEntry?: (ttftMs: number) => void + setStreamMode?: (mode: SpinnerMode) => void + onCompactProgress?: (event: CompactProgressEvent) => void + setSDKStatus?: (status: SDKStatus) => void + openMessageSelector?: () => void + updateFileHistoryState: ( + updater: (prev: FileHistoryState) => FileHistoryState, + ) => void + updateAttributionState: ( + updater: (prev: AttributionState) => AttributionState, + ) => void + setConversationId?: (id: UUID) => void + agentId?: AgentId // Only set for subagents; use getSessionId() for session ID. Hooks use this to distinguish subagent calls. + agentType?: string // Subagent type name. For the main thread's --agent type, hooks fall back to getMainThreadAgentType(). + /** When true, canUseTool must always be called even when hooks auto-approve. + * Used by speculation for overlay file path rewriting. */ + requireCanUseTool?: boolean + messages: Message[] + fileReadingLimits?: { + maxTokens?: number + maxSizeBytes?: number + } + globLimits?: { + maxResults?: number + } + toolDecisions?: Map< + string, + { + source: string + decision: 'accept' | 'reject' + timestamp: number + } + > + queryTracking?: QueryChainTracking + /** Callback factory for requesting interactive prompts from the user. + * Returns a prompt callback bound to the given source name. + * Only available in interactive (REPL) contexts. */ + requestPrompt?: ( + sourceName: string, + toolInputSummary?: string | null, + ) => (request: PromptRequest) => Promise + toolUseId?: string + criticalSystemReminder_EXPERIMENTAL?: string + /** When true, preserve toolUseResult on messages even for subagents. + * Used by in-process teammates whose transcripts are viewable by the user. */ + preserveToolUseResults?: boolean + /** Local denial tracking state for async subagents whose setAppState is a + * no-op. Without this, the denial counter never accumulates and the + * fallback-to-prompting threshold is never reached. Mutable — the + * permissions code updates it in place. */ + localDenialTracking?: DenialTrackingState + /** + * Per-conversation-thread content replacement state for the tool result + * budget. When present, query.ts applies the aggregate tool result budget. + * Main thread: REPL provisions once (never resets — stale UUID keys + * are inert). Subagents: createSubagentContext clones the parent's state + * by default (cache-sharing forks need identical decisions), or + * resumeAgentBackground threads one reconstructed from sidechain records. + */ + contentReplacementState?: ContentReplacementState + /** + * Parent's rendered system prompt bytes, frozen at turn start. + * Used by fork subagents to share the parent's prompt cache — re-calling + * getSystemPrompt() at fork-spawn time can diverge (GrowthBook cold→warm) + * and bust the cache. See forkSubagent.ts. + */ + renderedSystemPrompt?: SystemPrompt +} + +// Re-export ToolProgressData from centralized location +export type { ToolProgressData } + +export type Progress = ToolProgressData | HookProgress + +export type ToolProgress

= { + toolUseID: string + data: P +} + +export function filterToolProgressMessages( + progressMessagesForMessage: ProgressMessage[], +): ProgressMessage[] { + return progressMessagesForMessage.filter( + (msg): msg is ProgressMessage => + msg.data?.type !== 'hook_progress', + ) +} + +export type ToolResult = { + data: T + newMessages?: ( + | UserMessage + | AssistantMessage + | AttachmentMessage + | SystemMessage + )[] + // contextModifier is only honored for tools that aren't concurrency safe. + contextModifier?: (context: ToolUseContext) => ToolUseContext + /** MCP protocol metadata (structuredContent, _meta) to pass through to SDK consumers */ + mcpMeta?: { + _meta?: Record + structuredContent?: Record + } +} + +export type ToolCallProgress

= ( + progress: ToolProgress

, +) => void + +// Type for any schema that outputs an object with string keys +export type AnyObject = z.ZodType<{ [key: string]: unknown }> + +/** + * Checks if a tool matches the given name (primary name or alias). + */ +export function toolMatchesName( + tool: { name: string; aliases?: string[] }, + name: string, +): boolean { + return tool.name === name || (tool.aliases?.includes(name) ?? false) +} + +/** + * Finds a tool by name or alias from a list of tools. + */ +export function findToolByName(tools: Tools, name: string): Tool | undefined { + return tools.find(t => toolMatchesName(t, name)) +} + +export type Tool< + Input extends AnyObject = AnyObject, + Output = unknown, + P extends ToolProgressData = ToolProgressData, +> = { + /** + * Optional aliases for backwards compatibility when a tool is renamed. + * The tool can be looked up by any of these names in addition to its primary name. + */ + aliases?: string[] + /** + * One-line capability phrase used by ToolSearch for keyword matching. + * Helps the model find this tool via keyword search when it's deferred. + * 3–10 words, no trailing period. + * Prefer terms not already in the tool name (e.g. 'jupyter' for NotebookEdit). + */ + searchHint?: string + call( + args: z.infer, + context: ToolUseContext, + canUseTool: CanUseToolFn, + parentMessage: AssistantMessage, + onProgress?: ToolCallProgress

, + ): Promise> + description( + input: z.infer, + options: { + isNonInteractiveSession: boolean + toolPermissionContext: ToolPermissionContext + tools: Tools + }, + ): Promise + readonly inputSchema: Input + // Type for MCP tools that can specify their input schema directly in JSON Schema format + // rather than converting from Zod schema + readonly inputJSONSchema?: ToolInputJSONSchema + // Optional because TungstenTool doesn't define this. TODO: Make it required. + // When we do that, we can also go through and make this a bit more type-safe. + outputSchema?: z.ZodType + inputsEquivalent?(a: z.infer, b: z.infer): boolean + isConcurrencySafe(input: z.infer): boolean + isEnabled(): boolean + isReadOnly(input: z.infer): boolean + /** Defaults to false. Only set when the tool performs irreversible operations (delete, overwrite, send). */ + isDestructive?(input: z.infer): boolean + /** + * What should happen when the user submits a new message while this tool + * is running. + * + * - `'cancel'` — stop the tool and discard its result + * - `'block'` — keep running; the new message waits + * + * Defaults to `'block'` when not implemented. + */ + interruptBehavior?(): 'cancel' | 'block' + /** + * Returns information about whether this tool use is a search or read operation + * that should be collapsed into a condensed display in the UI. Examples include + * file searching (Grep, Glob), file reading (Read), and bash commands like find, + * grep, wc, etc. + * + * Returns an object indicating whether the operation is a search or read operation: + * - `isSearch: true` for search operations (grep, find, glob patterns) + * - `isRead: true` for read operations (cat, head, tail, file read) + * - `isList: true` for directory-listing operations (ls, tree, du) + * - All can be false if the operation shouldn't be collapsed + */ + isSearchOrReadCommand?(input: z.infer): { + isSearch: boolean + isRead: boolean + isList?: boolean + } + isOpenWorld?(input: z.infer): boolean + requiresUserInteraction?(): boolean + isMcp?: boolean + isLsp?: boolean + /** + * When true, this tool is deferred (sent with defer_loading: true) and requires + * ToolSearch to be used before it can be called. + */ + readonly shouldDefer?: boolean + /** + * When true, this tool is never deferred — its full schema appears in the + * initial prompt even when ToolSearch is enabled. For MCP tools, set via + * `_meta['anthropic/alwaysLoad']`. Use for tools the model must see on + * turn 1 without a ToolSearch round-trip. + */ + readonly alwaysLoad?: boolean + /** + * For MCP tools: the server and tool names as received from the MCP server (unnormalized). + * Present on all MCP tools regardless of whether `name` is prefixed (mcp__server__tool) + * or unprefixed (CLAUDE_AGENT_SDK_MCP_NO_PREFIX mode). + */ + mcpInfo?: { serverName: string; toolName: string } + readonly name: string + /** + * Maximum size in characters for tool result before it gets persisted to disk. + * When exceeded, the result is saved to a file and Claude receives a preview + * with the file path instead of the full content. + * + * Set to Infinity for tools whose output must never be persisted (e.g. Read, + * where persisting creates a circular Read→file→Read loop and the tool + * already self-bounds via its own limits). + */ + maxResultSizeChars: number + /** + * When true, enables strict mode for this tool, which causes the API to + * more strictly adhere to tool instructions and parameter schemas. + * Only applied when the tengu_tool_pear is enabled. + */ + readonly strict?: boolean + + /** + * Called on copies of tool_use input before observers see it (SDK stream, + * transcript, canUseTool, PreToolUse/PostToolUse hooks). Mutate in place + * to add legacy/derived fields. Must be idempotent. The original API-bound + * input is never mutated (preserves prompt cache). Not re-applied when a + * hook/permission returns a fresh updatedInput — those own their shape. + */ + backfillObservableInput?(input: Record): void + + /** + * Determines if this tool is allowed to run with this input in the current context. + * It informs the model of why the tool use failed, and does not directly display any UI. + * @param input + * @param context + */ + validateInput?( + input: z.infer, + context: ToolUseContext, + ): Promise + + /** + * Determines if the user is asked for permission. Only called after validateInput() passes. + * General permission logic is in permissions.ts. This method contains tool-specific logic. + * @param input + * @param context + */ + checkPermissions( + input: z.infer, + context: ToolUseContext, + ): Promise + + // Optional method for tools that operate on a file path + getPath?(input: z.infer): string + + /** + * Prepare a matcher for hook `if` conditions (permission-rule patterns like + * "git *" from "Bash(git *)"). Called once per hook-input pair; any + * expensive parsing happens here. Returns a closure that is called per + * hook pattern. If not implemented, only tool-name-level matching works. + */ + preparePermissionMatcher?( + input: z.infer, + ): Promise<(pattern: string) => boolean> + + prompt(options: { + getToolPermissionContext: () => Promise + tools: Tools + agents: AgentDefinition[] + allowedAgentTypes?: string[] + }): Promise + userFacingName(input: Partial> | undefined): string + userFacingNameBackgroundColor?( + input: Partial> | undefined, + ): keyof Theme | undefined + /** + * Transparent wrappers (e.g. REPL) delegate all rendering to their progress + * handler, which emits native-looking blocks for each inner tool call. + * The wrapper itself shows nothing. + */ + isTransparentWrapper?(): boolean + /** + * Returns a short string summary of this tool use for display in compact views. + * @param input The tool input + * @returns A short string summary, or null to not display + */ + getToolUseSummary?(input: Partial> | undefined): string | null + /** + * Returns a human-readable present-tense activity description for spinner display. + * Example: "Reading src/foo.ts", "Running bun test", "Searching for pattern" + * @param input The tool input + * @returns Activity description string, or null to fall back to tool name + */ + getActivityDescription?( + input: Partial> | undefined, + ): string | null + /** + * Returns a compact representation of this tool use for the auto-mode + * security classifier. Examples: `ls -la` for Bash, `/tmp/x: new content` + * for Edit. Return '' to skip this tool in the classifier transcript + * (e.g. tools with no security relevance). May return an object to avoid + * double-encoding when the caller JSON-wraps the value. + */ + toAutoClassifierInput(input: z.infer): unknown + mapToolResultToToolResultBlockParam( + content: Output, + toolUseID: string, + ): ToolResultBlockParam + /** + * Optional. When omitted, the tool result renders nothing (same as returning + * null). Omit for tools whose results are surfaced elsewhere (e.g., TodoWrite + * updates the todo panel, not the transcript). + */ + renderToolResultMessage?( + content: Output, + progressMessagesForMessage: ProgressMessage

[], + options: { + style?: 'condensed' + theme: ThemeName + tools: Tools + verbose: boolean + isTranscriptMode?: boolean + isBriefOnly?: boolean + /** Original tool_use input, when available. Useful for compact result + * summaries that reference what was requested (e.g. "Sent to #foo"). */ + input?: unknown + }, + ): React.ReactNode + /** + * Flattened text of what renderToolResultMessage shows IN TRANSCRIPT + * MODE (verbose=true, isTranscriptMode=true). For transcript search + * indexing: the index counts occurrences in this string, the highlight + * overlay scans the actual screen buffer. For count ≡ highlight, this + * must return the text that ends up visible — not the model-facing + * serialization from mapToolResultToToolResultBlockParam (which adds + * system-reminders, persisted-output wrappers). + * + * Chrome can be skipped (under-count is fine). "Found 3 files in 12ms" + * isn't worth indexing. Phantoms are not fine — text that's claimed + * here but doesn't render is a count≠highlight bug. + * + * Optional: omitted → field-name heuristic in transcriptSearch.ts. + * Drift caught by test/utils/transcriptSearch.renderFidelity.test.tsx + * which renders sample outputs and flags text that's indexed-but-not- + * rendered (phantom) or rendered-but-not-indexed (under-count warning). + */ + extractSearchText?(out: Output): string + /** + * Render the tool use message. Note that `input` is partial because we render + * the message as soon as possible, possibly before tool parameters have fully + * streamed in. + */ + renderToolUseMessage( + input: Partial>, + options: { theme: ThemeName; verbose: boolean; commands?: Command[] }, + ): React.ReactNode + /** + * Returns true when the non-verbose rendering of this output is truncated + * (i.e., clicking to expand would reveal more content). Gates + * click-to-expand in fullscreen — only messages where verbose actually + * shows more get a hover/click affordance. Unset means never truncated. + */ + isResultTruncated?(output: Output): boolean + /** + * Renders an optional tag to display after the tool use message. + * Used for additional metadata like timeout, model, resume ID, etc. + * Returns null to not display anything. + */ + renderToolUseTag?(input: Partial>): React.ReactNode + /** + * Optional. When omitted, no progress UI is shown while the tool runs. + */ + renderToolUseProgressMessage?( + progressMessagesForMessage: ProgressMessage

[], + options: { + tools: Tools + verbose: boolean + terminalSize?: { columns: number; rows: number } + inProgressToolCallCount?: number + isTranscriptMode?: boolean + }, + ): React.ReactNode + renderToolUseQueuedMessage?(): React.ReactNode + /** + * Optional. When omitted, falls back to . + * Only define this for tools that need custom rejection UI (e.g., file edits + * that show the rejected diff). + */ + renderToolUseRejectedMessage?( + input: z.infer, + options: { + columns: number + messages: Message[] + style?: 'condensed' + theme: ThemeName + tools: Tools + verbose: boolean + progressMessagesForMessage: ProgressMessage

[] + isTranscriptMode?: boolean + }, + ): React.ReactNode + /** + * Optional. When omitted, falls back to . + * Only define this for tools that need custom error UI (e.g., search tools + * that show "File not found" instead of the raw error). + */ + renderToolUseErrorMessage?( + result: ToolResultBlockParam['content'], + options: { + progressMessagesForMessage: ProgressMessage

[] + tools: Tools + verbose: boolean + isTranscriptMode?: boolean + }, + ): React.ReactNode + + /** + * Renders multiple parallel instances of this tool as a group. + * @returns React node to render, or null to fall back to individual rendering + */ + /** + * Renders multiple tool uses as a group (non-verbose mode only). + * In verbose mode, individual tool uses render at their original positions. + * @returns React node to render, or null to fall back to individual rendering + */ + renderGroupedToolUse?( + toolUses: Array<{ + param: ToolUseBlockParam + isResolved: boolean + isError: boolean + isInProgress: boolean + progressMessages: ProgressMessage

[] + result?: { + param: ToolResultBlockParam + output: unknown + } + }>, + options: { + shouldAnimate: boolean + tools: Tools + }, + ): React.ReactNode | null +} + +/** + * A collection of tools. Use this type instead of `Tool[]` to make it easier + * to track where tool sets are assembled, passed, and filtered across the codebase. + */ +export type Tools = readonly Tool[] + +/** + * Methods that `buildTool` supplies a default for. A `ToolDef` may omit these; + * the resulting `Tool` always has them. + */ +type DefaultableToolKeys = + | 'isEnabled' + | 'isConcurrencySafe' + | 'isReadOnly' + | 'isDestructive' + | 'checkPermissions' + | 'toAutoClassifierInput' + | 'userFacingName' + +/** + * Tool definition accepted by `buildTool`. Same shape as `Tool` but with the + * defaultable methods optional — `buildTool` fills them in so callers always + * see a complete `Tool`. + */ +export type ToolDef< + Input extends AnyObject = AnyObject, + Output = unknown, + P extends ToolProgressData = ToolProgressData, +> = Omit, DefaultableToolKeys> & + Partial, DefaultableToolKeys>> + +/** + * Type-level spread mirroring `{ ...TOOL_DEFAULTS, ...def }`. For each + * defaultable key: if D provides it (required), D's type wins; if D omits + * it or has it optional (inherited from Partial<> in the constraint), the + * default fills in. All other keys come from D verbatim — preserving arity, + * optional presence, and literal types exactly as `satisfies Tool` did. + */ +type BuiltTool = Omit & { + [K in DefaultableToolKeys]-?: K extends keyof D + ? undefined extends D[K] + ? ToolDefaults[K] + : D[K] + : ToolDefaults[K] +} + +/** + * Build a complete `Tool` from a partial definition, filling in safe defaults + * for the commonly-stubbed methods. All tool exports should go through this so + * that defaults live in one place and callers never need `?.() ?? default`. + * + * Defaults (fail-closed where it matters): + * - `isEnabled` → `true` + * - `isConcurrencySafe` → `false` (assume not safe) + * - `isReadOnly` → `false` (assume writes) + * - `isDestructive` → `false` + * - `checkPermissions` → `{ behavior: 'allow', updatedInput }` (defer to general permission system) + * - `toAutoClassifierInput` → `''` (skip classifier — security-relevant tools must override) + * - `userFacingName` → `name` + */ +const TOOL_DEFAULTS = { + isEnabled: () => true, + isConcurrencySafe: (_input?: unknown) => false, + isReadOnly: (_input?: unknown) => false, + isDestructive: (_input?: unknown) => false, + checkPermissions: ( + input: { [key: string]: unknown }, + _ctx?: ToolUseContext, + ): Promise => + Promise.resolve({ behavior: 'allow', updatedInput: input }), + toAutoClassifierInput: (_input?: unknown) => '', + userFacingName: (_input?: unknown) => '', +} + +// The defaults type is the ACTUAL shape of TOOL_DEFAULTS (optional params so +// both 0-arg and full-arg call sites type-check — stubs varied in arity and +// tests relied on that), not the interface's strict signatures. +type ToolDefaults = typeof TOOL_DEFAULTS + +// D infers the concrete object-literal type from the call site. The +// constraint provides contextual typing for method parameters; `any` in +// constraint position is structural and never leaks into the return type. +// BuiltTool mirrors runtime `{...TOOL_DEFAULTS, ...def}` at the type level. +// eslint-disable-next-line @typescript-eslint/no-explicit-any +type AnyToolDef = ToolDef + +export function buildTool(def: D): BuiltTool { + // The runtime spread is straightforward; the `as` bridges the gap between + // the structural-any constraint and the precise BuiltTool return. The + // type semantics are proven by the 0-error typecheck across all 60+ tools. + return { + ...TOOL_DEFAULTS, + userFacingName: () => def.name, + ...def, + } as BuiltTool +} diff --git a/src/assistant/sessionHistory.ts b/src/assistant/sessionHistory.ts new file mode 100644 index 0000000..9e1ddc5 --- /dev/null +++ b/src/assistant/sessionHistory.ts @@ -0,0 +1,87 @@ +import axios from 'axios' +import { getOauthConfig } from '../constants/oauth.js' +import type { SDKMessage } from '../entrypoints/agentSdkTypes.js' +import { logForDebugging } from '../utils/debug.js' +import { getOAuthHeaders, prepareApiRequest } from '../utils/teleport/api.js' + +export const HISTORY_PAGE_SIZE = 100 + +export type HistoryPage = { + /** Chronological order within the page. */ + events: SDKMessage[] + /** Oldest event ID in this page → before_id cursor for next-older page. */ + firstId: string | null + /** true = older events exist. */ + hasMore: boolean +} + +type SessionEventsResponse = { + data: SDKMessage[] + has_more: boolean + first_id: string | null + last_id: string | null +} + +export type HistoryAuthCtx = { + baseUrl: string + headers: Record +} + +/** Prepare auth + headers + base URL once, reuse across pages. */ +export async function createHistoryAuthCtx( + sessionId: string, +): Promise { + const { accessToken, orgUUID } = await prepareApiRequest() + return { + baseUrl: `${getOauthConfig().BASE_API_URL}/v1/sessions/${sessionId}/events`, + headers: { + ...getOAuthHeaders(accessToken), + 'anthropic-beta': 'ccr-byoc-2025-07-29', + 'x-organization-uuid': orgUUID, + }, + } +} + +async function fetchPage( + ctx: HistoryAuthCtx, + params: Record, + label: string, +): Promise { + const resp = await axios + .get(ctx.baseUrl, { + headers: ctx.headers, + params, + timeout: 15000, + validateStatus: () => true, + }) + .catch(() => null) + if (!resp || resp.status !== 200) { + logForDebugging(`[${label}] HTTP ${resp?.status ?? 'error'}`) + return null + } + return { + events: Array.isArray(resp.data.data) ? resp.data.data : [], + firstId: resp.data.first_id, + hasMore: resp.data.has_more, + } +} + +/** + * Newest page: last `limit` events, chronological, via anchor_to_latest. + * has_more=true means older events exist. + */ +export async function fetchLatestEvents( + ctx: HistoryAuthCtx, + limit = HISTORY_PAGE_SIZE, +): Promise { + return fetchPage(ctx, { limit, anchor_to_latest: true }, 'fetchLatestEvents') +} + +/** Older page: events immediately before `beforeId` cursor. */ +export async function fetchOlderEvents( + ctx: HistoryAuthCtx, + beforeId: string, + limit = HISTORY_PAGE_SIZE, +): Promise { + return fetchPage(ctx, { limit, before_id: beforeId }, 'fetchOlderEvents') +} diff --git a/src/bootstrap/state.ts b/src/bootstrap/state.ts new file mode 100644 index 0000000..d7199e5 --- /dev/null +++ b/src/bootstrap/state.ts @@ -0,0 +1,1758 @@ +import type { BetaMessageStreamParams } from '@anthropic-ai/sdk/resources/beta/messages/messages.mjs' +import type { Attributes, Meter, MetricOptions } from '@opentelemetry/api' +import type { logs } from '@opentelemetry/api-logs' +import type { LoggerProvider } from '@opentelemetry/sdk-logs' +import type { MeterProvider } from '@opentelemetry/sdk-metrics' +import type { BasicTracerProvider } from '@opentelemetry/sdk-trace-base' +import { realpathSync } from 'fs' +import sumBy from 'lodash-es/sumBy.js' +import { cwd } from 'process' +import type { HookEvent, ModelUsage } from 'src/entrypoints/agentSdkTypes.js' +import type { AgentColorName } from 'src/tools/AgentTool/agentColorManager.js' +import type { HookCallbackMatcher } from 'src/types/hooks.js' +// Indirection for browser-sdk build (package.json "browser" field swaps +// crypto.ts for crypto.browser.ts). Pure leaf re-export of node:crypto — +// zero circular-dep risk. Path-alias import bypasses bootstrap-isolation +// (rule only checks ./ and / prefixes); explicit disable documents intent. +// eslint-disable-next-line custom-rules/bootstrap-isolation +import { randomUUID } from 'src/utils/crypto.js' +import type { ModelSetting } from 'src/utils/model/model.js' +import type { ModelStrings } from 'src/utils/model/modelStrings.js' +import type { SettingSource } from 'src/utils/settings/constants.js' +import { resetSettingsCache } from 'src/utils/settings/settingsCache.js' +import type { PluginHookMatcher } from 'src/utils/settings/types.js' +import { createSignal } from 'src/utils/signal.js' + +// Union type for registered hooks - can be SDK callbacks or native plugin hooks +type RegisteredHookMatcher = HookCallbackMatcher | PluginHookMatcher + +import type { SessionId } from 'src/types/ids.js' + +// DO NOT ADD MORE STATE HERE - BE JUDICIOUS WITH GLOBAL STATE + +// dev: true on entries that came via --dangerously-load-development-channels. +// The allowlist gate checks this per-entry (not the session-wide +// hasDevChannels bit) so passing both flags doesn't let the dev dialog's +// acceptance leak allowlist-bypass to the --channels entries. +export type ChannelEntry = + | { kind: 'plugin'; name: string; marketplace: string; dev?: boolean } + | { kind: 'server'; name: string; dev?: boolean } + +export type AttributedCounter = { + add(value: number, additionalAttributes?: Attributes): void +} + +type State = { + originalCwd: string + // Stable project root - set once at startup (including by --worktree flag), + // never updated by mid-session EnterWorktreeTool. + // Use for project identity (history, skills, sessions) not file operations. + projectRoot: string + totalCostUSD: number + totalAPIDuration: number + totalAPIDurationWithoutRetries: number + totalToolDuration: number + turnHookDurationMs: number + turnToolDurationMs: number + turnClassifierDurationMs: number + turnToolCount: number + turnHookCount: number + turnClassifierCount: number + startTime: number + lastInteractionTime: number + totalLinesAdded: number + totalLinesRemoved: number + hasUnknownModelCost: boolean + cwd: string + modelUsage: { [modelName: string]: ModelUsage } + mainLoopModelOverride: ModelSetting | undefined + initialMainLoopModel: ModelSetting + modelStrings: ModelStrings | null + isInteractive: boolean + kairosActive: boolean + // When true, ensureToolResultPairing throws on mismatch instead of + // repairing with synthetic placeholders. HFI opts in at startup so + // trajectories fail fast rather than conditioning the model on fake + // tool_results. + strictToolResultPairing: boolean + sdkAgentProgressSummariesEnabled: boolean + userMsgOptIn: boolean + clientType: string + sessionSource: string | undefined + questionPreviewFormat: 'markdown' | 'html' | undefined + flagSettingsPath: string | undefined + flagSettingsInline: Record | null + allowedSettingSources: SettingSource[] + sessionIngressToken: string | null | undefined + oauthTokenFromFd: string | null | undefined + apiKeyFromFd: string | null | undefined + // Telemetry state + meter: Meter | null + sessionCounter: AttributedCounter | null + locCounter: AttributedCounter | null + prCounter: AttributedCounter | null + commitCounter: AttributedCounter | null + costCounter: AttributedCounter | null + tokenCounter: AttributedCounter | null + codeEditToolDecisionCounter: AttributedCounter | null + activeTimeCounter: AttributedCounter | null + statsStore: { observe(name: string, value: number): void } | null + sessionId: SessionId + // Parent session ID for tracking session lineage (e.g., plan mode -> implementation) + parentSessionId: SessionId | undefined + // Logger state + loggerProvider: LoggerProvider | null + eventLogger: ReturnType | null + // Meter provider state + meterProvider: MeterProvider | null + // Tracer provider state + tracerProvider: BasicTracerProvider | null + // Agent color state + agentColorMap: Map + agentColorIndex: number + // Last API request for bug reports + lastAPIRequest: Omit | null + // Messages from the last API request (ant-only; reference, not clone). + // Captures the exact post-compaction, CLAUDE.md-injected message set sent + // to the API so /share's serialized_conversation.json reflects reality. + lastAPIRequestMessages: BetaMessageStreamParams['messages'] | null + // Last auto-mode classifier request(s) for /share transcript + lastClassifierRequests: unknown[] | null + // CLAUDE.md content cached by context.ts for the auto-mode classifier. + // Breaks the yoloClassifier → claudemd → filesystem → permissions cycle. + cachedClaudeMdContent: string | null + // In-memory error log for recent errors + inMemoryErrorLog: Array<{ error: string; timestamp: string }> + // Session-only plugins from --plugin-dir flag + inlinePlugins: Array + // Explicit --chrome / --no-chrome flag value (undefined = not set on CLI) + chromeFlagOverride: boolean | undefined + // Use cowork_plugins directory instead of plugins (--cowork flag or env var) + useCoworkPlugins: boolean + // Session-only bypass permissions mode flag (not persisted) + sessionBypassPermissionsMode: boolean + // Session-only flag gating the .claude/scheduled_tasks.json watcher + // (useScheduledTasks). Set by cronScheduler.start() when the JSON has + // entries, or by CronCreateTool. Not persisted. + scheduledTasksEnabled: boolean + // Session-only cron tasks created via CronCreate with durable: false. + // Fire on schedule like file-backed tasks but are never written to + // .claude/scheduled_tasks.json — they die with the process. Typed via + // SessionCronTask below (not importing from cronTasks.ts keeps + // bootstrap a leaf of the import DAG). + sessionCronTasks: SessionCronTask[] + // Teams created this session via TeamCreate. cleanupSessionTeams() + // removes these on gracefulShutdown so subagent-created teams don't + // persist on disk forever (gh-32730). TeamDelete removes entries to + // avoid double-cleanup. Lives here (not teamHelpers.ts) so + // resetStateForTests() clears it between tests. + sessionCreatedTeams: Set + // Session-only trust flag for home directory (not persisted to disk) + // When running from home dir, trust dialog is shown but not saved to disk. + // This flag allows features requiring trust to work during the session. + sessionTrustAccepted: boolean + // Session-only flag to disable session persistence to disk + sessionPersistenceDisabled: boolean + // Track if user has exited plan mode in this session (for re-entry guidance) + hasExitedPlanMode: boolean + // Track if we need to show the plan mode exit attachment (one-time notification) + needsPlanModeExitAttachment: boolean + // Track if we need to show the auto mode exit attachment (one-time notification) + needsAutoModeExitAttachment: boolean + // Track if LSP plugin recommendation has been shown this session (only show once) + lspRecommendationShownThisSession: boolean + // SDK init event state - jsonSchema for structured output + initJsonSchema: Record | null + // Registered hooks - SDK callbacks and plugin native hooks + registeredHooks: Partial> | null + // Cache for plan slugs: sessionId -> wordSlug + planSlugCache: Map + // Track teleported session for reliability logging + teleportedSessionInfo: { + isTeleported: boolean + hasLoggedFirstMessage: boolean + sessionId: string | null + } | null + // Track invoked skills for preservation across compaction + // Keys are composite: `${agentId ?? ''}:${skillName}` to prevent cross-agent overwrites + invokedSkills: Map< + string, + { + skillName: string + skillPath: string + content: string + invokedAt: number + agentId: string | null + } + > + // Track slow operations for dev bar display (ant-only) + slowOperations: Array<{ + operation: string + durationMs: number + timestamp: number + }> + // SDK-provided betas (e.g., context-1m-2025-08-07) + sdkBetas: string[] | undefined + // Main thread agent type (from --agent flag or settings) + mainThreadAgentType: string | undefined + // Remote mode (--remote flag) + isRemoteMode: boolean + // Direct connect server URL (for display in header) + directConnectServerUrl: string | undefined + // System prompt section cache state + systemPromptSectionCache: Map + // Last date emitted to the model (for detecting midnight date changes) + lastEmittedDate: string | null + // Additional directories from --add-dir flag (for CLAUDE.md loading) + additionalDirectoriesForClaudeMd: string[] + // Channel server allowlist from --channels flag (servers whose channel + // notifications should register this session). Parsed once in main.tsx — + // the tag decides trust model: 'plugin' → marketplace verification + + // allowlist, 'server' → allowlist always fails (schema is plugin-only). + // Either kind needs entry.dev to bypass allowlist. + allowedChannels: ChannelEntry[] + // True if any entry in allowedChannels came from + // --dangerously-load-development-channels (so ChannelsNotice can name the + // right flag in policy-blocked messages) + hasDevChannels: boolean + // Dir containing the session's `.jsonl`; null = derive from originalCwd. + sessionProjectDir: string | null + // Cached prompt cache 1h TTL allowlist from GrowthBook (session-stable) + promptCache1hAllowlist: string[] | null + // Cached 1h TTL user eligibility (session-stable). Latched on first + // evaluation so mid-session overage flips don't change the cache_control + // TTL, which would bust the server-side prompt cache. + promptCache1hEligible: boolean | null + // Sticky-on latch for AFK_MODE_BETA_HEADER. Once auto mode is first + // activated, keep sending the header for the rest of the session so + // Shift+Tab toggles don't bust the ~50-70K token prompt cache. + afkModeHeaderLatched: boolean | null + // Sticky-on latch for FAST_MODE_BETA_HEADER. Once fast mode is first + // enabled, keep sending the header so cooldown enter/exit doesn't + // double-bust the prompt cache. The `speed` body param stays dynamic. + fastModeHeaderLatched: boolean | null + // Sticky-on latch for the cache-editing beta header. Once cached + // microcompact is first enabled, keep sending the header so mid-session + // GrowthBook/settings toggles don't bust the prompt cache. + cacheEditingHeaderLatched: boolean | null + // Sticky-on latch for clearing thinking from prior tool loops. Triggered + // when >1h since last API call (confirmed cache miss — no cache-hit + // benefit to keeping thinking). Once latched, stays on so the newly-warmed + // thinking-cleared cache isn't busted by flipping back to keep:'all'. + thinkingClearLatched: boolean | null + // Current prompt ID (UUID) correlating a user prompt with subsequent OTel events + promptId: string | null + // Last API requestId for the main conversation chain (not subagents). + // Updated after each successful API response for main-session queries. + // Read at shutdown to send cache eviction hints to inference. + lastMainRequestId: string | undefined + // Timestamp (Date.now()) of the last successful API call completion. + // Used to compute timeSinceLastApiCallMs in tengu_api_success for + // correlating cache misses with idle time (cache TTL is ~5min). + lastApiCompletionTimestamp: number | null + // Set to true after compaction (auto or manual /compact). Consumed by + // logAPISuccess to tag the first post-compaction API call so we can + // distinguish compaction-induced cache misses from TTL expiry. + pendingPostCompaction: boolean +} + +// ALSO HERE - THINK THRICE BEFORE MODIFYING +function getInitialState(): State { + // Resolve symlinks in cwd to match behavior of shell.ts setCwd + // This ensures consistency with how paths are sanitized for session storage + let resolvedCwd = '' + if ( + typeof process !== 'undefined' && + typeof process.cwd === 'function' && + typeof realpathSync === 'function' + ) { + const rawCwd = cwd() + try { + resolvedCwd = realpathSync(rawCwd).normalize('NFC') + } catch { + // File Provider EPERM on CloudStorage mounts (lstat per path component). + resolvedCwd = rawCwd.normalize('NFC') + } + } + const state: State = { + originalCwd: resolvedCwd, + projectRoot: resolvedCwd, + totalCostUSD: 0, + totalAPIDuration: 0, + totalAPIDurationWithoutRetries: 0, + totalToolDuration: 0, + turnHookDurationMs: 0, + turnToolDurationMs: 0, + turnClassifierDurationMs: 0, + turnToolCount: 0, + turnHookCount: 0, + turnClassifierCount: 0, + startTime: Date.now(), + lastInteractionTime: Date.now(), + totalLinesAdded: 0, + totalLinesRemoved: 0, + hasUnknownModelCost: false, + cwd: resolvedCwd, + modelUsage: {}, + mainLoopModelOverride: undefined, + initialMainLoopModel: null, + modelStrings: null, + isInteractive: false, + kairosActive: false, + strictToolResultPairing: false, + sdkAgentProgressSummariesEnabled: false, + userMsgOptIn: false, + clientType: 'cli', + sessionSource: undefined, + questionPreviewFormat: undefined, + sessionIngressToken: undefined, + oauthTokenFromFd: undefined, + apiKeyFromFd: undefined, + flagSettingsPath: undefined, + flagSettingsInline: null, + allowedSettingSources: [ + 'userSettings', + 'projectSettings', + 'localSettings', + 'flagSettings', + 'policySettings', + ], + // Telemetry state + meter: null, + sessionCounter: null, + locCounter: null, + prCounter: null, + commitCounter: null, + costCounter: null, + tokenCounter: null, + codeEditToolDecisionCounter: null, + activeTimeCounter: null, + statsStore: null, + sessionId: randomUUID() as SessionId, + parentSessionId: undefined, + // Logger state + loggerProvider: null, + eventLogger: null, + // Meter provider state + meterProvider: null, + tracerProvider: null, + // Agent color state + agentColorMap: new Map(), + agentColorIndex: 0, + // Last API request for bug reports + lastAPIRequest: null, + lastAPIRequestMessages: null, + // Last auto-mode classifier request(s) for /share transcript + lastClassifierRequests: null, + cachedClaudeMdContent: null, + // In-memory error log for recent errors + inMemoryErrorLog: [], + // Session-only plugins from --plugin-dir flag + inlinePlugins: [], + // Explicit --chrome / --no-chrome flag value (undefined = not set on CLI) + chromeFlagOverride: undefined, + // Use cowork_plugins directory instead of plugins + useCoworkPlugins: false, + // Session-only bypass permissions mode flag (not persisted) + sessionBypassPermissionsMode: false, + // Scheduled tasks disabled until flag or dialog enables them + scheduledTasksEnabled: false, + sessionCronTasks: [], + sessionCreatedTeams: new Set(), + // Session-only trust flag (not persisted to disk) + sessionTrustAccepted: false, + // Session-only flag to disable session persistence to disk + sessionPersistenceDisabled: false, + // Track if user has exited plan mode in this session + hasExitedPlanMode: false, + // Track if we need to show the plan mode exit attachment + needsPlanModeExitAttachment: false, + // Track if we need to show the auto mode exit attachment + needsAutoModeExitAttachment: false, + // Track if LSP plugin recommendation has been shown this session + lspRecommendationShownThisSession: false, + // SDK init event state + initJsonSchema: null, + registeredHooks: null, + // Cache for plan slugs + planSlugCache: new Map(), + // Track teleported session for reliability logging + teleportedSessionInfo: null, + // Track invoked skills for preservation across compaction + invokedSkills: new Map(), + // Track slow operations for dev bar display + slowOperations: [], + // SDK-provided betas + sdkBetas: undefined, + // Main thread agent type + mainThreadAgentType: undefined, + // Remote mode + isRemoteMode: false, + ...(process.env.USER_TYPE === 'ant' + ? { + replBridgeActive: false, + } + : {}), + // Direct connect server URL + directConnectServerUrl: undefined, + // System prompt section cache state + systemPromptSectionCache: new Map(), + // Last date emitted to the model + lastEmittedDate: null, + // Additional directories from --add-dir flag (for CLAUDE.md loading) + additionalDirectoriesForClaudeMd: [], + // Channel server allowlist from --channels flag + allowedChannels: [], + hasDevChannels: false, + // Session project dir (null = derive from originalCwd) + sessionProjectDir: null, + // Prompt cache 1h allowlist (null = not yet fetched from GrowthBook) + promptCache1hAllowlist: null, + // Prompt cache 1h eligibility (null = not yet evaluated) + promptCache1hEligible: null, + // Beta header latches (null = not yet triggered) + afkModeHeaderLatched: null, + fastModeHeaderLatched: null, + cacheEditingHeaderLatched: null, + thinkingClearLatched: null, + // Current prompt ID + promptId: null, + lastMainRequestId: undefined, + lastApiCompletionTimestamp: null, + pendingPostCompaction: false, + } + + return state +} + +// AND ESPECIALLY HERE +const STATE: State = getInitialState() + +export function getSessionId(): SessionId { + return STATE.sessionId +} + +export function regenerateSessionId( + options: { setCurrentAsParent?: boolean } = {}, +): SessionId { + if (options.setCurrentAsParent) { + STATE.parentSessionId = STATE.sessionId + } + // Drop the outgoing session's plan-slug entry so the Map doesn't + // accumulate stale keys. Callers that need to carry the slug across + // (REPL.tsx clearContext) read it before calling clearConversation. + STATE.planSlugCache.delete(STATE.sessionId) + // Regenerated sessions live in the current project: reset projectDir to + // null so getTranscriptPath() derives from originalCwd. + STATE.sessionId = randomUUID() as SessionId + STATE.sessionProjectDir = null + return STATE.sessionId +} + +export function getParentSessionId(): SessionId | undefined { + return STATE.parentSessionId +} + +/** + * Atomically switch the active session. `sessionId` and `sessionProjectDir` + * always change together — there is no separate setter for either, so they + * cannot drift out of sync (CC-34). + * + * @param projectDir — directory containing `.jsonl`. Omit (or + * pass `null`) for sessions in the current project — the path will derive + * from originalCwd at read time. Pass `dirname(transcriptPath)` when the + * session lives in a different project directory (git worktrees, + * cross-project resume). Every call resets the project dir; it never + * carries over from the previous session. + */ +export function switchSession( + sessionId: SessionId, + projectDir: string | null = null, +): void { + // Drop the outgoing session's plan-slug entry so the Map stays bounded + // across repeated /resume. Only the current session's slug is ever read + // (plans.ts getPlanSlug defaults to getSessionId()). + STATE.planSlugCache.delete(STATE.sessionId) + STATE.sessionId = sessionId + STATE.sessionProjectDir = projectDir + sessionSwitched.emit(sessionId) +} + +const sessionSwitched = createSignal<[id: SessionId]>() + +/** + * Register a callback that fires when switchSession changes the active + * sessionId. bootstrap can't import listeners directly (DAG leaf), so + * callers register themselves. concurrentSessions.ts uses this to keep the + * PID file's sessionId in sync with --resume. + */ +export const onSessionSwitch = sessionSwitched.subscribe + +/** + * Project directory the current session's transcript lives in, or `null` if + * the session was created in the current project (common case — derive from + * originalCwd). See `switchSession()`. + */ +export function getSessionProjectDir(): string | null { + return STATE.sessionProjectDir +} + +export function getOriginalCwd(): string { + return STATE.originalCwd +} + +/** + * Get the stable project root directory. + * Unlike getOriginalCwd(), this is never updated by mid-session EnterWorktreeTool + * (so skills/history stay stable when entering a throwaway worktree). + * It IS set at startup by --worktree, since that worktree is the session's project. + * Use for project identity (history, skills, sessions) not file operations. + */ +export function getProjectRoot(): string { + return STATE.projectRoot +} + +export function setOriginalCwd(cwd: string): void { + STATE.originalCwd = cwd.normalize('NFC') +} + +/** + * Only for --worktree startup flag. Mid-session EnterWorktreeTool must NOT + * call this — skills/history should stay anchored to where the session started. + */ +export function setProjectRoot(cwd: string): void { + STATE.projectRoot = cwd.normalize('NFC') +} + +export function getCwdState(): string { + return STATE.cwd +} + +export function setCwdState(cwd: string): void { + STATE.cwd = cwd.normalize('NFC') +} + +export function getDirectConnectServerUrl(): string | undefined { + return STATE.directConnectServerUrl +} + +export function setDirectConnectServerUrl(url: string): void { + STATE.directConnectServerUrl = url +} + +export function addToTotalDurationState( + duration: number, + durationWithoutRetries: number, +): void { + STATE.totalAPIDuration += duration + STATE.totalAPIDurationWithoutRetries += durationWithoutRetries +} + +export function resetTotalDurationStateAndCost_FOR_TESTS_ONLY(): void { + STATE.totalAPIDuration = 0 + STATE.totalAPIDurationWithoutRetries = 0 + STATE.totalCostUSD = 0 +} + +export function addToTotalCostState( + cost: number, + modelUsage: ModelUsage, + model: string, +): void { + STATE.modelUsage[model] = modelUsage + STATE.totalCostUSD += cost +} + +export function getTotalCostUSD(): number { + return STATE.totalCostUSD +} + +export function getTotalAPIDuration(): number { + return STATE.totalAPIDuration +} + +export function getTotalDuration(): number { + return Date.now() - STATE.startTime +} + +export function getTotalAPIDurationWithoutRetries(): number { + return STATE.totalAPIDurationWithoutRetries +} + +export function getTotalToolDuration(): number { + return STATE.totalToolDuration +} + +export function addToToolDuration(duration: number): void { + STATE.totalToolDuration += duration + STATE.turnToolDurationMs += duration + STATE.turnToolCount++ +} + +export function getTurnHookDurationMs(): number { + return STATE.turnHookDurationMs +} + +export function addToTurnHookDuration(duration: number): void { + STATE.turnHookDurationMs += duration + STATE.turnHookCount++ +} + +export function resetTurnHookDuration(): void { + STATE.turnHookDurationMs = 0 + STATE.turnHookCount = 0 +} + +export function getTurnHookCount(): number { + return STATE.turnHookCount +} + +export function getTurnToolDurationMs(): number { + return STATE.turnToolDurationMs +} + +export function resetTurnToolDuration(): void { + STATE.turnToolDurationMs = 0 + STATE.turnToolCount = 0 +} + +export function getTurnToolCount(): number { + return STATE.turnToolCount +} + +export function getTurnClassifierDurationMs(): number { + return STATE.turnClassifierDurationMs +} + +export function addToTurnClassifierDuration(duration: number): void { + STATE.turnClassifierDurationMs += duration + STATE.turnClassifierCount++ +} + +export function resetTurnClassifierDuration(): void { + STATE.turnClassifierDurationMs = 0 + STATE.turnClassifierCount = 0 +} + +export function getTurnClassifierCount(): number { + return STATE.turnClassifierCount +} + +export function getStatsStore(): { + observe(name: string, value: number): void +} | null { + return STATE.statsStore +} + +export function setStatsStore( + store: { observe(name: string, value: number): void } | null, +): void { + STATE.statsStore = store +} + +/** + * Marks that an interaction occurred. + * + * By default the actual Date.now() call is deferred until the next Ink render + * frame (via flushInteractionTime()) so we avoid calling Date.now() on every + * single keypress. + * + * Pass `immediate = true` when calling from React useEffect callbacks or + * other code that runs *after* the Ink render cycle has already flushed. + * Without it the timestamp stays stale until the next render, which may never + * come if the user is idle (e.g. permission dialog waiting for input). + */ +let interactionTimeDirty = false + +export function updateLastInteractionTime(immediate?: boolean): void { + if (immediate) { + flushInteractionTime_inner() + } else { + interactionTimeDirty = true + } +} + +/** + * If an interaction was recorded since the last flush, update the timestamp + * now. Called by Ink before each render cycle so we batch many keypresses into + * a single Date.now() call. + */ +export function flushInteractionTime(): void { + if (interactionTimeDirty) { + flushInteractionTime_inner() + } +} + +function flushInteractionTime_inner(): void { + STATE.lastInteractionTime = Date.now() + interactionTimeDirty = false +} + +export function addToTotalLinesChanged(added: number, removed: number): void { + STATE.totalLinesAdded += added + STATE.totalLinesRemoved += removed +} + +export function getTotalLinesAdded(): number { + return STATE.totalLinesAdded +} + +export function getTotalLinesRemoved(): number { + return STATE.totalLinesRemoved +} + +export function getTotalInputTokens(): number { + return sumBy(Object.values(STATE.modelUsage), 'inputTokens') +} + +export function getTotalOutputTokens(): number { + return sumBy(Object.values(STATE.modelUsage), 'outputTokens') +} + +export function getTotalCacheReadInputTokens(): number { + return sumBy(Object.values(STATE.modelUsage), 'cacheReadInputTokens') +} + +export function getTotalCacheCreationInputTokens(): number { + return sumBy(Object.values(STATE.modelUsage), 'cacheCreationInputTokens') +} + +export function getTotalWebSearchRequests(): number { + return sumBy(Object.values(STATE.modelUsage), 'webSearchRequests') +} + +let outputTokensAtTurnStart = 0 +let currentTurnTokenBudget: number | null = null +export function getTurnOutputTokens(): number { + return getTotalOutputTokens() - outputTokensAtTurnStart +} +export function getCurrentTurnTokenBudget(): number | null { + return currentTurnTokenBudget +} +let budgetContinuationCount = 0 +export function snapshotOutputTokensForTurn(budget: number | null): void { + outputTokensAtTurnStart = getTotalOutputTokens() + currentTurnTokenBudget = budget + budgetContinuationCount = 0 +} +export function getBudgetContinuationCount(): number { + return budgetContinuationCount +} +export function incrementBudgetContinuationCount(): void { + budgetContinuationCount++ +} + +export function setHasUnknownModelCost(): void { + STATE.hasUnknownModelCost = true +} + +export function hasUnknownModelCost(): boolean { + return STATE.hasUnknownModelCost +} + +export function getLastMainRequestId(): string | undefined { + return STATE.lastMainRequestId +} + +export function setLastMainRequestId(requestId: string): void { + STATE.lastMainRequestId = requestId +} + +export function getLastApiCompletionTimestamp(): number | null { + return STATE.lastApiCompletionTimestamp +} + +export function setLastApiCompletionTimestamp(timestamp: number): void { + STATE.lastApiCompletionTimestamp = timestamp +} + +/** Mark that a compaction just occurred. The next API success event will + * include isPostCompaction=true, then the flag auto-resets. */ +export function markPostCompaction(): void { + STATE.pendingPostCompaction = true +} + +/** Consume the post-compaction flag. Returns true once after compaction, + * then returns false until the next compaction. */ +export function consumePostCompaction(): boolean { + const was = STATE.pendingPostCompaction + STATE.pendingPostCompaction = false + return was +} + +export function getLastInteractionTime(): number { + return STATE.lastInteractionTime +} + +// Scroll drain suspension — background intervals check this before doing work +// so they don't compete with scroll frames for the event loop. Set by +// ScrollBox scrollBy/scrollTo, cleared SCROLL_DRAIN_IDLE_MS after the last +// scroll event. Module-scope (not in STATE) — ephemeral hot-path flag, no +// test-reset needed since the debounce timer self-clears. +let scrollDraining = false +let scrollDrainTimer: ReturnType | undefined +const SCROLL_DRAIN_IDLE_MS = 150 + +/** Mark that a scroll event just happened. Background intervals gate on + * getIsScrollDraining() and skip their work until the debounce clears. */ +export function markScrollActivity(): void { + scrollDraining = true + if (scrollDrainTimer) clearTimeout(scrollDrainTimer) + scrollDrainTimer = setTimeout(() => { + scrollDraining = false + scrollDrainTimer = undefined + }, SCROLL_DRAIN_IDLE_MS) + scrollDrainTimer.unref?.() +} + +/** True while scroll is actively draining (within 150ms of last event). + * Intervals should early-return when this is set — the work picks up next + * tick after scroll settles. */ +export function getIsScrollDraining(): boolean { + return scrollDraining +} + +/** Await this before expensive one-shot work (network, subprocess) that could + * coincide with scroll. Resolves immediately if not scrolling; otherwise + * polls at the idle interval until the flag clears. */ +export async function waitForScrollIdle(): Promise { + while (scrollDraining) { + // bootstrap-isolation forbids importing sleep() from src/utils/ + // eslint-disable-next-line no-restricted-syntax + await new Promise(r => setTimeout(r, SCROLL_DRAIN_IDLE_MS).unref?.()) + } +} + +export function getModelUsage(): { [modelName: string]: ModelUsage } { + return STATE.modelUsage +} + +export function getUsageForModel(model: string): ModelUsage | undefined { + return STATE.modelUsage[model] +} + +/** + * Gets the model override set from the --model CLI flag or after the user + * updates their configured model. + */ +export function getMainLoopModelOverride(): ModelSetting | undefined { + return STATE.mainLoopModelOverride +} + +export function getInitialMainLoopModel(): ModelSetting { + return STATE.initialMainLoopModel +} + +export function setMainLoopModelOverride( + model: ModelSetting | undefined, +): void { + STATE.mainLoopModelOverride = model +} + +export function setInitialMainLoopModel(model: ModelSetting): void { + STATE.initialMainLoopModel = model +} + +export function getSdkBetas(): string[] | undefined { + return STATE.sdkBetas +} + +export function setSdkBetas(betas: string[] | undefined): void { + STATE.sdkBetas = betas +} + +export function resetCostState(): void { + STATE.totalCostUSD = 0 + STATE.totalAPIDuration = 0 + STATE.totalAPIDurationWithoutRetries = 0 + STATE.totalToolDuration = 0 + STATE.startTime = Date.now() + STATE.totalLinesAdded = 0 + STATE.totalLinesRemoved = 0 + STATE.hasUnknownModelCost = false + STATE.modelUsage = {} + STATE.promptId = null +} + +/** + * Sets cost state values for session restore. + * Called by restoreCostStateForSession in cost-tracker.ts. + */ +export function setCostStateForRestore({ + totalCostUSD, + totalAPIDuration, + totalAPIDurationWithoutRetries, + totalToolDuration, + totalLinesAdded, + totalLinesRemoved, + lastDuration, + modelUsage, +}: { + totalCostUSD: number + totalAPIDuration: number + totalAPIDurationWithoutRetries: number + totalToolDuration: number + totalLinesAdded: number + totalLinesRemoved: number + lastDuration: number | undefined + modelUsage: { [modelName: string]: ModelUsage } | undefined +}): void { + STATE.totalCostUSD = totalCostUSD + STATE.totalAPIDuration = totalAPIDuration + STATE.totalAPIDurationWithoutRetries = totalAPIDurationWithoutRetries + STATE.totalToolDuration = totalToolDuration + STATE.totalLinesAdded = totalLinesAdded + STATE.totalLinesRemoved = totalLinesRemoved + + // Restore per-model usage breakdown + if (modelUsage) { + STATE.modelUsage = modelUsage + } + + // Adjust startTime to make wall duration accumulate + if (lastDuration) { + STATE.startTime = Date.now() - lastDuration + } +} + +// Only used in tests +export function resetStateForTests(): void { + if (process.env.NODE_ENV !== 'test') { + throw new Error('resetStateForTests can only be called in tests') + } + Object.entries(getInitialState()).forEach(([key, value]) => { + STATE[key as keyof State] = value as never + }) + outputTokensAtTurnStart = 0 + currentTurnTokenBudget = null + budgetContinuationCount = 0 + sessionSwitched.clear() +} + +// You shouldn't use this directly. See src/utils/model/modelStrings.ts::getModelStrings() +export function getModelStrings(): ModelStrings | null { + return STATE.modelStrings +} + +// You shouldn't use this directly. See src/utils/model/modelStrings.ts +export function setModelStrings(modelStrings: ModelStrings): void { + STATE.modelStrings = modelStrings +} + +// Test utility function to reset model strings for re-initialization. +// Separate from setModelStrings because we only want to accept 'null' in tests. +export function resetModelStringsForTestingOnly() { + STATE.modelStrings = null +} + +export function setMeter( + meter: Meter, + createCounter: (name: string, options: MetricOptions) => AttributedCounter, +): void { + STATE.meter = meter + + // Initialize all counters using the provided factory + STATE.sessionCounter = createCounter('claude_code.session.count', { + description: 'Count of CLI sessions started', + }) + STATE.locCounter = createCounter('claude_code.lines_of_code.count', { + description: + "Count of lines of code modified, with the 'type' attribute indicating whether lines were added or removed", + }) + STATE.prCounter = createCounter('claude_code.pull_request.count', { + description: 'Number of pull requests created', + }) + STATE.commitCounter = createCounter('claude_code.commit.count', { + description: 'Number of git commits created', + }) + STATE.costCounter = createCounter('claude_code.cost.usage', { + description: 'Cost of the Claude Code session', + unit: 'USD', + }) + STATE.tokenCounter = createCounter('claude_code.token.usage', { + description: 'Number of tokens used', + unit: 'tokens', + }) + STATE.codeEditToolDecisionCounter = createCounter( + 'claude_code.code_edit_tool.decision', + { + description: + 'Count of code editing tool permission decisions (accept/reject) for Edit, Write, and NotebookEdit tools', + }, + ) + STATE.activeTimeCounter = createCounter('claude_code.active_time.total', { + description: 'Total active time in seconds', + unit: 's', + }) +} + +export function getMeter(): Meter | null { + return STATE.meter +} + +export function getSessionCounter(): AttributedCounter | null { + return STATE.sessionCounter +} + +export function getLocCounter(): AttributedCounter | null { + return STATE.locCounter +} + +export function getPrCounter(): AttributedCounter | null { + return STATE.prCounter +} + +export function getCommitCounter(): AttributedCounter | null { + return STATE.commitCounter +} + +export function getCostCounter(): AttributedCounter | null { + return STATE.costCounter +} + +export function getTokenCounter(): AttributedCounter | null { + return STATE.tokenCounter +} + +export function getCodeEditToolDecisionCounter(): AttributedCounter | null { + return STATE.codeEditToolDecisionCounter +} + +export function getActiveTimeCounter(): AttributedCounter | null { + return STATE.activeTimeCounter +} + +export function getLoggerProvider(): LoggerProvider | null { + return STATE.loggerProvider +} + +export function setLoggerProvider(provider: LoggerProvider | null): void { + STATE.loggerProvider = provider +} + +export function getEventLogger(): ReturnType | null { + return STATE.eventLogger +} + +export function setEventLogger( + logger: ReturnType | null, +): void { + STATE.eventLogger = logger +} + +export function getMeterProvider(): MeterProvider | null { + return STATE.meterProvider +} + +export function setMeterProvider(provider: MeterProvider | null): void { + STATE.meterProvider = provider +} +export function getTracerProvider(): BasicTracerProvider | null { + return STATE.tracerProvider +} +export function setTracerProvider(provider: BasicTracerProvider | null): void { + STATE.tracerProvider = provider +} + +export function getIsNonInteractiveSession(): boolean { + return !STATE.isInteractive +} + +export function getIsInteractive(): boolean { + return STATE.isInteractive +} + +export function setIsInteractive(value: boolean): void { + STATE.isInteractive = value +} + +export function getClientType(): string { + return STATE.clientType +} + +export function setClientType(type: string): void { + STATE.clientType = type +} + +export function getSdkAgentProgressSummariesEnabled(): boolean { + return STATE.sdkAgentProgressSummariesEnabled +} + +export function setSdkAgentProgressSummariesEnabled(value: boolean): void { + STATE.sdkAgentProgressSummariesEnabled = value +} + +export function getKairosActive(): boolean { + return STATE.kairosActive +} + +export function setKairosActive(value: boolean): void { + STATE.kairosActive = value +} + +export function getStrictToolResultPairing(): boolean { + return STATE.strictToolResultPairing +} + +export function setStrictToolResultPairing(value: boolean): void { + STATE.strictToolResultPairing = value +} + +// Field name 'userMsgOptIn' avoids excluded-string substrings ('BriefTool', +// 'SendUserMessage' — case-insensitive). All callers are inside feature() +// guards so these accessors don't need their own (matches getKairosActive). +export function getUserMsgOptIn(): boolean { + return STATE.userMsgOptIn +} + +export function setUserMsgOptIn(value: boolean): void { + STATE.userMsgOptIn = value +} + +export function getSessionSource(): string | undefined { + return STATE.sessionSource +} + +export function setSessionSource(source: string): void { + STATE.sessionSource = source +} + +export function getQuestionPreviewFormat(): 'markdown' | 'html' | undefined { + return STATE.questionPreviewFormat +} + +export function setQuestionPreviewFormat(format: 'markdown' | 'html'): void { + STATE.questionPreviewFormat = format +} + +export function getAgentColorMap(): Map { + return STATE.agentColorMap +} + +export function getFlagSettingsPath(): string | undefined { + return STATE.flagSettingsPath +} + +export function setFlagSettingsPath(path: string | undefined): void { + STATE.flagSettingsPath = path +} + +export function getFlagSettingsInline(): Record | null { + return STATE.flagSettingsInline +} + +export function setFlagSettingsInline( + settings: Record | null, +): void { + STATE.flagSettingsInline = settings +} + +export function getSessionIngressToken(): string | null | undefined { + return STATE.sessionIngressToken +} + +export function setSessionIngressToken(token: string | null): void { + STATE.sessionIngressToken = token +} + +export function getOauthTokenFromFd(): string | null | undefined { + return STATE.oauthTokenFromFd +} + +export function setOauthTokenFromFd(token: string | null): void { + STATE.oauthTokenFromFd = token +} + +export function getApiKeyFromFd(): string | null | undefined { + return STATE.apiKeyFromFd +} + +export function setApiKeyFromFd(key: string | null): void { + STATE.apiKeyFromFd = key +} + +export function setLastAPIRequest( + params: Omit | null, +): void { + STATE.lastAPIRequest = params +} + +export function getLastAPIRequest(): Omit< + BetaMessageStreamParams, + 'messages' +> | null { + return STATE.lastAPIRequest +} + +export function setLastAPIRequestMessages( + messages: BetaMessageStreamParams['messages'] | null, +): void { + STATE.lastAPIRequestMessages = messages +} + +export function getLastAPIRequestMessages(): + | BetaMessageStreamParams['messages'] + | null { + return STATE.lastAPIRequestMessages +} + +export function setLastClassifierRequests(requests: unknown[] | null): void { + STATE.lastClassifierRequests = requests +} + +export function getLastClassifierRequests(): unknown[] | null { + return STATE.lastClassifierRequests +} + +export function setCachedClaudeMdContent(content: string | null): void { + STATE.cachedClaudeMdContent = content +} + +export function getCachedClaudeMdContent(): string | null { + return STATE.cachedClaudeMdContent +} + +export function addToInMemoryErrorLog(errorInfo: { + error: string + timestamp: string +}): void { + const MAX_IN_MEMORY_ERRORS = 100 + if (STATE.inMemoryErrorLog.length >= MAX_IN_MEMORY_ERRORS) { + STATE.inMemoryErrorLog.shift() // Remove oldest error + } + STATE.inMemoryErrorLog.push(errorInfo) +} + +export function getAllowedSettingSources(): SettingSource[] { + return STATE.allowedSettingSources +} + +export function setAllowedSettingSources(sources: SettingSource[]): void { + STATE.allowedSettingSources = sources +} + +export function preferThirdPartyAuthentication(): boolean { + // IDE extension should behave as 1P for authentication reasons. + return getIsNonInteractiveSession() && STATE.clientType !== 'claude-vscode' +} + +export function setInlinePlugins(plugins: Array): void { + STATE.inlinePlugins = plugins +} + +export function getInlinePlugins(): Array { + return STATE.inlinePlugins +} + +export function setChromeFlagOverride(value: boolean | undefined): void { + STATE.chromeFlagOverride = value +} + +export function getChromeFlagOverride(): boolean | undefined { + return STATE.chromeFlagOverride +} + +export function setUseCoworkPlugins(value: boolean): void { + STATE.useCoworkPlugins = value + resetSettingsCache() +} + +export function getUseCoworkPlugins(): boolean { + return STATE.useCoworkPlugins +} + +export function setSessionBypassPermissionsMode(enabled: boolean): void { + STATE.sessionBypassPermissionsMode = enabled +} + +export function getSessionBypassPermissionsMode(): boolean { + return STATE.sessionBypassPermissionsMode +} + +export function setScheduledTasksEnabled(enabled: boolean): void { + STATE.scheduledTasksEnabled = enabled +} + +export function getScheduledTasksEnabled(): boolean { + return STATE.scheduledTasksEnabled +} + +export type SessionCronTask = { + id: string + cron: string + prompt: string + createdAt: number + recurring?: boolean + /** + * When set, the task was created by an in-process teammate (not the team lead). + * The scheduler routes fires to that teammate's pendingUserMessages queue + * instead of the main REPL command queue. Session-only — never written to disk. + */ + agentId?: string +} + +export function getSessionCronTasks(): SessionCronTask[] { + return STATE.sessionCronTasks +} + +export function addSessionCronTask(task: SessionCronTask): void { + STATE.sessionCronTasks.push(task) +} + +/** + * Returns the number of tasks actually removed. Callers use this to skip + * downstream work (e.g. the disk read in removeCronTasks) when all ids + * were accounted for here. + */ +export function removeSessionCronTasks(ids: readonly string[]): number { + if (ids.length === 0) return 0 + const idSet = new Set(ids) + const remaining = STATE.sessionCronTasks.filter(t => !idSet.has(t.id)) + const removed = STATE.sessionCronTasks.length - remaining.length + if (removed === 0) return 0 + STATE.sessionCronTasks = remaining + return removed +} + +export function setSessionTrustAccepted(accepted: boolean): void { + STATE.sessionTrustAccepted = accepted +} + +export function getSessionTrustAccepted(): boolean { + return STATE.sessionTrustAccepted +} + +export function setSessionPersistenceDisabled(disabled: boolean): void { + STATE.sessionPersistenceDisabled = disabled +} + +export function isSessionPersistenceDisabled(): boolean { + return STATE.sessionPersistenceDisabled +} + +export function hasExitedPlanModeInSession(): boolean { + return STATE.hasExitedPlanMode +} + +export function setHasExitedPlanMode(value: boolean): void { + STATE.hasExitedPlanMode = value +} + +export function needsPlanModeExitAttachment(): boolean { + return STATE.needsPlanModeExitAttachment +} + +export function setNeedsPlanModeExitAttachment(value: boolean): void { + STATE.needsPlanModeExitAttachment = value +} + +export function handlePlanModeTransition( + fromMode: string, + toMode: string, +): void { + // If switching TO plan mode, clear any pending exit attachment + // This prevents sending both plan_mode and plan_mode_exit when user toggles quickly + if (toMode === 'plan' && fromMode !== 'plan') { + STATE.needsPlanModeExitAttachment = false + } + + // If switching out of plan mode, trigger the plan_mode_exit attachment + if (fromMode === 'plan' && toMode !== 'plan') { + STATE.needsPlanModeExitAttachment = true + } +} + +export function needsAutoModeExitAttachment(): boolean { + return STATE.needsAutoModeExitAttachment +} + +export function setNeedsAutoModeExitAttachment(value: boolean): void { + STATE.needsAutoModeExitAttachment = value +} + +export function handleAutoModeTransition( + fromMode: string, + toMode: string, +): void { + // Auto↔plan transitions are handled by prepareContextForPlanMode (auto may + // stay active through plan if opted in) and ExitPlanMode (restores mode). + // Skip both directions so this function only handles direct auto transitions. + if ( + (fromMode === 'auto' && toMode === 'plan') || + (fromMode === 'plan' && toMode === 'auto') + ) { + return + } + const fromIsAuto = fromMode === 'auto' + const toIsAuto = toMode === 'auto' + + // If switching TO auto mode, clear any pending exit attachment + // This prevents sending both auto_mode and auto_mode_exit when user toggles quickly + if (toIsAuto && !fromIsAuto) { + STATE.needsAutoModeExitAttachment = false + } + + // If switching out of auto mode, trigger the auto_mode_exit attachment + if (fromIsAuto && !toIsAuto) { + STATE.needsAutoModeExitAttachment = true + } +} + +// LSP plugin recommendation session tracking +export function hasShownLspRecommendationThisSession(): boolean { + return STATE.lspRecommendationShownThisSession +} + +export function setLspRecommendationShownThisSession(value: boolean): void { + STATE.lspRecommendationShownThisSession = value +} + +// SDK init event state +export function setInitJsonSchema(schema: Record): void { + STATE.initJsonSchema = schema +} + +export function getInitJsonSchema(): Record | null { + return STATE.initJsonSchema +} + +export function registerHookCallbacks( + hooks: Partial>, +): void { + if (!STATE.registeredHooks) { + STATE.registeredHooks = {} + } + + // `registerHookCallbacks` may be called multiple times, so we need to merge (not overwrite) + for (const [event, matchers] of Object.entries(hooks)) { + const eventKey = event as HookEvent + if (!STATE.registeredHooks[eventKey]) { + STATE.registeredHooks[eventKey] = [] + } + STATE.registeredHooks[eventKey]!.push(...matchers) + } +} + +export function getRegisteredHooks(): Partial< + Record +> | null { + return STATE.registeredHooks +} + +export function clearRegisteredHooks(): void { + STATE.registeredHooks = null +} + +export function clearRegisteredPluginHooks(): void { + if (!STATE.registeredHooks) { + return + } + + const filtered: Partial> = {} + for (const [event, matchers] of Object.entries(STATE.registeredHooks)) { + // Keep only callback hooks (those without pluginRoot) + const callbackHooks = matchers.filter(m => !('pluginRoot' in m)) + if (callbackHooks.length > 0) { + filtered[event as HookEvent] = callbackHooks + } + } + + STATE.registeredHooks = Object.keys(filtered).length > 0 ? filtered : null +} + +export function resetSdkInitState(): void { + STATE.initJsonSchema = null + STATE.registeredHooks = null +} + +export function getPlanSlugCache(): Map { + return STATE.planSlugCache +} + +export function getSessionCreatedTeams(): Set { + return STATE.sessionCreatedTeams +} + +// Teleported session tracking for reliability logging +export function setTeleportedSessionInfo(info: { + sessionId: string | null +}): void { + STATE.teleportedSessionInfo = { + isTeleported: true, + hasLoggedFirstMessage: false, + sessionId: info.sessionId, + } +} + +export function getTeleportedSessionInfo(): { + isTeleported: boolean + hasLoggedFirstMessage: boolean + sessionId: string | null +} | null { + return STATE.teleportedSessionInfo +} + +export function markFirstTeleportMessageLogged(): void { + if (STATE.teleportedSessionInfo) { + STATE.teleportedSessionInfo.hasLoggedFirstMessage = true + } +} + +// Invoked skills tracking for preservation across compaction +export type InvokedSkillInfo = { + skillName: string + skillPath: string + content: string + invokedAt: number + agentId: string | null +} + +export function addInvokedSkill( + skillName: string, + skillPath: string, + content: string, + agentId: string | null = null, +): void { + const key = `${agentId ?? ''}:${skillName}` + STATE.invokedSkills.set(key, { + skillName, + skillPath, + content, + invokedAt: Date.now(), + agentId, + }) +} + +export function getInvokedSkills(): Map { + return STATE.invokedSkills +} + +export function getInvokedSkillsForAgent( + agentId: string | undefined | null, +): Map { + const normalizedId = agentId ?? null + const filtered = new Map() + for (const [key, skill] of STATE.invokedSkills) { + if (skill.agentId === normalizedId) { + filtered.set(key, skill) + } + } + return filtered +} + +export function clearInvokedSkills( + preservedAgentIds?: ReadonlySet, +): void { + if (!preservedAgentIds || preservedAgentIds.size === 0) { + STATE.invokedSkills.clear() + return + } + for (const [key, skill] of STATE.invokedSkills) { + if (skill.agentId === null || !preservedAgentIds.has(skill.agentId)) { + STATE.invokedSkills.delete(key) + } + } +} + +export function clearInvokedSkillsForAgent(agentId: string): void { + for (const [key, skill] of STATE.invokedSkills) { + if (skill.agentId === agentId) { + STATE.invokedSkills.delete(key) + } + } +} + +// Slow operations tracking for dev bar +const MAX_SLOW_OPERATIONS = 10 +const SLOW_OPERATION_TTL_MS = 10000 + +export function addSlowOperation(operation: string, durationMs: number): void { + if (process.env.USER_TYPE !== 'ant') return + // Skip tracking for editor sessions (user editing a prompt file in $EDITOR) + // These are intentionally slow since the user is drafting text + if (operation.includes('exec') && operation.includes('claude-prompt-')) { + return + } + const now = Date.now() + // Remove stale operations + STATE.slowOperations = STATE.slowOperations.filter( + op => now - op.timestamp < SLOW_OPERATION_TTL_MS, + ) + // Add new operation + STATE.slowOperations.push({ operation, durationMs, timestamp: now }) + // Keep only the most recent operations + if (STATE.slowOperations.length > MAX_SLOW_OPERATIONS) { + STATE.slowOperations = STATE.slowOperations.slice(-MAX_SLOW_OPERATIONS) + } +} + +const EMPTY_SLOW_OPERATIONS: ReadonlyArray<{ + operation: string + durationMs: number + timestamp: number +}> = [] + +export function getSlowOperations(): ReadonlyArray<{ + operation: string + durationMs: number + timestamp: number +}> { + // Most common case: nothing tracked. Return a stable reference so the + // caller's setState() can bail via Object.is instead of re-rendering at 2fps. + if (STATE.slowOperations.length === 0) { + return EMPTY_SLOW_OPERATIONS + } + const now = Date.now() + // Only allocate a new array when something actually expired; otherwise keep + // the reference stable across polls while ops are still fresh. + if ( + STATE.slowOperations.some(op => now - op.timestamp >= SLOW_OPERATION_TTL_MS) + ) { + STATE.slowOperations = STATE.slowOperations.filter( + op => now - op.timestamp < SLOW_OPERATION_TTL_MS, + ) + if (STATE.slowOperations.length === 0) { + return EMPTY_SLOW_OPERATIONS + } + } + // Safe to return directly: addSlowOperation() reassigns STATE.slowOperations + // before pushing, so the array held in React state is never mutated. + return STATE.slowOperations +} + +export function getMainThreadAgentType(): string | undefined { + return STATE.mainThreadAgentType +} + +export function setMainThreadAgentType(agentType: string | undefined): void { + STATE.mainThreadAgentType = agentType +} + +export function getIsRemoteMode(): boolean { + return STATE.isRemoteMode +} + +export function setIsRemoteMode(value: boolean): void { + STATE.isRemoteMode = value +} + +// System prompt section accessors + +export function getSystemPromptSectionCache(): Map { + return STATE.systemPromptSectionCache +} + +export function setSystemPromptSectionCacheEntry( + name: string, + value: string | null, +): void { + STATE.systemPromptSectionCache.set(name, value) +} + +export function clearSystemPromptSectionState(): void { + STATE.systemPromptSectionCache.clear() +} + +// Last emitted date accessors (for detecting midnight date changes) + +export function getLastEmittedDate(): string | null { + return STATE.lastEmittedDate +} + +export function setLastEmittedDate(date: string | null): void { + STATE.lastEmittedDate = date +} + +export function getAdditionalDirectoriesForClaudeMd(): string[] { + return STATE.additionalDirectoriesForClaudeMd +} + +export function setAdditionalDirectoriesForClaudeMd( + directories: string[], +): void { + STATE.additionalDirectoriesForClaudeMd = directories +} + +export function getAllowedChannels(): ChannelEntry[] { + return STATE.allowedChannels +} + +export function setAllowedChannels(entries: ChannelEntry[]): void { + STATE.allowedChannels = entries +} + +export function getHasDevChannels(): boolean { + return STATE.hasDevChannels +} + +export function setHasDevChannels(value: boolean): void { + STATE.hasDevChannels = value +} + +export function getPromptCache1hAllowlist(): string[] | null { + return STATE.promptCache1hAllowlist +} + +export function setPromptCache1hAllowlist(allowlist: string[] | null): void { + STATE.promptCache1hAllowlist = allowlist +} + +export function getPromptCache1hEligible(): boolean | null { + return STATE.promptCache1hEligible +} + +export function setPromptCache1hEligible(eligible: boolean | null): void { + STATE.promptCache1hEligible = eligible +} + +export function getAfkModeHeaderLatched(): boolean | null { + return STATE.afkModeHeaderLatched +} + +export function setAfkModeHeaderLatched(v: boolean): void { + STATE.afkModeHeaderLatched = v +} + +export function getFastModeHeaderLatched(): boolean | null { + return STATE.fastModeHeaderLatched +} + +export function setFastModeHeaderLatched(v: boolean): void { + STATE.fastModeHeaderLatched = v +} + +export function getCacheEditingHeaderLatched(): boolean | null { + return STATE.cacheEditingHeaderLatched +} + +export function setCacheEditingHeaderLatched(v: boolean): void { + STATE.cacheEditingHeaderLatched = v +} + +export function getThinkingClearLatched(): boolean | null { + return STATE.thinkingClearLatched +} + +export function setThinkingClearLatched(v: boolean): void { + STATE.thinkingClearLatched = v +} + +/** + * Reset beta header latches to null. Called on /clear and /compact so a + * fresh conversation gets fresh header evaluation. + */ +export function clearBetaHeaderLatches(): void { + STATE.afkModeHeaderLatched = null + STATE.fastModeHeaderLatched = null + STATE.cacheEditingHeaderLatched = null + STATE.thinkingClearLatched = null +} + +export function getPromptId(): string | null { + return STATE.promptId +} + +export function setPromptId(id: string | null): void { + STATE.promptId = id +} + diff --git a/src/bridge/bridgeApi.ts b/src/bridge/bridgeApi.ts new file mode 100644 index 0000000..052bd4f --- /dev/null +++ b/src/bridge/bridgeApi.ts @@ -0,0 +1,539 @@ +import axios from 'axios' + +import { debugBody, extractErrorDetail } from './debugUtils.js' +import { + BRIDGE_LOGIN_INSTRUCTION, + type BridgeApiClient, + type BridgeConfig, + type PermissionResponseEvent, + type WorkResponse, +} from './types.js' + +type BridgeApiDeps = { + baseUrl: string + getAccessToken: () => string | undefined + runnerVersion: string + onDebug?: (msg: string) => void + /** + * Called on 401 to attempt OAuth token refresh. Returns true if refreshed, + * in which case the request is retried once. Injected because + * handleOAuth401Error from utils/auth.ts transitively pulls in config.ts → + * file.ts → permissions/filesystem.ts → sessionStorage.ts → commands.ts + * (~1300 modules). Daemon callers using env-var tokens omit this — their + * tokens don't refresh, so 401 goes straight to BridgeFatalError. + */ + onAuth401?: (staleAccessToken: string) => Promise + /** + * Returns the trusted device token to send as X-Trusted-Device-Token on + * bridge API calls. Bridge sessions have SecurityTier=ELEVATED on the + * server (CCR v2); when the server's enforcement flag is on, + * ConnectBridgeWorker requires a trusted device at JWT-issuance. + * Optional — when absent or returning undefined, the header is omitted + * and the server falls through to its flag-off/no-op path. The CLI-side + * gate is tengu_sessions_elevated_auth_enforcement (see trustedDevice.ts). + */ + getTrustedDeviceToken?: () => string | undefined +} + +const BETA_HEADER = 'environments-2025-11-01' + +/** Allowlist pattern for server-provided IDs used in URL path segments. */ +const SAFE_ID_PATTERN = /^[a-zA-Z0-9_-]+$/ + +/** + * Validate that a server-provided ID is safe to interpolate into a URL path. + * Prevents path traversal (e.g. `../../admin`) and injection via IDs that + * contain slashes, dots, or other special characters. + */ +export function validateBridgeId(id: string, label: string): string { + if (!id || !SAFE_ID_PATTERN.test(id)) { + throw new Error(`Invalid ${label}: contains unsafe characters`) + } + return id +} + +/** Fatal bridge errors that should not be retried (e.g. auth failures). */ +export class BridgeFatalError extends Error { + readonly status: number + /** Server-provided error type, e.g. "environment_expired". */ + readonly errorType: string | undefined + constructor(message: string, status: number, errorType?: string) { + super(message) + this.name = 'BridgeFatalError' + this.status = status + this.errorType = errorType + } +} + +export function createBridgeApiClient(deps: BridgeApiDeps): BridgeApiClient { + function debug(msg: string): void { + deps.onDebug?.(msg) + } + + let consecutiveEmptyPolls = 0 + const EMPTY_POLL_LOG_INTERVAL = 100 + + function getHeaders(accessToken: string): Record { + const headers: Record = { + Authorization: `Bearer ${accessToken}`, + 'Content-Type': 'application/json', + 'anthropic-version': '2023-06-01', + 'anthropic-beta': BETA_HEADER, + 'x-environment-runner-version': deps.runnerVersion, + } + const deviceToken = deps.getTrustedDeviceToken?.() + if (deviceToken) { + headers['X-Trusted-Device-Token'] = deviceToken + } + return headers + } + + function resolveAuth(): string { + const accessToken = deps.getAccessToken() + if (!accessToken) { + throw new Error(BRIDGE_LOGIN_INSTRUCTION) + } + return accessToken + } + + /** + * Execute an OAuth-authenticated request with a single retry on 401. + * On 401, attempts token refresh via handleOAuth401Error (same pattern as + * withRetry.ts for v1/messages). If refresh succeeds, retries the request + * once with the new token. If refresh fails or the retry also returns 401, + * the 401 response is returned for handleErrorStatus to throw BridgeFatalError. + */ + async function withOAuthRetry( + fn: (accessToken: string) => Promise<{ status: number; data: T }>, + context: string, + ): Promise<{ status: number; data: T }> { + const accessToken = resolveAuth() + const response = await fn(accessToken) + + if (response.status !== 401) { + return response + } + + if (!deps.onAuth401) { + debug(`[bridge:api] ${context}: 401 received, no refresh handler`) + return response + } + + // Attempt token refresh — matches the pattern in withRetry.ts + debug(`[bridge:api] ${context}: 401 received, attempting token refresh`) + const refreshed = await deps.onAuth401(accessToken) + if (refreshed) { + debug(`[bridge:api] ${context}: Token refreshed, retrying request`) + const newToken = resolveAuth() + const retryResponse = await fn(newToken) + if (retryResponse.status !== 401) { + return retryResponse + } + debug(`[bridge:api] ${context}: Retry after refresh also got 401`) + } else { + debug(`[bridge:api] ${context}: Token refresh failed`) + } + + // Refresh failed — return 401 for handleErrorStatus to throw + return response + } + + return { + async registerBridgeEnvironment( + config: BridgeConfig, + ): Promise<{ environment_id: string; environment_secret: string }> { + debug( + `[bridge:api] POST /v1/environments/bridge bridgeId=${config.bridgeId}`, + ) + + const response = await withOAuthRetry( + (token: string) => + axios.post<{ + environment_id: string + environment_secret: string + }>( + `${deps.baseUrl}/v1/environments/bridge`, + { + machine_name: config.machineName, + directory: config.dir, + branch: config.branch, + git_repo_url: config.gitRepoUrl, + // Advertise session capacity so claude.ai/code can show + // "2/4 sessions" badges and only block the picker when + // actually at capacity. Backends that don't yet accept + // this field will silently ignore it. + max_sessions: config.maxSessions, + // worker_type lets claude.ai filter environments by origin + // (e.g. assistant picker only shows assistant-mode workers). + // Desktop cowork app sends "cowork"; we send a distinct value. + metadata: { worker_type: config.workerType }, + // Idempotent re-registration: if we have a backend-issued + // environment_id from a prior session (--session-id resume), + // send it back so the backend reattaches instead of creating + // a new env. The backend may still hand back a fresh ID if + // the old one expired — callers must compare the response. + ...(config.reuseEnvironmentId && { + environment_id: config.reuseEnvironmentId, + }), + }, + { + headers: getHeaders(token), + timeout: 15_000, + validateStatus: status => status < 500, + }, + ), + 'Registration', + ) + + handleErrorStatus(response.status, response.data, 'Registration') + debug( + `[bridge:api] POST /v1/environments/bridge -> ${response.status} environment_id=${response.data.environment_id}`, + ) + debug( + `[bridge:api] >>> ${debugBody({ machine_name: config.machineName, directory: config.dir, branch: config.branch, git_repo_url: config.gitRepoUrl, max_sessions: config.maxSessions, metadata: { worker_type: config.workerType } })}`, + ) + debug(`[bridge:api] <<< ${debugBody(response.data)}`) + return response.data + }, + + async pollForWork( + environmentId: string, + environmentSecret: string, + signal?: AbortSignal, + reclaimOlderThanMs?: number, + ): Promise { + validateBridgeId(environmentId, 'environmentId') + + // Save and reset so errors break the "consecutive empty" streak. + // Restored below when the response is truly empty. + const prevEmptyPolls = consecutiveEmptyPolls + consecutiveEmptyPolls = 0 + + const response = await axios.get( + `${deps.baseUrl}/v1/environments/${environmentId}/work/poll`, + { + headers: getHeaders(environmentSecret), + params: + reclaimOlderThanMs !== undefined + ? { reclaim_older_than_ms: reclaimOlderThanMs } + : undefined, + timeout: 10_000, + signal, + validateStatus: status => status < 500, + }, + ) + + handleErrorStatus(response.status, response.data, 'Poll') + + // Empty body or null = no work available + if (!response.data) { + consecutiveEmptyPolls = prevEmptyPolls + 1 + if ( + consecutiveEmptyPolls === 1 || + consecutiveEmptyPolls % EMPTY_POLL_LOG_INTERVAL === 0 + ) { + debug( + `[bridge:api] GET .../work/poll -> ${response.status} (no work, ${consecutiveEmptyPolls} consecutive empty polls)`, + ) + } + return null + } + + debug( + `[bridge:api] GET .../work/poll -> ${response.status} workId=${response.data.id} type=${response.data.data?.type}${response.data.data?.id ? ` sessionId=${response.data.data.id}` : ''}`, + ) + debug(`[bridge:api] <<< ${debugBody(response.data)}`) + return response.data + }, + + async acknowledgeWork( + environmentId: string, + workId: string, + sessionToken: string, + ): Promise { + validateBridgeId(environmentId, 'environmentId') + validateBridgeId(workId, 'workId') + + debug(`[bridge:api] POST .../work/${workId}/ack`) + + const response = await axios.post( + `${deps.baseUrl}/v1/environments/${environmentId}/work/${workId}/ack`, + {}, + { + headers: getHeaders(sessionToken), + timeout: 10_000, + validateStatus: s => s < 500, + }, + ) + + handleErrorStatus(response.status, response.data, 'Acknowledge') + debug(`[bridge:api] POST .../work/${workId}/ack -> ${response.status}`) + }, + + async stopWork( + environmentId: string, + workId: string, + force: boolean, + ): Promise { + validateBridgeId(environmentId, 'environmentId') + validateBridgeId(workId, 'workId') + + debug(`[bridge:api] POST .../work/${workId}/stop force=${force}`) + + const response = await withOAuthRetry( + (token: string) => + axios.post( + `${deps.baseUrl}/v1/environments/${environmentId}/work/${workId}/stop`, + { force }, + { + headers: getHeaders(token), + timeout: 10_000, + validateStatus: s => s < 500, + }, + ), + 'StopWork', + ) + + handleErrorStatus(response.status, response.data, 'StopWork') + debug(`[bridge:api] POST .../work/${workId}/stop -> ${response.status}`) + }, + + async deregisterEnvironment(environmentId: string): Promise { + validateBridgeId(environmentId, 'environmentId') + + debug(`[bridge:api] DELETE /v1/environments/bridge/${environmentId}`) + + const response = await withOAuthRetry( + (token: string) => + axios.delete( + `${deps.baseUrl}/v1/environments/bridge/${environmentId}`, + { + headers: getHeaders(token), + timeout: 10_000, + validateStatus: s => s < 500, + }, + ), + 'Deregister', + ) + + handleErrorStatus(response.status, response.data, 'Deregister') + debug( + `[bridge:api] DELETE /v1/environments/bridge/${environmentId} -> ${response.status}`, + ) + }, + + async archiveSession(sessionId: string): Promise { + validateBridgeId(sessionId, 'sessionId') + + debug(`[bridge:api] POST /v1/sessions/${sessionId}/archive`) + + const response = await withOAuthRetry( + (token: string) => + axios.post( + `${deps.baseUrl}/v1/sessions/${sessionId}/archive`, + {}, + { + headers: getHeaders(token), + timeout: 10_000, + validateStatus: s => s < 500, + }, + ), + 'ArchiveSession', + ) + + // 409 = already archived (idempotent, not an error) + if (response.status === 409) { + debug( + `[bridge:api] POST /v1/sessions/${sessionId}/archive -> 409 (already archived)`, + ) + return + } + + handleErrorStatus(response.status, response.data, 'ArchiveSession') + debug( + `[bridge:api] POST /v1/sessions/${sessionId}/archive -> ${response.status}`, + ) + }, + + async reconnectSession( + environmentId: string, + sessionId: string, + ): Promise { + validateBridgeId(environmentId, 'environmentId') + validateBridgeId(sessionId, 'sessionId') + + debug( + `[bridge:api] POST /v1/environments/${environmentId}/bridge/reconnect session_id=${sessionId}`, + ) + + const response = await withOAuthRetry( + (token: string) => + axios.post( + `${deps.baseUrl}/v1/environments/${environmentId}/bridge/reconnect`, + { session_id: sessionId }, + { + headers: getHeaders(token), + timeout: 10_000, + validateStatus: s => s < 500, + }, + ), + 'ReconnectSession', + ) + + handleErrorStatus(response.status, response.data, 'ReconnectSession') + debug(`[bridge:api] POST .../bridge/reconnect -> ${response.status}`) + }, + + async heartbeatWork( + environmentId: string, + workId: string, + sessionToken: string, + ): Promise<{ lease_extended: boolean; state: string }> { + validateBridgeId(environmentId, 'environmentId') + validateBridgeId(workId, 'workId') + + debug(`[bridge:api] POST .../work/${workId}/heartbeat`) + + const response = await axios.post<{ + lease_extended: boolean + state: string + last_heartbeat: string + ttl_seconds: number + }>( + `${deps.baseUrl}/v1/environments/${environmentId}/work/${workId}/heartbeat`, + {}, + { + headers: getHeaders(sessionToken), + timeout: 10_000, + validateStatus: s => s < 500, + }, + ) + + handleErrorStatus(response.status, response.data, 'Heartbeat') + debug( + `[bridge:api] POST .../work/${workId}/heartbeat -> ${response.status} lease_extended=${response.data.lease_extended} state=${response.data.state}`, + ) + return response.data + }, + + async sendPermissionResponseEvent( + sessionId: string, + event: PermissionResponseEvent, + sessionToken: string, + ): Promise { + validateBridgeId(sessionId, 'sessionId') + + debug( + `[bridge:api] POST /v1/sessions/${sessionId}/events type=${event.type}`, + ) + + const response = await axios.post( + `${deps.baseUrl}/v1/sessions/${sessionId}/events`, + { events: [event] }, + { + headers: getHeaders(sessionToken), + timeout: 10_000, + validateStatus: s => s < 500, + }, + ) + + handleErrorStatus( + response.status, + response.data, + 'SendPermissionResponseEvent', + ) + debug( + `[bridge:api] POST /v1/sessions/${sessionId}/events -> ${response.status}`, + ) + debug(`[bridge:api] >>> ${debugBody({ events: [event] })}`) + debug(`[bridge:api] <<< ${debugBody(response.data)}`) + }, + } +} + +function handleErrorStatus( + status: number, + data: unknown, + context: string, +): void { + if (status === 200 || status === 204) { + return + } + const detail = extractErrorDetail(data) + const errorType = extractErrorTypeFromData(data) + switch (status) { + case 401: + throw new BridgeFatalError( + `${context}: Authentication failed (401)${detail ? `: ${detail}` : ''}. ${BRIDGE_LOGIN_INSTRUCTION}`, + 401, + errorType, + ) + case 403: + throw new BridgeFatalError( + isExpiredErrorType(errorType) + ? 'Remote Control session has expired. Please restart with `claude remote-control` or /remote-control.' + : `${context}: Access denied (403)${detail ? `: ${detail}` : ''}. Check your organization permissions.`, + 403, + errorType, + ) + case 404: + throw new BridgeFatalError( + detail ?? + `${context}: Not found (404). Remote Control may not be available for this organization.`, + 404, + errorType, + ) + case 410: + throw new BridgeFatalError( + detail ?? + 'Remote Control session has expired. Please restart with `claude remote-control` or /remote-control.', + 410, + errorType ?? 'environment_expired', + ) + case 429: + throw new Error(`${context}: Rate limited (429). Polling too frequently.`) + default: + throw new Error( + `${context}: Failed with status ${status}${detail ? `: ${detail}` : ''}`, + ) + } +} + +/** Check whether an error type string indicates a session/environment expiry. */ +export function isExpiredErrorType(errorType: string | undefined): boolean { + if (!errorType) { + return false + } + return errorType.includes('expired') || errorType.includes('lifetime') +} + +/** + * Check whether a BridgeFatalError is a suppressible 403 permission error. + * These are 403 errors for scopes like 'external_poll_sessions' or operations + * like StopWork that fail because the user's role lacks 'environments:manage'. + * They don't affect core functionality and shouldn't be shown to users. + */ +export function isSuppressible403(err: BridgeFatalError): boolean { + if (err.status !== 403) { + return false + } + return ( + err.message.includes('external_poll_sessions') || + err.message.includes('environments:manage') + ) +} + +function extractErrorTypeFromData(data: unknown): string | undefined { + if (data && typeof data === 'object') { + if ( + 'error' in data && + data.error && + typeof data.error === 'object' && + 'type' in data.error && + typeof data.error.type === 'string' + ) { + return data.error.type + } + } + return undefined +} diff --git a/src/bridge/bridgeConfig.ts b/src/bridge/bridgeConfig.ts new file mode 100644 index 0000000..02f0876 --- /dev/null +++ b/src/bridge/bridgeConfig.ts @@ -0,0 +1,48 @@ +/** + * Shared bridge auth/URL resolution. Consolidates the ant-only + * CLAUDE_BRIDGE_* dev overrides that were previously copy-pasted across + * a dozen files — inboundAttachments, BriefTool/upload, bridgeMain, + * initReplBridge, remoteBridgeCore, daemon workers, /rename, + * /remote-control. + * + * Two layers: *Override() returns the ant-only env var (or undefined); + * the non-Override versions fall through to the real OAuth store/config. + * Callers that compose with a different auth source (e.g. daemon workers + * using IPC auth) use the Override getters directly. + */ + +import { getOauthConfig } from '../constants/oauth.js' +import { getClaudeAIOAuthTokens } from '../utils/auth.js' + +/** Ant-only dev override: CLAUDE_BRIDGE_OAUTH_TOKEN, else undefined. */ +export function getBridgeTokenOverride(): string | undefined { + return ( + (process.env.USER_TYPE === 'ant' && + process.env.CLAUDE_BRIDGE_OAUTH_TOKEN) || + undefined + ) +} + +/** Ant-only dev override: CLAUDE_BRIDGE_BASE_URL, else undefined. */ +export function getBridgeBaseUrlOverride(): string | undefined { + return ( + (process.env.USER_TYPE === 'ant' && process.env.CLAUDE_BRIDGE_BASE_URL) || + undefined + ) +} + +/** + * Access token for bridge API calls: dev override first, then the OAuth + * keychain. Undefined means "not logged in". + */ +export function getBridgeAccessToken(): string | undefined { + return getBridgeTokenOverride() ?? getClaudeAIOAuthTokens()?.accessToken +} + +/** + * Base URL for bridge API calls: dev override first, then the production + * OAuth config. Always returns a URL. + */ +export function getBridgeBaseUrl(): string { + return getBridgeBaseUrlOverride() ?? getOauthConfig().BASE_API_URL +} diff --git a/src/bridge/bridgeDebug.ts b/src/bridge/bridgeDebug.ts new file mode 100644 index 0000000..4d0f422 --- /dev/null +++ b/src/bridge/bridgeDebug.ts @@ -0,0 +1,135 @@ +import { logForDebugging } from '../utils/debug.js' +import { BridgeFatalError } from './bridgeApi.js' +import type { BridgeApiClient } from './types.js' + +/** + * Ant-only fault injection for manually testing bridge recovery paths. + * + * Real failure modes this targets (BQ 2026-03-12, 7-day window): + * poll 404 not_found_error — 147K sessions/week, dead onEnvironmentLost gate + * ws_closed 1002/1006 — 22K sessions/week, zombie poll after close + * register transient failure — residual: network blips during doReconnect + * + * Usage: /bridge-kick from the REPL while Remote Control is + * connected, then tail debug.log to watch the recovery machinery react. + * + * Module-level state is intentional here: one bridge per REPL process, the + * /bridge-kick slash command has no other way to reach into initBridgeCore's + * closures, and teardown clears the slot. + */ + +/** One-shot fault to inject on the next matching api call. */ +type BridgeFault = { + method: + | 'pollForWork' + | 'registerBridgeEnvironment' + | 'reconnectSession' + | 'heartbeatWork' + /** Fatal errors go through handleErrorStatus → BridgeFatalError. Transient + * errors surface as plain axios rejections (5xx / network). Recovery code + * distinguishes the two: fatal → teardown, transient → retry/backoff. */ + kind: 'fatal' | 'transient' + status: number + errorType?: string + /** Remaining injections. Decremented on consume; removed at 0. */ + count: number +} + +export type BridgeDebugHandle = { + /** Invoke the transport's permanent-close handler directly. Tests the + * ws_closed → reconnectEnvironmentWithSession escalation (#22148). */ + fireClose: (code: number) => void + /** Call reconnectEnvironmentWithSession() — same as SIGUSR2 but + * reachable from the slash command. */ + forceReconnect: () => void + /** Queue a fault for the next N calls to the named api method. */ + injectFault: (fault: BridgeFault) => void + /** Abort the at-capacity sleep so an injected poll fault lands + * immediately instead of up to 10min later. */ + wakePollLoop: () => void + /** env/session IDs for the debug.log grep. */ + describe: () => string +} + +let debugHandle: BridgeDebugHandle | null = null +const faultQueue: BridgeFault[] = [] + +export function registerBridgeDebugHandle(h: BridgeDebugHandle): void { + debugHandle = h +} + +export function clearBridgeDebugHandle(): void { + debugHandle = null + faultQueue.length = 0 +} + +export function getBridgeDebugHandle(): BridgeDebugHandle | null { + return debugHandle +} + +export function injectBridgeFault(fault: BridgeFault): void { + faultQueue.push(fault) + logForDebugging( + `[bridge:debug] Queued fault: ${fault.method} ${fault.kind}/${fault.status}${fault.errorType ? `/${fault.errorType}` : ''} ×${fault.count}`, + ) +} + +/** + * Wrap a BridgeApiClient so each call first checks the fault queue. If a + * matching fault is queued, throw the specified error instead of calling + * through. Delegates everything else to the real client. + * + * Only called when USER_TYPE === 'ant' — zero overhead in external builds. + */ +export function wrapApiForFaultInjection( + api: BridgeApiClient, +): BridgeApiClient { + function consume(method: BridgeFault['method']): BridgeFault | null { + const idx = faultQueue.findIndex(f => f.method === method) + if (idx === -1) return null + const fault = faultQueue[idx]! + fault.count-- + if (fault.count <= 0) faultQueue.splice(idx, 1) + return fault + } + + function throwFault(fault: BridgeFault, context: string): never { + logForDebugging( + `[bridge:debug] Injecting ${fault.kind} fault into ${context}: status=${fault.status} errorType=${fault.errorType ?? 'none'}`, + ) + if (fault.kind === 'fatal') { + throw new BridgeFatalError( + `[injected] ${context} ${fault.status}`, + fault.status, + fault.errorType, + ) + } + // Transient: mimic an axios rejection (5xx / network). No .status on + // the error itself — that's how the catch blocks distinguish. + throw new Error(`[injected transient] ${context} ${fault.status}`) + } + + return { + ...api, + async pollForWork(envId, secret, signal, reclaimMs) { + const f = consume('pollForWork') + if (f) throwFault(f, 'Poll') + return api.pollForWork(envId, secret, signal, reclaimMs) + }, + async registerBridgeEnvironment(config) { + const f = consume('registerBridgeEnvironment') + if (f) throwFault(f, 'Registration') + return api.registerBridgeEnvironment(config) + }, + async reconnectSession(envId, sessionId) { + const f = consume('reconnectSession') + if (f) throwFault(f, 'ReconnectSession') + return api.reconnectSession(envId, sessionId) + }, + async heartbeatWork(envId, workId, token) { + const f = consume('heartbeatWork') + if (f) throwFault(f, 'Heartbeat') + return api.heartbeatWork(envId, workId, token) + }, + } +} diff --git a/src/bridge/bridgeEnabled.ts b/src/bridge/bridgeEnabled.ts new file mode 100644 index 0000000..b6eec41 --- /dev/null +++ b/src/bridge/bridgeEnabled.ts @@ -0,0 +1,202 @@ +import { feature } from 'bun:bundle' +import { + checkGate_CACHED_OR_BLOCKING, + getDynamicConfig_CACHED_MAY_BE_STALE, + getFeatureValue_CACHED_MAY_BE_STALE, +} from '../services/analytics/growthbook.js' +// Namespace import breaks the bridgeEnabled → auth → config → bridgeEnabled +// cycle — authModule.foo is a live binding, so by the time the helpers below +// call it, auth.js is fully loaded. Previously used require() for the same +// deferral, but require() hits a CJS cache that diverges from the ESM +// namespace after mock.module() (daemon/auth.test.ts), breaking spyOn. +import * as authModule from '../utils/auth.js' +import { isEnvTruthy } from '../utils/envUtils.js' +import { lt } from '../utils/semver.js' + +/** + * Runtime check for bridge mode entitlement. + * + * Remote Control requires a claude.ai subscription (the bridge auths to CCR + * with the claude.ai OAuth token). isClaudeAISubscriber() excludes + * Bedrock/Vertex/Foundry, apiKeyHelper/gateway deployments, env-var API keys, + * and Console API logins — none of which have the OAuth token CCR needs. + * See github.com/deshaw/anthropic-issues/issues/24. + * + * The `feature('BRIDGE_MODE')` guard ensures the GrowthBook string literal + * is only referenced when bridge mode is enabled at build time. + */ +export function isBridgeEnabled(): boolean { + // Positive ternary pattern — see docs/feature-gating.md. + // Negative pattern (if (!feature(...)) return) does not eliminate + // inline string literals from external builds. + return feature('BRIDGE_MODE') + ? isClaudeAISubscriber() && + getFeatureValue_CACHED_MAY_BE_STALE('tengu_ccr_bridge', false) + : false +} + +/** + * Blocking entitlement check for Remote Control. + * + * Returns cached `true` immediately (fast path). If the disk cache says + * `false` or is missing, awaits GrowthBook init and fetches the fresh + * server value (slow path, max ~5s), then writes it to disk. + * + * Use at entitlement gates where a stale `false` would unfairly block access. + * For user-facing error paths, prefer `getBridgeDisabledReason()` which gives + * a specific diagnostic. For render-body UI visibility checks, use + * `isBridgeEnabled()` instead. + */ +export async function isBridgeEnabledBlocking(): Promise { + return feature('BRIDGE_MODE') + ? isClaudeAISubscriber() && + (await checkGate_CACHED_OR_BLOCKING('tengu_ccr_bridge')) + : false +} + +/** + * Diagnostic message for why Remote Control is unavailable, or null if + * it's enabled. Call this instead of a bare `isBridgeEnabledBlocking()` + * check when you need to show the user an actionable error. + * + * The GrowthBook gate targets on organizationUUID, which comes from + * config.oauthAccount — populated by /api/oauth/profile during login. + * That endpoint requires the user:profile scope. Tokens without it + * (setup-token, CLAUDE_CODE_OAUTH_TOKEN env var, or pre-scope-expansion + * logins) leave oauthAccount unpopulated, so the gate falls back to + * false and users see a dead-end "not enabled" message with no hint + * that re-login would fix it. See CC-1165 / gh-33105. + */ +export async function getBridgeDisabledReason(): Promise { + if (feature('BRIDGE_MODE')) { + if (!isClaudeAISubscriber()) { + return 'Remote Control requires a claude.ai subscription. Run `claude auth login` to sign in with your claude.ai account.' + } + if (!hasProfileScope()) { + return 'Remote Control requires a full-scope login token. Long-lived tokens (from `claude setup-token` or CLAUDE_CODE_OAUTH_TOKEN) are limited to inference-only for security reasons. Run `claude auth login` to use Remote Control.' + } + if (!getOauthAccountInfo()?.organizationUuid) { + return 'Unable to determine your organization for Remote Control eligibility. Run `claude auth login` to refresh your account information.' + } + if (!(await checkGate_CACHED_OR_BLOCKING('tengu_ccr_bridge'))) { + return 'Remote Control is not yet enabled for your account.' + } + return null + } + return 'Remote Control is not available in this build.' +} + +// try/catch: main.tsx:5698 calls isBridgeEnabled() while defining the Commander +// program, before enableConfigs() runs. isClaudeAISubscriber() → getGlobalConfig() +// throws "Config accessed before allowed" there. Pre-config, no OAuth token can +// exist anyway — false is correct. Same swallow getFeatureValue_CACHED_MAY_BE_STALE +// already does at growthbook.ts:775-780. +function isClaudeAISubscriber(): boolean { + try { + return authModule.isClaudeAISubscriber() + } catch { + return false + } +} +function hasProfileScope(): boolean { + try { + return authModule.hasProfileScope() + } catch { + return false + } +} +function getOauthAccountInfo(): ReturnType< + typeof authModule.getOauthAccountInfo +> { + try { + return authModule.getOauthAccountInfo() + } catch { + return undefined + } +} + +/** + * Runtime check for the env-less (v2) REPL bridge path. + * Returns true when the GrowthBook flag `tengu_bridge_repl_v2` is enabled. + * + * This gates which implementation initReplBridge uses — NOT whether bridge + * is available at all (see isBridgeEnabled above). Daemon/print paths stay + * on the env-based implementation regardless of this gate. + */ +export function isEnvLessBridgeEnabled(): boolean { + return feature('BRIDGE_MODE') + ? getFeatureValue_CACHED_MAY_BE_STALE('tengu_bridge_repl_v2', false) + : false +} + +/** + * Kill-switch for the `cse_*` → `session_*` client-side retag shim. + * + * The shim exists because compat/convert.go:27 validates TagSession and the + * claude.ai frontend routes on `session_*`, while v2 worker endpoints hand out + * `cse_*`. Once the server tags by environment_kind and the frontend accepts + * `cse_*` directly, flip this to false to make toCompatSessionId a no-op. + * Defaults to true — the shim stays active until explicitly disabled. + */ +export function isCseShimEnabled(): boolean { + return feature('BRIDGE_MODE') + ? getFeatureValue_CACHED_MAY_BE_STALE( + 'tengu_bridge_repl_v2_cse_shim_enabled', + true, + ) + : true +} + +/** + * Returns an error message if the current CLI version is below the + * minimum required for the v1 (env-based) Remote Control path, or null if the + * version is fine. The v2 (env-less) path uses checkEnvLessBridgeMinVersion() + * in envLessBridgeConfig.ts instead — the two implementations have independent + * version floors. + * + * Uses cached (non-blocking) GrowthBook config. If GrowthBook hasn't + * loaded yet, the default '0.0.0' means the check passes — a safe fallback. + */ +export function checkBridgeMinVersion(): string | null { + // Positive pattern — see docs/feature-gating.md. + // Negative pattern (if (!feature(...)) return) does not eliminate + // inline string literals from external builds. + if (feature('BRIDGE_MODE')) { + const config = getDynamicConfig_CACHED_MAY_BE_STALE<{ + minVersion: string + }>('tengu_bridge_min_version', { minVersion: '0.0.0' }) + if (config.minVersion && lt(MACRO.VERSION, config.minVersion)) { + return `Your version of Claude Code (${MACRO.VERSION}) is too old for Remote Control.\nVersion ${config.minVersion} or higher is required. Run \`claude update\` to update.` + } + } + return null +} + +/** + * Default for remoteControlAtStartup when the user hasn't explicitly set it. + * When the CCR_AUTO_CONNECT build flag is present (ant-only) and the + * tengu_cobalt_harbor GrowthBook gate is on, all sessions connect to CCR by + * default — the user can still opt out by setting remoteControlAtStartup=false + * in config (explicit settings always win over this default). + * + * Defined here rather than in config.ts to avoid a direct + * config.ts → growthbook.ts import cycle (growthbook.ts → user.ts → config.ts). + */ +export function getCcrAutoConnectDefault(): boolean { + return feature('CCR_AUTO_CONNECT') + ? getFeatureValue_CACHED_MAY_BE_STALE('tengu_cobalt_harbor', false) + : false +} + +/** + * Opt-in CCR mirror mode — every local session spawns an outbound-only + * Remote Control session that receives forwarded events. Separate from + * getCcrAutoConnectDefault (bidirectional Remote Control). Env var wins for + * local opt-in; GrowthBook controls rollout. + */ +export function isCcrMirrorEnabled(): boolean { + return feature('CCR_MIRROR') + ? isEnvTruthy(process.env.CLAUDE_CODE_CCR_MIRROR) || + getFeatureValue_CACHED_MAY_BE_STALE('tengu_ccr_mirror', false) + : false +} diff --git a/src/bridge/bridgeMain.ts b/src/bridge/bridgeMain.ts new file mode 100644 index 0000000..7aeacaf --- /dev/null +++ b/src/bridge/bridgeMain.ts @@ -0,0 +1,2999 @@ +import { feature } from 'bun:bundle' +import { randomUUID } from 'crypto' +import { hostname, tmpdir } from 'os' +import { basename, join, resolve } from 'path' +import { getRemoteSessionUrl } from '../constants/product.js' +import { shutdownDatadog } from '../services/analytics/datadog.js' +import { shutdown1PEventLogging } from '../services/analytics/firstPartyEventLogger.js' +import { checkGate_CACHED_OR_BLOCKING } from '../services/analytics/growthbook.js' +import { + type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + logEvent, + logEventAsync, +} from '../services/analytics/index.js' +import { isInBundledMode } from '../utils/bundledMode.js' +import { logForDebugging } from '../utils/debug.js' +import { logForDiagnosticsNoPII } from '../utils/diagLogs.js' +import { isEnvTruthy, isInProtectedNamespace } from '../utils/envUtils.js' +import { errorMessage } from '../utils/errors.js' +import { truncateToWidth } from '../utils/format.js' +import { logError } from '../utils/log.js' +import { sleep } from '../utils/sleep.js' +import { createAgentWorktree, removeAgentWorktree } from '../utils/worktree.js' +import { + BridgeFatalError, + createBridgeApiClient, + isExpiredErrorType, + isSuppressible403, + validateBridgeId, +} from './bridgeApi.js' +import { formatDuration } from './bridgeStatusUtil.js' +import { createBridgeLogger } from './bridgeUI.js' +import { createCapacityWake } from './capacityWake.js' +import { describeAxiosError } from './debugUtils.js' +import { createTokenRefreshScheduler } from './jwtUtils.js' +import { getPollIntervalConfig } from './pollConfig.js' +import { toCompatSessionId, toInfraSessionId } from './sessionIdCompat.js' +import { createSessionSpawner, safeFilenameId } from './sessionRunner.js' +import { getTrustedDeviceToken } from './trustedDevice.js' +import { + BRIDGE_LOGIN_ERROR, + type BridgeApiClient, + type BridgeConfig, + type BridgeLogger, + DEFAULT_SESSION_TIMEOUT_MS, + type SessionDoneStatus, + type SessionHandle, + type SessionSpawner, + type SessionSpawnOpts, + type SpawnMode, +} from './types.js' +import { + buildCCRv2SdkUrl, + buildSdkUrl, + decodeWorkSecret, + registerWorker, + sameSessionId, +} from './workSecret.js' + +export type BackoffConfig = { + connInitialMs: number + connCapMs: number + connGiveUpMs: number + generalInitialMs: number + generalCapMs: number + generalGiveUpMs: number + /** SIGTERM→SIGKILL grace period on shutdown. Default 30s. */ + shutdownGraceMs?: number + /** stopWorkWithRetry base delay (1s/2s/4s backoff). Default 1000ms. */ + stopWorkBaseDelayMs?: number +} + +const DEFAULT_BACKOFF: BackoffConfig = { + connInitialMs: 2_000, + connCapMs: 120_000, // 2 minutes + connGiveUpMs: 600_000, // 10 minutes + generalInitialMs: 500, + generalCapMs: 30_000, + generalGiveUpMs: 600_000, // 10 minutes +} + +/** Status update interval for the live display (ms). */ +const STATUS_UPDATE_INTERVAL_MS = 1_000 +const SPAWN_SESSIONS_DEFAULT = 32 + +/** + * GrowthBook gate for multi-session spawn modes (--spawn / --capacity / --create-session-in-dir). + * Sibling of tengu_ccr_bridge_multi_environment (multiple envs per host:dir) — + * this one enables multiple sessions per environment. + * Rollout staged via targeting rules: ants first, then gradual external. + * + * Uses the blocking gate check so a stale disk-cache miss doesn't unfairly + * deny access. The fast path (cache has true) is still instant; only the + * cold-start path awaits the server fetch, and that fetch also seeds the + * disk cache for next time. + */ +async function isMultiSessionSpawnEnabled(): Promise { + return checkGate_CACHED_OR_BLOCKING('tengu_ccr_bridge_multi_session') +} + +/** + * Returns the threshold for detecting system sleep/wake in the poll loop. + * Must exceed the max backoff cap — otherwise normal backoff delays trigger + * false sleep detection (resetting the error budget indefinitely). Using + * 2× the connection backoff cap, matching the pattern in WebSocketTransport + * and replBridge. + */ +function pollSleepDetectionThresholdMs(backoff: BackoffConfig): number { + return backoff.connCapMs * 2 +} + +/** + * Returns the args that must precede CLI flags when spawning a child claude + * process. In compiled binaries, process.execPath is the claude binary itself + * and args go directly to it. In npm installs (node running cli.js), + * process.execPath is the node runtime — the child spawn must pass the script + * path as the first arg, otherwise node interprets --sdk-url as a node option + * and exits with "bad option: --sdk-url". See anthropics/claude-code#28334. + */ +function spawnScriptArgs(): string[] { + if (isInBundledMode() || !process.argv[1]) { + return [] + } + return [process.argv[1]] +} + +/** Attempt to spawn a session; returns error string if spawn throws. */ +function safeSpawn( + spawner: SessionSpawner, + opts: SessionSpawnOpts, + dir: string, +): SessionHandle | string { + try { + return spawner.spawn(opts, dir) + } catch (err) { + const errMsg = errorMessage(err) + logError(new Error(`Session spawn failed: ${errMsg}`)) + return errMsg + } +} + +export async function runBridgeLoop( + config: BridgeConfig, + environmentId: string, + environmentSecret: string, + api: BridgeApiClient, + spawner: SessionSpawner, + logger: BridgeLogger, + signal: AbortSignal, + backoffConfig: BackoffConfig = DEFAULT_BACKOFF, + initialSessionId?: string, + getAccessToken?: () => string | undefined | Promise, +): Promise { + // Local abort controller so that onSessionDone can stop the poll loop. + // Linked to the incoming signal so external aborts also work. + const controller = new AbortController() + if (signal.aborted) { + controller.abort() + } else { + signal.addEventListener('abort', () => controller.abort(), { once: true }) + } + const loopSignal = controller.signal + + const activeSessions = new Map() + const sessionStartTimes = new Map() + const sessionWorkIds = new Map() + // Compat-surface ID (session_*) computed once at spawn and cached so + // cleanup and status-update ticks use the same key regardless of whether + // the tengu_bridge_repl_v2_cse_shim_enabled gate flips mid-session. + const sessionCompatIds = new Map() + // Session ingress JWTs for heartbeat auth, keyed by sessionId. + // Stored separately from handle.accessToken because the token refresh + // scheduler overwrites that field with the OAuth token (~3h55m in). + const sessionIngressTokens = new Map() + const sessionTimers = new Map>() + const completedWorkIds = new Set() + const sessionWorktrees = new Map< + string, + { + worktreePath: string + worktreeBranch?: string + gitRoot?: string + hookBased?: boolean + } + >() + // Track sessions killed by the timeout watchdog so onSessionDone can + // distinguish them from server-initiated or shutdown interrupts. + const timedOutSessions = new Set() + // Sessions that already have a title (server-set or bridge-derived) so + // onFirstUserMessage doesn't clobber a user-assigned --name / web rename. + // Keyed by compatSessionId to match logger.setSessionTitle's key. + const titledSessions = new Set() + // Signal to wake the at-capacity sleep early when a session completes, + // so the bridge can immediately accept new work. + const capacityWake = createCapacityWake(loopSignal) + + /** + * Heartbeat all active work items. + * Returns 'ok' if at least one heartbeat succeeded, 'auth_failed' if any + * got a 401/403 (JWT expired — re-queued via reconnectSession so the next + * poll delivers fresh work), or 'failed' if all failed for other reasons. + */ + async function heartbeatActiveWorkItems(): Promise< + 'ok' | 'auth_failed' | 'fatal' | 'failed' + > { + let anySuccess = false + let anyFatal = false + const authFailedSessions: string[] = [] + for (const [sessionId] of activeSessions) { + const workId = sessionWorkIds.get(sessionId) + const ingressToken = sessionIngressTokens.get(sessionId) + if (!workId || !ingressToken) { + continue + } + try { + await api.heartbeatWork(environmentId, workId, ingressToken) + anySuccess = true + } catch (err) { + logForDebugging( + `[bridge:heartbeat] Failed for sessionId=${sessionId} workId=${workId}: ${errorMessage(err)}`, + ) + if (err instanceof BridgeFatalError) { + logEvent('tengu_bridge_heartbeat_error', { + status: + err.status as unknown as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + error_type: (err.status === 401 || err.status === 403 + ? 'auth_failed' + : 'fatal') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + if (err.status === 401 || err.status === 403) { + authFailedSessions.push(sessionId) + } else { + // 404/410 = environment expired or deleted — no point retrying + anyFatal = true + } + } + } + } + // JWT expired → trigger server-side re-dispatch. Without this, work stays + // ACK'd out of the Redis PEL and poll returns empty forever (CC-1263). + // The existingHandle path below delivers the fresh token to the child. + // sessionId is already in the format /bridge/reconnect expects: it comes + // from work.data.id, which matches the server's EnvironmentInstance store + // (cse_* under the compat gate, session_* otherwise). + for (const sessionId of authFailedSessions) { + logger.logVerbose( + `Session ${sessionId} token expired — re-queuing via bridge/reconnect`, + ) + try { + await api.reconnectSession(environmentId, sessionId) + logForDebugging( + `[bridge:heartbeat] Re-queued sessionId=${sessionId} via bridge/reconnect`, + ) + } catch (err) { + logger.logError( + `Failed to refresh session ${sessionId} token: ${errorMessage(err)}`, + ) + logForDebugging( + `[bridge:heartbeat] reconnectSession(${sessionId}) failed: ${errorMessage(err)}`, + { level: 'error' }, + ) + } + } + if (anyFatal) { + return 'fatal' + } + if (authFailedSessions.length > 0) { + return 'auth_failed' + } + return anySuccess ? 'ok' : 'failed' + } + + // Sessions spawned with CCR v2 env vars. v2 children cannot use OAuth + // tokens (CCR worker endpoints validate the JWT's session_id claim, + // register_worker.go:32), so onRefresh triggers server re-dispatch + // instead — the next poll delivers fresh work with a new JWT via the + // existingHandle path below. + const v2Sessions = new Set() + + // Proactive token refresh: schedules a timer 5min before the session + // ingress JWT expires. v1 delivers OAuth directly; v2 calls + // reconnectSession to trigger server re-dispatch (CC-1263: without + // this, v2 daemon sessions silently die at ~5h since the server does + // not auto-re-dispatch ACK'd work on lease expiry). + const tokenRefresh = getAccessToken + ? createTokenRefreshScheduler({ + getAccessToken, + onRefresh: (sessionId, oauthToken) => { + const handle = activeSessions.get(sessionId) + if (!handle) { + return + } + if (v2Sessions.has(sessionId)) { + logger.logVerbose( + `Refreshing session ${sessionId} token via bridge/reconnect`, + ) + void api + .reconnectSession(environmentId, sessionId) + .catch((err: unknown) => { + logger.logError( + `Failed to refresh session ${sessionId} token: ${errorMessage(err)}`, + ) + logForDebugging( + `[bridge:token] reconnectSession(${sessionId}) failed: ${errorMessage(err)}`, + { level: 'error' }, + ) + }) + } else { + handle.updateAccessToken(oauthToken) + } + }, + label: 'bridge', + }) + : null + const loopStartTime = Date.now() + // Track all in-flight cleanup promises (stopWork, worktree removal) so + // the shutdown sequence can await them before process.exit(). + const pendingCleanups = new Set>() + function trackCleanup(p: Promise): void { + pendingCleanups.add(p) + void p.finally(() => pendingCleanups.delete(p)) + } + let connBackoff = 0 + let generalBackoff = 0 + let connErrorStart: number | null = null + let generalErrorStart: number | null = null + let lastPollErrorTime: number | null = null + let statusUpdateTimer: ReturnType | null = null + // Set by BridgeFatalError and give-up paths so the shutdown block can + // skip the resume message (resume is impossible after env expiry/auth + // failure/sustained connection errors). + let fatalExit = false + + logForDebugging( + `[bridge:work] Starting poll loop spawnMode=${config.spawnMode} maxSessions=${config.maxSessions} environmentId=${environmentId}`, + ) + logForDiagnosticsNoPII('info', 'bridge_loop_started', { + max_sessions: config.maxSessions, + spawn_mode: config.spawnMode, + }) + + // For ant users, show where session debug logs will land so they can tail them. + // sessionRunner.ts uses the same base path. File appears once a session spawns. + if (process.env.USER_TYPE === 'ant') { + let debugGlob: string + if (config.debugFile) { + const ext = config.debugFile.lastIndexOf('.') + debugGlob = + ext > 0 + ? `${config.debugFile.slice(0, ext)}-*${config.debugFile.slice(ext)}` + : `${config.debugFile}-*` + } else { + debugGlob = join(tmpdir(), 'claude', 'bridge-session-*.log') + } + logger.setDebugLogPath(debugGlob) + } + + logger.printBanner(config, environmentId) + + // Seed the logger's session count + spawn mode before any render. Without + // this, setAttached() below renders with the logger's default sessionMax=1, + // showing "Capacity: 0/1" until the status ticker kicks in (which is gated + // by !initialSessionId and only starts after the poll loop picks up work). + logger.updateSessionCount(0, config.maxSessions, config.spawnMode) + + // If an initial session was pre-created, show its URL from the start so + // the user can click through immediately (matching /remote-control behavior). + if (initialSessionId) { + logger.setAttached(initialSessionId) + } + + /** Refresh the inline status display. Shows idle or active depending on state. */ + function updateStatusDisplay(): void { + // Push the session count (no-op when maxSessions === 1) so the + // next renderStatusLine tick shows the current count. + logger.updateSessionCount( + activeSessions.size, + config.maxSessions, + config.spawnMode, + ) + + // Push per-session activity into the multi-session display. + for (const [sid, handle] of activeSessions) { + const act = handle.currentActivity + if (act) { + logger.updateSessionActivity(sessionCompatIds.get(sid) ?? sid, act) + } + } + + if (activeSessions.size === 0) { + logger.updateIdleStatus() + return + } + + // Show the most recently started session that is still actively working. + // Sessions whose current activity is 'result' or 'error' are between + // turns — the CLI emitted its result but the process stays alive waiting + // for the next user message. Skip updating so the status line keeps + // whatever state it had (Attached / session title). + const [sessionId, handle] = [...activeSessions.entries()].pop()! + const startTime = sessionStartTimes.get(sessionId) + if (!startTime) return + + const activity = handle.currentActivity + if (!activity || activity.type === 'result' || activity.type === 'error') { + // Session is between turns — keep current status (Attached/titled). + // In multi-session mode, still refresh so bullet-list activities stay current. + if (config.maxSessions > 1) logger.refreshDisplay() + return + } + + const elapsed = formatDuration(Date.now() - startTime) + + // Build trail from recent tool activities (last 5) + const trail = handle.activities + .filter(a => a.type === 'tool_start') + .slice(-5) + .map(a => a.summary) + + logger.updateSessionStatus(sessionId, elapsed, activity, trail) + } + + /** Start the status display update ticker. */ + function startStatusUpdates(): void { + stopStatusUpdates() + // Call immediately so the first transition (e.g. Connecting → Ready) + // happens without delay, avoiding concurrent timer races. + updateStatusDisplay() + statusUpdateTimer = setInterval( + updateStatusDisplay, + STATUS_UPDATE_INTERVAL_MS, + ) + } + + /** Stop the status display update ticker. */ + function stopStatusUpdates(): void { + if (statusUpdateTimer) { + clearInterval(statusUpdateTimer) + statusUpdateTimer = null + } + } + + function onSessionDone( + sessionId: string, + startTime: number, + handle: SessionHandle, + ): (status: SessionDoneStatus) => void { + return (rawStatus: SessionDoneStatus): void => { + const workId = sessionWorkIds.get(sessionId) + activeSessions.delete(sessionId) + sessionStartTimes.delete(sessionId) + sessionWorkIds.delete(sessionId) + sessionIngressTokens.delete(sessionId) + const compatId = sessionCompatIds.get(sessionId) ?? sessionId + sessionCompatIds.delete(sessionId) + logger.removeSession(compatId) + titledSessions.delete(compatId) + v2Sessions.delete(sessionId) + // Clear per-session timeout timer + const timer = sessionTimers.get(sessionId) + if (timer) { + clearTimeout(timer) + sessionTimers.delete(sessionId) + } + // Clear token refresh timer + tokenRefresh?.cancel(sessionId) + // Wake the at-capacity sleep so the bridge can accept new work immediately + capacityWake.wake() + + // If the session was killed by the timeout watchdog, treat it as a + // failed session (not a server/shutdown interrupt) so we still call + // stopWork and archiveSession below. + const wasTimedOut = timedOutSessions.delete(sessionId) + const status: SessionDoneStatus = + wasTimedOut && rawStatus === 'interrupted' ? 'failed' : rawStatus + const durationMs = Date.now() - startTime + + logForDebugging( + `[bridge:session] sessionId=${sessionId} workId=${workId ?? 'unknown'} exited status=${status} duration=${formatDuration(durationMs)}`, + ) + logEvent('tengu_bridge_session_done', { + status: + status as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + duration_ms: durationMs, + }) + logForDiagnosticsNoPII('info', 'bridge_session_done', { + status, + duration_ms: durationMs, + }) + + // Clear the status display before printing final log + logger.clearStatus() + stopStatusUpdates() + + // Build error message from stderr if available + const stderrSummary = + handle.lastStderr.length > 0 ? handle.lastStderr.join('\n') : undefined + let failureMessage: string | undefined + + switch (status) { + case 'completed': + logger.logSessionComplete(sessionId, durationMs) + break + case 'failed': + // Skip failure log during shutdown — the child exits non-zero when + // killed, which is expected and not a real failure. + // Also skip for timeout-killed sessions — the timeout watchdog + // already logged a clear timeout message. + if (!wasTimedOut && !loopSignal.aborted) { + failureMessage = stderrSummary ?? 'Process exited with error' + logger.logSessionFailed(sessionId, failureMessage) + logError(new Error(`Bridge session failed: ${failureMessage}`)) + } + break + case 'interrupted': + logger.logVerbose(`Session ${sessionId} interrupted`) + break + } + + // Notify the server that this work item is done. Skip for interrupted + // sessions — interrupts are either server-initiated (the server already + // knows) or caused by bridge shutdown (which calls stopWork() separately). + if (status !== 'interrupted' && workId) { + trackCleanup( + stopWorkWithRetry( + api, + environmentId, + workId, + logger, + backoffConfig.stopWorkBaseDelayMs, + ), + ) + completedWorkIds.add(workId) + } + + // Clean up worktree if one was created for this session + const wt = sessionWorktrees.get(sessionId) + if (wt) { + sessionWorktrees.delete(sessionId) + trackCleanup( + removeAgentWorktree( + wt.worktreePath, + wt.worktreeBranch, + wt.gitRoot, + wt.hookBased, + ).catch((err: unknown) => + logger.logVerbose( + `Failed to remove worktree ${wt.worktreePath}: ${errorMessage(err)}`, + ), + ), + ) + } + + // Lifecycle decision: in multi-session mode, keep the bridge running + // after a session completes. In single-session mode, abort the poll + // loop so the bridge exits cleanly. + if (status !== 'interrupted' && !loopSignal.aborted) { + if (config.spawnMode !== 'single-session') { + // Multi-session: archive the completed session so it doesn't linger + // as stale in the web UI. archiveSession is idempotent (409 if already + // archived), so double-archiving at shutdown is safe. + // sessionId arrived as cse_* from the work poll (infrastructure-layer + // tag). archiveSession hits /v1/sessions/{id}/archive which is the + // compat surface and validates TagSession (session_*). Re-tag — same + // UUID underneath. + trackCleanup( + api + .archiveSession(compatId) + .catch((err: unknown) => + logger.logVerbose( + `Failed to archive session ${sessionId}: ${errorMessage(err)}`, + ), + ), + ) + logForDebugging( + `[bridge:session] Session ${status}, returning to idle (multi-session mode)`, + ) + } else { + // Single-session: coupled lifecycle — tear down environment + logForDebugging( + `[bridge:session] Session ${status}, aborting poll loop to tear down environment`, + ) + controller.abort() + return + } + } + + if (!loopSignal.aborted) { + startStatusUpdates() + } + } + } + + // Start the idle status display immediately — unless we have a pre-created + // session, in which case setAttached() already set up the display and the + // poll loop will start status updates when it picks up the session. + if (!initialSessionId) { + startStatusUpdates() + } + + while (!loopSignal.aborted) { + // Fetched once per iteration — the GrowthBook cache refreshes every + // 5 min, so a loop running at the at-capacity rate picks up config + // changes within one sleep cycle. + const pollConfig = getPollIntervalConfig() + + try { + const work = await api.pollForWork( + environmentId, + environmentSecret, + loopSignal, + pollConfig.reclaim_older_than_ms, + ) + + // Log reconnection if we were previously disconnected + const wasDisconnected = + connErrorStart !== null || generalErrorStart !== null + if (wasDisconnected) { + const disconnectedMs = + Date.now() - (connErrorStart ?? generalErrorStart ?? Date.now()) + logger.logReconnected(disconnectedMs) + logForDebugging( + `[bridge:poll] Reconnected after ${formatDuration(disconnectedMs)}`, + ) + logEvent('tengu_bridge_reconnected', { + disconnected_ms: disconnectedMs, + }) + } + + connBackoff = 0 + generalBackoff = 0 + connErrorStart = null + generalErrorStart = null + lastPollErrorTime = null + + // Null response = no work available in the queue. + // Add a minimum delay to avoid hammering the server. + if (!work) { + // Use live check (not a snapshot) since sessions can end during poll. + const atCap = activeSessions.size >= config.maxSessions + if (atCap) { + const atCapMs = pollConfig.multisession_poll_interval_ms_at_capacity + // Heartbeat loops WITHOUT polling. When at-capacity polling is also + // enabled (atCapMs > 0), the loop tracks a deadline and breaks out + // to poll at that interval — heartbeat and poll compose instead of + // one suppressing the other. We break out to poll when: + // - Poll deadline reached (atCapMs > 0 only) + // - Auth fails (JWT expired → poll refreshes tokens) + // - Capacity wake fires (session ended → poll for new work) + // - Loop aborted (shutdown) + if (pollConfig.non_exclusive_heartbeat_interval_ms > 0) { + logEvent('tengu_bridge_heartbeat_mode_entered', { + active_sessions: activeSessions.size, + heartbeat_interval_ms: + pollConfig.non_exclusive_heartbeat_interval_ms, + }) + // Deadline computed once at entry — GB updates to atCapMs don't + // shift an in-flight deadline (next entry picks up the new value). + const pollDeadline = atCapMs > 0 ? Date.now() + atCapMs : null + let hbResult: 'ok' | 'auth_failed' | 'fatal' | 'failed' = 'ok' + let hbCycles = 0 + while ( + !loopSignal.aborted && + activeSessions.size >= config.maxSessions && + (pollDeadline === null || Date.now() < pollDeadline) + ) { + // Re-read config each cycle so GrowthBook updates take effect + const hbConfig = getPollIntervalConfig() + if (hbConfig.non_exclusive_heartbeat_interval_ms <= 0) break + + // Capture capacity signal BEFORE the async heartbeat call so + // a session ending during the HTTP request is caught by the + // subsequent sleep (instead of being lost to a replaced controller). + const cap = capacityWake.signal() + + hbResult = await heartbeatActiveWorkItems() + if (hbResult === 'auth_failed' || hbResult === 'fatal') { + cap.cleanup() + break + } + + hbCycles++ + await sleep( + hbConfig.non_exclusive_heartbeat_interval_ms, + cap.signal, + ) + cap.cleanup() + } + + // Determine exit reason for telemetry + const exitReason = + hbResult === 'auth_failed' || hbResult === 'fatal' + ? hbResult + : loopSignal.aborted + ? 'shutdown' + : activeSessions.size < config.maxSessions + ? 'capacity_changed' + : pollDeadline !== null && Date.now() >= pollDeadline + ? 'poll_due' + : 'config_disabled' + logEvent('tengu_bridge_heartbeat_mode_exited', { + reason: + exitReason as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + heartbeat_cycles: hbCycles, + active_sessions: activeSessions.size, + }) + if (exitReason === 'poll_due') { + // bridgeApi throttles empty-poll logs (EMPTY_POLL_LOG_INTERVAL=100) + // so the once-per-10min poll_due poll is invisible at counter=2. + // Log it here so verification runs see both endpoints in the debug log. + logForDebugging( + `[bridge:poll] Heartbeat poll_due after ${hbCycles} cycles — falling through to pollForWork`, + ) + } + + // On auth_failed or fatal, sleep before polling to avoid a tight + // poll+heartbeat loop. Auth_failed: heartbeatActiveWorkItems + // already called reconnectSession — the sleep gives the server + // time to propagate the re-queue. Fatal (404/410): may be a + // single work item GCd while the environment is still valid. + // Use atCapMs if enabled, else the heartbeat interval as a floor + // (guaranteed > 0 here) so heartbeat-only configs don't tight-loop. + if (hbResult === 'auth_failed' || hbResult === 'fatal') { + const cap = capacityWake.signal() + await sleep( + atCapMs > 0 + ? atCapMs + : pollConfig.non_exclusive_heartbeat_interval_ms, + cap.signal, + ) + cap.cleanup() + } + } else if (atCapMs > 0) { + // Heartbeat disabled: slow poll as liveness signal. + const cap = capacityWake.signal() + await sleep(atCapMs, cap.signal) + cap.cleanup() + } + } else { + const interval = + activeSessions.size > 0 + ? pollConfig.multisession_poll_interval_ms_partial_capacity + : pollConfig.multisession_poll_interval_ms_not_at_capacity + await sleep(interval, loopSignal) + } + continue + } + + // At capacity — we polled to keep the heartbeat alive, but cannot + // accept new work right now. We still enter the switch below so that + // token refreshes for existing sessions are processed (the case + // 'session' handler checks for existing sessions before the inner + // capacity guard). + const atCapacityBeforeSwitch = activeSessions.size >= config.maxSessions + + // Skip work items that have already been completed and stopped. + // The server may re-deliver stale work before processing our stop + // request, which would otherwise cause a duplicate session spawn. + if (completedWorkIds.has(work.id)) { + logForDebugging( + `[bridge:work] Skipping already-completed workId=${work.id}`, + ) + // Respect capacity throttle — without a sleep here, persistent stale + // redeliveries would tight-loop at poll-request speed (the !work + // branch above is the only sleep, and work != null skips it). + if (atCapacityBeforeSwitch) { + const cap = capacityWake.signal() + if (pollConfig.non_exclusive_heartbeat_interval_ms > 0) { + await heartbeatActiveWorkItems() + await sleep( + pollConfig.non_exclusive_heartbeat_interval_ms, + cap.signal, + ) + } else if (pollConfig.multisession_poll_interval_ms_at_capacity > 0) { + await sleep( + pollConfig.multisession_poll_interval_ms_at_capacity, + cap.signal, + ) + } + cap.cleanup() + } else { + await sleep(1000, loopSignal) + } + continue + } + + // Decode the work secret for session spawning and to extract the JWT + // used for the ack call below. + let secret + try { + secret = decodeWorkSecret(work.secret) + } catch (err) { + const errMsg = errorMessage(err) + logger.logError( + `Failed to decode work secret for workId=${work.id}: ${errMsg}`, + ) + logEvent('tengu_bridge_work_secret_failed', {}) + // Can't ack (needs the JWT we failed to decode). stopWork uses OAuth, + // so it's callable here — prevents XAUTOCLAIM from re-delivering this + // poisoned item every reclaim_older_than_ms cycle. + completedWorkIds.add(work.id) + trackCleanup( + stopWorkWithRetry( + api, + environmentId, + work.id, + logger, + backoffConfig.stopWorkBaseDelayMs, + ), + ) + // Respect capacity throttle before retrying — without a sleep here, + // repeated decode failures at capacity would tight-loop at + // poll-request speed (work != null skips the !work sleep above). + if (atCapacityBeforeSwitch) { + const cap = capacityWake.signal() + if (pollConfig.non_exclusive_heartbeat_interval_ms > 0) { + await heartbeatActiveWorkItems() + await sleep( + pollConfig.non_exclusive_heartbeat_interval_ms, + cap.signal, + ) + } else if (pollConfig.multisession_poll_interval_ms_at_capacity > 0) { + await sleep( + pollConfig.multisession_poll_interval_ms_at_capacity, + cap.signal, + ) + } + cap.cleanup() + } + continue + } + + // Explicitly acknowledge after committing to handle the work — NOT + // before. The at-capacity guard inside case 'session' can break + // without spawning; acking there would permanently lose the work. + // Ack failures are non-fatal: server re-delivers, and existingHandle + // / completedWorkIds paths handle the dedup. + const ackWork = async (): Promise => { + logForDebugging(`[bridge:work] Acknowledging workId=${work.id}`) + try { + await api.acknowledgeWork( + environmentId, + work.id, + secret.session_ingress_token, + ) + } catch (err) { + logForDebugging( + `[bridge:work] Acknowledge failed workId=${work.id}: ${errorMessage(err)}`, + ) + } + } + + const workType: string = work.data.type + switch (work.data.type) { + case 'healthcheck': + await ackWork() + logForDebugging('[bridge:work] Healthcheck received') + logger.logVerbose('Healthcheck received') + break + case 'session': { + const sessionId = work.data.id + try { + validateBridgeId(sessionId, 'session_id') + } catch { + await ackWork() + logger.logError(`Invalid session_id received: ${sessionId}`) + break + } + + // If the session is already running, deliver the fresh token so + // the child process can reconnect its WebSocket with the new + // session ingress token. This handles the case where the server + // re-dispatches work for an existing session after the WS drops. + const existingHandle = activeSessions.get(sessionId) + if (existingHandle) { + existingHandle.updateAccessToken(secret.session_ingress_token) + sessionIngressTokens.set(sessionId, secret.session_ingress_token) + sessionWorkIds.set(sessionId, work.id) + // Re-schedule next refresh from the fresh JWT's expiry. onRefresh + // branches on v2Sessions so both v1 and v2 are safe here. + tokenRefresh?.schedule(sessionId, secret.session_ingress_token) + logForDebugging( + `[bridge:work] Updated access token for existing sessionId=${sessionId} workId=${work.id}`, + ) + await ackWork() + break + } + + // At capacity — token refresh for existing sessions is handled + // above, but we cannot spawn new ones. The post-switch capacity + // sleep will throttle the loop; just break here. + if (activeSessions.size >= config.maxSessions) { + logForDebugging( + `[bridge:work] At capacity (${activeSessions.size}/${config.maxSessions}), cannot spawn new session for workId=${work.id}`, + ) + break + } + + await ackWork() + const spawnStartTime = Date.now() + + // CCR v2 path: register this bridge as the session worker, get the + // epoch, and point the child at /v1/code/sessions/{id}. The child + // already has the full v2 client (SSETransport + CCRClient) — same + // code path environment-manager launches in containers. + // + // v1 path: Session-Ingress WebSocket. Uses config.sessionIngressUrl + // (not secret.api_base_url, which may point to a remote proxy tunnel + // that doesn't know about locally-created sessions). + let sdkUrl: string + let useCcrV2 = false + let workerEpoch: number | undefined + // Server decides per-session via the work secret; env var is the + // ant-dev override (e.g. forcing v2 before the server flag is on). + if ( + secret.use_code_sessions === true || + isEnvTruthy(process.env.CLAUDE_BRIDGE_USE_CCR_V2) + ) { + sdkUrl = buildCCRv2SdkUrl(config.apiBaseUrl, sessionId) + // Retry once on transient failure (network blip, 500) before + // permanently giving up and killing the session. + for (let attempt = 1; attempt <= 2; attempt++) { + try { + workerEpoch = await registerWorker( + sdkUrl, + secret.session_ingress_token, + ) + useCcrV2 = true + logForDebugging( + `[bridge:session] CCR v2: registered worker sessionId=${sessionId} epoch=${workerEpoch} attempt=${attempt}`, + ) + break + } catch (err) { + const errMsg = errorMessage(err) + if (attempt < 2) { + logForDebugging( + `[bridge:session] CCR v2: registerWorker attempt ${attempt} failed, retrying: ${errMsg}`, + ) + await sleep(2_000, loopSignal) + if (loopSignal.aborted) break + continue + } + logger.logError( + `CCR v2 worker registration failed for session ${sessionId}: ${errMsg}`, + ) + logError(new Error(`registerWorker failed: ${errMsg}`)) + completedWorkIds.add(work.id) + trackCleanup( + stopWorkWithRetry( + api, + environmentId, + work.id, + logger, + backoffConfig.stopWorkBaseDelayMs, + ), + ) + } + } + if (!useCcrV2) break + } else { + sdkUrl = buildSdkUrl(config.sessionIngressUrl, sessionId) + } + + // In worktree mode, on-demand sessions get an isolated git worktree + // so concurrent sessions don't interfere with each other's file + // changes. The pre-created initial session (if any) runs in + // config.dir so the user's first session lands in the directory they + // invoked `rc` from — matching the old single-session UX. + // In same-dir and single-session modes, all sessions share config.dir. + // Capture spawnMode before the await below — the `w` key handler + // mutates config.spawnMode directly, and createAgentWorktree can + // take 1-2s, so reading config.spawnMode after the await can + // produce contradictory analytics (spawn_mode:'same-dir', in_worktree:true). + const spawnModeAtDecision = config.spawnMode + let sessionDir = config.dir + let worktreeCreateMs = 0 + if ( + spawnModeAtDecision === 'worktree' && + (initialSessionId === undefined || + !sameSessionId(sessionId, initialSessionId)) + ) { + const wtStart = Date.now() + try { + const wt = await createAgentWorktree( + `bridge-${safeFilenameId(sessionId)}`, + ) + worktreeCreateMs = Date.now() - wtStart + sessionWorktrees.set(sessionId, { + worktreePath: wt.worktreePath, + worktreeBranch: wt.worktreeBranch, + gitRoot: wt.gitRoot, + hookBased: wt.hookBased, + }) + sessionDir = wt.worktreePath + logForDebugging( + `[bridge:session] Created worktree for sessionId=${sessionId} at ${wt.worktreePath}`, + ) + } catch (err) { + const errMsg = errorMessage(err) + logger.logError( + `Failed to create worktree for session ${sessionId}: ${errMsg}`, + ) + logError(new Error(`Worktree creation failed: ${errMsg}`)) + completedWorkIds.add(work.id) + trackCleanup( + stopWorkWithRetry( + api, + environmentId, + work.id, + logger, + backoffConfig.stopWorkBaseDelayMs, + ), + ) + break + } + } + + logForDebugging( + `[bridge:session] Spawning sessionId=${sessionId} sdkUrl=${sdkUrl}`, + ) + + // compat-surface session_* form for logger/Sessions-API calls. + // Work poll returns cse_* under v2 compat; convert before spawn so + // the onFirstUserMessage callback can close over it. + const compatSessionId = toCompatSessionId(sessionId) + + const spawnResult = safeSpawn( + spawner, + { + sessionId, + sdkUrl, + accessToken: secret.session_ingress_token, + useCcrV2, + workerEpoch, + onFirstUserMessage: text => { + // Server-set titles (--name, web rename) win. fetchSessionTitle + // runs concurrently; if it already populated titledSessions, + // skip. If it hasn't resolved yet, the derived title sticks — + // acceptable since the server had no title at spawn time. + if (titledSessions.has(compatSessionId)) return + titledSessions.add(compatSessionId) + const title = deriveSessionTitle(text) + logger.setSessionTitle(compatSessionId, title) + logForDebugging( + `[bridge:title] derived title for ${compatSessionId}: ${title}`, + ) + void import('./createSession.js') + .then(({ updateBridgeSessionTitle }) => + updateBridgeSessionTitle(compatSessionId, title, { + baseUrl: config.apiBaseUrl, + }), + ) + .catch(err => + logForDebugging( + `[bridge:title] failed to update title for ${compatSessionId}: ${err}`, + { level: 'error' }, + ), + ) + }, + }, + sessionDir, + ) + if (typeof spawnResult === 'string') { + logger.logError( + `Failed to spawn session ${sessionId}: ${spawnResult}`, + ) + // Clean up worktree if one was created for this session + const wt = sessionWorktrees.get(sessionId) + if (wt) { + sessionWorktrees.delete(sessionId) + trackCleanup( + removeAgentWorktree( + wt.worktreePath, + wt.worktreeBranch, + wt.gitRoot, + wt.hookBased, + ).catch((err: unknown) => + logger.logVerbose( + `Failed to remove worktree ${wt.worktreePath}: ${errorMessage(err)}`, + ), + ), + ) + } + completedWorkIds.add(work.id) + trackCleanup( + stopWorkWithRetry( + api, + environmentId, + work.id, + logger, + backoffConfig.stopWorkBaseDelayMs, + ), + ) + break + } + const handle = spawnResult + + const spawnDurationMs = Date.now() - spawnStartTime + logEvent('tengu_bridge_session_started', { + active_sessions: activeSessions.size, + spawn_mode: + spawnModeAtDecision as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + in_worktree: sessionWorktrees.has(sessionId), + spawn_duration_ms: spawnDurationMs, + worktree_create_ms: worktreeCreateMs, + inProtectedNamespace: isInProtectedNamespace(), + }) + logForDiagnosticsNoPII('info', 'bridge_session_started', { + spawn_mode: spawnModeAtDecision, + in_worktree: sessionWorktrees.has(sessionId), + spawn_duration_ms: spawnDurationMs, + worktree_create_ms: worktreeCreateMs, + }) + + activeSessions.set(sessionId, handle) + sessionWorkIds.set(sessionId, work.id) + sessionIngressTokens.set(sessionId, secret.session_ingress_token) + sessionCompatIds.set(sessionId, compatSessionId) + + const startTime = Date.now() + sessionStartTimes.set(sessionId, startTime) + + // Use a generic prompt description since we no longer get startup_context + logger.logSessionStart(sessionId, `Session ${sessionId}`) + + // Compute the actual debug file path (mirrors sessionRunner.ts logic) + const safeId = safeFilenameId(sessionId) + let sessionDebugFile: string | undefined + if (config.debugFile) { + const ext = config.debugFile.lastIndexOf('.') + if (ext > 0) { + sessionDebugFile = `${config.debugFile.slice(0, ext)}-${safeId}${config.debugFile.slice(ext)}` + } else { + sessionDebugFile = `${config.debugFile}-${safeId}` + } + } else if (config.verbose || process.env.USER_TYPE === 'ant') { + sessionDebugFile = join( + tmpdir(), + 'claude', + `bridge-session-${safeId}.log`, + ) + } + + if (sessionDebugFile) { + logger.logVerbose(`Debug log: ${sessionDebugFile}`) + } + + // Register in the sessions Map before starting status updates so the + // first render tick shows the correct count and bullet list in sync. + logger.addSession( + compatSessionId, + getRemoteSessionUrl(compatSessionId, config.sessionIngressUrl), + ) + + // Start live status updates and transition to "Attached" state. + startStatusUpdates() + logger.setAttached(compatSessionId) + + // One-shot title fetch. If the session already has a title (set via + // --name, web rename, or /remote-control), display it and mark as + // titled so the first-user-message fallback doesn't overwrite it. + // Otherwise onFirstUserMessage derives one from the first prompt. + void fetchSessionTitle(compatSessionId, config.apiBaseUrl) + .then(title => { + if (title && activeSessions.has(sessionId)) { + titledSessions.add(compatSessionId) + logger.setSessionTitle(compatSessionId, title) + logForDebugging( + `[bridge:title] server title for ${compatSessionId}: ${title}`, + ) + } + }) + .catch(err => + logForDebugging( + `[bridge:title] failed to fetch title for ${compatSessionId}: ${err}`, + { level: 'error' }, + ), + ) + + // Start per-session timeout watchdog + const timeoutMs = + config.sessionTimeoutMs ?? DEFAULT_SESSION_TIMEOUT_MS + if (timeoutMs > 0) { + const timer = setTimeout( + onSessionTimeout, + timeoutMs, + sessionId, + timeoutMs, + logger, + timedOutSessions, + handle, + ) + sessionTimers.set(sessionId, timer) + } + + // Schedule proactive token refresh before the JWT expires. + // onRefresh branches on v2Sessions: v1 delivers OAuth to the + // child, v2 triggers server re-dispatch via reconnectSession. + if (useCcrV2) { + v2Sessions.add(sessionId) + } + tokenRefresh?.schedule(sessionId, secret.session_ingress_token) + + void handle.done.then(onSessionDone(sessionId, startTime, handle)) + break + } + default: + await ackWork() + // Gracefully ignore unknown work types. The backend may send new + // types before the bridge client is updated. + logForDebugging( + `[bridge:work] Unknown work type: ${workType}, skipping`, + ) + break + } + + // When at capacity, throttle the loop. The switch above still runs so + // existing-session token refreshes are processed, but we sleep here + // to avoid busy-looping. Include the capacity wake signal so the + // sleep is interrupted immediately when a session completes. + if (atCapacityBeforeSwitch) { + const cap = capacityWake.signal() + if (pollConfig.non_exclusive_heartbeat_interval_ms > 0) { + await heartbeatActiveWorkItems() + await sleep( + pollConfig.non_exclusive_heartbeat_interval_ms, + cap.signal, + ) + } else if (pollConfig.multisession_poll_interval_ms_at_capacity > 0) { + await sleep( + pollConfig.multisession_poll_interval_ms_at_capacity, + cap.signal, + ) + } + cap.cleanup() + } + } catch (err) { + if (loopSignal.aborted) { + break + } + + // Fatal errors (401/403) — no point retrying, auth won't fix itself + if (err instanceof BridgeFatalError) { + fatalExit = true + // Server-enforced expiry gets a clean status message, not an error + if (isExpiredErrorType(err.errorType)) { + logger.logStatus(err.message) + } else if (isSuppressible403(err)) { + // Cosmetic 403 errors (e.g., external_poll_sessions scope, + // environments:manage permission) — don't show to user + logForDebugging(`[bridge:work] Suppressed 403 error: ${err.message}`) + } else { + logger.logError(err.message) + logError(err) + } + logEvent('tengu_bridge_fatal_error', { + status: err.status, + error_type: + err.errorType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + logForDiagnosticsNoPII( + isExpiredErrorType(err.errorType) ? 'info' : 'error', + 'bridge_fatal_error', + { status: err.status, error_type: err.errorType }, + ) + break + } + + const errMsg = describeAxiosError(err) + + if (isConnectionError(err) || isServerError(err)) { + const now = Date.now() + + // Detect system sleep/wake: if the gap since the last poll error + // greatly exceeds the expected backoff, the machine likely slept. + // Reset error tracking so the bridge retries with a fresh budget. + if ( + lastPollErrorTime !== null && + now - lastPollErrorTime > pollSleepDetectionThresholdMs(backoffConfig) + ) { + logForDebugging( + `[bridge:work] Detected system sleep (${Math.round((now - lastPollErrorTime) / 1000)}s gap), resetting error budget`, + ) + logForDiagnosticsNoPII('info', 'bridge_poll_sleep_detected', { + gapMs: now - lastPollErrorTime, + }) + connErrorStart = null + connBackoff = 0 + generalErrorStart = null + generalBackoff = 0 + } + lastPollErrorTime = now + + if (!connErrorStart) { + connErrorStart = now + } + const elapsed = now - connErrorStart + if (elapsed >= backoffConfig.connGiveUpMs) { + logger.logError( + `Server unreachable for ${Math.round(elapsed / 60_000)} minutes, giving up.`, + ) + logEvent('tengu_bridge_poll_give_up', { + error_type: + 'connection' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + elapsed_ms: elapsed, + }) + logForDiagnosticsNoPII('error', 'bridge_poll_give_up', { + error_type: 'connection', + elapsed_ms: elapsed, + }) + fatalExit = true + break + } + + // Reset the other track when switching error types + generalErrorStart = null + generalBackoff = 0 + + connBackoff = connBackoff + ? Math.min(connBackoff * 2, backoffConfig.connCapMs) + : backoffConfig.connInitialMs + const delay = addJitter(connBackoff) + logger.logVerbose( + `Connection error, retrying in ${formatDelay(delay)} (${Math.round(elapsed / 1000)}s elapsed): ${errMsg}`, + ) + logger.updateReconnectingStatus( + formatDelay(delay), + formatDuration(elapsed), + ) + // The poll_due heartbeat-loop exit leaves a healthy lease exposed to + // this backoff path. Heartbeat before each sleep so /poll outages + // (the VerifyEnvironmentSecretAuth DB path heartbeat was introduced + // to avoid) don't kill the 300s lease TTL. No-op when activeSessions + // is empty or heartbeat is disabled. + if (getPollIntervalConfig().non_exclusive_heartbeat_interval_ms > 0) { + await heartbeatActiveWorkItems() + } + await sleep(delay, loopSignal) + } else { + const now = Date.now() + + // Sleep detection for general errors (same logic as connection errors) + if ( + lastPollErrorTime !== null && + now - lastPollErrorTime > pollSleepDetectionThresholdMs(backoffConfig) + ) { + logForDebugging( + `[bridge:work] Detected system sleep (${Math.round((now - lastPollErrorTime) / 1000)}s gap), resetting error budget`, + ) + logForDiagnosticsNoPII('info', 'bridge_poll_sleep_detected', { + gapMs: now - lastPollErrorTime, + }) + connErrorStart = null + connBackoff = 0 + generalErrorStart = null + generalBackoff = 0 + } + lastPollErrorTime = now + + if (!generalErrorStart) { + generalErrorStart = now + } + const elapsed = now - generalErrorStart + if (elapsed >= backoffConfig.generalGiveUpMs) { + logger.logError( + `Persistent errors for ${Math.round(elapsed / 60_000)} minutes, giving up.`, + ) + logEvent('tengu_bridge_poll_give_up', { + error_type: + 'general' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + elapsed_ms: elapsed, + }) + logForDiagnosticsNoPII('error', 'bridge_poll_give_up', { + error_type: 'general', + elapsed_ms: elapsed, + }) + fatalExit = true + break + } + + // Reset the other track when switching error types + connErrorStart = null + connBackoff = 0 + + generalBackoff = generalBackoff + ? Math.min(generalBackoff * 2, backoffConfig.generalCapMs) + : backoffConfig.generalInitialMs + const delay = addJitter(generalBackoff) + logger.logVerbose( + `Poll failed, retrying in ${formatDelay(delay)} (${Math.round(elapsed / 1000)}s elapsed): ${errMsg}`, + ) + logger.updateReconnectingStatus( + formatDelay(delay), + formatDuration(elapsed), + ) + if (getPollIntervalConfig().non_exclusive_heartbeat_interval_ms > 0) { + await heartbeatActiveWorkItems() + } + await sleep(delay, loopSignal) + } + } + } + + // Clean up + stopStatusUpdates() + logger.clearStatus() + + const loopDurationMs = Date.now() - loopStartTime + logEvent('tengu_bridge_shutdown', { + active_sessions: activeSessions.size, + loop_duration_ms: loopDurationMs, + }) + logForDiagnosticsNoPII('info', 'bridge_shutdown', { + active_sessions: activeSessions.size, + loop_duration_ms: loopDurationMs, + }) + + // Graceful shutdown: kill active sessions, report them as interrupted, + // archive sessions, then deregister the environment so the web UI shows + // the bridge as offline. + + // Collect all session IDs to archive on exit. This includes: + // 1. Active sessions (snapshot before killing — onSessionDone clears maps) + // 2. The initial auto-created session (may never have had work dispatched) + // api.archiveSession is idempotent (409 if already archived), so + // double-archiving is safe. + const sessionsToArchive = new Set(activeSessions.keys()) + if (initialSessionId) { + sessionsToArchive.add(initialSessionId) + } + // Snapshot before killing — onSessionDone clears sessionCompatIds. + const compatIdSnapshot = new Map(sessionCompatIds) + + if (activeSessions.size > 0) { + logForDebugging( + `[bridge:shutdown] Shutting down ${activeSessions.size} active session(s)`, + ) + logger.logStatus( + `Shutting down ${activeSessions.size} active session(s)\u2026`, + ) + + // Snapshot work IDs before killing — onSessionDone clears the maps when + // each child exits, so we need a copy for the stopWork calls below. + const shutdownWorkIds = new Map(sessionWorkIds) + + for (const [sessionId, handle] of activeSessions.entries()) { + logForDebugging( + `[bridge:shutdown] Sending SIGTERM to sessionId=${sessionId}`, + ) + handle.kill() + } + + const timeout = new AbortController() + await Promise.race([ + Promise.allSettled([...activeSessions.values()].map(h => h.done)), + sleep(backoffConfig.shutdownGraceMs ?? 30_000, timeout.signal), + ]) + timeout.abort() + + // SIGKILL any processes that didn't respond to SIGTERM within the grace window + for (const [sid, handle] of activeSessions.entries()) { + logForDebugging(`[bridge:shutdown] Force-killing stuck sessionId=${sid}`) + handle.forceKill() + } + + // Clear any remaining session timeout and refresh timers + for (const timer of sessionTimers.values()) { + clearTimeout(timer) + } + sessionTimers.clear() + tokenRefresh?.cancelAll() + + // Clean up any remaining worktrees from active sessions. + // Snapshot and clear the map first so onSessionDone (which may fire + // during the await below when handle.done resolves) won't try to + // remove the same worktrees again. + if (sessionWorktrees.size > 0) { + const remainingWorktrees = [...sessionWorktrees.values()] + sessionWorktrees.clear() + logForDebugging( + `[bridge:shutdown] Cleaning up ${remainingWorktrees.length} worktree(s)`, + ) + await Promise.allSettled( + remainingWorktrees.map(wt => + removeAgentWorktree( + wt.worktreePath, + wt.worktreeBranch, + wt.gitRoot, + wt.hookBased, + ), + ), + ) + } + + // Stop all active work items so the server knows they're done + await Promise.allSettled( + [...shutdownWorkIds.entries()].map(([sessionId, workId]) => { + return api + .stopWork(environmentId, workId, true) + .catch(err => + logger.logVerbose( + `Failed to stop work ${workId} for session ${sessionId}: ${errorMessage(err)}`, + ), + ) + }), + ) + } + + // Ensure all in-flight cleanup (stopWork, worktree removal) from + // onSessionDone completes before deregistering — otherwise + // process.exit() can kill them mid-flight. + if (pendingCleanups.size > 0) { + await Promise.allSettled([...pendingCleanups]) + } + + // In single-session mode with a known session, leave the session and + // environment alive so `claude remote-control --session-id=` can resume. + // The backend GCs stale environments via a 4h TTL (BRIDGE_LAST_POLL_TTL). + // Archiving the session or deregistering the environment would make the + // printed resume command a lie — deregister deletes Firestore + Redis stream. + // Skip when the loop exited fatally (env expired, auth failed, give-up) — + // resume is impossible in those cases and the message would contradict the + // error already printed. + // feature('KAIROS') gate: --session-id is ant-only; without the gate, + // revert to the pre-PR behavior (archive + deregister on every shutdown). + if ( + feature('KAIROS') && + config.spawnMode === 'single-session' && + initialSessionId && + !fatalExit + ) { + logger.logStatus( + `Resume this session by running \`claude remote-control --continue\``, + ) + logForDebugging( + `[bridge:shutdown] Skipping archive+deregister to allow resume of session ${initialSessionId}`, + ) + return + } + + // Archive all known sessions so they don't linger as idle/running on the + // server after the bridge goes offline. + if (sessionsToArchive.size > 0) { + logForDebugging( + `[bridge:shutdown] Archiving ${sessionsToArchive.size} session(s)`, + ) + await Promise.allSettled( + [...sessionsToArchive].map(sessionId => + api + .archiveSession( + compatIdSnapshot.get(sessionId) ?? toCompatSessionId(sessionId), + ) + .catch(err => + logger.logVerbose( + `Failed to archive session ${sessionId}: ${errorMessage(err)}`, + ), + ), + ), + ) + } + + // Deregister the environment so the web UI shows the bridge as offline + // and the Redis stream is cleaned up. + try { + await api.deregisterEnvironment(environmentId) + logForDebugging( + `[bridge:shutdown] Environment deregistered, bridge offline`, + ) + logger.logVerbose('Environment deregistered.') + } catch (err) { + logger.logVerbose(`Failed to deregister environment: ${errorMessage(err)}`) + } + + // Clear the crash-recovery pointer — the env is gone, pointer would be + // stale. The early return above (resumable SIGINT shutdown) skips this, + // leaving the pointer as a backup for the printed --session-id hint. + const { clearBridgePointer } = await import('./bridgePointer.js') + await clearBridgePointer(config.dir) + + logger.logVerbose('Environment offline.') +} + +const CONNECTION_ERROR_CODES = new Set([ + 'ECONNREFUSED', + 'ECONNRESET', + 'ETIMEDOUT', + 'ENETUNREACH', + 'EHOSTUNREACH', +]) + +export function isConnectionError(err: unknown): boolean { + if ( + err && + typeof err === 'object' && + 'code' in err && + typeof err.code === 'string' && + CONNECTION_ERROR_CODES.has(err.code) + ) { + return true + } + return false +} + +/** Detect HTTP 5xx errors from axios (code: 'ERR_BAD_RESPONSE'). */ +export function isServerError(err: unknown): boolean { + return ( + !!err && + typeof err === 'object' && + 'code' in err && + typeof err.code === 'string' && + err.code === 'ERR_BAD_RESPONSE' + ) +} + +/** Add ±25% jitter to a delay value. */ +function addJitter(ms: number): number { + return Math.max(0, ms + ms * 0.25 * (2 * Math.random() - 1)) +} + +function formatDelay(ms: number): string { + return ms >= 1000 ? `${(ms / 1000).toFixed(1)}s` : `${Math.round(ms)}ms` +} + +/** + * Retry stopWork with exponential backoff (3 attempts, 1s/2s/4s). + * Ensures the server learns the work item ended, preventing server-side zombies. + */ +async function stopWorkWithRetry( + api: BridgeApiClient, + environmentId: string, + workId: string, + logger: BridgeLogger, + baseDelayMs = 1000, +): Promise { + const MAX_ATTEMPTS = 3 + + for (let attempt = 1; attempt <= MAX_ATTEMPTS; attempt++) { + try { + await api.stopWork(environmentId, workId, false) + logForDebugging( + `[bridge:work] stopWork succeeded for workId=${workId} on attempt ${attempt}/${MAX_ATTEMPTS}`, + ) + return + } catch (err) { + // Auth/permission errors won't be fixed by retrying + if (err instanceof BridgeFatalError) { + if (isSuppressible403(err)) { + logForDebugging( + `[bridge:work] Suppressed stopWork 403 for ${workId}: ${err.message}`, + ) + } else { + logger.logError(`Failed to stop work ${workId}: ${err.message}`) + } + logForDiagnosticsNoPII('error', 'bridge_stop_work_failed', { + attempts: attempt, + fatal: true, + }) + return + } + const errMsg = errorMessage(err) + if (attempt < MAX_ATTEMPTS) { + const delay = addJitter(baseDelayMs * Math.pow(2, attempt - 1)) + logger.logVerbose( + `Failed to stop work ${workId} (attempt ${attempt}/${MAX_ATTEMPTS}), retrying in ${formatDelay(delay)}: ${errMsg}`, + ) + await sleep(delay) + } else { + logger.logError( + `Failed to stop work ${workId} after ${MAX_ATTEMPTS} attempts: ${errMsg}`, + ) + logForDiagnosticsNoPII('error', 'bridge_stop_work_failed', { + attempts: MAX_ATTEMPTS, + }) + } + } + } +} + +function onSessionTimeout( + sessionId: string, + timeoutMs: number, + logger: BridgeLogger, + timedOutSessions: Set, + handle: SessionHandle, +): void { + logForDebugging( + `[bridge:session] sessionId=${sessionId} timed out after ${formatDuration(timeoutMs)}`, + ) + logEvent('tengu_bridge_session_timeout', { + timeout_ms: timeoutMs, + }) + logger.logSessionFailed( + sessionId, + `Session timed out after ${formatDuration(timeoutMs)}`, + ) + timedOutSessions.add(sessionId) + handle.kill() +} + +export type ParsedArgs = { + verbose: boolean + sandbox: boolean + debugFile?: string + sessionTimeoutMs?: number + permissionMode?: string + name?: string + /** Value passed to --spawn (if any); undefined if no --spawn flag was given. */ + spawnMode: SpawnMode | undefined + /** Value passed to --capacity (if any); undefined if no --capacity flag was given. */ + capacity: number | undefined + /** --[no-]create-session-in-dir override; undefined = use default (on). */ + createSessionInDir: boolean | undefined + /** Resume an existing session instead of creating a new one. */ + sessionId?: string + /** Resume the last session in this directory (reads bridge-pointer.json). */ + continueSession: boolean + help: boolean + error?: string +} + +const SPAWN_FLAG_VALUES = ['session', 'same-dir', 'worktree'] as const + +function parseSpawnValue(raw: string | undefined): SpawnMode | string { + if (raw === 'session') return 'single-session' + if (raw === 'same-dir') return 'same-dir' + if (raw === 'worktree') return 'worktree' + return `--spawn requires one of: ${SPAWN_FLAG_VALUES.join(', ')} (got: ${raw ?? ''})` +} + +function parseCapacityValue(raw: string | undefined): number | string { + const n = raw === undefined ? NaN : parseInt(raw, 10) + if (isNaN(n) || n < 1) { + return `--capacity requires a positive integer (got: ${raw ?? ''})` + } + return n +} + +export function parseArgs(args: string[]): ParsedArgs { + let verbose = false + let sandbox = false + let debugFile: string | undefined + let sessionTimeoutMs: number | undefined + let permissionMode: string | undefined + let name: string | undefined + let help = false + let spawnMode: SpawnMode | undefined + let capacity: number | undefined + let createSessionInDir: boolean | undefined + let sessionId: string | undefined + let continueSession = false + + for (let i = 0; i < args.length; i++) { + const arg = args[i]! + if (arg === '--help' || arg === '-h') { + help = true + } else if (arg === '--verbose' || arg === '-v') { + verbose = true + } else if (arg === '--sandbox') { + sandbox = true + } else if (arg === '--no-sandbox') { + sandbox = false + } else if (arg === '--debug-file' && i + 1 < args.length) { + debugFile = resolve(args[++i]!) + } else if (arg.startsWith('--debug-file=')) { + debugFile = resolve(arg.slice('--debug-file='.length)) + } else if (arg === '--session-timeout' && i + 1 < args.length) { + sessionTimeoutMs = parseInt(args[++i]!, 10) * 1000 + } else if (arg.startsWith('--session-timeout=')) { + sessionTimeoutMs = + parseInt(arg.slice('--session-timeout='.length), 10) * 1000 + } else if (arg === '--permission-mode' && i + 1 < args.length) { + permissionMode = args[++i]! + } else if (arg.startsWith('--permission-mode=')) { + permissionMode = arg.slice('--permission-mode='.length) + } else if (arg === '--name' && i + 1 < args.length) { + name = args[++i]! + } else if (arg.startsWith('--name=')) { + name = arg.slice('--name='.length) + } else if ( + feature('KAIROS') && + arg === '--session-id' && + i + 1 < args.length + ) { + sessionId = args[++i]! + if (!sessionId) { + return makeError('--session-id requires a value') + } + } else if (feature('KAIROS') && arg.startsWith('--session-id=')) { + sessionId = arg.slice('--session-id='.length) + if (!sessionId) { + return makeError('--session-id requires a value') + } + } else if (feature('KAIROS') && (arg === '--continue' || arg === '-c')) { + continueSession = true + } else if (arg === '--spawn' || arg.startsWith('--spawn=')) { + if (spawnMode !== undefined) { + return makeError('--spawn may only be specified once') + } + const raw = arg.startsWith('--spawn=') + ? arg.slice('--spawn='.length) + : args[++i] + const v = parseSpawnValue(raw) + if (v === 'single-session' || v === 'same-dir' || v === 'worktree') { + spawnMode = v + } else { + return makeError(v) + } + } else if (arg === '--capacity' || arg.startsWith('--capacity=')) { + if (capacity !== undefined) { + return makeError('--capacity may only be specified once') + } + const raw = arg.startsWith('--capacity=') + ? arg.slice('--capacity='.length) + : args[++i] + const v = parseCapacityValue(raw) + if (typeof v === 'number') capacity = v + else return makeError(v) + } else if (arg === '--create-session-in-dir') { + createSessionInDir = true + } else if (arg === '--no-create-session-in-dir') { + createSessionInDir = false + } else { + return makeError( + `Unknown argument: ${arg}\nRun 'claude remote-control --help' for usage.`, + ) + } + } + + // Note: gate check for --spawn/--capacity/--create-session-in-dir is in bridgeMain + // (gate-aware error). Flag cross-validation happens here. + + // --capacity only makes sense for multi-session modes. + if (spawnMode === 'single-session' && capacity !== undefined) { + return makeError( + `--capacity cannot be used with --spawn=session (single-session mode has fixed capacity 1).`, + ) + } + + // --session-id / --continue resume a specific session on its original + // environment; incompatible with spawn-related flags (which configure + // fresh session creation), and mutually exclusive with each other. + if ( + (sessionId || continueSession) && + (spawnMode !== undefined || + capacity !== undefined || + createSessionInDir !== undefined) + ) { + return makeError( + `--session-id and --continue cannot be used with --spawn, --capacity, or --create-session-in-dir.`, + ) + } + if (sessionId && continueSession) { + return makeError(`--session-id and --continue cannot be used together.`) + } + + return { + verbose, + sandbox, + debugFile, + sessionTimeoutMs, + permissionMode, + name, + spawnMode, + capacity, + createSessionInDir, + sessionId, + continueSession, + help, + } + + function makeError(error: string): ParsedArgs { + return { + verbose, + sandbox, + debugFile, + sessionTimeoutMs, + permissionMode, + name, + spawnMode, + capacity, + createSessionInDir, + sessionId, + continueSession, + help, + error, + } + } +} + +async function printHelp(): Promise { + // Use EXTERNAL_PERMISSION_MODES for help text — internal modes (bubble) + // are ant-only and auto is feature-gated; they're still accepted by validation. + const { EXTERNAL_PERMISSION_MODES } = await import('../types/permissions.js') + const modes = EXTERNAL_PERMISSION_MODES.join(', ') + const showServer = await isMultiSessionSpawnEnabled() + const serverOptions = showServer + ? ` --spawn Spawn mode: same-dir, worktree, session + (default: same-dir) + --capacity Max concurrent sessions in worktree or + same-dir mode (default: ${SPAWN_SESSIONS_DEFAULT}) + --[no-]create-session-in-dir Pre-create a session in the current + directory; in worktree mode this session + stays in cwd while on-demand sessions get + isolated worktrees (default: on) +` + : '' + const serverDescription = showServer + ? ` + Remote Control runs as a persistent server that accepts multiple concurrent + sessions in the current directory. One session is pre-created on start so + you have somewhere to type immediately. Use --spawn=worktree to isolate + each on-demand session in its own git worktree, or --spawn=session for + the classic single-session mode (exits when that session ends). Press 'w' + during runtime to toggle between same-dir and worktree. +` + : '' + const serverNote = showServer + ? ` - Worktree mode requires a git repository or WorktreeCreate/WorktreeRemove hooks +` + : '' + const help = ` +Remote Control - Connect your local environment to claude.ai/code + +USAGE + claude remote-control [options] +OPTIONS + --name Name for the session (shown in claude.ai/code) +${ + feature('KAIROS') + ? ` -c, --continue Resume the last session in this directory + --session-id Resume a specific session by ID (cannot be + used with spawn flags or --continue) +` + : '' +} --permission-mode Permission mode for spawned sessions + (${modes}) + --debug-file Write debug logs to file + -v, --verbose Enable verbose output + -h, --help Show this help +${serverOptions} +DESCRIPTION + Remote Control allows you to control sessions on your local device from + claude.ai/code (https://claude.ai/code). Run this command in the + directory you want to work in, then connect from the Claude app or web. +${serverDescription} +NOTES + - You must be logged in with a Claude account that has a subscription + - Run \`claude\` first in the directory to accept the workspace trust dialog +${serverNote}` + // biome-ignore lint/suspicious/noConsole: intentional help output + console.log(help) +} + +const TITLE_MAX_LEN = 80 + +/** Derive a session title from a user message: first line, truncated. */ +function deriveSessionTitle(text: string): string { + // Collapse whitespace — newlines/tabs would break the single-line status display. + const flat = text.replace(/\s+/g, ' ').trim() + return truncateToWidth(flat, TITLE_MAX_LEN) +} + +/** + * One-shot fetch of a session's title via GET /v1/sessions/{id}. + * + * Uses `getBridgeSession` from createSession.ts (ccr-byoc headers + org UUID) + * rather than the environments-level bridgeApi client, whose headers make the + * Sessions API return 404. Returns undefined if the session has no title yet + * or the fetch fails — the caller falls back to deriving a title from the + * first user message. + */ +async function fetchSessionTitle( + compatSessionId: string, + baseUrl: string, +): Promise { + const { getBridgeSession } = await import('./createSession.js') + const session = await getBridgeSession(compatSessionId, { baseUrl }) + return session?.title || undefined +} + +export async function bridgeMain(args: string[]): Promise { + const parsed = parseArgs(args) + + if (parsed.help) { + await printHelp() + return + } + if (parsed.error) { + // biome-ignore lint/suspicious/noConsole: intentional error output + console.error(`Error: ${parsed.error}`) + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(1) + } + + const { + verbose, + sandbox, + debugFile, + sessionTimeoutMs, + permissionMode, + name, + spawnMode: parsedSpawnMode, + capacity: parsedCapacity, + createSessionInDir: parsedCreateSessionInDir, + sessionId: parsedSessionId, + continueSession, + } = parsed + // Mutable so --continue can set it from the pointer file. The #20460 + // resume flow below then treats it the same as an explicit --session-id. + let resumeSessionId = parsedSessionId + // When --continue found a pointer, this is the directory it came from + // (may be a worktree sibling, not `dir`). On resume-flow deterministic + // failure, clear THIS file so --continue doesn't keep hitting the same + // dead session. Undefined for explicit --session-id (leaves pointer alone). + let resumePointerDir: string | undefined + + const usedMultiSessionFeature = + parsedSpawnMode !== undefined || + parsedCapacity !== undefined || + parsedCreateSessionInDir !== undefined + + // Validate permission mode early so the user gets an error before + // the bridge starts polling for work. + if (permissionMode !== undefined) { + const { PERMISSION_MODES } = await import('../types/permissions.js') + const valid: readonly string[] = PERMISSION_MODES + if (!valid.includes(permissionMode)) { + // biome-ignore lint/suspicious/noConsole: intentional error output + console.error( + `Error: Invalid permission mode '${permissionMode}'. Valid modes: ${valid.join(', ')}`, + ) + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(1) + } + } + + const dir = resolve('.') + + // The bridge fast-path bypasses init.ts, so we must enable config reading + // before any code that transitively calls getGlobalConfig() + const { enableConfigs, checkHasTrustDialogAccepted } = await import( + '../utils/config.js' + ) + enableConfigs() + + // Initialize analytics and error reporting sinks. The bridge bypasses the + // setup() init flow, so we call initSinks() directly to attach sinks here. + const { initSinks } = await import('../utils/sinks.js') + initSinks() + + // Gate-aware validation: --spawn / --capacity / --create-session-in-dir require + // the multi-session gate. parseArgs has already validated flag combinations; + // here we only check the gate since that requires an async GrowthBook call. + // Runs after enableConfigs() (GrowthBook cache reads global config) and after + // initSinks() so the denial event can be enqueued. + const multiSessionEnabled = await isMultiSessionSpawnEnabled() + if (usedMultiSessionFeature && !multiSessionEnabled) { + await logEventAsync('tengu_bridge_multi_session_denied', { + used_spawn: parsedSpawnMode !== undefined, + used_capacity: parsedCapacity !== undefined, + used_create_session_in_dir: parsedCreateSessionInDir !== undefined, + }) + // logEventAsync only enqueues — process.exit() discards buffered events. + // Flush explicitly, capped at 500ms to match gracefulShutdown.ts. + // (sleep() doesn't unref its timer, but process.exit() follows immediately + // so the ref'd timer can't delay shutdown.) + await Promise.race([ + Promise.all([shutdown1PEventLogging(), shutdownDatadog()]), + sleep(500, undefined, { unref: true }), + ]).catch(() => {}) + // biome-ignore lint/suspicious/noConsole: intentional error output + console.error( + 'Error: Multi-session Remote Control is not enabled for your account yet.', + ) + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(1) + } + + // Set the bootstrap CWD so that trust checks, project config lookups, and + // git utilities (getBranch, getRemoteUrl) resolve against the correct path. + const { setOriginalCwd, setCwdState } = await import('../bootstrap/state.js') + setOriginalCwd(dir) + setCwdState(dir) + + // The bridge bypasses main.tsx (which renders the interactive TrustDialog via showSetupScreens), + // so we must verify trust was previously established by a normal `claude` session. + if (!checkHasTrustDialogAccepted()) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.error( + `Error: Workspace not trusted. Please run \`claude\` in ${dir} first to review and accept the workspace trust dialog.`, + ) + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(1) + } + + // Resolve auth + const { clearOAuthTokenCache, checkAndRefreshOAuthTokenIfNeeded } = + await import('../utils/auth.js') + const { getBridgeAccessToken, getBridgeBaseUrl } = await import( + './bridgeConfig.js' + ) + + const bridgeToken = getBridgeAccessToken() + if (!bridgeToken) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.error(BRIDGE_LOGIN_ERROR) + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(1) + } + + // First-time remote dialog — explain what bridge does and get consent + const { + getGlobalConfig, + saveGlobalConfig, + getCurrentProjectConfig, + saveCurrentProjectConfig, + } = await import('../utils/config.js') + if (!getGlobalConfig().remoteDialogSeen) { + const readline = await import('readline') + const rl = readline.createInterface({ + input: process.stdin, + output: process.stdout, + }) + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log( + '\nRemote Control lets you access this CLI session from the web (claude.ai/code)\nor the Claude app, so you can pick up where you left off on any device.\n\nYou can disconnect remote access anytime by running /remote-control again.\n', + ) + const answer = await new Promise(resolve => { + rl.question('Enable Remote Control? (y/n) ', resolve) + }) + rl.close() + saveGlobalConfig(current => { + if (current.remoteDialogSeen) return current + return { ...current, remoteDialogSeen: true } + }) + if (answer.toLowerCase() !== 'y' && answer.toLowerCase() !== 'yes') { + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(0) + } + } + + // --continue: resolve the most recent session from the crash-recovery + // pointer and chain into the #20460 --session-id flow. Worktree-aware: + // checks current dir first (fast path, zero exec), then fans out to git + // worktree siblings if that misses — the REPL bridge writes to + // getOriginalCwd() which EnterWorktreeTool/activeWorktreeSession can + // point at a worktree while the user's shell is at the repo root. + // KAIROS-gated at parseArgs — continueSession is always false in external + // builds, so this block tree-shakes. + if (feature('KAIROS') && continueSession) { + const { readBridgePointerAcrossWorktrees } = await import( + './bridgePointer.js' + ) + const found = await readBridgePointerAcrossWorktrees(dir) + if (!found) { + // biome-ignore lint/suspicious/noConsole: intentional error output + console.error( + `Error: No recent session found in this directory or its worktrees. Run \`claude remote-control\` to start a new one.`, + ) + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(1) + } + const { pointer, dir: pointerDir } = found + const ageMin = Math.round(pointer.ageMs / 60_000) + const ageStr = ageMin < 60 ? `${ageMin}m` : `${Math.round(ageMin / 60)}h` + const fromWt = pointerDir !== dir ? ` from worktree ${pointerDir}` : '' + // biome-ignore lint/suspicious/noConsole: intentional info output + console.error( + `Resuming session ${pointer.sessionId} (${ageStr} ago)${fromWt}\u2026`, + ) + resumeSessionId = pointer.sessionId + // Track where the pointer came from so the #20460 exit(1) paths below + // clear the RIGHT file on deterministic failure — otherwise --continue + // would keep hitting the same dead session. May be a worktree sibling. + resumePointerDir = pointerDir + } + + // In production, baseUrl is the Anthropic API (from OAuth config). + // CLAUDE_BRIDGE_BASE_URL overrides this for ant local dev only. + const baseUrl = getBridgeBaseUrl() + + // For non-localhost targets, require HTTPS to protect credentials. + if ( + baseUrl.startsWith('http://') && + !baseUrl.includes('localhost') && + !baseUrl.includes('127.0.0.1') + ) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.error( + 'Error: Remote Control base URL uses HTTP. Only HTTPS or localhost HTTP is allowed.', + ) + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(1) + } + + // Session ingress URL for WebSocket connections. In production this is the + // same as baseUrl (Envoy routes /v1/session_ingress/* to session-ingress). + // Locally, session-ingress runs on a different port (9413) than the + // contain-provide-api (8211), so CLAUDE_BRIDGE_SESSION_INGRESS_URL must be + // set explicitly. Ant-only, matching CLAUDE_BRIDGE_BASE_URL. + const sessionIngressUrl = + process.env.USER_TYPE === 'ant' && + process.env.CLAUDE_BRIDGE_SESSION_INGRESS_URL + ? process.env.CLAUDE_BRIDGE_SESSION_INGRESS_URL + : baseUrl + + const { getBranch, getRemoteUrl, findGitRoot } = await import( + '../utils/git.js' + ) + + // Precheck worktree availability for the first-run dialog and the `w` + // toggle. Unconditional so we know upfront whether worktree is an option. + const { hasWorktreeCreateHook } = await import('../utils/hooks.js') + const worktreeAvailable = hasWorktreeCreateHook() || findGitRoot(dir) !== null + + // Load saved per-project spawn-mode preference. Gated by multiSessionEnabled + // so a GrowthBook rollback cleanly reverts users to single-session — + // otherwise a saved pref would silently re-enable multi-session behavior + // (worktree isolation, 32 max sessions, w toggle) despite the gate being off. + // Also guard against a stale worktree pref left over from when this dir WAS + // a git repo (or the user copied config) — clear it on disk so the warning + // doesn't repeat on every launch. + let savedSpawnMode = multiSessionEnabled + ? getCurrentProjectConfig().remoteControlSpawnMode + : undefined + if (savedSpawnMode === 'worktree' && !worktreeAvailable) { + // biome-ignore lint/suspicious/noConsole: intentional warning output + console.error( + 'Warning: Saved spawn mode is worktree but this directory is not a git repository. Falling back to same-dir.', + ) + savedSpawnMode = undefined + saveCurrentProjectConfig(current => { + if (current.remoteControlSpawnMode === undefined) return current + return { ...current, remoteControlSpawnMode: undefined } + }) + } + + // First-run spawn-mode choice: ask once per project when the choice is + // meaningful (gate on, both modes available, no explicit override, not + // resuming). Saves to ProjectConfig so subsequent runs skip this. + if ( + multiSessionEnabled && + !savedSpawnMode && + worktreeAvailable && + parsedSpawnMode === undefined && + !resumeSessionId && + process.stdin.isTTY + ) { + const readline = await import('readline') + const rl = readline.createInterface({ + input: process.stdin, + output: process.stdout, + }) + // biome-ignore lint/suspicious/noConsole: intentional dialog output + console.log( + `\nClaude Remote Control is launching in spawn mode which lets you create new sessions in this project from Claude Code on Web or your Mobile app. Learn more here: https://code.claude.com/docs/en/remote-control\n\n` + + `Spawn mode for this project:\n` + + ` [1] same-dir \u2014 sessions share the current directory (default)\n` + + ` [2] worktree \u2014 each session gets an isolated git worktree\n\n` + + `This can be changed later or explicitly set with --spawn=same-dir or --spawn=worktree.\n`, + ) + const answer = await new Promise(resolve => { + rl.question('Choose [1/2] (default: 1): ', resolve) + }) + rl.close() + const chosen: 'same-dir' | 'worktree' = + answer.trim() === '2' ? 'worktree' : 'same-dir' + savedSpawnMode = chosen + logEvent('tengu_bridge_spawn_mode_chosen', { + spawn_mode: + chosen as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + saveCurrentProjectConfig(current => { + if (current.remoteControlSpawnMode === chosen) return current + return { ...current, remoteControlSpawnMode: chosen } + }) + } + + // Determine effective spawn mode. + // Precedence: resume > explicit --spawn > saved project pref > gate default + // - resuming via --continue / --session-id: always single-session (resume + // targets one specific session in its original directory) + // - explicit --spawn flag: use that value directly (does not persist) + // - saved ProjectConfig.remoteControlSpawnMode: set by first-run dialog or `w` + // - default with gate on: same-dir (persistent multi-session, shared cwd) + // - default with gate off: single-session (unchanged legacy behavior) + // Track how spawn mode was determined, for rollout analytics. + type SpawnModeSource = 'resume' | 'flag' | 'saved' | 'gate_default' + let spawnModeSource: SpawnModeSource + let spawnMode: SpawnMode + if (resumeSessionId) { + spawnMode = 'single-session' + spawnModeSource = 'resume' + } else if (parsedSpawnMode !== undefined) { + spawnMode = parsedSpawnMode + spawnModeSource = 'flag' + } else if (savedSpawnMode !== undefined) { + spawnMode = savedSpawnMode + spawnModeSource = 'saved' + } else { + spawnMode = multiSessionEnabled ? 'same-dir' : 'single-session' + spawnModeSource = 'gate_default' + } + const maxSessions = + spawnMode === 'single-session' + ? 1 + : (parsedCapacity ?? SPAWN_SESSIONS_DEFAULT) + // Pre-create an empty session on start so the user has somewhere to type + // immediately, running in the current directory (exempted from worktree + // creation in the spawn loop). On by default; --no-create-session-in-dir + // opts out for a pure on-demand server where every session is isolated. + // The effectiveResumeSessionId guard at the creation site handles the + // resume case (skip creation when resume succeeded; fall through to + // fresh creation on env-mismatch fallback). + const preCreateSession = parsedCreateSessionInDir ?? true + + // Without --continue: a leftover pointer means the previous run didn't + // shut down cleanly (crash, kill -9, terminal closed). Clear it so the + // stale env doesn't linger past its relevance. Runs in all modes + // (clearBridgePointer is a no-op when no file exists) — covers the + // gate-transition case where a user crashed in single-session mode then + // starts fresh in worktree mode. Only single-session mode writes new + // pointers. + if (!resumeSessionId) { + const { clearBridgePointer } = await import('./bridgePointer.js') + await clearBridgePointer(dir) + } + + // Worktree mode requires either git or WorktreeCreate/WorktreeRemove hooks. + // Only reachable via explicit --spawn=worktree (default is same-dir); + // saved worktree pref was already guarded above. + if (spawnMode === 'worktree' && !worktreeAvailable) { + // biome-ignore lint/suspicious/noConsole: intentional error output + console.error( + `Error: Worktree mode requires a git repository or WorktreeCreate hooks configured. Use --spawn=session for single-session mode.`, + ) + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(1) + } + + const branch = await getBranch() + const gitRepoUrl = await getRemoteUrl() + const machineName = hostname() + const bridgeId = randomUUID() + + const { handleOAuth401Error } = await import('../utils/auth.js') + const api = createBridgeApiClient({ + baseUrl, + getAccessToken: getBridgeAccessToken, + runnerVersion: MACRO.VERSION, + onDebug: logForDebugging, + onAuth401: handleOAuth401Error, + getTrustedDeviceToken, + }) + + // When resuming a session via --session-id, fetch it to learn its + // environment_id and reuse that for registration (idempotent on the + // backend). Left undefined otherwise — the backend rejects + // client-generated UUIDs and will allocate a fresh environment. + // feature('KAIROS') gate: --session-id is ant-only; parseArgs already + // rejects the flag when the gate is off, so resumeSessionId is always + // undefined here in external builds — this guard is for tree-shaking. + let reuseEnvironmentId: string | undefined + if (feature('KAIROS') && resumeSessionId) { + try { + validateBridgeId(resumeSessionId, 'sessionId') + } catch { + // biome-ignore lint/suspicious/noConsole: intentional error output + console.error( + `Error: Invalid session ID "${resumeSessionId}". Session IDs must not contain unsafe characters.`, + ) + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(1) + } + // Proactively refresh the OAuth token — getBridgeSession uses raw axios + // without the withOAuthRetry 401-refresh logic. An expired-but-present + // token would otherwise produce a misleading "not found" error. + await checkAndRefreshOAuthTokenIfNeeded() + clearOAuthTokenCache() + const { getBridgeSession } = await import('./createSession.js') + const session = await getBridgeSession(resumeSessionId, { + baseUrl, + getAccessToken: getBridgeAccessToken, + }) + if (!session) { + // Session gone on server → pointer is stale. Clear it so the user + // isn't re-prompted next launch. (Explicit --session-id leaves the + // pointer alone — it's an independent file they may not even have.) + // resumePointerDir may be a worktree sibling — clear THAT file. + if (resumePointerDir) { + const { clearBridgePointer } = await import('./bridgePointer.js') + await clearBridgePointer(resumePointerDir) + } + // biome-ignore lint/suspicious/noConsole: intentional error output + console.error( + `Error: Session ${resumeSessionId} not found. It may have been archived or expired, or your login may have lapsed (run \`claude /login\`).`, + ) + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(1) + } + if (!session.environment_id) { + if (resumePointerDir) { + const { clearBridgePointer } = await import('./bridgePointer.js') + await clearBridgePointer(resumePointerDir) + } + // biome-ignore lint/suspicious/noConsole: intentional error output + console.error( + `Error: Session ${resumeSessionId} has no environment_id. It may never have been attached to a bridge.`, + ) + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(1) + } + reuseEnvironmentId = session.environment_id + logForDebugging( + `[bridge:init] Resuming session ${resumeSessionId} on environment ${reuseEnvironmentId}`, + ) + } + + const config: BridgeConfig = { + dir, + machineName, + branch, + gitRepoUrl, + maxSessions, + spawnMode, + verbose, + sandbox, + bridgeId, + workerType: 'claude_code', + environmentId: randomUUID(), + reuseEnvironmentId, + apiBaseUrl: baseUrl, + sessionIngressUrl, + debugFile, + sessionTimeoutMs, + } + + logForDebugging( + `[bridge:init] bridgeId=${bridgeId}${reuseEnvironmentId ? ` reuseEnvironmentId=${reuseEnvironmentId}` : ''} dir=${dir} branch=${branch} gitRepoUrl=${gitRepoUrl} machine=${machineName}`, + ) + logForDebugging( + `[bridge:init] apiBaseUrl=${baseUrl} sessionIngressUrl=${sessionIngressUrl}`, + ) + logForDebugging( + `[bridge:init] sandbox=${sandbox}${debugFile ? ` debugFile=${debugFile}` : ''}`, + ) + + // Register the bridge environment before entering the poll loop. + let environmentId: string + let environmentSecret: string + try { + const reg = await api.registerBridgeEnvironment(config) + environmentId = reg.environment_id + environmentSecret = reg.environment_secret + } catch (err) { + logEvent('tengu_bridge_registration_failed', { + status: err instanceof BridgeFatalError ? err.status : undefined, + }) + // Registration failures are fatal — print a clean message instead of a stack trace. + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.error( + err instanceof BridgeFatalError && err.status === 404 + ? 'Remote Control environments are not available for your account.' + : `Error: ${errorMessage(err)}`, + ) + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(1) + } + + // Tracks whether the --session-id resume flow completed successfully. + // Used below to skip fresh session creation and seed initialSessionId. + // Cleared on env mismatch so we gracefully fall back to a new session. + let effectiveResumeSessionId: string | undefined + if (feature('KAIROS') && resumeSessionId) { + if (reuseEnvironmentId && environmentId !== reuseEnvironmentId) { + // Backend returned a different environment_id — the original env + // expired or was reaped. Reconnect won't work against the new env + // (session is bound to the old one). Log to sentry for visibility + // and fall through to fresh session creation on the new env. + logError( + new Error( + `Bridge resume env mismatch: requested ${reuseEnvironmentId}, backend returned ${environmentId}. Falling back to fresh session.`, + ), + ) + // biome-ignore lint/suspicious/noConsole: intentional warning output + console.warn( + `Warning: Could not resume session ${resumeSessionId} — its environment has expired. Creating a fresh session instead.`, + ) + // Don't deregister — we're going to use this new environment. + // effectiveResumeSessionId stays undefined → fresh session path below. + } else { + // Force-stop any stale worker instances for this session and re-queue + // it so our poll loop picks it up. Must happen after registration so + // the backend knows a live worker exists for the environment. + // + // The pointer stores a session_* ID but /bridge/reconnect looks + // sessions up by their infra tag (cse_*) when ccr_v2_compat_enabled + // is on. Try both; the conversion is a no-op if already cse_*. + const infraResumeId = toInfraSessionId(resumeSessionId) + const reconnectCandidates = + infraResumeId === resumeSessionId + ? [resumeSessionId] + : [resumeSessionId, infraResumeId] + let reconnected = false + let lastReconnectErr: unknown + for (const candidateId of reconnectCandidates) { + try { + await api.reconnectSession(environmentId, candidateId) + logForDebugging( + `[bridge:init] Session ${candidateId} re-queued via bridge/reconnect`, + ) + effectiveResumeSessionId = resumeSessionId + reconnected = true + break + } catch (err) { + lastReconnectErr = err + logForDebugging( + `[bridge:init] reconnectSession(${candidateId}) failed: ${errorMessage(err)}`, + ) + } + } + if (!reconnected) { + const err = lastReconnectErr + + // Do NOT deregister on transient reconnect failure — at this point + // environmentId IS the session's own environment. Deregistering + // would make retry impossible. The backend's 4h TTL cleans up. + const isFatal = err instanceof BridgeFatalError + // Clear pointer only on fatal reconnect failure. Transient failures + // ("try running the same command again") should keep the pointer so + // next launch re-prompts — that IS the retry mechanism. + if (resumePointerDir && isFatal) { + const { clearBridgePointer } = await import('./bridgePointer.js') + await clearBridgePointer(resumePointerDir) + } + // biome-ignore lint/suspicious/noConsole: intentional error output + console.error( + isFatal + ? `Error: ${errorMessage(err)}` + : `Error: Failed to reconnect session ${resumeSessionId}: ${errorMessage(err)}\nThe session may still be resumable — try running the same command again.`, + ) + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(1) + } + } + } + + logForDebugging( + `[bridge:init] Registered, server environmentId=${environmentId}`, + ) + const startupPollConfig = getPollIntervalConfig() + logEvent('tengu_bridge_started', { + max_sessions: config.maxSessions, + has_debug_file: !!config.debugFile, + sandbox: config.sandbox, + verbose: config.verbose, + heartbeat_interval_ms: + startupPollConfig.non_exclusive_heartbeat_interval_ms, + spawn_mode: + config.spawnMode as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + spawn_mode_source: + spawnModeSource as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + multi_session_gate: multiSessionEnabled, + pre_create_session: preCreateSession, + worktree_available: worktreeAvailable, + }) + logForDiagnosticsNoPII('info', 'bridge_started', { + max_sessions: config.maxSessions, + sandbox: config.sandbox, + spawn_mode: config.spawnMode, + }) + + const spawner = createSessionSpawner({ + execPath: process.execPath, + scriptArgs: spawnScriptArgs(), + env: process.env, + verbose, + sandbox, + debugFile, + permissionMode, + onDebug: logForDebugging, + onActivity: (sessionId, activity) => { + logForDebugging( + `[bridge:activity] sessionId=${sessionId} ${activity.type} ${activity.summary}`, + ) + }, + onPermissionRequest: (sessionId, request, _accessToken) => { + logForDebugging( + `[bridge:perm] sessionId=${sessionId} tool=${request.request.tool_name} request_id=${request.request_id} (not auto-approving)`, + ) + }, + }) + + const logger = createBridgeLogger({ verbose }) + const { parseGitHubRepository } = await import('../utils/detectRepository.js') + const ownerRepo = gitRepoUrl ? parseGitHubRepository(gitRepoUrl) : null + // Use the repo name from the parsed owner/repo, or fall back to the dir basename + const repoName = ownerRepo ? ownerRepo.split('/').pop()! : basename(dir) + logger.setRepoInfo(repoName, branch) + + // `w` toggle is available iff we're in a multi-session mode AND worktree + // is a valid option. When unavailable, the mode suffix and hint are hidden. + const toggleAvailable = spawnMode !== 'single-session' && worktreeAvailable + if (toggleAvailable) { + // Safe cast: spawnMode is not single-session (checked above), and the + // saved-worktree-in-non-git guard + exit check above ensure worktree + // is only reached when available. + logger.setSpawnModeDisplay(spawnMode as 'same-dir' | 'worktree') + } + + // Listen for keys: space toggles QR code, w toggles spawn mode + const onStdinData = (data: Buffer): void => { + if (data[0] === 0x03 || data[0] === 0x04) { + // Ctrl+C / Ctrl+D — trigger graceful shutdown + process.emit('SIGINT') + return + } + if (data[0] === 0x20 /* space */) { + logger.toggleQr() + return + } + if (data[0] === 0x77 /* 'w' */) { + if (!toggleAvailable) return + const newMode: 'same-dir' | 'worktree' = + config.spawnMode === 'same-dir' ? 'worktree' : 'same-dir' + config.spawnMode = newMode + logEvent('tengu_bridge_spawn_mode_toggled', { + spawn_mode: + newMode as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + logger.logStatus( + newMode === 'worktree' + ? 'Spawn mode: worktree (new sessions get isolated git worktrees)' + : 'Spawn mode: same-dir (new sessions share the current directory)', + ) + logger.setSpawnModeDisplay(newMode) + logger.refreshDisplay() + saveCurrentProjectConfig(current => { + if (current.remoteControlSpawnMode === newMode) return current + return { ...current, remoteControlSpawnMode: newMode } + }) + return + } + } + if (process.stdin.isTTY) { + process.stdin.setRawMode(true) + process.stdin.resume() + process.stdin.on('data', onStdinData) + } + + const controller = new AbortController() + const onSigint = (): void => { + logForDebugging('[bridge:shutdown] SIGINT received, shutting down') + controller.abort() + } + const onSigterm = (): void => { + logForDebugging('[bridge:shutdown] SIGTERM received, shutting down') + controller.abort() + } + process.on('SIGINT', onSigint) + process.on('SIGTERM', onSigterm) + + // Auto-create an empty session so the user has somewhere to type + // immediately (matching /remote-control behavior). Controlled by + // preCreateSession: on by default; --no-create-session-in-dir opts out. + // When a --session-id resume succeeded, skip creation entirely — the + // session already exists and bridge/reconnect has re-queued it. + // When resume was requested but failed on env mismatch, effectiveResumeSessionId + // is undefined, so we fall through to fresh session creation (honoring the + // "Creating a fresh session instead" warning printed above). + let initialSessionId: string | null = + feature('KAIROS') && effectiveResumeSessionId + ? effectiveResumeSessionId + : null + if (preCreateSession && !(feature('KAIROS') && effectiveResumeSessionId)) { + const { createBridgeSession } = await import('./createSession.js') + try { + initialSessionId = await createBridgeSession({ + environmentId, + title: name, + events: [], + gitRepoUrl, + branch, + signal: controller.signal, + baseUrl, + getAccessToken: getBridgeAccessToken, + permissionMode, + }) + if (initialSessionId) { + logForDebugging( + `[bridge:init] Created initial session ${initialSessionId}`, + ) + } + } catch (err) { + logForDebugging( + `[bridge:init] Session creation failed (non-fatal): ${errorMessage(err)}`, + ) + } + } + + // Crash-recovery pointer: write immediately so kill -9 at any point + // after this leaves a recoverable trail. Covers both fresh sessions and + // resumed ones (so a second crash after resume is still recoverable). + // Cleared when runBridgeLoop falls through to archive+deregister; left in + // place on the SIGINT resumable-shutdown return (backup for when the user + // closes the terminal before copying the printed --session-id hint). + // Refreshed hourly so a 5h+ session that crashes still has a fresh + // pointer (staleness checks file mtime, backend TTL is rolling-from-poll). + let pointerRefreshTimer: ReturnType | null = null + // Single-session only: --continue forces single-session mode on resume, + // so a pointer written in multi-session mode would contradict the user's + // config when they try to resume. The resumable-shutdown path is also + // gated to single-session (line ~1254) so the pointer would be orphaned. + if (initialSessionId && spawnMode === 'single-session') { + const { writeBridgePointer } = await import('./bridgePointer.js') + const pointerPayload = { + sessionId: initialSessionId, + environmentId, + source: 'standalone' as const, + } + await writeBridgePointer(config.dir, pointerPayload) + pointerRefreshTimer = setInterval( + writeBridgePointer, + 60 * 60 * 1000, + config.dir, + pointerPayload, + ) + // Don't let the interval keep the process alive on its own. + pointerRefreshTimer.unref?.() + } + + try { + await runBridgeLoop( + config, + environmentId, + environmentSecret, + api, + spawner, + logger, + controller.signal, + undefined, + initialSessionId ?? undefined, + async () => { + // Clear the memoized OAuth token cache so we re-read from secure + // storage, picking up tokens refreshed by child processes. + clearOAuthTokenCache() + // Proactively refresh the token if it's expired on disk too. + await checkAndRefreshOAuthTokenIfNeeded() + return getBridgeAccessToken() + }, + ) + } finally { + if (pointerRefreshTimer !== null) { + clearInterval(pointerRefreshTimer) + } + process.off('SIGINT', onSigint) + process.off('SIGTERM', onSigterm) + process.stdin.off('data', onStdinData) + if (process.stdin.isTTY) { + process.stdin.setRawMode(false) + } + process.stdin.pause() + } + + // The bridge bypasses init.ts (and its graceful shutdown handler), so we + // must exit explicitly. + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(0) +} + +// ─── Headless bridge (daemon worker) ──────────────────────────────────────── + +/** + * Thrown by runBridgeHeadless for configuration issues the supervisor should + * NOT retry (trust not accepted, worktree unavailable, http-not-https). The + * daemon worker catches this and exits with EXIT_CODE_PERMANENT so the + * supervisor parks the worker instead of respawning it on backoff. + */ +export class BridgeHeadlessPermanentError extends Error { + constructor(message: string) { + super(message) + this.name = 'BridgeHeadlessPermanentError' + } +} + +export type HeadlessBridgeOpts = { + dir: string + name?: string + spawnMode: 'same-dir' | 'worktree' + capacity: number + permissionMode?: string + sandbox: boolean + sessionTimeoutMs?: number + createSessionOnStart: boolean + getAccessToken: () => string | undefined + onAuth401: (failedToken: string) => Promise + log: (s: string) => void +} + +/** + * Non-interactive bridge entrypoint for the `remoteControl` daemon worker. + * + * Linear subset of bridgeMain(): no readline dialogs, no stdin key handlers, + * no TUI, no process.exit(). Config comes from the caller (daemon.json), auth + * comes via IPC (supervisor's AuthManager), logs go to the worker's stdout + * pipe. Throws on fatal errors — the worker catches and maps permanent vs + * transient to the right exit code. + * + * Resolves cleanly when `signal` aborts and the poll loop tears down. + */ +export async function runBridgeHeadless( + opts: HeadlessBridgeOpts, + signal: AbortSignal, +): Promise { + const { dir, log } = opts + + // Worker inherits the supervisor's CWD. chdir first so git utilities + // (getBranch/getRemoteUrl) — which read from bootstrap CWD state set + // below — resolve against the right repo. + process.chdir(dir) + const { setOriginalCwd, setCwdState } = await import('../bootstrap/state.js') + setOriginalCwd(dir) + setCwdState(dir) + + const { enableConfigs, checkHasTrustDialogAccepted } = await import( + '../utils/config.js' + ) + enableConfigs() + const { initSinks } = await import('../utils/sinks.js') + initSinks() + + if (!checkHasTrustDialogAccepted()) { + throw new BridgeHeadlessPermanentError( + `Workspace not trusted: ${dir}. Run \`claude\` in that directory first to accept the trust dialog.`, + ) + } + + if (!opts.getAccessToken()) { + // Transient — supervisor's AuthManager may pick up a token on next cycle. + throw new Error(BRIDGE_LOGIN_ERROR) + } + + const { getBridgeBaseUrl } = await import('./bridgeConfig.js') + const baseUrl = getBridgeBaseUrl() + if ( + baseUrl.startsWith('http://') && + !baseUrl.includes('localhost') && + !baseUrl.includes('127.0.0.1') + ) { + throw new BridgeHeadlessPermanentError( + 'Remote Control base URL uses HTTP. Only HTTPS or localhost HTTP is allowed.', + ) + } + const sessionIngressUrl = + process.env.USER_TYPE === 'ant' && + process.env.CLAUDE_BRIDGE_SESSION_INGRESS_URL + ? process.env.CLAUDE_BRIDGE_SESSION_INGRESS_URL + : baseUrl + + const { getBranch, getRemoteUrl, findGitRoot } = await import( + '../utils/git.js' + ) + const { hasWorktreeCreateHook } = await import('../utils/hooks.js') + + if (opts.spawnMode === 'worktree') { + const worktreeAvailable = + hasWorktreeCreateHook() || findGitRoot(dir) !== null + if (!worktreeAvailable) { + throw new BridgeHeadlessPermanentError( + `Worktree mode requires a git repository or WorktreeCreate hooks. Directory ${dir} has neither.`, + ) + } + } + + const branch = await getBranch() + const gitRepoUrl = await getRemoteUrl() + const machineName = hostname() + const bridgeId = randomUUID() + + const config: BridgeConfig = { + dir, + machineName, + branch, + gitRepoUrl, + maxSessions: opts.capacity, + spawnMode: opts.spawnMode, + verbose: false, + sandbox: opts.sandbox, + bridgeId, + workerType: 'claude_code', + environmentId: randomUUID(), + apiBaseUrl: baseUrl, + sessionIngressUrl, + sessionTimeoutMs: opts.sessionTimeoutMs, + } + + const api = createBridgeApiClient({ + baseUrl, + getAccessToken: opts.getAccessToken, + runnerVersion: MACRO.VERSION, + onDebug: log, + onAuth401: opts.onAuth401, + getTrustedDeviceToken, + }) + + let environmentId: string + let environmentSecret: string + try { + const reg = await api.registerBridgeEnvironment(config) + environmentId = reg.environment_id + environmentSecret = reg.environment_secret + } catch (err) { + // Transient — let supervisor backoff-retry. + throw new Error(`Bridge registration failed: ${errorMessage(err)}`) + } + + const spawner = createSessionSpawner({ + execPath: process.execPath, + scriptArgs: spawnScriptArgs(), + env: process.env, + verbose: false, + sandbox: opts.sandbox, + permissionMode: opts.permissionMode, + onDebug: log, + }) + + const logger = createHeadlessBridgeLogger(log) + logger.printBanner(config, environmentId) + + let initialSessionId: string | undefined + if (opts.createSessionOnStart) { + const { createBridgeSession } = await import('./createSession.js') + try { + const sid = await createBridgeSession({ + environmentId, + title: opts.name, + events: [], + gitRepoUrl, + branch, + signal, + baseUrl, + getAccessToken: opts.getAccessToken, + permissionMode: opts.permissionMode, + }) + if (sid) { + initialSessionId = sid + log(`created initial session ${sid}`) + } + } catch (err) { + log(`session pre-creation failed (non-fatal): ${errorMessage(err)}`) + } + } + + await runBridgeLoop( + config, + environmentId, + environmentSecret, + api, + spawner, + logger, + signal, + undefined, + initialSessionId, + async () => opts.getAccessToken(), + ) +} + +/** BridgeLogger adapter that routes everything to a single line-log fn. */ +function createHeadlessBridgeLogger(log: (s: string) => void): BridgeLogger { + const noop = (): void => {} + return { + printBanner: (cfg, envId) => + log( + `registered environmentId=${envId} dir=${cfg.dir} spawnMode=${cfg.spawnMode} capacity=${cfg.maxSessions}`, + ), + logSessionStart: (id, _prompt) => log(`session start ${id}`), + logSessionComplete: (id, ms) => log(`session complete ${id} (${ms}ms)`), + logSessionFailed: (id, err) => log(`session failed ${id}: ${err}`), + logStatus: log, + logVerbose: log, + logError: s => log(`error: ${s}`), + logReconnected: ms => log(`reconnected after ${ms}ms`), + addSession: (id, _url) => log(`session attached ${id}`), + removeSession: id => log(`session detached ${id}`), + updateIdleStatus: noop, + updateReconnectingStatus: noop, + updateSessionStatus: noop, + updateSessionActivity: noop, + updateSessionCount: noop, + updateFailedStatus: noop, + setSpawnModeDisplay: noop, + setRepoInfo: noop, + setDebugLogPath: noop, + setAttached: noop, + setSessionTitle: noop, + clearStatus: noop, + toggleQr: noop, + refreshDisplay: noop, + } +} diff --git a/src/bridge/bridgeMessaging.ts b/src/bridge/bridgeMessaging.ts new file mode 100644 index 0000000..98ece03 --- /dev/null +++ b/src/bridge/bridgeMessaging.ts @@ -0,0 +1,461 @@ +/** + * Shared transport-layer helpers for bridge message handling. + * + * Extracted from replBridge.ts so both the env-based core (initBridgeCore) + * and the env-less core (initEnvLessBridgeCore) can use the same ingress + * parsing, control-request handling, and echo-dedup machinery. + * + * Everything here is pure — no closure over bridge-specific state. All + * collaborators (transport, sessionId, UUID sets, callbacks) are passed + * as params. + */ + +import { randomUUID } from 'crypto' +import type { SDKMessage } from '../entrypoints/agentSdkTypes.js' +import type { + SDKControlRequest, + SDKControlResponse, +} from '../entrypoints/sdk/controlTypes.js' +import type { SDKResultSuccess } from '../entrypoints/sdk/coreTypes.js' +import { logEvent } from '../services/analytics/index.js' +import { EMPTY_USAGE } from '../services/api/emptyUsage.js' +import type { Message } from '../types/message.js' +import { normalizeControlMessageKeys } from '../utils/controlMessageCompat.js' +import { logForDebugging } from '../utils/debug.js' +import { stripDisplayTagsAllowEmpty } from '../utils/displayTags.js' +import { errorMessage } from '../utils/errors.js' +import type { PermissionMode } from '../utils/permissions/PermissionMode.js' +import { jsonParse } from '../utils/slowOperations.js' +import type { ReplBridgeTransport } from './replBridgeTransport.js' + +// ─── Type guards ───────────────────────────────────────────────────────────── + +/** Type predicate for parsed WebSocket messages. SDKMessage is a + * discriminated union on `type` — validating the discriminant is + * sufficient for the predicate; callers narrow further via the union. */ +export function isSDKMessage(value: unknown): value is SDKMessage { + return ( + value !== null && + typeof value === 'object' && + 'type' in value && + typeof value.type === 'string' + ) +} + +/** Type predicate for control_response messages from the server. */ +export function isSDKControlResponse( + value: unknown, +): value is SDKControlResponse { + return ( + value !== null && + typeof value === 'object' && + 'type' in value && + value.type === 'control_response' && + 'response' in value + ) +} + +/** Type predicate for control_request messages from the server. */ +export function isSDKControlRequest( + value: unknown, +): value is SDKControlRequest { + return ( + value !== null && + typeof value === 'object' && + 'type' in value && + value.type === 'control_request' && + 'request_id' in value && + 'request' in value + ) +} + +/** + * True for message types that should be forwarded to the bridge transport. + * The server only wants user/assistant turns and slash-command system events; + * everything else (tool_result, progress, etc.) is internal REPL chatter. + */ +export function isEligibleBridgeMessage(m: Message): boolean { + // Virtual messages (REPL inner calls) are display-only — bridge/SDK + // consumers see the REPL tool_use/result which summarizes the work. + if ((m.type === 'user' || m.type === 'assistant') && m.isVirtual) { + return false + } + return ( + m.type === 'user' || + m.type === 'assistant' || + (m.type === 'system' && m.subtype === 'local_command') + ) +} + +/** + * Extract title-worthy text from a Message for onUserMessage. Returns + * undefined for messages that shouldn't title the session: non-user, meta + * (nudges), tool results, compact summaries, non-human origins (task + * notifications, channel messages), or pure display-tag content + * (, , etc.). + * + * Synthetic interrupts ([Request interrupted by user]) are NOT filtered here — + * isSyntheticMessage lives in messages.ts (heavy import, pulls command + * registry). The initialMessages path in initReplBridge checks it; the + * writeMessages path reaching an interrupt as the *first* message is + * implausible (an interrupt implies a prior prompt already flowed through). + */ +export function extractTitleText(m: Message): string | undefined { + if (m.type !== 'user' || m.isMeta || m.toolUseResult || m.isCompactSummary) + return undefined + if (m.origin && m.origin.kind !== 'human') return undefined + const content = m.message.content + let raw: string | undefined + if (typeof content === 'string') { + raw = content + } else { + for (const block of content) { + if (block.type === 'text') { + raw = block.text + break + } + } + } + if (!raw) return undefined + const clean = stripDisplayTagsAllowEmpty(raw) + return clean || undefined +} + +// ─── Ingress routing ───────────────────────────────────────────────────────── + +/** + * Parse an ingress WebSocket message and route it to the appropriate handler. + * Ignores messages whose UUID is in recentPostedUUIDs (echoes of what we sent) + * or in recentInboundUUIDs (re-deliveries we've already forwarded — e.g. + * server replayed history after a transport swap lost the seq-num cursor). + */ +export function handleIngressMessage( + data: string, + recentPostedUUIDs: BoundedUUIDSet, + recentInboundUUIDs: BoundedUUIDSet, + onInboundMessage: ((msg: SDKMessage) => void | Promise) | undefined, + onPermissionResponse?: ((response: SDKControlResponse) => void) | undefined, + onControlRequest?: ((request: SDKControlRequest) => void) | undefined, +): void { + try { + const parsed: unknown = normalizeControlMessageKeys(jsonParse(data)) + + // control_response is not an SDKMessage — check before the type guard + if (isSDKControlResponse(parsed)) { + logForDebugging('[bridge:repl] Ingress message type=control_response') + onPermissionResponse?.(parsed) + return + } + + // control_request from the server (initialize, set_model, can_use_tool). + // Must respond promptly or the server kills the WS (~10-14s timeout). + if (isSDKControlRequest(parsed)) { + logForDebugging( + `[bridge:repl] Inbound control_request subtype=${parsed.request.subtype}`, + ) + onControlRequest?.(parsed) + return + } + + if (!isSDKMessage(parsed)) return + + // Check for UUID to detect echoes of our own messages + const uuid = + 'uuid' in parsed && typeof parsed.uuid === 'string' + ? parsed.uuid + : undefined + + if (uuid && recentPostedUUIDs.has(uuid)) { + logForDebugging( + `[bridge:repl] Ignoring echo: type=${parsed.type} uuid=${uuid}`, + ) + return + } + + // Defensive dedup: drop inbound prompts we've already forwarded. The + // SSE seq-num carryover (lastTransportSequenceNum) is the primary fix + // for history-replay; this catches edge cases where that negotiation + // fails (server ignores from_sequence_num, transport died before + // receiving any frames, etc). + if (uuid && recentInboundUUIDs.has(uuid)) { + logForDebugging( + `[bridge:repl] Ignoring re-delivered inbound: type=${parsed.type} uuid=${uuid}`, + ) + return + } + + logForDebugging( + `[bridge:repl] Ingress message type=${parsed.type}${uuid ? ` uuid=${uuid}` : ''}`, + ) + + if (parsed.type === 'user') { + if (uuid) recentInboundUUIDs.add(uuid) + logEvent('tengu_bridge_message_received', { + is_repl: true, + }) + // Fire-and-forget — handler may be async (attachment resolution). + void onInboundMessage?.(parsed) + } else { + logForDebugging( + `[bridge:repl] Ignoring non-user inbound message: type=${parsed.type}`, + ) + } + } catch (err) { + logForDebugging( + `[bridge:repl] Failed to parse ingress message: ${errorMessage(err)}`, + ) + } +} + +// ─── Server-initiated control requests ─────────────────────────────────────── + +export type ServerControlRequestHandlers = { + transport: ReplBridgeTransport | null + sessionId: string + /** + * When true, all mutable requests (interrupt, set_model, set_permission_mode, + * set_max_thinking_tokens) reply with an error instead of false-success. + * initialize still replies success — the server kills the connection otherwise. + * Used by the outbound-only bridge mode and the SDK's /bridge subpath so claude.ai sees a + * proper error instead of "action succeeded but nothing happened locally". + */ + outboundOnly?: boolean + onInterrupt?: () => void + onSetModel?: (model: string | undefined) => void + onSetMaxThinkingTokens?: (maxTokens: number | null) => void + onSetPermissionMode?: ( + mode: PermissionMode, + ) => { ok: true } | { ok: false; error: string } +} + +const OUTBOUND_ONLY_ERROR = + 'This session is outbound-only. Enable Remote Control locally to allow inbound control.' + +/** + * Respond to inbound control_request messages from the server. The server + * sends these for session lifecycle events (initialize, set_model) and + * for turn-level coordination (interrupt, set_max_thinking_tokens). If we + * don't respond, the server hangs and kills the WS after ~10-14s. + * + * Previously a closure inside initBridgeCore's onWorkReceived; now takes + * collaborators as params so both cores can use it. + */ +export function handleServerControlRequest( + request: SDKControlRequest, + handlers: ServerControlRequestHandlers, +): void { + const { + transport, + sessionId, + outboundOnly, + onInterrupt, + onSetModel, + onSetMaxThinkingTokens, + onSetPermissionMode, + } = handlers + if (!transport) { + logForDebugging( + '[bridge:repl] Cannot respond to control_request: transport not configured', + ) + return + } + + let response: SDKControlResponse + + // Outbound-only: reply error for mutable requests so claude.ai doesn't show + // false success. initialize must still succeed (server kills the connection + // if it doesn't — see comment above). + if (outboundOnly && request.request.subtype !== 'initialize') { + response = { + type: 'control_response', + response: { + subtype: 'error', + request_id: request.request_id, + error: OUTBOUND_ONLY_ERROR, + }, + } + const event = { ...response, session_id: sessionId } + void transport.write(event) + logForDebugging( + `[bridge:repl] Rejected ${request.request.subtype} (outbound-only) request_id=${request.request_id}`, + ) + return + } + + switch (request.request.subtype) { + case 'initialize': + // Respond with minimal capabilities — the REPL handles + // commands, models, and account info itself. + response = { + type: 'control_response', + response: { + subtype: 'success', + request_id: request.request_id, + response: { + commands: [], + output_style: 'normal', + available_output_styles: ['normal'], + models: [], + account: {}, + pid: process.pid, + }, + }, + } + break + + case 'set_model': + onSetModel?.(request.request.model) + response = { + type: 'control_response', + response: { + subtype: 'success', + request_id: request.request_id, + }, + } + break + + case 'set_max_thinking_tokens': + onSetMaxThinkingTokens?.(request.request.max_thinking_tokens) + response = { + type: 'control_response', + response: { + subtype: 'success', + request_id: request.request_id, + }, + } + break + + case 'set_permission_mode': { + // The callback returns a policy verdict so we can send an error + // control_response without importing isAutoModeGateEnabled / + // isBypassPermissionsModeDisabled here (bootstrap-isolation). If no + // callback is registered (daemon context, which doesn't wire this — + // see daemonBridge.ts), return an error verdict rather than a silent + // false-success: the mode is never actually applied in that context, + // so success would lie to the client. + const verdict = onSetPermissionMode?.(request.request.mode) ?? { + ok: false, + error: + 'set_permission_mode is not supported in this context (onSetPermissionMode callback not registered)', + } + if (verdict.ok) { + response = { + type: 'control_response', + response: { + subtype: 'success', + request_id: request.request_id, + }, + } + } else { + response = { + type: 'control_response', + response: { + subtype: 'error', + request_id: request.request_id, + error: verdict.error, + }, + } + } + break + } + + case 'interrupt': + onInterrupt?.() + response = { + type: 'control_response', + response: { + subtype: 'success', + request_id: request.request_id, + }, + } + break + + default: + // Unknown subtype — respond with error so the server doesn't + // hang waiting for a reply that never comes. + response = { + type: 'control_response', + response: { + subtype: 'error', + request_id: request.request_id, + error: `REPL bridge does not handle control_request subtype: ${request.request.subtype}`, + }, + } + } + + const event = { ...response, session_id: sessionId } + void transport.write(event) + logForDebugging( + `[bridge:repl] Sent control_response for ${request.request.subtype} request_id=${request.request_id} result=${response.response.subtype}`, + ) +} + +// ─── Result message (for session archival on teardown) ─────────────────────── + +/** + * Build a minimal `SDKResultSuccess` message for session archival. + * The server needs this event before a WS close to trigger archival. + */ +export function makeResultMessage(sessionId: string): SDKResultSuccess { + return { + type: 'result', + subtype: 'success', + duration_ms: 0, + duration_api_ms: 0, + is_error: false, + num_turns: 0, + result: '', + stop_reason: null, + total_cost_usd: 0, + usage: { ...EMPTY_USAGE }, + modelUsage: {}, + permission_denials: [], + session_id: sessionId, + uuid: randomUUID(), + } +} + +// ─── BoundedUUIDSet (echo-dedup ring buffer) ───────────────────────────────── + +/** + * FIFO-bounded set backed by a circular buffer. Evicts the oldest entry + * when capacity is reached, keeping memory usage constant at O(capacity). + * + * Messages are added in chronological order, so evicted entries are always + * the oldest. The caller relies on external ordering (the hook's + * lastWrittenIndexRef) as the primary dedup — this set is a secondary + * safety net for echo filtering and race-condition dedup. + */ +export class BoundedUUIDSet { + private readonly capacity: number + private readonly ring: (string | undefined)[] + private readonly set = new Set() + private writeIdx = 0 + + constructor(capacity: number) { + this.capacity = capacity + this.ring = new Array(capacity) + } + + add(uuid: string): void { + if (this.set.has(uuid)) return + // Evict the entry at the current write position (if occupied) + const evicted = this.ring[this.writeIdx] + if (evicted !== undefined) { + this.set.delete(evicted) + } + this.ring[this.writeIdx] = uuid + this.set.add(uuid) + this.writeIdx = (this.writeIdx + 1) % this.capacity + } + + has(uuid: string): boolean { + return this.set.has(uuid) + } + + clear(): void { + this.set.clear() + this.ring.fill(undefined) + this.writeIdx = 0 + } +} diff --git a/src/bridge/bridgePermissionCallbacks.ts b/src/bridge/bridgePermissionCallbacks.ts new file mode 100644 index 0000000..feaee66 --- /dev/null +++ b/src/bridge/bridgePermissionCallbacks.ts @@ -0,0 +1,43 @@ +import type { PermissionUpdate } from '../utils/permissions/PermissionUpdateSchema.js' + +type BridgePermissionResponse = { + behavior: 'allow' | 'deny' + updatedInput?: Record + updatedPermissions?: PermissionUpdate[] + message?: string +} + +type BridgePermissionCallbacks = { + sendRequest( + requestId: string, + toolName: string, + input: Record, + toolUseId: string, + description: string, + permissionSuggestions?: PermissionUpdate[], + blockedPath?: string, + ): void + sendResponse(requestId: string, response: BridgePermissionResponse): void + /** Cancel a pending control_request so the web app can dismiss its prompt. */ + cancelRequest(requestId: string): void + onResponse( + requestId: string, + handler: (response: BridgePermissionResponse) => void, + ): () => void // returns unsubscribe +} + +/** Type predicate for validating a parsed control_response payload + * as a BridgePermissionResponse. Checks the required `behavior` + * discriminant rather than using an unsafe `as` cast. */ +function isBridgePermissionResponse( + value: unknown, +): value is BridgePermissionResponse { + if (!value || typeof value !== 'object') return false + return ( + 'behavior' in value && + (value.behavior === 'allow' || value.behavior === 'deny') + ) +} + +export { isBridgePermissionResponse } +export type { BridgePermissionCallbacks, BridgePermissionResponse } diff --git a/src/bridge/bridgePointer.ts b/src/bridge/bridgePointer.ts new file mode 100644 index 0000000..c32befc --- /dev/null +++ b/src/bridge/bridgePointer.ts @@ -0,0 +1,210 @@ +import { mkdir, readFile, stat, unlink, writeFile } from 'fs/promises' +import { dirname, join } from 'path' +import { z } from 'zod/v4' +import { logForDebugging } from '../utils/debug.js' +import { isENOENT } from '../utils/errors.js' +import { getWorktreePathsPortable } from '../utils/getWorktreePathsPortable.js' +import { lazySchema } from '../utils/lazySchema.js' +import { + getProjectsDir, + sanitizePath, +} from '../utils/sessionStoragePortable.js' +import { jsonParse, jsonStringify } from '../utils/slowOperations.js' + +/** + * Upper bound on worktree fanout. git worktree list is naturally bounded + * (50 is a LOT), but this caps the parallel stat() burst and guards against + * pathological setups. Above this, --continue falls back to current-dir-only. + */ +const MAX_WORKTREE_FANOUT = 50 + +/** + * Crash-recovery pointer for Remote Control sessions. + * + * Written immediately after a bridge session is created, periodically + * refreshed during the session, and cleared on clean shutdown. If the + * process dies unclean (crash, kill -9, terminal closed), the pointer + * persists. On next startup, `claude remote-control` detects it and offers + * to resume via the --session-id flow from #20460. + * + * Staleness is checked against the file's mtime (not an embedded timestamp) + * so that a periodic re-write with the same content serves as a refresh — + * matches the backend's rolling BRIDGE_LAST_POLL_TTL (4h) semantics. A + * bridge that's been polling for 5+ hours and then crashes still has a + * fresh pointer as long as the refresh ran within the window. + * + * Scoped per working directory (alongside transcript JSONL files) so two + * concurrent bridges in different repos don't clobber each other. + */ + +export const BRIDGE_POINTER_TTL_MS = 4 * 60 * 60 * 1000 + +const BridgePointerSchema = lazySchema(() => + z.object({ + sessionId: z.string(), + environmentId: z.string(), + source: z.enum(['standalone', 'repl']), + }), +) + +export type BridgePointer = z.infer> + +export function getBridgePointerPath(dir: string): string { + return join(getProjectsDir(), sanitizePath(dir), 'bridge-pointer.json') +} + +/** + * Write the pointer. Also used to refresh mtime during long sessions — + * calling with the same IDs is a cheap no-content-change write that bumps + * the staleness clock. Best-effort — a crash-recovery file must never + * itself cause a crash. Logs and swallows on error. + */ +export async function writeBridgePointer( + dir: string, + pointer: BridgePointer, +): Promise { + const path = getBridgePointerPath(dir) + try { + await mkdir(dirname(path), { recursive: true }) + await writeFile(path, jsonStringify(pointer), 'utf8') + logForDebugging(`[bridge:pointer] wrote ${path}`) + } catch (err: unknown) { + logForDebugging(`[bridge:pointer] write failed: ${err}`, { level: 'warn' }) + } +} + +/** + * Read the pointer and its age (ms since last write). Operates directly + * and handles errors — no existence check (CLAUDE.md TOCTOU rule). Returns + * null on any failure: missing file, corrupted JSON, schema mismatch, or + * stale (mtime > 4h ago). Stale/invalid pointers are deleted so they don't + * keep re-prompting after the backend has already GC'd the env. + */ +export async function readBridgePointer( + dir: string, +): Promise<(BridgePointer & { ageMs: number }) | null> { + const path = getBridgePointerPath(dir) + let raw: string + let mtimeMs: number + try { + // stat for mtime (staleness anchor), then read. Two syscalls, but both + // are needed — mtime IS the data we return, not a TOCTOU guard. + mtimeMs = (await stat(path)).mtimeMs + raw = await readFile(path, 'utf8') + } catch { + return null + } + + const parsed = BridgePointerSchema().safeParse(safeJsonParse(raw)) + if (!parsed.success) { + logForDebugging(`[bridge:pointer] invalid schema, clearing: ${path}`) + await clearBridgePointer(dir) + return null + } + + const ageMs = Math.max(0, Date.now() - mtimeMs) + if (ageMs > BRIDGE_POINTER_TTL_MS) { + logForDebugging(`[bridge:pointer] stale (>4h mtime), clearing: ${path}`) + await clearBridgePointer(dir) + return null + } + + return { ...parsed.data, ageMs } +} + +/** + * Worktree-aware read for `--continue`. The REPL bridge writes its pointer + * to `getOriginalCwd()` which EnterWorktreeTool/activeWorktreeSession can + * mutate to a worktree path — but `claude remote-control --continue` runs + * with `resolve('.')` = shell CWD. This fans out across git worktree + * siblings to find the freshest pointer, matching /resume's semantics. + * + * Fast path: checks `dir` first. Only shells out to `git worktree list` if + * that misses — the common case (pointer in launch dir) is one stat, zero + * exec. Fanout reads run in parallel; capped at MAX_WORKTREE_FANOUT. + * + * Returns the pointer AND the dir it was found in, so the caller can clear + * the right file on resume failure. + */ +export async function readBridgePointerAcrossWorktrees( + dir: string, +): Promise<{ pointer: BridgePointer & { ageMs: number }; dir: string } | null> { + // Fast path: current dir. Covers standalone bridge (always matches) and + // REPL bridge when no worktree mutation happened. + const here = await readBridgePointer(dir) + if (here) { + return { pointer: here, dir } + } + + // Fanout: scan worktree siblings. getWorktreePathsPortable has a 5s + // timeout and returns [] on any error (not a git repo, git not installed). + const worktrees = await getWorktreePathsPortable(dir) + if (worktrees.length <= 1) return null + if (worktrees.length > MAX_WORKTREE_FANOUT) { + logForDebugging( + `[bridge:pointer] ${worktrees.length} worktrees exceeds fanout cap ${MAX_WORKTREE_FANOUT}, skipping`, + ) + return null + } + + // Dedupe against `dir` so we don't re-stat it. sanitizePath normalizes + // case/separators so worktree-list output matches our fast-path key even + // on Windows where git may emit C:/ vs stored c:/. + const dirKey = sanitizePath(dir) + const candidates = worktrees.filter(wt => sanitizePath(wt) !== dirKey) + + // Parallel stat+read. Each readBridgePointer is a stat() that ENOENTs + // for worktrees with no pointer (cheap) plus a ~100-byte read for the + // rare ones that have one. Promise.all → latency ≈ slowest single stat. + const results = await Promise.all( + candidates.map(async wt => { + const p = await readBridgePointer(wt) + return p ? { pointer: p, dir: wt } : null + }), + ) + + // Pick freshest (lowest ageMs). The pointer stores environmentId so + // resume reconnects to the right env regardless of which worktree + // --continue was invoked from. + let freshest: { + pointer: BridgePointer & { ageMs: number } + dir: string + } | null = null + for (const r of results) { + if (r && (!freshest || r.pointer.ageMs < freshest.pointer.ageMs)) { + freshest = r + } + } + if (freshest) { + logForDebugging( + `[bridge:pointer] fanout found pointer in worktree ${freshest.dir} (ageMs=${freshest.pointer.ageMs})`, + ) + } + return freshest +} + +/** + * Delete the pointer. Idempotent — ENOENT is expected when the process + * shut down clean previously. + */ +export async function clearBridgePointer(dir: string): Promise { + const path = getBridgePointerPath(dir) + try { + await unlink(path) + logForDebugging(`[bridge:pointer] cleared ${path}`) + } catch (err: unknown) { + if (!isENOENT(err)) { + logForDebugging(`[bridge:pointer] clear failed: ${err}`, { + level: 'warn', + }) + } + } +} + +function safeJsonParse(raw: string): unknown { + try { + return jsonParse(raw) + } catch { + return null + } +} diff --git a/src/bridge/bridgeStatusUtil.ts b/src/bridge/bridgeStatusUtil.ts new file mode 100644 index 0000000..90de462 --- /dev/null +++ b/src/bridge/bridgeStatusUtil.ts @@ -0,0 +1,163 @@ +import { + getClaudeAiBaseUrl, + getRemoteSessionUrl, +} from '../constants/product.js' +import { stringWidth } from '../ink/stringWidth.js' +import { formatDuration, truncateToWidth } from '../utils/format.js' +import { getGraphemeSegmenter } from '../utils/intl.js' + +/** Bridge status state machine states. */ +export type StatusState = + | 'idle' + | 'attached' + | 'titled' + | 'reconnecting' + | 'failed' + +/** How long a tool activity line stays visible after last tool_start (ms). */ +export const TOOL_DISPLAY_EXPIRY_MS = 30_000 + +/** Interval for the shimmer animation tick (ms). */ +export const SHIMMER_INTERVAL_MS = 150 + +export function timestamp(): string { + const now = new Date() + const h = String(now.getHours()).padStart(2, '0') + const m = String(now.getMinutes()).padStart(2, '0') + const s = String(now.getSeconds()).padStart(2, '0') + return `${h}:${m}:${s}` +} + +export { formatDuration, truncateToWidth as truncatePrompt } + +/** Abbreviate a tool activity summary for the trail display. */ +export function abbreviateActivity(summary: string): string { + return truncateToWidth(summary, 30) +} + +/** Build the connect URL shown when the bridge is idle. */ +export function buildBridgeConnectUrl( + environmentId: string, + ingressUrl?: string, +): string { + const baseUrl = getClaudeAiBaseUrl(undefined, ingressUrl) + return `${baseUrl}/code?bridge=${environmentId}` +} + +/** + * Build the session URL shown when a session is attached. Delegates to + * getRemoteSessionUrl for the cse_→session_ prefix translation, then appends + * the v1-specific ?bridge={environmentId} query. + */ +export function buildBridgeSessionUrl( + sessionId: string, + environmentId: string, + ingressUrl?: string, +): string { + return `${getRemoteSessionUrl(sessionId, ingressUrl)}?bridge=${environmentId}` +} + +/** Compute the glimmer index for a reverse-sweep shimmer animation. */ +export function computeGlimmerIndex( + tick: number, + messageWidth: number, +): number { + const cycleLength = messageWidth + 20 + return messageWidth + 10 - (tick % cycleLength) +} + +/** + * Split text into three segments by visual column position for shimmer rendering. + * + * Uses grapheme segmentation and `stringWidth` so the split is correct for + * multi-byte characters, emoji, and CJK glyphs. + * + * Returns `{ before, shimmer, after }` strings. Both renderers (chalk in + * bridgeUI.ts and React/Ink in bridge.tsx) apply their own coloring to + * these segments. + */ +export function computeShimmerSegments( + text: string, + glimmerIndex: number, +): { before: string; shimmer: string; after: string } { + const messageWidth = stringWidth(text) + const shimmerStart = glimmerIndex - 1 + const shimmerEnd = glimmerIndex + 1 + + // When shimmer is offscreen, return all text as "before" + if (shimmerStart >= messageWidth || shimmerEnd < 0) { + return { before: text, shimmer: '', after: '' } + } + + // Split into at most 3 segments by visual column position + const clampedStart = Math.max(0, shimmerStart) + let colPos = 0 + let before = '' + let shimmer = '' + let after = '' + for (const { segment } of getGraphemeSegmenter().segment(text)) { + const segWidth = stringWidth(segment) + if (colPos + segWidth <= clampedStart) { + before += segment + } else if (colPos > shimmerEnd) { + after += segment + } else { + shimmer += segment + } + colPos += segWidth + } + + return { before, shimmer, after } +} + +/** Computed bridge status label and color from connection state. */ +export type BridgeStatusInfo = { + label: + | 'Remote Control failed' + | 'Remote Control reconnecting' + | 'Remote Control active' + | 'Remote Control connecting\u2026' + color: 'error' | 'warning' | 'success' +} + +/** Derive a status label and color from the bridge connection state. */ +export function getBridgeStatus({ + error, + connected, + sessionActive, + reconnecting, +}: { + error: string | undefined + connected: boolean + sessionActive: boolean + reconnecting: boolean +}): BridgeStatusInfo { + if (error) return { label: 'Remote Control failed', color: 'error' } + if (reconnecting) + return { label: 'Remote Control reconnecting', color: 'warning' } + if (sessionActive || connected) + return { label: 'Remote Control active', color: 'success' } + return { label: 'Remote Control connecting\u2026', color: 'warning' } +} + +/** Footer text shown when bridge is idle (Ready state). */ +export function buildIdleFooterText(url: string): string { + return `Code everywhere with the Claude app or ${url}` +} + +/** Footer text shown when a session is active (Connected state). */ +export function buildActiveFooterText(url: string): string { + return `Continue coding in the Claude app or ${url}` +} + +/** Footer text shown when the bridge has failed. */ +export const FAILED_FOOTER_TEXT = 'Something went wrong, please try again' + +/** + * Wrap text in an OSC 8 terminal hyperlink. Zero visual width for layout purposes. + * strip-ansi (used by stringWidth) correctly strips these sequences, so + * countVisualLines in bridgeUI.ts remains accurate. + */ +export function wrapWithOsc8Link(text: string, url: string): string { + return `\x1b]8;;${url}\x07${text}\x1b]8;;\x07` +} diff --git a/src/bridge/bridgeUI.ts b/src/bridge/bridgeUI.ts new file mode 100644 index 0000000..5149839 --- /dev/null +++ b/src/bridge/bridgeUI.ts @@ -0,0 +1,530 @@ +import chalk from 'chalk' +import { toString as qrToString } from 'qrcode' +import { + BRIDGE_FAILED_INDICATOR, + BRIDGE_READY_INDICATOR, + BRIDGE_SPINNER_FRAMES, +} from '../constants/figures.js' +import { stringWidth } from '../ink/stringWidth.js' +import { logForDebugging } from '../utils/debug.js' +import { + buildActiveFooterText, + buildBridgeConnectUrl, + buildBridgeSessionUrl, + buildIdleFooterText, + FAILED_FOOTER_TEXT, + formatDuration, + type StatusState, + TOOL_DISPLAY_EXPIRY_MS, + timestamp, + truncatePrompt, + wrapWithOsc8Link, +} from './bridgeStatusUtil.js' +import type { + BridgeConfig, + BridgeLogger, + SessionActivity, + SpawnMode, +} from './types.js' + +const QR_OPTIONS = { + type: 'utf8' as const, + errorCorrectionLevel: 'L' as const, + small: true, +} + +/** Generate a QR code and return its lines. */ +async function generateQr(url: string): Promise { + const qr = await qrToString(url, QR_OPTIONS) + return qr.split('\n').filter((line: string) => line.length > 0) +} + +export function createBridgeLogger(options: { + verbose: boolean + write?: (s: string) => void +}): BridgeLogger { + const write = options.write ?? ((s: string) => process.stdout.write(s)) + const verbose = options.verbose + + // Track how many status lines are currently displayed at the bottom + let statusLineCount = 0 + + // Status state machine + let currentState: StatusState = 'idle' + let currentStateText = 'Ready' + let repoName = '' + let branch = '' + let debugLogPath = '' + + // Connect URL (built in printBanner with correct base for staging/prod) + let connectUrl = '' + let cachedIngressUrl = '' + let cachedEnvironmentId = '' + let activeSessionUrl: string | null = null + + // QR code lines for the current URL + let qrLines: string[] = [] + let qrVisible = false + + // Tool activity for the second status line + let lastToolSummary: string | null = null + let lastToolTime = 0 + + // Session count indicator (shown when multi-session mode is enabled) + let sessionActive = 0 + let sessionMax = 1 + // Spawn mode shown in the session-count line + gates the `w` hint + let spawnModeDisplay: 'same-dir' | 'worktree' | null = null + let spawnMode: SpawnMode = 'single-session' + + // Per-session display info for the multi-session bullet list (keyed by compat sessionId) + const sessionDisplayInfo = new Map< + string, + { title?: string; url: string; activity?: SessionActivity } + >() + + // Connecting spinner state + let connectingTimer: ReturnType | null = null + let connectingTick = 0 + + /** + * Count how many visual terminal rows a string occupies, accounting for + * line wrapping. Each `\n` is one row, and content wider than the terminal + * wraps to additional rows. + */ + function countVisualLines(text: string): number { + // eslint-disable-next-line custom-rules/prefer-use-terminal-size + const cols = process.stdout.columns || 80 // non-React CLI context + let count = 0 + // Split on newlines to get logical lines + for (const logical of text.split('\n')) { + if (logical.length === 0) { + // Empty segment between consecutive \n — counts as 1 row + count++ + continue + } + const width = stringWidth(logical) + count += Math.max(1, Math.ceil(width / cols)) + } + // The trailing \n in "line\n" produces an empty last element — don't count it + // because the cursor sits at the start of the next line, not a new visual row. + if (text.endsWith('\n')) { + count-- + } + return count + } + + /** Write a status line and track its visual line count. */ + function writeStatus(text: string): void { + write(text) + statusLineCount += countVisualLines(text) + } + + /** Clear any currently displayed status lines. */ + function clearStatusLines(): void { + if (statusLineCount <= 0) return + logForDebugging(`[bridge:ui] clearStatusLines count=${statusLineCount}`) + // Move cursor up to the start of the status block, then erase everything below + write(`\x1b[${statusLineCount}A`) // cursor up N lines + write('\x1b[J') // erase from cursor to end of screen + statusLineCount = 0 + } + + /** Print a permanent log line, clearing status first and restoring after. */ + function printLog(line: string): void { + clearStatusLines() + write(line) + } + + /** Regenerate the QR code with the given URL. */ + function regenerateQr(url: string): void { + generateQr(url) + .then(lines => { + qrLines = lines + renderStatusLine() + }) + .catch(e => { + logForDebugging(`QR code generation failed: ${e}`, { level: 'error' }) + }) + } + + /** Render the connecting spinner line (shown before first updateIdleStatus). */ + function renderConnectingLine(): void { + clearStatusLines() + + const frame = + BRIDGE_SPINNER_FRAMES[connectingTick % BRIDGE_SPINNER_FRAMES.length]! + let suffix = '' + if (repoName) { + suffix += chalk.dim(' \u00b7 ') + chalk.dim(repoName) + } + if (branch) { + suffix += chalk.dim(' \u00b7 ') + chalk.dim(branch) + } + writeStatus( + `${chalk.yellow(frame)} ${chalk.yellow('Connecting')}${suffix}\n`, + ) + } + + /** Start the connecting spinner. Stopped by first updateIdleStatus(). */ + function startConnecting(): void { + stopConnecting() + renderConnectingLine() + connectingTimer = setInterval(() => { + connectingTick++ + renderConnectingLine() + }, 150) + } + + /** Stop the connecting spinner. */ + function stopConnecting(): void { + if (connectingTimer) { + clearInterval(connectingTimer) + connectingTimer = null + } + } + + /** Render and write the current status lines based on state. */ + function renderStatusLine(): void { + if (currentState === 'reconnecting' || currentState === 'failed') { + // These states are handled separately (updateReconnectingStatus / + // updateFailedStatus). Return before clearing so callers like toggleQr + // and setSpawnModeDisplay don't blank the display during these states. + return + } + + clearStatusLines() + + const isIdle = currentState === 'idle' + + // QR code above the status line + if (qrVisible) { + for (const line of qrLines) { + writeStatus(`${chalk.dim(line)}\n`) + } + } + + // Determine indicator and colors based on state + const indicator = BRIDGE_READY_INDICATOR + const indicatorColor = isIdle ? chalk.green : chalk.cyan + const baseColor = isIdle ? chalk.green : chalk.cyan + const stateText = baseColor(currentStateText) + + // Build the suffix with repo and branch + let suffix = '' + if (repoName) { + suffix += chalk.dim(' \u00b7 ') + chalk.dim(repoName) + } + // In worktree mode each session gets its own branch, so showing the + // bridge's branch would be misleading. + if (branch && spawnMode !== 'worktree') { + suffix += chalk.dim(' \u00b7 ') + chalk.dim(branch) + } + + if (process.env.USER_TYPE === 'ant' && debugLogPath) { + writeStatus( + `${chalk.yellow('[ANT-ONLY] Logs:')} ${chalk.dim(debugLogPath)}\n`, + ) + } + writeStatus(`${indicatorColor(indicator)} ${stateText}${suffix}\n`) + + // Session count and per-session list (multi-session mode only) + if (sessionMax > 1) { + const modeHint = + spawnMode === 'worktree' + ? 'New sessions will be created in an isolated worktree' + : 'New sessions will be created in the current directory' + writeStatus( + ` ${chalk.dim(`Capacity: ${sessionActive}/${sessionMax} \u00b7 ${modeHint}`)}\n`, + ) + for (const [, info] of sessionDisplayInfo) { + const titleText = info.title + ? truncatePrompt(info.title, 35) + : chalk.dim('Attached') + const titleLinked = wrapWithOsc8Link(titleText, info.url) + const act = info.activity + const showAct = act && act.type !== 'result' && act.type !== 'error' + const actText = showAct + ? chalk.dim(` ${truncatePrompt(act.summary, 40)}`) + : '' + writeStatus(` ${titleLinked}${actText} +`) + } + } + + // Mode line for spawn modes with a single slot (or true single-session mode) + if (sessionMax === 1) { + const modeText = + spawnMode === 'single-session' + ? 'Single session \u00b7 exits when complete' + : spawnMode === 'worktree' + ? `Capacity: ${sessionActive}/1 \u00b7 New sessions will be created in an isolated worktree` + : `Capacity: ${sessionActive}/1 \u00b7 New sessions will be created in the current directory` + writeStatus(` ${chalk.dim(modeText)}\n`) + } + + // Tool activity line for single-session mode + if ( + sessionMax === 1 && + !isIdle && + lastToolSummary && + Date.now() - lastToolTime < TOOL_DISPLAY_EXPIRY_MS + ) { + writeStatus(` ${chalk.dim(truncatePrompt(lastToolSummary, 60))}\n`) + } + + // Blank line separator before footer + const url = activeSessionUrl ?? connectUrl + if (url) { + writeStatus('\n') + const footerText = isIdle + ? buildIdleFooterText(url) + : buildActiveFooterText(url) + const qrHint = qrVisible + ? chalk.dim.italic('space to hide QR code') + : chalk.dim.italic('space to show QR code') + const toggleHint = spawnModeDisplay + ? chalk.dim.italic(' \u00b7 w to toggle spawn mode') + : '' + writeStatus(`${chalk.dim(footerText)}\n`) + writeStatus(`${qrHint}${toggleHint}\n`) + } + } + + return { + printBanner(config: BridgeConfig, environmentId: string): void { + cachedIngressUrl = config.sessionIngressUrl + cachedEnvironmentId = environmentId + connectUrl = buildBridgeConnectUrl(environmentId, cachedIngressUrl) + regenerateQr(connectUrl) + + if (verbose) { + write(chalk.dim(`Remote Control`) + ` v${MACRO.VERSION}\n`) + } + if (verbose) { + if (config.spawnMode !== 'single-session') { + write(chalk.dim(`Spawn mode: `) + `${config.spawnMode}\n`) + write( + chalk.dim(`Max concurrent sessions: `) + `${config.maxSessions}\n`, + ) + } + write(chalk.dim(`Environment ID: `) + `${environmentId}\n`) + } + if (config.sandbox) { + write(chalk.dim(`Sandbox: `) + `${chalk.green('Enabled')}\n`) + } + write('\n') + + // Start connecting spinner — first updateIdleStatus() will stop it + startConnecting() + }, + + logSessionStart(sessionId: string, prompt: string): void { + if (verbose) { + const short = truncatePrompt(prompt, 80) + printLog( + chalk.dim(`[${timestamp()}]`) + + ` Session started: ${chalk.white(`"${short}"`)} (${chalk.dim(sessionId)})\n`, + ) + } + }, + + logSessionComplete(sessionId: string, durationMs: number): void { + printLog( + chalk.dim(`[${timestamp()}]`) + + ` Session ${chalk.green('completed')} (${formatDuration(durationMs)}) ${chalk.dim(sessionId)}\n`, + ) + }, + + logSessionFailed(sessionId: string, error: string): void { + printLog( + chalk.dim(`[${timestamp()}]`) + + ` Session ${chalk.red('failed')}: ${error} ${chalk.dim(sessionId)}\n`, + ) + }, + + logStatus(message: string): void { + printLog(chalk.dim(`[${timestamp()}]`) + ` ${message}\n`) + }, + + logVerbose(message: string): void { + if (verbose) { + printLog(chalk.dim(`[${timestamp()}] ${message}`) + '\n') + } + }, + + logError(message: string): void { + printLog(chalk.red(`[${timestamp()}] Error: ${message}`) + '\n') + }, + + logReconnected(disconnectedMs: number): void { + printLog( + chalk.dim(`[${timestamp()}]`) + + ` ${chalk.green('Reconnected')} after ${formatDuration(disconnectedMs)}\n`, + ) + }, + + setRepoInfo(repo: string, branchName: string): void { + repoName = repo + branch = branchName + }, + + setDebugLogPath(path: string): void { + debugLogPath = path + }, + + updateIdleStatus(): void { + stopConnecting() + + currentState = 'idle' + currentStateText = 'Ready' + lastToolSummary = null + lastToolTime = 0 + activeSessionUrl = null + regenerateQr(connectUrl) + renderStatusLine() + }, + + setAttached(sessionId: string): void { + stopConnecting() + currentState = 'attached' + currentStateText = 'Connected' + lastToolSummary = null + lastToolTime = 0 + // Multi-session: keep footer/QR on the environment connect URL so users + // can spawn more sessions. Per-session links are in the bullet list. + if (sessionMax <= 1) { + activeSessionUrl = buildBridgeSessionUrl( + sessionId, + cachedEnvironmentId, + cachedIngressUrl, + ) + regenerateQr(activeSessionUrl) + } + renderStatusLine() + }, + + updateReconnectingStatus(delayStr: string, elapsedStr: string): void { + stopConnecting() + clearStatusLines() + currentState = 'reconnecting' + + // QR code above the status line + if (qrVisible) { + for (const line of qrLines) { + writeStatus(`${chalk.dim(line)}\n`) + } + } + + const frame = + BRIDGE_SPINNER_FRAMES[connectingTick % BRIDGE_SPINNER_FRAMES.length]! + connectingTick++ + writeStatus( + `${chalk.yellow(frame)} ${chalk.yellow('Reconnecting')} ${chalk.dim('\u00b7')} ${chalk.dim(`retrying in ${delayStr}`)} ${chalk.dim('\u00b7')} ${chalk.dim(`disconnected ${elapsedStr}`)}\n`, + ) + }, + + updateFailedStatus(error: string): void { + stopConnecting() + clearStatusLines() + currentState = 'failed' + + let suffix = '' + if (repoName) { + suffix += chalk.dim(' \u00b7 ') + chalk.dim(repoName) + } + if (branch) { + suffix += chalk.dim(' \u00b7 ') + chalk.dim(branch) + } + + writeStatus( + `${chalk.red(BRIDGE_FAILED_INDICATOR)} ${chalk.red('Remote Control Failed')}${suffix}\n`, + ) + writeStatus(`${chalk.dim(FAILED_FOOTER_TEXT)}\n`) + + if (error) { + writeStatus(`${chalk.red(error)}\n`) + } + }, + + updateSessionStatus( + _sessionId: string, + _elapsed: string, + activity: SessionActivity, + _trail: string[], + ): void { + // Cache tool activity for the second status line + if (activity.type === 'tool_start') { + lastToolSummary = activity.summary + lastToolTime = Date.now() + } + renderStatusLine() + }, + + clearStatus(): void { + stopConnecting() + clearStatusLines() + }, + + toggleQr(): void { + qrVisible = !qrVisible + renderStatusLine() + }, + + updateSessionCount(active: number, max: number, mode: SpawnMode): void { + if (sessionActive === active && sessionMax === max && spawnMode === mode) + return + sessionActive = active + sessionMax = max + spawnMode = mode + // Don't re-render here — the status ticker calls renderStatusLine + // on its own cadence, and the next tick will pick up the new values. + }, + + setSpawnModeDisplay(mode: 'same-dir' | 'worktree' | null): void { + if (spawnModeDisplay === mode) return + spawnModeDisplay = mode + // Also sync the #21118-added spawnMode so the next render shows correct + // mode hint + branch visibility. Don't render here — matches + // updateSessionCount: called before printBanner (initial setup) and + // again from the `w` handler (which follows with refreshDisplay). + if (mode) spawnMode = mode + }, + + addSession(sessionId: string, url: string): void { + sessionDisplayInfo.set(sessionId, { url }) + }, + + updateSessionActivity(sessionId: string, activity: SessionActivity): void { + const info = sessionDisplayInfo.get(sessionId) + if (!info) return + info.activity = activity + }, + + setSessionTitle(sessionId: string, title: string): void { + const info = sessionDisplayInfo.get(sessionId) + if (!info) return + info.title = title + // Guard against reconnecting/failed — renderStatusLine clears then returns + // early for those states, which would erase the spinner/error. + if (currentState === 'reconnecting' || currentState === 'failed') return + if (sessionMax === 1) { + // Single-session: show title in the main status line too. + currentState = 'titled' + currentStateText = truncatePrompt(title, 40) + } + renderStatusLine() + }, + + removeSession(sessionId: string): void { + sessionDisplayInfo.delete(sessionId) + }, + + refreshDisplay(): void { + // Skip during reconnecting/failed — renderStatusLine clears then returns + // early for those states, which would erase the spinner/error. + if (currentState === 'reconnecting' || currentState === 'failed') return + renderStatusLine() + }, + } +} diff --git a/src/bridge/capacityWake.ts b/src/bridge/capacityWake.ts new file mode 100644 index 0000000..e58c50d --- /dev/null +++ b/src/bridge/capacityWake.ts @@ -0,0 +1,56 @@ +/** + * Shared capacity-wake primitive for bridge poll loops. + * + * Both replBridge.ts and bridgeMain.ts need to sleep while "at capacity" + * but wake early when either (a) the outer loop signal aborts (shutdown), + * or (b) capacity frees up (session done / transport lost). This module + * encapsulates the mutable wake-controller + two-signal merger that both + * poll loops previously duplicated byte-for-byte. + */ + +export type CapacitySignal = { signal: AbortSignal; cleanup: () => void } + +export type CapacityWake = { + /** + * Create a signal that aborts when either the outer loop signal or the + * capacity-wake controller fires. Returns the merged signal and a cleanup + * function that removes listeners when the sleep resolves normally + * (without abort). + */ + signal(): CapacitySignal + /** + * Abort the current at-capacity sleep and arm a fresh controller so the + * poll loop immediately re-checks for new work. + */ + wake(): void +} + +export function createCapacityWake(outerSignal: AbortSignal): CapacityWake { + let wakeController = new AbortController() + + function wake(): void { + wakeController.abort() + wakeController = new AbortController() + } + + function signal(): CapacitySignal { + const merged = new AbortController() + const abort = (): void => merged.abort() + if (outerSignal.aborted || wakeController.signal.aborted) { + merged.abort() + return { signal: merged.signal, cleanup: () => {} } + } + outerSignal.addEventListener('abort', abort, { once: true }) + const capSig = wakeController.signal + capSig.addEventListener('abort', abort, { once: true }) + return { + signal: merged.signal, + cleanup: () => { + outerSignal.removeEventListener('abort', abort) + capSig.removeEventListener('abort', abort) + }, + } + } + + return { signal, wake } +} diff --git a/src/bridge/codeSessionApi.ts b/src/bridge/codeSessionApi.ts new file mode 100644 index 0000000..65b46a3 --- /dev/null +++ b/src/bridge/codeSessionApi.ts @@ -0,0 +1,168 @@ +/** + * Thin HTTP wrappers for the CCR v2 code-session API. + * + * Separate file from remoteBridgeCore.ts so the SDK /bridge subpath can + * export createCodeSession + fetchRemoteCredentials without bundling the + * heavy CLI tree (analytics, transport, etc.). Callers supply explicit + * accessToken + baseUrl — no implicit auth or config reads. + */ + +import axios from 'axios' +import { logForDebugging } from '../utils/debug.js' +import { errorMessage } from '../utils/errors.js' +import { jsonStringify } from '../utils/slowOperations.js' +import { extractErrorDetail } from './debugUtils.js' + +const ANTHROPIC_VERSION = '2023-06-01' + +function oauthHeaders(accessToken: string): Record { + return { + Authorization: `Bearer ${accessToken}`, + 'Content-Type': 'application/json', + 'anthropic-version': ANTHROPIC_VERSION, + } +} + +export async function createCodeSession( + baseUrl: string, + accessToken: string, + title: string, + timeoutMs: number, + tags?: string[], +): Promise { + const url = `${baseUrl}/v1/code/sessions` + let response + try { + response = await axios.post( + url, + // bridge: {} is the positive signal for the oneof runner — omitting it + // (or sending environment_id: "") now 400s. BridgeRunner is an empty + // message today; it's a placeholder for future bridge-specific options. + { title, bridge: {}, ...(tags?.length ? { tags } : {}) }, + { + headers: oauthHeaders(accessToken), + timeout: timeoutMs, + validateStatus: s => s < 500, + }, + ) + } catch (err: unknown) { + logForDebugging( + `[code-session] Session create request failed: ${errorMessage(err)}`, + ) + return null + } + + if (response.status !== 200 && response.status !== 201) { + const detail = extractErrorDetail(response.data) + logForDebugging( + `[code-session] Session create failed ${response.status}${detail ? `: ${detail}` : ''}`, + ) + return null + } + + const data: unknown = response.data + if ( + !data || + typeof data !== 'object' || + !('session' in data) || + !data.session || + typeof data.session !== 'object' || + !('id' in data.session) || + typeof data.session.id !== 'string' || + !data.session.id.startsWith('cse_') + ) { + logForDebugging( + `[code-session] No session.id (cse_*) in response: ${jsonStringify(data).slice(0, 200)}`, + ) + return null + } + return data.session.id +} + +/** + * Credentials from POST /bridge. JWT is opaque — do not decode. + * Each /bridge call bumps worker_epoch server-side (it IS the register). + */ +export type RemoteCredentials = { + worker_jwt: string + api_base_url: string + expires_in: number + worker_epoch: number +} + +export async function fetchRemoteCredentials( + sessionId: string, + baseUrl: string, + accessToken: string, + timeoutMs: number, + trustedDeviceToken?: string, +): Promise { + const url = `${baseUrl}/v1/code/sessions/${sessionId}/bridge` + const headers = oauthHeaders(accessToken) + if (trustedDeviceToken) { + headers['X-Trusted-Device-Token'] = trustedDeviceToken + } + let response + try { + response = await axios.post( + url, + {}, + { + headers, + timeout: timeoutMs, + validateStatus: s => s < 500, + }, + ) + } catch (err: unknown) { + logForDebugging( + `[code-session] /bridge request failed: ${errorMessage(err)}`, + ) + return null + } + + if (response.status !== 200) { + const detail = extractErrorDetail(response.data) + logForDebugging( + `[code-session] /bridge failed ${response.status}${detail ? `: ${detail}` : ''}`, + ) + return null + } + + const data: unknown = response.data + if ( + data === null || + typeof data !== 'object' || + !('worker_jwt' in data) || + typeof data.worker_jwt !== 'string' || + !('expires_in' in data) || + typeof data.expires_in !== 'number' || + !('api_base_url' in data) || + typeof data.api_base_url !== 'string' || + !('worker_epoch' in data) + ) { + logForDebugging( + `[code-session] /bridge response malformed (need worker_jwt, expires_in, api_base_url, worker_epoch): ${jsonStringify(data).slice(0, 200)}`, + ) + return null + } + // protojson serializes int64 as a string to avoid JS precision loss; + // Go may also return a number depending on encoder settings. + const rawEpoch = data.worker_epoch + const epoch = typeof rawEpoch === 'string' ? Number(rawEpoch) : rawEpoch + if ( + typeof epoch !== 'number' || + !Number.isFinite(epoch) || + !Number.isSafeInteger(epoch) + ) { + logForDebugging( + `[code-session] /bridge worker_epoch invalid: ${jsonStringify(rawEpoch)}`, + ) + return null + } + return { + worker_jwt: data.worker_jwt, + api_base_url: data.api_base_url, + expires_in: data.expires_in, + worker_epoch: epoch, + } +} diff --git a/src/bridge/createSession.ts b/src/bridge/createSession.ts new file mode 100644 index 0000000..d5bc83a --- /dev/null +++ b/src/bridge/createSession.ts @@ -0,0 +1,384 @@ +import type { SDKMessage } from '../entrypoints/agentSdkTypes.js' +import { logForDebugging } from '../utils/debug.js' +import { errorMessage } from '../utils/errors.js' +import { extractErrorDetail } from './debugUtils.js' +import { toCompatSessionId } from './sessionIdCompat.js' + +type GitSource = { + type: 'git_repository' + url: string + revision?: string +} + +type GitOutcome = { + type: 'git_repository' + git_info: { type: 'github'; repo: string; branches: string[] } +} + +// Events must be wrapped in { type: 'event', data: } for the +// POST /v1/sessions endpoint (discriminated union format). +type SessionEvent = { + type: 'event' + data: SDKMessage +} + +/** + * Create a session on a bridge environment via POST /v1/sessions. + * + * Used by both `claude remote-control` (empty session so the user has somewhere to + * type immediately) and `/remote-control` (session pre-populated with conversation + * history). + * + * Returns the session ID on success, or null if creation fails (non-fatal). + */ +export async function createBridgeSession({ + environmentId, + title, + events, + gitRepoUrl, + branch, + signal, + baseUrl: baseUrlOverride, + getAccessToken, + permissionMode, +}: { + environmentId: string + title?: string + events: SessionEvent[] + gitRepoUrl: string | null + branch: string + signal: AbortSignal + baseUrl?: string + getAccessToken?: () => string | undefined + permissionMode?: string +}): Promise { + const { getClaudeAIOAuthTokens } = await import('../utils/auth.js') + const { getOrganizationUUID } = await import('../services/oauth/client.js') + const { getOauthConfig } = await import('../constants/oauth.js') + const { getOAuthHeaders } = await import('../utils/teleport/api.js') + const { parseGitHubRepository } = await import('../utils/detectRepository.js') + const { getDefaultBranch } = await import('../utils/git.js') + const { getMainLoopModel } = await import('../utils/model/model.js') + const { default: axios } = await import('axios') + + const accessToken = + getAccessToken?.() ?? getClaudeAIOAuthTokens()?.accessToken + if (!accessToken) { + logForDebugging('[bridge] No access token for session creation') + return null + } + + const orgUUID = await getOrganizationUUID() + if (!orgUUID) { + logForDebugging('[bridge] No org UUID for session creation') + return null + } + + // Build git source and outcome context + let gitSource: GitSource | null = null + let gitOutcome: GitOutcome | null = null + + if (gitRepoUrl) { + const { parseGitRemote } = await import('../utils/detectRepository.js') + const parsed = parseGitRemote(gitRepoUrl) + if (parsed) { + const { host, owner, name } = parsed + const revision = branch || (await getDefaultBranch()) || undefined + gitSource = { + type: 'git_repository', + url: `https://${host}/${owner}/${name}`, + revision, + } + gitOutcome = { + type: 'git_repository', + git_info: { + type: 'github', + repo: `${owner}/${name}`, + branches: [`claude/${branch || 'task'}`], + }, + } + } else { + // Fallback: try parseGitHubRepository for owner/repo format + const ownerRepo = parseGitHubRepository(gitRepoUrl) + if (ownerRepo) { + const [owner, name] = ownerRepo.split('/') + if (owner && name) { + const revision = branch || (await getDefaultBranch()) || undefined + gitSource = { + type: 'git_repository', + url: `https://github.com/${owner}/${name}`, + revision, + } + gitOutcome = { + type: 'git_repository', + git_info: { + type: 'github', + repo: `${owner}/${name}`, + branches: [`claude/${branch || 'task'}`], + }, + } + } + } + } + } + + const requestBody = { + ...(title !== undefined && { title }), + events, + session_context: { + sources: gitSource ? [gitSource] : [], + outcomes: gitOutcome ? [gitOutcome] : [], + model: getMainLoopModel(), + }, + environment_id: environmentId, + source: 'remote-control', + ...(permissionMode && { permission_mode: permissionMode }), + } + + const headers = { + ...getOAuthHeaders(accessToken), + 'anthropic-beta': 'ccr-byoc-2025-07-29', + 'x-organization-uuid': orgUUID, + } + + const url = `${baseUrlOverride ?? getOauthConfig().BASE_API_URL}/v1/sessions` + let response + try { + response = await axios.post(url, requestBody, { + headers, + signal, + validateStatus: s => s < 500, + }) + } catch (err: unknown) { + logForDebugging( + `[bridge] Session creation request failed: ${errorMessage(err)}`, + ) + return null + } + const isSuccess = response.status === 200 || response.status === 201 + + if (!isSuccess) { + const detail = extractErrorDetail(response.data) + logForDebugging( + `[bridge] Session creation failed with status ${response.status}${detail ? `: ${detail}` : ''}`, + ) + return null + } + + const sessionData: unknown = response.data + if ( + !sessionData || + typeof sessionData !== 'object' || + !('id' in sessionData) || + typeof sessionData.id !== 'string' + ) { + logForDebugging('[bridge] No session ID in response') + return null + } + + return sessionData.id +} + +/** + * Fetch a bridge session via GET /v1/sessions/{id}. + * + * Returns the session's environment_id (for `--session-id` resume) and title. + * Uses the same org-scoped headers as create/archive — the environments-level + * client in bridgeApi.ts uses a different beta header and no org UUID, which + * makes the Sessions API return 404. + */ +export async function getBridgeSession( + sessionId: string, + opts?: { baseUrl?: string; getAccessToken?: () => string | undefined }, +): Promise<{ environment_id?: string; title?: string } | null> { + const { getClaudeAIOAuthTokens } = await import('../utils/auth.js') + const { getOrganizationUUID } = await import('../services/oauth/client.js') + const { getOauthConfig } = await import('../constants/oauth.js') + const { getOAuthHeaders } = await import('../utils/teleport/api.js') + const { default: axios } = await import('axios') + + const accessToken = + opts?.getAccessToken?.() ?? getClaudeAIOAuthTokens()?.accessToken + if (!accessToken) { + logForDebugging('[bridge] No access token for session fetch') + return null + } + + const orgUUID = await getOrganizationUUID() + if (!orgUUID) { + logForDebugging('[bridge] No org UUID for session fetch') + return null + } + + const headers = { + ...getOAuthHeaders(accessToken), + 'anthropic-beta': 'ccr-byoc-2025-07-29', + 'x-organization-uuid': orgUUID, + } + + const url = `${opts?.baseUrl ?? getOauthConfig().BASE_API_URL}/v1/sessions/${sessionId}` + logForDebugging(`[bridge] Fetching session ${sessionId}`) + + let response + try { + response = await axios.get<{ environment_id?: string; title?: string }>( + url, + { headers, timeout: 10_000, validateStatus: s => s < 500 }, + ) + } catch (err: unknown) { + logForDebugging( + `[bridge] Session fetch request failed: ${errorMessage(err)}`, + ) + return null + } + + if (response.status !== 200) { + const detail = extractErrorDetail(response.data) + logForDebugging( + `[bridge] Session fetch failed with status ${response.status}${detail ? `: ${detail}` : ''}`, + ) + return null + } + + return response.data +} + +/** + * Archive a bridge session via POST /v1/sessions/{id}/archive. + * + * The CCR server never auto-archives sessions — archival is always an + * explicit client action. Both `claude remote-control` (standalone bridge) and the + * always-on `/remote-control` REPL bridge call this during shutdown to archive any + * sessions that are still alive. + * + * The archive endpoint accepts sessions in any status (running, idle, + * requires_action, pending) and returns 409 if already archived, making + * it safe to call even if the server-side runner already archived the + * session. + * + * Callers must handle errors — this function has no try/catch; 5xx, + * timeouts, and network errors throw. Archival is best-effort during + * cleanup; call sites wrap with .catch(). + */ +export async function archiveBridgeSession( + sessionId: string, + opts?: { + baseUrl?: string + getAccessToken?: () => string | undefined + timeoutMs?: number + }, +): Promise { + const { getClaudeAIOAuthTokens } = await import('../utils/auth.js') + const { getOrganizationUUID } = await import('../services/oauth/client.js') + const { getOauthConfig } = await import('../constants/oauth.js') + const { getOAuthHeaders } = await import('../utils/teleport/api.js') + const { default: axios } = await import('axios') + + const accessToken = + opts?.getAccessToken?.() ?? getClaudeAIOAuthTokens()?.accessToken + if (!accessToken) { + logForDebugging('[bridge] No access token for session archive') + return + } + + const orgUUID = await getOrganizationUUID() + if (!orgUUID) { + logForDebugging('[bridge] No org UUID for session archive') + return + } + + const headers = { + ...getOAuthHeaders(accessToken), + 'anthropic-beta': 'ccr-byoc-2025-07-29', + 'x-organization-uuid': orgUUID, + } + + const url = `${opts?.baseUrl ?? getOauthConfig().BASE_API_URL}/v1/sessions/${sessionId}/archive` + logForDebugging(`[bridge] Archiving session ${sessionId}`) + + const response = await axios.post( + url, + {}, + { + headers, + timeout: opts?.timeoutMs ?? 10_000, + validateStatus: s => s < 500, + }, + ) + + if (response.status === 200) { + logForDebugging(`[bridge] Session ${sessionId} archived successfully`) + } else { + const detail = extractErrorDetail(response.data) + logForDebugging( + `[bridge] Session archive failed with status ${response.status}${detail ? `: ${detail}` : ''}`, + ) + } +} + +/** + * Update the title of a bridge session via PATCH /v1/sessions/{id}. + * + * Called when the user renames a session via /rename while a bridge + * connection is active, so the title stays in sync on claude.ai/code. + * + * Errors are swallowed — title sync is best-effort. + */ +export async function updateBridgeSessionTitle( + sessionId: string, + title: string, + opts?: { baseUrl?: string; getAccessToken?: () => string | undefined }, +): Promise { + const { getClaudeAIOAuthTokens } = await import('../utils/auth.js') + const { getOrganizationUUID } = await import('../services/oauth/client.js') + const { getOauthConfig } = await import('../constants/oauth.js') + const { getOAuthHeaders } = await import('../utils/teleport/api.js') + const { default: axios } = await import('axios') + + const accessToken = + opts?.getAccessToken?.() ?? getClaudeAIOAuthTokens()?.accessToken + if (!accessToken) { + logForDebugging('[bridge] No access token for session title update') + return + } + + const orgUUID = await getOrganizationUUID() + if (!orgUUID) { + logForDebugging('[bridge] No org UUID for session title update') + return + } + + const headers = { + ...getOAuthHeaders(accessToken), + 'anthropic-beta': 'ccr-byoc-2025-07-29', + 'x-organization-uuid': orgUUID, + } + + // Compat gateway only accepts session_* (compat/convert.go:27). v2 callers + // pass raw cse_*; retag here so all callers can pass whatever they hold. + // Idempotent for v1's session_* and bridgeMain's pre-converted compatSessionId. + const compatId = toCompatSessionId(sessionId) + const url = `${opts?.baseUrl ?? getOauthConfig().BASE_API_URL}/v1/sessions/${compatId}` + logForDebugging(`[bridge] Updating session title: ${compatId} → ${title}`) + + try { + const response = await axios.patch( + url, + { title }, + { headers, timeout: 10_000, validateStatus: s => s < 500 }, + ) + + if (response.status === 200) { + logForDebugging(`[bridge] Session title updated successfully`) + } else { + const detail = extractErrorDetail(response.data) + logForDebugging( + `[bridge] Session title update failed with status ${response.status}${detail ? `: ${detail}` : ''}`, + ) + } + } catch (err: unknown) { + logForDebugging( + `[bridge] Session title update request failed: ${errorMessage(err)}`, + ) + } +} diff --git a/src/bridge/debugUtils.ts b/src/bridge/debugUtils.ts new file mode 100644 index 0000000..e9f7293 --- /dev/null +++ b/src/bridge/debugUtils.ts @@ -0,0 +1,141 @@ +import { + type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + logEvent, +} from '../services/analytics/index.js' +import { logForDebugging } from '../utils/debug.js' +import { errorMessage } from '../utils/errors.js' +import { jsonStringify } from '../utils/slowOperations.js' + +const DEBUG_MSG_LIMIT = 2000 + +const SECRET_FIELD_NAMES = [ + 'session_ingress_token', + 'environment_secret', + 'access_token', + 'secret', + 'token', +] + +const SECRET_PATTERN = new RegExp( + `"(${SECRET_FIELD_NAMES.join('|')})"\\s*:\\s*"([^"]*)"`, + 'g', +) + +const REDACT_MIN_LENGTH = 16 + +export function redactSecrets(s: string): string { + return s.replace(SECRET_PATTERN, (_match, field: string, value: string) => { + if (value.length < REDACT_MIN_LENGTH) { + return `"${field}":"[REDACTED]"` + } + const redacted = `${value.slice(0, 8)}...${value.slice(-4)}` + return `"${field}":"${redacted}"` + }) +} + +/** Truncate a string for debug logging, collapsing newlines. */ +export function debugTruncate(s: string): string { + const flat = s.replace(/\n/g, '\\n') + if (flat.length <= DEBUG_MSG_LIMIT) { + return flat + } + return flat.slice(0, DEBUG_MSG_LIMIT) + `... (${flat.length} chars)` +} + +/** Truncate a JSON-serializable value for debug logging. */ +export function debugBody(data: unknown): string { + const raw = typeof data === 'string' ? data : jsonStringify(data) + const s = redactSecrets(raw) + if (s.length <= DEBUG_MSG_LIMIT) { + return s + } + return s.slice(0, DEBUG_MSG_LIMIT) + `... (${s.length} chars)` +} + +/** + * Extract a descriptive error message from an axios error (or any error). + * For HTTP errors, appends the server's response body message if available, + * since axios's default message only includes the status code. + */ +export function describeAxiosError(err: unknown): string { + const msg = errorMessage(err) + if (err && typeof err === 'object' && 'response' in err) { + const response = (err as { response?: { data?: unknown } }).response + if (response?.data && typeof response.data === 'object') { + const data = response.data as Record + const detail = + typeof data.message === 'string' + ? data.message + : typeof data.error === 'object' && + data.error && + 'message' in data.error && + typeof (data.error as Record).message === + 'string' + ? (data.error as Record).message + : undefined + if (detail) { + return `${msg}: ${detail}` + } + } + } + return msg +} + +/** + * Extract the HTTP status code from an axios error, if present. + * Returns undefined for non-HTTP errors (e.g. network failures). + */ +export function extractHttpStatus(err: unknown): number | undefined { + if ( + err && + typeof err === 'object' && + 'response' in err && + (err as { response?: { status?: unknown } }).response && + typeof (err as { response: { status?: unknown } }).response.status === + 'number' + ) { + return (err as { response: { status: number } }).response.status + } + return undefined +} + +/** + * Pull a human-readable message out of an API error response body. + * Checks `data.message` first, then `data.error.message`. + */ +export function extractErrorDetail(data: unknown): string | undefined { + if (!data || typeof data !== 'object') return undefined + if ('message' in data && typeof data.message === 'string') { + return data.message + } + if ( + 'error' in data && + data.error !== null && + typeof data.error === 'object' && + 'message' in data.error && + typeof data.error.message === 'string' + ) { + return data.error.message + } + return undefined +} + +/** + * Log a bridge init skip — debug message + `tengu_bridge_repl_skipped` + * analytics event. Centralizes the event name and the AnalyticsMetadata + * cast so call sites don't each repeat the 5-line boilerplate. + */ +export function logBridgeSkip( + reason: string, + debugMsg?: string, + v2?: boolean, +): void { + if (debugMsg) { + logForDebugging(debugMsg) + } + logEvent('tengu_bridge_repl_skipped', { + reason: + reason as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + ...(v2 !== undefined && { v2 }), + }) +} diff --git a/src/bridge/envLessBridgeConfig.ts b/src/bridge/envLessBridgeConfig.ts new file mode 100644 index 0000000..de0cb5e --- /dev/null +++ b/src/bridge/envLessBridgeConfig.ts @@ -0,0 +1,165 @@ +import { z } from 'zod/v4' +import { getFeatureValue_DEPRECATED } from '../services/analytics/growthbook.js' +import { lazySchema } from '../utils/lazySchema.js' +import { lt } from '../utils/semver.js' +import { isEnvLessBridgeEnabled } from './bridgeEnabled.js' + +export type EnvLessBridgeConfig = { + // withRetry — init-phase backoff (createSession, POST /bridge, recovery /bridge) + init_retry_max_attempts: number + init_retry_base_delay_ms: number + init_retry_jitter_fraction: number + init_retry_max_delay_ms: number + // axios timeout for POST /sessions, POST /bridge, POST /archive + http_timeout_ms: number + // BoundedUUIDSet ring size (echo + re-delivery dedup) + uuid_dedup_buffer_size: number + // CCRClient worker heartbeat cadence. Server TTL is 60s — 20s gives 3× margin. + heartbeat_interval_ms: number + // ±fraction of interval — per-beat jitter to spread fleet load. + heartbeat_jitter_fraction: number + // Fire proactive JWT refresh this long before expires_in. Larger buffer = + // more frequent refresh (refresh cadence ≈ expires_in - buffer). + token_refresh_buffer_ms: number + // Archive POST timeout in teardown(). Distinct from http_timeout_ms because + // gracefulShutdown races runCleanupFunctions() against a 2s cap — a 10s + // axios timeout on a slow/stalled archive burns the whole budget on a + // request that forceExit will kill anyway. + teardown_archive_timeout_ms: number + // Deadline for onConnect after transport.connect(). If neither onConnect + // nor onClose fires before this, emit tengu_bridge_repl_connect_timeout + // — the only telemetry for the ~1% of sessions that emit `started` then + // go silent (no error, no event, just nothing). + connect_timeout_ms: number + // Semver floor for the env-less bridge path. Separate from the v1 + // tengu_bridge_min_version config so a v2-specific bug can force upgrades + // without blocking v1 (env-based) clients, and vice versa. + min_version: string + // When true, tell users their claude.ai app may be too old to see v2 + // sessions — lets us roll the v2 bridge before the app ships the new + // session-list query. + should_show_app_upgrade_message: boolean +} + +export const DEFAULT_ENV_LESS_BRIDGE_CONFIG: EnvLessBridgeConfig = { + init_retry_max_attempts: 3, + init_retry_base_delay_ms: 500, + init_retry_jitter_fraction: 0.25, + init_retry_max_delay_ms: 4000, + http_timeout_ms: 10_000, + uuid_dedup_buffer_size: 2000, + heartbeat_interval_ms: 20_000, + heartbeat_jitter_fraction: 0.1, + token_refresh_buffer_ms: 300_000, + teardown_archive_timeout_ms: 1500, + connect_timeout_ms: 15_000, + min_version: '0.0.0', + should_show_app_upgrade_message: false, +} + +// Floors reject the whole object on violation (fall back to DEFAULT) rather +// than partially trusting — same defense-in-depth as pollConfig.ts. +const envLessBridgeConfigSchema = lazySchema(() => + z.object({ + init_retry_max_attempts: z.number().int().min(1).max(10).default(3), + init_retry_base_delay_ms: z.number().int().min(100).default(500), + init_retry_jitter_fraction: z.number().min(0).max(1).default(0.25), + init_retry_max_delay_ms: z.number().int().min(500).default(4000), + http_timeout_ms: z.number().int().min(2000).default(10_000), + uuid_dedup_buffer_size: z.number().int().min(100).max(50_000).default(2000), + // Server TTL is 60s. Floor 5s prevents thrash; cap 30s keeps ≥2× margin. + heartbeat_interval_ms: z + .number() + .int() + .min(5000) + .max(30_000) + .default(20_000), + // ±fraction per beat. Cap 0.5: at max interval (30s) × 1.5 = 45s worst case, + // still under the 60s TTL. + heartbeat_jitter_fraction: z.number().min(0).max(0.5).default(0.1), + // Floor 30s prevents tight-looping. Cap 30min rejects buffer-vs-delay + // semantic inversion: ops entering expires_in-5min (the *delay until + // refresh*) instead of 5min (the *buffer before expiry*) yields + // delayMs = expires_in - buffer ≈ 5min instead of ≈4h. Both are positive + // durations so .min() alone can't distinguish; .max() catches the + // inverted value since buffer ≥ 30min is nonsensical for a multi-hour JWT. + token_refresh_buffer_ms: z + .number() + .int() + .min(30_000) + .max(1_800_000) + .default(300_000), + // Cap 2000 keeps this under gracefulShutdown's 2s cleanup race — a higher + // timeout just lies to axios since forceExit kills the socket regardless. + teardown_archive_timeout_ms: z + .number() + .int() + .min(500) + .max(2000) + .default(1500), + // Observed p99 connect is ~2-3s; 15s is ~5× headroom. Floor 5s bounds + // false-positive rate under transient slowness; cap 60s bounds how long + // a truly-stalled session stays dark. + connect_timeout_ms: z.number().int().min(5_000).max(60_000).default(15_000), + min_version: z + .string() + .refine(v => { + try { + lt(v, '0.0.0') + return true + } catch { + return false + } + }) + .default('0.0.0'), + should_show_app_upgrade_message: z.boolean().default(false), + }), +) + +/** + * Fetch the env-less bridge timing config from GrowthBook. Read once per + * initEnvLessBridgeCore call — config is fixed for the lifetime of a bridge + * session. + * + * Uses the blocking getter (not _CACHED_MAY_BE_STALE) because /remote-control + * runs well after GrowthBook init — initializeGrowthBook() resolves instantly, + * so there's no startup penalty, and we get the fresh in-memory remoteEval + * value instead of the stale-on-first-read disk cache. The _DEPRECATED suffix + * warns against startup-path usage, which this isn't. + */ +export async function getEnvLessBridgeConfig(): Promise { + const raw = await getFeatureValue_DEPRECATED( + 'tengu_bridge_repl_v2_config', + DEFAULT_ENV_LESS_BRIDGE_CONFIG, + ) + const parsed = envLessBridgeConfigSchema().safeParse(raw) + return parsed.success ? parsed.data : DEFAULT_ENV_LESS_BRIDGE_CONFIG +} + +/** + * Returns an error message if the current CLI version is below the minimum + * required for the env-less (v2) bridge path, or null if the version is fine. + * + * v2 analogue of checkBridgeMinVersion() — reads from tengu_bridge_repl_v2_config + * instead of tengu_bridge_min_version so the two implementations can enforce + * independent floors. + */ +export async function checkEnvLessBridgeMinVersion(): Promise { + const cfg = await getEnvLessBridgeConfig() + if (cfg.min_version && lt(MACRO.VERSION, cfg.min_version)) { + return `Your version of Claude Code (${MACRO.VERSION}) is too old for Remote Control.\nVersion ${cfg.min_version} or higher is required. Run \`claude update\` to update.` + } + return null +} + +/** + * Whether to nudge users toward upgrading their claude.ai app when a + * Remote Control session starts. True only when the v2 bridge is active + * AND the should_show_app_upgrade_message config bit is set — lets us + * roll the v2 bridge before the app ships the new session-list query. + */ +export async function shouldShowAppUpgradeMessage(): Promise { + if (!isEnvLessBridgeEnabled()) return false + const cfg = await getEnvLessBridgeConfig() + return cfg.should_show_app_upgrade_message +} diff --git a/src/bridge/flushGate.ts b/src/bridge/flushGate.ts new file mode 100644 index 0000000..6216334 --- /dev/null +++ b/src/bridge/flushGate.ts @@ -0,0 +1,71 @@ +/** + * State machine for gating message writes during an initial flush. + * + * When a bridge session starts, historical messages are flushed to the + * server via a single HTTP POST. During that flush, new messages must + * be queued to prevent them from arriving at the server interleaved + * with the historical messages. + * + * Lifecycle: + * start() → enqueue() returns true, items are queued + * end() → returns queued items for draining, enqueue() returns false + * drop() → discards queued items (permanent transport close) + * deactivate() → clears active flag without dropping items + * (transport replacement — new transport will drain) + */ +export class FlushGate { + private _active = false + private _pending: T[] = [] + + get active(): boolean { + return this._active + } + + get pendingCount(): number { + return this._pending.length + } + + /** Mark flush as in-progress. enqueue() will start queuing items. */ + start(): void { + this._active = true + } + + /** + * End the flush and return any queued items for draining. + * Caller is responsible for sending the returned items. + */ + end(): T[] { + this._active = false + return this._pending.splice(0) + } + + /** + * If flush is active, queue the items and return true. + * If flush is not active, return false (caller should send directly). + */ + enqueue(...items: T[]): boolean { + if (!this._active) return false + this._pending.push(...items) + return true + } + + /** + * Discard all queued items (permanent transport close). + * Returns the number of items dropped. + */ + drop(): number { + this._active = false + const count = this._pending.length + this._pending.length = 0 + return count + } + + /** + * Clear the active flag without dropping queued items. + * Used when the transport is replaced (onWorkReceived) — the new + * transport's flush will drain the pending items. + */ + deactivate(): void { + this._active = false + } +} diff --git a/src/bridge/inboundAttachments.ts b/src/bridge/inboundAttachments.ts new file mode 100644 index 0000000..f7c13c8 --- /dev/null +++ b/src/bridge/inboundAttachments.ts @@ -0,0 +1,175 @@ +/** + * Resolve file_uuid attachments on inbound bridge user messages. + * + * Web composer uploads via cookie-authed /api/{org}/upload, sends file_uuid + * alongside the message. Here we fetch each via GET /api/oauth/files/{uuid}/content + * (oauth-authed, same store), write to ~/.claude/uploads/{sessionId}/, and + * return @path refs to prepend. Claude's Read tool takes it from there. + * + * Best-effort: any failure (no token, network, non-2xx, disk) logs debug and + * skips that attachment. The message still reaches Claude, just without @path. + */ + +import type { ContentBlockParam } from '@anthropic-ai/sdk/resources/messages.mjs' +import axios from 'axios' +import { randomUUID } from 'crypto' +import { mkdir, writeFile } from 'fs/promises' +import { basename, join } from 'path' +import { z } from 'zod/v4' +import { getSessionId } from '../bootstrap/state.js' +import { logForDebugging } from '../utils/debug.js' +import { getClaudeConfigHomeDir } from '../utils/envUtils.js' +import { lazySchema } from '../utils/lazySchema.js' +import { getBridgeAccessToken, getBridgeBaseUrl } from './bridgeConfig.js' + +const DOWNLOAD_TIMEOUT_MS = 30_000 + +function debug(msg: string): void { + logForDebugging(`[bridge:inbound-attach] ${msg}`) +} + +const attachmentSchema = lazySchema(() => + z.object({ + file_uuid: z.string(), + file_name: z.string(), + }), +) +const attachmentsArraySchema = lazySchema(() => z.array(attachmentSchema())) + +export type InboundAttachment = z.infer> + +/** Pull file_attachments off a loosely-typed inbound message. */ +export function extractInboundAttachments(msg: unknown): InboundAttachment[] { + if (typeof msg !== 'object' || msg === null || !('file_attachments' in msg)) { + return [] + } + const parsed = attachmentsArraySchema().safeParse(msg.file_attachments) + return parsed.success ? parsed.data : [] +} + +/** + * Strip path components and keep only filename-safe chars. file_name comes + * from the network (web composer), so treat it as untrusted even though the + * composer controls it. + */ +function sanitizeFileName(name: string): string { + const base = basename(name).replace(/[^a-zA-Z0-9._-]/g, '_') + return base || 'attachment' +} + +function uploadsDir(): string { + return join(getClaudeConfigHomeDir(), 'uploads', getSessionId()) +} + +/** + * Fetch + write one attachment. Returns the absolute path on success, + * undefined on any failure. + */ +async function resolveOne(att: InboundAttachment): Promise { + const token = getBridgeAccessToken() + if (!token) { + debug('skip: no oauth token') + return undefined + } + + let data: Buffer + try { + // getOauthConfig() (via getBridgeBaseUrl) throws on a non-allowlisted + // CLAUDE_CODE_CUSTOM_OAUTH_URL — keep it inside the try so a bad + // FedStart URL degrades to "no @path" instead of crashing print.ts's + // reader loop (which has no catch around the await). + const url = `${getBridgeBaseUrl()}/api/oauth/files/${encodeURIComponent(att.file_uuid)}/content` + const response = await axios.get(url, { + headers: { Authorization: `Bearer ${token}` }, + responseType: 'arraybuffer', + timeout: DOWNLOAD_TIMEOUT_MS, + validateStatus: () => true, + }) + if (response.status !== 200) { + debug(`fetch ${att.file_uuid} failed: status=${response.status}`) + return undefined + } + data = Buffer.from(response.data) + } catch (e) { + debug(`fetch ${att.file_uuid} threw: ${e}`) + return undefined + } + + // uuid-prefix makes collisions impossible across messages and within one + // (same filename, different files). 8 chars is enough — this isn't security. + const safeName = sanitizeFileName(att.file_name) + const prefix = ( + att.file_uuid.slice(0, 8) || randomUUID().slice(0, 8) + ).replace(/[^a-zA-Z0-9_-]/g, '_') + const dir = uploadsDir() + const outPath = join(dir, `${prefix}-${safeName}`) + + try { + await mkdir(dir, { recursive: true }) + await writeFile(outPath, data) + } catch (e) { + debug(`write ${outPath} failed: ${e}`) + return undefined + } + + debug(`resolved ${att.file_uuid} → ${outPath} (${data.length} bytes)`) + return outPath +} + +/** + * Resolve all attachments on an inbound message to a prefix string of + * @path refs. Empty string if none resolved. + */ +export async function resolveInboundAttachments( + attachments: InboundAttachment[], +): Promise { + if (attachments.length === 0) return '' + debug(`resolving ${attachments.length} attachment(s)`) + const paths = await Promise.all(attachments.map(resolveOne)) + const ok = paths.filter((p): p is string => p !== undefined) + if (ok.length === 0) return '' + // Quoted form — extractAtMentionedFiles truncates unquoted @refs at the + // first space, which breaks any home dir with spaces (/Users/John Smith/). + return ok.map(p => `@"${p}"`).join(' ') + ' ' +} + +/** + * Prepend @path refs to content, whichever form it's in. + * Targets the LAST text block — processUserInputBase reads inputString + * from processedBlocks[processedBlocks.length - 1], so putting refs in + * block[0] means they're silently ignored for [text, image] content. + */ +export function prependPathRefs( + content: string | Array, + prefix: string, +): string | Array { + if (!prefix) return content + if (typeof content === 'string') return prefix + content + const i = content.findLastIndex(b => b.type === 'text') + if (i !== -1) { + const b = content[i]! + if (b.type === 'text') { + return [ + ...content.slice(0, i), + { ...b, text: prefix + b.text }, + ...content.slice(i + 1), + ] + } + } + // No text block — append one at the end so it's last. + return [...content, { type: 'text', text: prefix.trimEnd() }] +} + +/** + * Convenience: extract + resolve + prepend. No-op when the message has no + * file_attachments field (fast path — no network, returns same reference). + */ +export async function resolveAndPrepend( + msg: unknown, + content: string | Array, +): Promise> { + const attachments = extractInboundAttachments(msg) + if (attachments.length === 0) return content + const prefix = await resolveInboundAttachments(attachments) + return prependPathRefs(content, prefix) +} diff --git a/src/bridge/inboundMessages.ts b/src/bridge/inboundMessages.ts new file mode 100644 index 0000000..2c02f50 --- /dev/null +++ b/src/bridge/inboundMessages.ts @@ -0,0 +1,80 @@ +import type { + Base64ImageSource, + ContentBlockParam, + ImageBlockParam, +} from '@anthropic-ai/sdk/resources/messages.mjs' +import type { UUID } from 'crypto' +import type { SDKMessage } from '../entrypoints/agentSdkTypes.js' +import { detectImageFormatFromBase64 } from '../utils/imageResizer.js' + +/** + * Process an inbound user message from the bridge, extracting content + * and UUID for enqueueing. Supports both string content and + * ContentBlockParam[] (e.g. messages containing images). + * + * Normalizes image blocks from bridge clients that may use camelCase + * `mediaType` instead of snake_case `media_type` (mobile-apps#5825). + * + * Returns the extracted fields, or undefined if the message should be + * skipped (non-user type, missing/empty content). + */ +export function extractInboundMessageFields( + msg: SDKMessage, +): + | { content: string | Array; uuid: UUID | undefined } + | undefined { + if (msg.type !== 'user') return undefined + const content = msg.message?.content + if (!content) return undefined + if (Array.isArray(content) && content.length === 0) return undefined + + const uuid = + 'uuid' in msg && typeof msg.uuid === 'string' + ? (msg.uuid as UUID) + : undefined + + return { + content: Array.isArray(content) ? normalizeImageBlocks(content) : content, + uuid, + } +} + +/** + * Normalize image content blocks from bridge clients. iOS/web clients may + * send `mediaType` (camelCase) instead of `media_type` (snake_case), or + * omit the field entirely. Without normalization, the bad block poisons + * the session — every subsequent API call fails with + * "media_type: Field required". + * + * Fast-path scan returns the original array reference when no + * normalization is needed (zero allocation on the happy path). + */ +export function normalizeImageBlocks( + blocks: Array, +): Array { + if (!blocks.some(isMalformedBase64Image)) return blocks + + return blocks.map(block => { + if (!isMalformedBase64Image(block)) return block + const src = block.source as unknown as Record + const mediaType = + typeof src.mediaType === 'string' && src.mediaType + ? src.mediaType + : detectImageFormatFromBase64(block.source.data) + return { + ...block, + source: { + type: 'base64' as const, + media_type: mediaType as Base64ImageSource['media_type'], + data: block.source.data, + }, + } + }) +} + +function isMalformedBase64Image( + block: ContentBlockParam, +): block is ImageBlockParam & { source: Base64ImageSource } { + if (block.type !== 'image' || block.source?.type !== 'base64') return false + return !(block.source as unknown as Record).media_type +} diff --git a/src/bridge/initReplBridge.ts b/src/bridge/initReplBridge.ts new file mode 100644 index 0000000..85e403d --- /dev/null +++ b/src/bridge/initReplBridge.ts @@ -0,0 +1,569 @@ +/** + * REPL-specific wrapper around initBridgeCore. Owns the parts that read + * bootstrap state — gates, cwd, session ID, git context, OAuth, title + * derivation — then delegates to the bootstrap-free core. + * + * Split out of replBridge.ts because the sessionStorage import + * (getCurrentSessionTitle) transitively pulls in src/commands.ts → the + * entire slash command + React component tree (~1300 modules). Keeping + * initBridgeCore in a file that doesn't touch sessionStorage lets + * daemonBridge.ts import the core without bloating the Agent SDK bundle. + * + * Called via dynamic import by useReplBridge (auto-start) and print.ts + * (SDK -p mode via query.enableRemoteControl). + */ + +import { feature } from 'bun:bundle' +import { hostname } from 'os' +import { getOriginalCwd, getSessionId } from '../bootstrap/state.js' +import type { SDKMessage } from '../entrypoints/agentSdkTypes.js' +import type { SDKControlResponse } from '../entrypoints/sdk/controlTypes.js' +import { getFeatureValue_CACHED_WITH_REFRESH } from '../services/analytics/growthbook.js' +import { getOrganizationUUID } from '../services/oauth/client.js' +import { + isPolicyAllowed, + waitForPolicyLimitsToLoad, +} from '../services/policyLimits/index.js' +import type { Message } from '../types/message.js' +import { + checkAndRefreshOAuthTokenIfNeeded, + getClaudeAIOAuthTokens, + handleOAuth401Error, +} from '../utils/auth.js' +import { getGlobalConfig, saveGlobalConfig } from '../utils/config.js' +import { logForDebugging } from '../utils/debug.js' +import { stripDisplayTagsAllowEmpty } from '../utils/displayTags.js' +import { errorMessage } from '../utils/errors.js' +import { getBranch, getRemoteUrl } from '../utils/git.js' +import { toSDKMessages } from '../utils/messages/mappers.js' +import { + getContentText, + getMessagesAfterCompactBoundary, + isSyntheticMessage, +} from '../utils/messages.js' +import type { PermissionMode } from '../utils/permissions/PermissionMode.js' +import { getCurrentSessionTitle } from '../utils/sessionStorage.js' +import { + extractConversationText, + generateSessionTitle, +} from '../utils/sessionTitle.js' +import { generateShortWordSlug } from '../utils/words.js' +import { + getBridgeAccessToken, + getBridgeBaseUrl, + getBridgeTokenOverride, +} from './bridgeConfig.js' +import { + checkBridgeMinVersion, + isBridgeEnabledBlocking, + isCseShimEnabled, + isEnvLessBridgeEnabled, +} from './bridgeEnabled.js' +import { + archiveBridgeSession, + createBridgeSession, + updateBridgeSessionTitle, +} from './createSession.js' +import { logBridgeSkip } from './debugUtils.js' +import { checkEnvLessBridgeMinVersion } from './envLessBridgeConfig.js' +import { getPollIntervalConfig } from './pollConfig.js' +import type { BridgeState, ReplBridgeHandle } from './replBridge.js' +import { initBridgeCore } from './replBridge.js' +import { setCseShimGate } from './sessionIdCompat.js' +import type { BridgeWorkerType } from './types.js' + +export type InitBridgeOptions = { + onInboundMessage?: (msg: SDKMessage) => void | Promise + onPermissionResponse?: (response: SDKControlResponse) => void + onInterrupt?: () => void + onSetModel?: (model: string | undefined) => void + onSetMaxThinkingTokens?: (maxTokens: number | null) => void + onSetPermissionMode?: ( + mode: PermissionMode, + ) => { ok: true } | { ok: false; error: string } + onStateChange?: (state: BridgeState, detail?: string) => void + initialMessages?: Message[] + // Explicit session name from `/remote-control `. When set, overrides + // the title derived from the conversation or /rename. + initialName?: string + // Fresh view of the full conversation at call time. Used by onUserMessage's + // count-3 derivation to call generateSessionTitle over the full conversation. + // Optional — print.ts's SDK enableRemoteControl path has no REPL message + // array; count-3 falls back to the single message text when absent. + getMessages?: () => Message[] + // UUIDs already flushed in a prior bridge session. Messages with these + // UUIDs are excluded from the initial flush to avoid poisoning the + // server (duplicate UUIDs across sessions cause the WS to be killed). + // Mutated in place — newly flushed UUIDs are added after each flush. + previouslyFlushedUUIDs?: Set + /** See BridgeCoreParams.perpetual. */ + perpetual?: boolean + /** + * When true, the bridge only forwards events outbound (no SSE inbound + * stream). Used by CCR mirror mode — local sessions visible on claude.ai + * without enabling inbound control. + */ + outboundOnly?: boolean + tags?: string[] +} + +export async function initReplBridge( + options?: InitBridgeOptions, +): Promise { + const { + onInboundMessage, + onPermissionResponse, + onInterrupt, + onSetModel, + onSetMaxThinkingTokens, + onSetPermissionMode, + onStateChange, + initialMessages, + getMessages, + previouslyFlushedUUIDs, + initialName, + perpetual, + outboundOnly, + tags, + } = options ?? {} + + // Wire the cse_ shim kill switch so toCompatSessionId respects the + // GrowthBook gate. Daemon/SDK paths skip this — shim defaults to active. + setCseShimGate(isCseShimEnabled) + + // 1. Runtime gate + if (!(await isBridgeEnabledBlocking())) { + logBridgeSkip('not_enabled', '[bridge:repl] Skipping: bridge not enabled') + return null + } + + // 1b. Minimum version check — deferred to after the v1/v2 branch below, + // since each implementation has its own floor (tengu_bridge_min_version + // for v1, tengu_bridge_repl_v2_config.min_version for v2). + + // 2. Check OAuth — must be signed in with claude.ai. Runs before the + // policy check so console-auth users get the actionable "/login" hint + // instead of a misleading policy error from a stale/wrong-org cache. + if (!getBridgeAccessToken()) { + logBridgeSkip('no_oauth', '[bridge:repl] Skipping: no OAuth tokens') + onStateChange?.('failed', '/login') + return null + } + + // 3. Check organization policy — remote control may be disabled + await waitForPolicyLimitsToLoad() + if (!isPolicyAllowed('allow_remote_control')) { + logBridgeSkip( + 'policy_denied', + '[bridge:repl] Skipping: allow_remote_control policy not allowed', + ) + onStateChange?.('failed', "disabled by your organization's policy") + return null + } + + // When CLAUDE_BRIDGE_OAUTH_TOKEN is set (ant-only local dev), the bridge + // uses that token directly via getBridgeAccessToken() — keychain state is + // irrelevant. Skip 2b/2c to preserve that decoupling: an expired keychain + // token shouldn't block a bridge connection that doesn't use it. + if (!getBridgeTokenOverride()) { + // 2a. Cross-process backoff. If N prior processes already saw this exact + // dead token (matched by expiresAt), skip silently — no event, no refresh + // attempt. The count threshold tolerates transient refresh failures (auth + // server 5xx, lockfile errors per auth.ts:1437/1444/1485): each process + // independently retries until 3 consecutive failures prove the token dead. + // Mirrors useReplBridge's MAX_CONSECUTIVE_INIT_FAILURES for in-process. + // The expiresAt key is content-addressed: /login → new token → new expiresAt + // → this stops matching without any explicit clear. + const cfg = getGlobalConfig() + if ( + cfg.bridgeOauthDeadExpiresAt != null && + (cfg.bridgeOauthDeadFailCount ?? 0) >= 3 && + getClaudeAIOAuthTokens()?.expiresAt === cfg.bridgeOauthDeadExpiresAt + ) { + logForDebugging( + `[bridge:repl] Skipping: cross-process backoff (dead token seen ${cfg.bridgeOauthDeadFailCount} times)`, + ) + return null + } + + // 2b. Proactively refresh if expired. Mirrors bridgeMain.ts:2096 — the REPL + // bridge fires at useEffect mount BEFORE any v1/messages call, making this + // usually the first OAuth request of the session. Without this, ~9% of + // registrations hit the server with a >8h-expired token → 401 → withOAuthRetry + // recovers, but the server logs a 401 we can avoid. VPN egress IPs observed + // at 30:1 401:200 when many unrelated users cluster at the 8h TTL boundary. + // + // Fresh-token cost: one memoized read + one Date.now() comparison (~µs). + // checkAndRefreshOAuthTokenIfNeeded clears its own cache in every path that + // touches the keychain (refresh success, lockfile race, throw), so no + // explicit clearOAuthTokenCache() here — that would force a blocking + // keychain spawn on the 91%+ fresh-token path. + await checkAndRefreshOAuthTokenIfNeeded() + + // 2c. Skip if token is still expired post-refresh-attempt. Env-var / FD + // tokens (auth.ts:894-917) have expiresAt=null → never trip this. But a + // keychain token whose refresh token is dead (password change, org left, + // token GC'd) has expiresAt ({ + ...c, + bridgeOauthDeadExpiresAt: deadExpiresAt, + bridgeOauthDeadFailCount: + c.bridgeOauthDeadExpiresAt === deadExpiresAt + ? (c.bridgeOauthDeadFailCount ?? 0) + 1 + : 1, + })) + return null + } + } + + // 4. Compute baseUrl — needed by both v1 (env-based) and v2 (env-less) + // paths. Hoisted above the v2 gate so both can use it. + const baseUrl = getBridgeBaseUrl() + + // 5. Derive session title. Precedence: explicit initialName → /rename + // (session storage) → last meaningful user message → generated slug. + // Cosmetic only (claude.ai session list); the model never sees it. + // Two flags: `hasExplicitTitle` (initialName or /rename — never auto- + // overwrite) vs. `hasTitle` (any title, including auto-derived — blocks + // the count-1 re-derivation but not count-3). The onUserMessage callback + // (wired to both v1 and v2 below) derives from the 1st prompt and again + // from the 3rd so mobile/web show a title that reflects more context. + // The slug fallback (e.g. "remote-control-graceful-unicorn") makes + // auto-started sessions distinguishable in the claude.ai list before the + // first prompt. + let title = `remote-control-${generateShortWordSlug()}` + let hasTitle = false + let hasExplicitTitle = false + if (initialName) { + title = initialName + hasTitle = true + hasExplicitTitle = true + } else { + const sessionId = getSessionId() + const customTitle = sessionId + ? getCurrentSessionTitle(sessionId) + : undefined + if (customTitle) { + title = customTitle + hasTitle = true + hasExplicitTitle = true + } else if (initialMessages && initialMessages.length > 0) { + // Find the last user message that has meaningful content. Skip meta + // (nudges), tool results, compact summaries ("This session is being + // continued…"), non-human origins (task notifications, channel pushes), + // and synthetic interrupts ([Request interrupted by user]) — none are + // human-authored. Same filter as extractTitleText + isSyntheticMessage. + for (let i = initialMessages.length - 1; i >= 0; i--) { + const msg = initialMessages[i]! + if ( + msg.type !== 'user' || + msg.isMeta || + msg.toolUseResult || + msg.isCompactSummary || + (msg.origin && msg.origin.kind !== 'human') || + isSyntheticMessage(msg) + ) + continue + const rawContent = getContentText(msg.message.content) + if (!rawContent) continue + const derived = deriveTitle(rawContent) + if (!derived) continue + title = derived + hasTitle = true + break + } + } + } + + // Shared by both v1 and v2 — fires on every title-worthy user message until + // it returns true. At count 1: deriveTitle placeholder immediately, then + // generateSessionTitle (Haiku, sentence-case) fire-and-forget upgrade. At + // count 3: re-generate over the full conversation. Skips entirely if the + // title is explicit (/remote-control or /rename) — re-checks + // sessionStorage at call time so /rename between messages isn't clobbered. + // Skips count 1 if initialMessages already derived (that title is fresh); + // still refreshes at count 3. v2 passes cse_*; updateBridgeSessionTitle + // retags internally. + let userMessageCount = 0 + let lastBridgeSessionId: string | undefined + let genSeq = 0 + const patch = ( + derived: string, + bridgeSessionId: string, + atCount: number, + ): void => { + hasTitle = true + title = derived + logForDebugging( + `[bridge:repl] derived title from message ${atCount}: ${derived}`, + ) + void updateBridgeSessionTitle(bridgeSessionId, derived, { + baseUrl, + getAccessToken: getBridgeAccessToken, + }).catch(() => {}) + } + // Fire-and-forget Haiku generation with post-await guards. Re-checks /rename + // (sessionStorage), v1 env-lost (lastBridgeSessionId), and same-session + // out-of-order resolution (genSeq — count-1's Haiku resolving after count-3 + // would clobber the richer title). generateSessionTitle never rejects. + const generateAndPatch = (input: string, bridgeSessionId: string): void => { + const gen = ++genSeq + const atCount = userMessageCount + void generateSessionTitle(input, AbortSignal.timeout(15_000)).then( + generated => { + if ( + generated && + gen === genSeq && + lastBridgeSessionId === bridgeSessionId && + !getCurrentSessionTitle(getSessionId()) + ) { + patch(generated, bridgeSessionId, atCount) + } + }, + ) + } + const onUserMessage = (text: string, bridgeSessionId: string): boolean => { + if (hasExplicitTitle || getCurrentSessionTitle(getSessionId())) { + return true + } + // v1 env-lost re-creates the session with a new ID. Reset the count so + // the new session gets its own count-3 derivation; hasTitle stays true + // (new session was created via getCurrentTitle(), which reads the count-1 + // title from this closure), so count-1 of the fresh cycle correctly skips. + if ( + lastBridgeSessionId !== undefined && + lastBridgeSessionId !== bridgeSessionId + ) { + userMessageCount = 0 + } + lastBridgeSessionId = bridgeSessionId + userMessageCount++ + if (userMessageCount === 1 && !hasTitle) { + const placeholder = deriveTitle(text) + if (placeholder) patch(placeholder, bridgeSessionId, userMessageCount) + generateAndPatch(text, bridgeSessionId) + } else if (userMessageCount === 3) { + const msgs = getMessages?.() + const input = msgs + ? extractConversationText(getMessagesAfterCompactBoundary(msgs)) + : text + generateAndPatch(input, bridgeSessionId) + } + // Also re-latches if v1 env-lost resets the transport's done flag past 3. + return userMessageCount >= 3 + } + + const initialHistoryCap = getFeatureValue_CACHED_WITH_REFRESH( + 'tengu_bridge_initial_history_cap', + 200, + 5 * 60 * 1000, + ) + + // Fetch orgUUID before the v1/v2 branch — both paths need it. v1 for + // environment registration; v2 for archive (which lives at the compat + // /v1/sessions/{id}/archive, not /v1/code/sessions). Without it, v2 + // archive 404s and sessions stay alive in CCR after /exit. + const orgUUID = await getOrganizationUUID() + if (!orgUUID) { + logBridgeSkip('no_org_uuid', '[bridge:repl] Skipping: no org UUID') + onStateChange?.('failed', '/login') + return null + } + + // ── GrowthBook gate: env-less bridge ────────────────────────────────── + // When enabled, skips the Environments API layer entirely (no register/ + // poll/ack/heartbeat) and connects directly via POST /bridge → worker_jwt. + // See server PR #292605 (renamed in #293280). REPL-only — daemon/print stay + // on env-based. + // + // NAMING: "env-less" is distinct from "CCR v2" (the /worker/* transport). + // The env-based path below can ALSO use CCR v2 via CLAUDE_CODE_USE_CCR_V2. + // tengu_bridge_repl_v2 gates env-less (no poll loop), not transport version. + // + // perpetual (assistant-mode session continuity via bridge-pointer.json) is + // env-coupled and not yet implemented here — fall back to env-based when set + // so KAIROS users don't silently lose cross-restart continuity. + if (isEnvLessBridgeEnabled() && !perpetual) { + const versionError = await checkEnvLessBridgeMinVersion() + if (versionError) { + logBridgeSkip( + 'version_too_old', + `[bridge:repl] Skipping: ${versionError}`, + true, + ) + onStateChange?.('failed', 'run `claude update` to upgrade') + return null + } + logForDebugging( + '[bridge:repl] Using env-less bridge path (tengu_bridge_repl_v2)', + ) + const { initEnvLessBridgeCore } = await import('./remoteBridgeCore.js') + return initEnvLessBridgeCore({ + baseUrl, + orgUUID, + title, + getAccessToken: getBridgeAccessToken, + onAuth401: handleOAuth401Error, + toSDKMessages, + initialHistoryCap, + initialMessages, + // v2 always creates a fresh server session (new cse_* id), so + // previouslyFlushedUUIDs is not passed — there's no cross-session + // UUID collision risk, and the ref persists across enable→disable→ + // re-enable cycles which would cause the new session to receive zero + // history (all UUIDs already in the set from the prior enable). + // v1 handles this by calling previouslyFlushedUUIDs.clear() on fresh + // session creation (replBridge.ts:768); v2 skips the param entirely. + onInboundMessage, + onUserMessage, + onPermissionResponse, + onInterrupt, + onSetModel, + onSetMaxThinkingTokens, + onSetPermissionMode, + onStateChange, + outboundOnly, + tags, + }) + } + + // ── v1 path: env-based (register/poll/ack/heartbeat) ────────────────── + + const versionError = checkBridgeMinVersion() + if (versionError) { + logBridgeSkip('version_too_old', `[bridge:repl] Skipping: ${versionError}`) + onStateChange?.('failed', 'run `claude update` to upgrade') + return null + } + + // Gather git context — this is the bootstrap-read boundary. + // Everything from here down is passed explicitly to bridgeCore. + const branch = await getBranch() + const gitRepoUrl = await getRemoteUrl() + const sessionIngressUrl = + process.env.USER_TYPE === 'ant' && + process.env.CLAUDE_BRIDGE_SESSION_INGRESS_URL + ? process.env.CLAUDE_BRIDGE_SESSION_INGRESS_URL + : baseUrl + + // Assistant-mode sessions advertise a distinct worker_type so the web UI + // can filter them into a dedicated picker. KAIROS guard keeps the + // assistant module out of external builds entirely. + let workerType: BridgeWorkerType = 'claude_code' + if (feature('KAIROS')) { + /* eslint-disable @typescript-eslint/no-require-imports */ + const { isAssistantMode } = + require('../assistant/index.js') as typeof import('../assistant/index.js') + /* eslint-enable @typescript-eslint/no-require-imports */ + if (isAssistantMode()) { + workerType = 'claude_code_assistant' + } + } + + // 6. Delegate. BridgeCoreHandle is a structural superset of + // ReplBridgeHandle (adds writeSdkMessages which REPL callers don't use), + // so no adapter needed — just the narrower type on the way out. + return initBridgeCore({ + dir: getOriginalCwd(), + machineName: hostname(), + branch, + gitRepoUrl, + title, + baseUrl, + sessionIngressUrl, + workerType, + getAccessToken: getBridgeAccessToken, + createSession: opts => + createBridgeSession({ + ...opts, + events: [], + baseUrl, + getAccessToken: getBridgeAccessToken, + }), + archiveSession: sessionId => + archiveBridgeSession(sessionId, { + baseUrl, + getAccessToken: getBridgeAccessToken, + // gracefulShutdown.ts:407 races runCleanupFunctions against 2s. + // Teardown also does stopWork (parallel) + deregister (sequential), + // so archive can't have the full budget. 1.5s matches v2's + // teardown_archive_timeout_ms default. + timeoutMs: 1500, + }).catch((err: unknown) => { + // archiveBridgeSession has no try/catch — 5xx/timeout/network throw + // straight through. Previously swallowed silently, making archive + // failures BQ-invisible and undiagnosable from debug logs. + logForDebugging( + `[bridge:repl] archiveBridgeSession threw: ${errorMessage(err)}`, + { level: 'error' }, + ) + }), + // getCurrentTitle is read on reconnect-after-env-lost to re-title the new + // session. /rename writes to session storage; onUserMessage mutates + // `title` directly — both paths are picked up here. + getCurrentTitle: () => getCurrentSessionTitle(getSessionId()) ?? title, + onUserMessage, + toSDKMessages, + onAuth401: handleOAuth401Error, + getPollIntervalConfig, + initialHistoryCap, + initialMessages, + previouslyFlushedUUIDs, + onInboundMessage, + onPermissionResponse, + onInterrupt, + onSetModel, + onSetMaxThinkingTokens, + onSetPermissionMode, + onStateChange, + perpetual, + }) +} + +const TITLE_MAX_LEN = 50 + +/** + * Quick placeholder title: strip display tags, take the first sentence, + * collapse whitespace, truncate to 50 chars. Returns undefined if the result + * is empty (e.g. message was only ). Replaced by + * generateSessionTitle once Haiku resolves (~1-15s). + */ +function deriveTitle(raw: string): string | undefined { + // Strip , , etc. — these appear in + // user messages when IDE/hooks inject context. stripDisplayTagsAllowEmpty + // returns '' (not the original) so pure-tag messages are skipped. + const clean = stripDisplayTagsAllowEmpty(raw) + // First sentence is usually the intent; rest is often context/detail. + // Capture group instead of lookbehind — keeps YARR JIT happy. + const firstSentence = /^(.*?[.!?])\s/.exec(clean)?.[1] ?? clean + // Collapse newlines/tabs — titles are single-line in the claude.ai list. + const flat = firstSentence.replace(/\s+/g, ' ').trim() + if (!flat) return undefined + return flat.length > TITLE_MAX_LEN + ? flat.slice(0, TITLE_MAX_LEN - 1) + '\u2026' + : flat +} diff --git a/src/bridge/jwtUtils.ts b/src/bridge/jwtUtils.ts new file mode 100644 index 0000000..030c001 --- /dev/null +++ b/src/bridge/jwtUtils.ts @@ -0,0 +1,256 @@ +import { logEvent } from '../services/analytics/index.js' +import { logForDebugging } from '../utils/debug.js' +import { logForDiagnosticsNoPII } from '../utils/diagLogs.js' +import { errorMessage } from '../utils/errors.js' +import { jsonParse } from '../utils/slowOperations.js' + +/** Format a millisecond duration as a human-readable string (e.g. "5m 30s"). */ +function formatDuration(ms: number): string { + if (ms < 60_000) return `${Math.round(ms / 1000)}s` + const m = Math.floor(ms / 60_000) + const s = Math.round((ms % 60_000) / 1000) + return s > 0 ? `${m}m ${s}s` : `${m}m` +} + +/** + * Decode a JWT's payload segment without verifying the signature. + * Strips the `sk-ant-si-` session-ingress prefix if present. + * Returns the parsed JSON payload as `unknown`, or `null` if the + * token is malformed or the payload is not valid JSON. + */ +export function decodeJwtPayload(token: string): unknown | null { + const jwt = token.startsWith('sk-ant-si-') + ? token.slice('sk-ant-si-'.length) + : token + const parts = jwt.split('.') + if (parts.length !== 3 || !parts[1]) return null + try { + return jsonParse(Buffer.from(parts[1], 'base64url').toString('utf8')) + } catch { + return null + } +} + +/** + * Decode the `exp` (expiry) claim from a JWT without verifying the signature. + * @returns The `exp` value in Unix seconds, or `null` if unparseable + */ +export function decodeJwtExpiry(token: string): number | null { + const payload = decodeJwtPayload(token) + if ( + payload !== null && + typeof payload === 'object' && + 'exp' in payload && + typeof payload.exp === 'number' + ) { + return payload.exp + } + return null +} + +/** Refresh buffer: request a new token before expiry. */ +const TOKEN_REFRESH_BUFFER_MS = 5 * 60 * 1000 + +/** Fallback refresh interval when the new token's expiry is unknown. */ +const FALLBACK_REFRESH_INTERVAL_MS = 30 * 60 * 1000 // 30 minutes + +/** Max consecutive failures before giving up on the refresh chain. */ +const MAX_REFRESH_FAILURES = 3 + +/** Retry delay when getAccessToken returns undefined. */ +const REFRESH_RETRY_DELAY_MS = 60_000 + +/** + * Creates a token refresh scheduler that proactively refreshes session tokens + * before they expire. Used by both the standalone bridge and the REPL bridge. + * + * When a token is about to expire, the scheduler calls `onRefresh` with the + * session ID and the bridge's OAuth access token. The caller is responsible + * for delivering the token to the appropriate transport (child process stdin + * for standalone bridge, WebSocket reconnect for REPL bridge). + */ +export function createTokenRefreshScheduler({ + getAccessToken, + onRefresh, + label, + refreshBufferMs = TOKEN_REFRESH_BUFFER_MS, +}: { + getAccessToken: () => string | undefined | Promise + onRefresh: (sessionId: string, oauthToken: string) => void + label: string + /** How long before expiry to fire refresh. Defaults to 5 min. */ + refreshBufferMs?: number +}): { + schedule: (sessionId: string, token: string) => void + scheduleFromExpiresIn: (sessionId: string, expiresInSeconds: number) => void + cancel: (sessionId: string) => void + cancelAll: () => void +} { + const timers = new Map>() + const failureCounts = new Map() + // Generation counter per session — incremented by schedule() and cancel() + // so that in-flight async doRefresh() calls can detect when they've been + // superseded and should skip setting follow-up timers. + const generations = new Map() + + function nextGeneration(sessionId: string): number { + const gen = (generations.get(sessionId) ?? 0) + 1 + generations.set(sessionId, gen) + return gen + } + + function schedule(sessionId: string, token: string): void { + const expiry = decodeJwtExpiry(token) + if (!expiry) { + // Token is not a decodable JWT (e.g. an OAuth token passed from the + // REPL bridge WebSocket open handler). Preserve any existing timer + // (such as the follow-up refresh set by doRefresh) so the refresh + // chain is not broken. + logForDebugging( + `[${label}:token] Could not decode JWT expiry for sessionId=${sessionId}, token prefix=${token.slice(0, 15)}…, keeping existing timer`, + ) + return + } + + // Clear any existing refresh timer — we have a concrete expiry to replace it. + const existing = timers.get(sessionId) + if (existing) { + clearTimeout(existing) + } + + // Bump generation to invalidate any in-flight async doRefresh. + const gen = nextGeneration(sessionId) + + const expiryDate = new Date(expiry * 1000).toISOString() + const delayMs = expiry * 1000 - Date.now() - refreshBufferMs + if (delayMs <= 0) { + logForDebugging( + `[${label}:token] Token for sessionId=${sessionId} expires=${expiryDate} (past or within buffer), refreshing immediately`, + ) + void doRefresh(sessionId, gen) + return + } + + logForDebugging( + `[${label}:token] Scheduled token refresh for sessionId=${sessionId} in ${formatDuration(delayMs)} (expires=${expiryDate}, buffer=${refreshBufferMs / 1000}s)`, + ) + + const timer = setTimeout(doRefresh, delayMs, sessionId, gen) + timers.set(sessionId, timer) + } + + /** + * Schedule refresh using an explicit TTL (seconds until expiry) rather + * than decoding a JWT's exp claim. Used by callers whose JWT is opaque + * (e.g. POST /v1/code/sessions/{id}/bridge returns expires_in directly). + */ + function scheduleFromExpiresIn( + sessionId: string, + expiresInSeconds: number, + ): void { + const existing = timers.get(sessionId) + if (existing) clearTimeout(existing) + const gen = nextGeneration(sessionId) + // Clamp to 30s floor — if refreshBufferMs exceeds the server's expires_in + // (e.g. very large buffer for frequent-refresh testing, or server shortens + // expires_in unexpectedly), unclamped delayMs ≤ 0 would tight-loop. + const delayMs = Math.max(expiresInSeconds * 1000 - refreshBufferMs, 30_000) + logForDebugging( + `[${label}:token] Scheduled token refresh for sessionId=${sessionId} in ${formatDuration(delayMs)} (expires_in=${expiresInSeconds}s, buffer=${refreshBufferMs / 1000}s)`, + ) + const timer = setTimeout(doRefresh, delayMs, sessionId, gen) + timers.set(sessionId, timer) + } + + async function doRefresh(sessionId: string, gen: number): Promise { + let oauthToken: string | undefined + try { + oauthToken = await getAccessToken() + } catch (err) { + logForDebugging( + `[${label}:token] getAccessToken threw for sessionId=${sessionId}: ${errorMessage(err)}`, + { level: 'error' }, + ) + } + + // If the session was cancelled or rescheduled while we were awaiting, + // the generation will have changed — bail out to avoid orphaned timers. + if (generations.get(sessionId) !== gen) { + logForDebugging( + `[${label}:token] doRefresh for sessionId=${sessionId} stale (gen ${gen} vs ${generations.get(sessionId)}), skipping`, + ) + return + } + + if (!oauthToken) { + const failures = (failureCounts.get(sessionId) ?? 0) + 1 + failureCounts.set(sessionId, failures) + logForDebugging( + `[${label}:token] No OAuth token available for refresh, sessionId=${sessionId} (failure ${failures}/${MAX_REFRESH_FAILURES})`, + { level: 'error' }, + ) + logForDiagnosticsNoPII('error', 'bridge_token_refresh_no_oauth') + // Schedule a retry so the refresh chain can recover if the token + // becomes available again (e.g. transient cache clear during refresh). + // Cap retries to avoid spamming on genuine failures. + if (failures < MAX_REFRESH_FAILURES) { + const retryTimer = setTimeout( + doRefresh, + REFRESH_RETRY_DELAY_MS, + sessionId, + gen, + ) + timers.set(sessionId, retryTimer) + } + return + } + + // Reset failure counter on successful token retrieval + failureCounts.delete(sessionId) + + logForDebugging( + `[${label}:token] Refreshing token for sessionId=${sessionId}: new token prefix=${oauthToken.slice(0, 15)}…`, + ) + logEvent('tengu_bridge_token_refreshed', {}) + onRefresh(sessionId, oauthToken) + + // Schedule a follow-up refresh so long-running sessions stay authenticated. + // Without this, the initial one-shot timer leaves the session vulnerable + // to token expiry if it runs past the first refresh window. + const timer = setTimeout( + doRefresh, + FALLBACK_REFRESH_INTERVAL_MS, + sessionId, + gen, + ) + timers.set(sessionId, timer) + logForDebugging( + `[${label}:token] Scheduled follow-up refresh for sessionId=${sessionId} in ${formatDuration(FALLBACK_REFRESH_INTERVAL_MS)}`, + ) + } + + function cancel(sessionId: string): void { + // Bump generation to invalidate any in-flight async doRefresh. + nextGeneration(sessionId) + const timer = timers.get(sessionId) + if (timer) { + clearTimeout(timer) + timers.delete(sessionId) + } + failureCounts.delete(sessionId) + } + + function cancelAll(): void { + // Bump all generations so in-flight doRefresh calls are invalidated. + for (const sessionId of generations.keys()) { + nextGeneration(sessionId) + } + for (const timer of timers.values()) { + clearTimeout(timer) + } + timers.clear() + failureCounts.clear() + } + + return { schedule, scheduleFromExpiresIn, cancel, cancelAll } +} diff --git a/src/bridge/pollConfig.ts b/src/bridge/pollConfig.ts new file mode 100644 index 0000000..024b476 --- /dev/null +++ b/src/bridge/pollConfig.ts @@ -0,0 +1,110 @@ +import { z } from 'zod/v4' +import { getFeatureValue_CACHED_WITH_REFRESH } from '../services/analytics/growthbook.js' +import { lazySchema } from '../utils/lazySchema.js' +import { + DEFAULT_POLL_CONFIG, + type PollIntervalConfig, +} from './pollConfigDefaults.js' + +// .min(100) on the seek-work intervals restores the old Math.max(..., 100) +// defense-in-depth floor against fat-fingered GrowthBook values. Unlike a +// clamp, Zod rejects the whole object on violation — a config with one bad +// field falls back to DEFAULT_POLL_CONFIG entirely rather than being +// partially trusted. +// +// The at_capacity intervals use a 0-or-≥100 refinement: 0 means "disabled" +// (heartbeat-only mode), ≥100 is the fat-finger floor. Values 1–99 are +// rejected so unit confusion (ops thinks seconds, enters 10) doesn't poll +// every 10ms against the VerifyEnvironmentSecretAuth DB path. +// +// The object-level refines require at least one at-capacity liveness +// mechanism enabled: heartbeat OR the relevant poll interval. Without this, +// the hb=0, atCapMs=0 drift config (ops disables heartbeat without +// restoring at_capacity) falls through every throttle site with no sleep — +// tight-looping /poll at HTTP-round-trip speed. +const zeroOrAtLeast100 = { + message: 'must be 0 (disabled) or ≥100ms', +} +const pollIntervalConfigSchema = lazySchema(() => + z + .object({ + poll_interval_ms_not_at_capacity: z.number().int().min(100), + // 0 = no at-capacity polling. Independent of heartbeat — both can be + // enabled (heartbeat runs, periodically breaks out to poll). + poll_interval_ms_at_capacity: z + .number() + .int() + .refine(v => v === 0 || v >= 100, zeroOrAtLeast100), + // 0 = disabled; positive value = heartbeat at this interval while at + // capacity. Runs alongside at-capacity polling, not instead of it. + // Named non_exclusive to distinguish from the old heartbeat_interval_ms + // (either-or semantics in pre-#22145 clients). .default(0) so existing + // GrowthBook configs without this field parse successfully. + non_exclusive_heartbeat_interval_ms: z.number().int().min(0).default(0), + // Multisession (bridgeMain.ts) intervals. Defaults match the + // single-session values so existing configs without these fields + // preserve current behavior. + multisession_poll_interval_ms_not_at_capacity: z + .number() + .int() + .min(100) + .default( + DEFAULT_POLL_CONFIG.multisession_poll_interval_ms_not_at_capacity, + ), + multisession_poll_interval_ms_partial_capacity: z + .number() + .int() + .min(100) + .default( + DEFAULT_POLL_CONFIG.multisession_poll_interval_ms_partial_capacity, + ), + multisession_poll_interval_ms_at_capacity: z + .number() + .int() + .refine(v => v === 0 || v >= 100, zeroOrAtLeast100) + .default(DEFAULT_POLL_CONFIG.multisession_poll_interval_ms_at_capacity), + // .min(1) matches the server's ge=1 constraint (work_v1.py:230). + reclaim_older_than_ms: z.number().int().min(1).default(5000), + session_keepalive_interval_v2_ms: z + .number() + .int() + .min(0) + .default(120_000), + }) + .refine( + cfg => + cfg.non_exclusive_heartbeat_interval_ms > 0 || + cfg.poll_interval_ms_at_capacity > 0, + { + message: + 'at-capacity liveness requires non_exclusive_heartbeat_interval_ms > 0 or poll_interval_ms_at_capacity > 0', + }, + ) + .refine( + cfg => + cfg.non_exclusive_heartbeat_interval_ms > 0 || + cfg.multisession_poll_interval_ms_at_capacity > 0, + { + message: + 'at-capacity liveness requires non_exclusive_heartbeat_interval_ms > 0 or multisession_poll_interval_ms_at_capacity > 0', + }, + ), +) + +/** + * Fetch the bridge poll interval config from GrowthBook with a 5-minute + * refresh window. Validates the served JSON against the schema; falls back + * to defaults if the flag is absent, malformed, or partially-specified. + * + * Shared by bridgeMain.ts (standalone) and replBridge.ts (REPL) so ops + * can tune both poll rates fleet-wide with a single config push. + */ +export function getPollIntervalConfig(): PollIntervalConfig { + const raw = getFeatureValue_CACHED_WITH_REFRESH( + 'tengu_bridge_poll_interval_config', + DEFAULT_POLL_CONFIG, + 5 * 60 * 1000, + ) + const parsed = pollIntervalConfigSchema().safeParse(raw) + return parsed.success ? parsed.data : DEFAULT_POLL_CONFIG +} diff --git a/src/bridge/pollConfigDefaults.ts b/src/bridge/pollConfigDefaults.ts new file mode 100644 index 0000000..7a4e6d8 --- /dev/null +++ b/src/bridge/pollConfigDefaults.ts @@ -0,0 +1,82 @@ +/** + * Bridge poll interval defaults. Extracted from pollConfig.ts so callers + * that don't need live GrowthBook tuning (daemon via Agent SDK) can avoid + * the growthbook.ts → config.ts → file.ts → sessionStorage.ts → commands.ts + * transitive dependency chain. + */ + +/** + * Poll interval when actively seeking work (no transport / below maxSessions). + * Governs user-visible "connecting…" latency on initial work pickup and + * recovery speed after the server re-dispatches a work item. + */ +const POLL_INTERVAL_MS_NOT_AT_CAPACITY = 2000 + +/** + * Poll interval when the transport is connected. Runs independently of + * heartbeat — when both are enabled, the heartbeat loop breaks out to poll + * at this interval. Set to 0 to disable at-capacity polling entirely. + * + * Server-side constraints that bound this value: + * - BRIDGE_LAST_POLL_TTL = 4h (Redis key expiry → environment auto-archived) + * - max_poll_stale_seconds = 24h (session-creation health gate, currently disabled) + * + * 10 minutes gives 24× headroom on the Redis TTL while still picking up + * server-initiated token-rotation redispatches within one poll cycle. + * The transport auto-reconnects internally for 10 minutes on transient WS + * failures, so poll is not the recovery path — it's strictly a liveness + * signal plus a backstop for permanent close. + */ +const POLL_INTERVAL_MS_AT_CAPACITY = 600_000 + +/** + * Multisession bridge (bridgeMain.ts) poll intervals. Defaults match the + * single-session values so existing GrowthBook configs without these fields + * preserve current behavior. Ops can tune these independently via the + * tengu_bridge_poll_interval_config GB flag. + */ +const MULTISESSION_POLL_INTERVAL_MS_NOT_AT_CAPACITY = + POLL_INTERVAL_MS_NOT_AT_CAPACITY +const MULTISESSION_POLL_INTERVAL_MS_PARTIAL_CAPACITY = + POLL_INTERVAL_MS_NOT_AT_CAPACITY +const MULTISESSION_POLL_INTERVAL_MS_AT_CAPACITY = POLL_INTERVAL_MS_AT_CAPACITY + +export type PollIntervalConfig = { + poll_interval_ms_not_at_capacity: number + poll_interval_ms_at_capacity: number + non_exclusive_heartbeat_interval_ms: number + multisession_poll_interval_ms_not_at_capacity: number + multisession_poll_interval_ms_partial_capacity: number + multisession_poll_interval_ms_at_capacity: number + reclaim_older_than_ms: number + session_keepalive_interval_v2_ms: number +} + +export const DEFAULT_POLL_CONFIG: PollIntervalConfig = { + poll_interval_ms_not_at_capacity: POLL_INTERVAL_MS_NOT_AT_CAPACITY, + poll_interval_ms_at_capacity: POLL_INTERVAL_MS_AT_CAPACITY, + // 0 = disabled. When > 0, at-capacity loops send per-work-item heartbeats + // at this interval. Independent of poll_interval_ms_at_capacity — both may + // run (heartbeat periodically yields to poll). 60s gives 5× headroom under + // the server's 300s heartbeat TTL. Named non_exclusive to distinguish from + // the old heartbeat_interval_ms field (either-or semantics in pre-#22145 + // clients — heartbeat suppressed poll). Old clients ignore this key; ops + // can set both fields during rollout. + non_exclusive_heartbeat_interval_ms: 0, + multisession_poll_interval_ms_not_at_capacity: + MULTISESSION_POLL_INTERVAL_MS_NOT_AT_CAPACITY, + multisession_poll_interval_ms_partial_capacity: + MULTISESSION_POLL_INTERVAL_MS_PARTIAL_CAPACITY, + multisession_poll_interval_ms_at_capacity: + MULTISESSION_POLL_INTERVAL_MS_AT_CAPACITY, + // Poll query param: reclaim unacknowledged work items older than this. + // Matches the server's DEFAULT_RECLAIM_OLDER_THAN_MS (work_service.py:24). + // Enables picking up stale-pending work after JWT expiry, when the prior + // ack failed because the session_ingress_token was already stale. + reclaim_older_than_ms: 5000, + // 0 = disabled. When > 0, push a silent {type:'keep_alive'} frame to + // session-ingress at this interval so upstream proxies don't GC an idle + // remote-control session. 2 min is the default. _v2: bridge-only gate + // (pre-v2 clients read the old key, new clients ignore it). + session_keepalive_interval_v2_ms: 120_000, +} diff --git a/src/bridge/remoteBridgeCore.ts b/src/bridge/remoteBridgeCore.ts new file mode 100644 index 0000000..76545f6 --- /dev/null +++ b/src/bridge/remoteBridgeCore.ts @@ -0,0 +1,1008 @@ +// biome-ignore-all assist/source/organizeImports: ANT-ONLY import markers must not be reordered +/** + * Env-less Remote Control bridge core. + * + * "Env-less" = no Environments API layer. Distinct from "CCR v2" (the + * /worker/* transport protocol) — the env-based path (replBridge.ts) can also + * use CCR v2 transport via CLAUDE_CODE_USE_CCR_V2. This file is about removing + * the poll/dispatch layer, not about which transport protocol is underneath. + * + * Unlike initBridgeCore (env-based, ~2400 lines), this connects directly + * to the session-ingress layer without the Environments API work-dispatch + * layer: + * + * 1. POST /v1/code/sessions (OAuth, no env_id) → session.id + * 2. POST /v1/code/sessions/{id}/bridge (OAuth) → {worker_jwt, expires_in, api_base_url, worker_epoch} + * Each /bridge call bumps epoch — it IS the register. No separate /worker/register. + * 3. createV2ReplTransport(worker_jwt, worker_epoch) → SSE + CCRClient + * 4. createTokenRefreshScheduler → proactive /bridge re-call (new JWT + new epoch) + * 5. 401 on SSE → rebuild transport with fresh /bridge credentials (same seq-num) + * + * No register/poll/ack/stop/heartbeat/deregister environment lifecycle. + * The Environments API historically existed because CCR's /worker/* + * endpoints required a session_id+role=worker JWT that only the work-dispatch + * layer could mint. Server PR #292605 (renamed in #293280) adds the /bridge endpoint as a direct + * OAuth→worker_jwt exchange, making the env layer optional for REPL sessions. + * + * Gated by `tengu_bridge_repl_v2` GrowthBook flag in initReplBridge.ts. + * REPL-only — daemon/print stay on env-based. + */ + +import { feature } from 'bun:bundle' +import axios from 'axios' +import { + createV2ReplTransport, + type ReplBridgeTransport, +} from './replBridgeTransport.js' +import { buildCCRv2SdkUrl } from './workSecret.js' +import { toCompatSessionId } from './sessionIdCompat.js' +import { FlushGate } from './flushGate.js' +import { createTokenRefreshScheduler } from './jwtUtils.js' +import { getTrustedDeviceToken } from './trustedDevice.js' +import { + getEnvLessBridgeConfig, + type EnvLessBridgeConfig, +} from './envLessBridgeConfig.js' +import { + handleIngressMessage, + handleServerControlRequest, + makeResultMessage, + isEligibleBridgeMessage, + extractTitleText, + BoundedUUIDSet, +} from './bridgeMessaging.js' +import { logBridgeSkip } from './debugUtils.js' +import { logForDebugging } from '../utils/debug.js' +import { logForDiagnosticsNoPII } from '../utils/diagLogs.js' +import { isInProtectedNamespace } from '../utils/envUtils.js' +import { errorMessage } from '../utils/errors.js' +import { sleep } from '../utils/sleep.js' +import { registerCleanup } from '../utils/cleanupRegistry.js' +import { + type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + logEvent, +} from '../services/analytics/index.js' +import type { ReplBridgeHandle, BridgeState } from './replBridge.js' +import type { Message } from '../types/message.js' +import type { SDKMessage } from '../entrypoints/agentSdkTypes.js' +import type { + SDKControlRequest, + SDKControlResponse, +} from '../entrypoints/sdk/controlTypes.js' +import type { PermissionMode } from '../utils/permissions/PermissionMode.js' + +const ANTHROPIC_VERSION = '2023-06-01' + +// Telemetry discriminator for ws_connected. 'initial' is the default and +// never passed to rebuildTransport (which can only be called post-init); +// Exclude<> makes that constraint explicit at both signatures. +type ConnectCause = 'initial' | 'proactive_refresh' | 'auth_401_recovery' + +function oauthHeaders(accessToken: string): Record { + return { + Authorization: `Bearer ${accessToken}`, + 'Content-Type': 'application/json', + 'anthropic-version': ANTHROPIC_VERSION, + } +} + +export type EnvLessBridgeParams = { + baseUrl: string + orgUUID: string + title: string + getAccessToken: () => string | undefined + onAuth401?: (staleAccessToken: string) => Promise + /** + * Converts internal Message[] → SDKMessage[] for writeMessages() and the + * initial-flush/drain paths. Injected rather than imported — mappers.ts + * transitively pulls in src/commands.ts (entire command registry + React + * tree) which would bloat bundles that don't already have it. + */ + toSDKMessages: (messages: Message[]) => SDKMessage[] + initialHistoryCap: number + initialMessages?: Message[] + onInboundMessage?: (msg: SDKMessage) => void | Promise + /** + * Fired on each title-worthy user message seen in writeMessages() until + * the callback returns true (done). Mirrors replBridge.ts's onUserMessage — + * caller derives a title and PATCHes /v1/sessions/{id} so auto-started + * sessions don't stay at the generic fallback. The caller owns the + * derive-at-count-1-and-3 policy; the transport just keeps calling until + * told to stop. sessionId is the raw cse_* — updateBridgeSessionTitle + * retags internally. + */ + onUserMessage?: (text: string, sessionId: string) => boolean + onPermissionResponse?: (response: SDKControlResponse) => void + onInterrupt?: () => void + onSetModel?: (model: string | undefined) => void + onSetMaxThinkingTokens?: (maxTokens: number | null) => void + onSetPermissionMode?: ( + mode: PermissionMode, + ) => { ok: true } | { ok: false; error: string } + onStateChange?: (state: BridgeState, detail?: string) => void + /** + * When true, skip opening the SSE read stream — only the CCRClient write + * path is activated. Threaded to createV2ReplTransport and + * handleServerControlRequest. + */ + outboundOnly?: boolean + /** Free-form tags for session categorization (e.g. ['ccr-mirror']). */ + tags?: string[] +} + +/** + * Create a session, fetch a worker JWT, connect the v2 transport. + * + * Returns null on any pre-flight failure (session create failed, /bridge + * failed, transport setup failed). Caller (initReplBridge) surfaces this + * as a generic "initialization failed" state. + */ +export async function initEnvLessBridgeCore( + params: EnvLessBridgeParams, +): Promise { + const { + baseUrl, + orgUUID, + title, + getAccessToken, + onAuth401, + toSDKMessages, + initialHistoryCap, + initialMessages, + onInboundMessage, + onUserMessage, + onPermissionResponse, + onInterrupt, + onSetModel, + onSetMaxThinkingTokens, + onSetPermissionMode, + onStateChange, + outboundOnly, + tags, + } = params + + const cfg = await getEnvLessBridgeConfig() + + // ── 1. Create session (POST /v1/code/sessions, no env_id) ─────────────── + const accessToken = getAccessToken() + if (!accessToken) { + logForDebugging('[remote-bridge] No OAuth token') + return null + } + + const createdSessionId = await withRetry( + () => + createCodeSession(baseUrl, accessToken, title, cfg.http_timeout_ms, tags), + 'createCodeSession', + cfg, + ) + if (!createdSessionId) { + onStateChange?.('failed', 'Session creation failed — see debug log') + logBridgeSkip('v2_session_create_failed', undefined, true) + return null + } + const sessionId: string = createdSessionId + logForDebugging(`[remote-bridge] Created session ${sessionId}`) + logForDiagnosticsNoPII('info', 'bridge_repl_v2_session_created') + + // ── 2. Fetch bridge credentials (POST /bridge → worker_jwt, expires_in, api_base_url) ── + const credentials = await withRetry( + () => + fetchRemoteCredentials( + sessionId, + baseUrl, + accessToken, + cfg.http_timeout_ms, + ), + 'fetchRemoteCredentials', + cfg, + ) + if (!credentials) { + onStateChange?.('failed', 'Remote credentials fetch failed — see debug log') + logBridgeSkip('v2_remote_creds_failed', undefined, true) + void archiveSession( + sessionId, + baseUrl, + accessToken, + orgUUID, + cfg.http_timeout_ms, + ) + return null + } + logForDebugging( + `[remote-bridge] Fetched bridge credentials (expires_in=${credentials.expires_in}s)`, + ) + + // ── 3. Build v2 transport (SSETransport + CCRClient) ──────────────────── + const sessionUrl = buildCCRv2SdkUrl(credentials.api_base_url, sessionId) + logForDebugging(`[remote-bridge] v2 session URL: ${sessionUrl}`) + + let transport: ReplBridgeTransport + try { + transport = await createV2ReplTransport({ + sessionUrl, + ingressToken: credentials.worker_jwt, + sessionId, + epoch: credentials.worker_epoch, + heartbeatIntervalMs: cfg.heartbeat_interval_ms, + heartbeatJitterFraction: cfg.heartbeat_jitter_fraction, + // Per-instance closure — keeps the worker JWT out of + // process.env.CLAUDE_CODE_SESSION_ACCESS_TOKEN, which mcp/client.ts + // reads ungatedly and would otherwise send to user-configured ws/http + // MCP servers. Frozen-at-construction is correct: transport is fully + // rebuilt on refresh (rebuildTransport below). + getAuthToken: () => credentials.worker_jwt, + outboundOnly, + }) + } catch (err) { + logForDebugging( + `[remote-bridge] v2 transport setup failed: ${errorMessage(err)}`, + { level: 'error' }, + ) + onStateChange?.('failed', `Transport setup failed: ${errorMessage(err)}`) + logBridgeSkip('v2_transport_setup_failed', undefined, true) + void archiveSession( + sessionId, + baseUrl, + accessToken, + orgUUID, + cfg.http_timeout_ms, + ) + return null + } + logForDebugging( + `[remote-bridge] v2 transport created (epoch=${credentials.worker_epoch})`, + ) + onStateChange?.('ready') + + // ── 4. State ──────────────────────────────────────────────────────────── + + // Echo dedup: messages we POST come back on the read stream. Seeded with + // initial message UUIDs so server echoes of flushed history are recognized. + // Both sets cover initial UUIDs — recentPostedUUIDs is a 2000-cap ring buffer + // and could evict them after enough live writes; initialMessageUUIDs is the + // unbounded fallback. Defense-in-depth; mirrors replBridge.ts. + const recentPostedUUIDs = new BoundedUUIDSet(cfg.uuid_dedup_buffer_size) + const initialMessageUUIDs = new Set() + if (initialMessages) { + for (const msg of initialMessages) { + initialMessageUUIDs.add(msg.uuid) + recentPostedUUIDs.add(msg.uuid) + } + } + + // Defensive dedup for re-delivered inbound prompts (seq-num negotiation + // edge cases, server history replay after transport swap). + const recentInboundUUIDs = new BoundedUUIDSet(cfg.uuid_dedup_buffer_size) + + // FlushGate: queue live writes while the history flush POST is in flight, + // so the server receives [history..., live...] in order. + const flushGate = new FlushGate() + + let initialFlushDone = false + let tornDown = false + let authRecoveryInFlight = false + // Latch for onUserMessage — flips true when the callback returns true + // (policy says "done deriving"). sessionId is const (no re-create path — + // rebuildTransport swaps JWT/epoch, same session), so no reset needed. + let userMessageCallbackDone = !onUserMessage + + // Telemetry: why did onConnect fire? Set by rebuildTransport before + // wireTransportCallbacks; read asynchronously by onConnect. Race-safe + // because authRecoveryInFlight serializes rebuild callers, and a fresh + // initEnvLessBridgeCore() call gets a fresh closure defaulting to 'initial'. + let connectCause: ConnectCause = 'initial' + + // Deadline for onConnect after transport.connect(). Cleared by onConnect + // (connected) and onClose (got a close — not silent). If neither fires + // before cfg.connect_timeout_ms, onConnectTimeout emits — the only + // signal for the `started → (silence)` gap. + let connectDeadline: ReturnType | undefined + function onConnectTimeout(cause: ConnectCause): void { + if (tornDown) return + logEvent('tengu_bridge_repl_connect_timeout', { + v2: true, + elapsed_ms: cfg.connect_timeout_ms, + cause: + cause as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + } + + // ── 5. JWT refresh scheduler ──────────────────────────────────────────── + // Schedule a callback 5min before expiry (per response.expires_in). On fire, + // re-fetch /bridge with OAuth → rebuild transport with fresh credentials. + // Each /bridge call bumps epoch server-side, so a JWT-only swap would leave + // the old CCRClient heartbeating with a stale epoch → 409 within 20s. + // JWT is opaque — do not decode. + const refresh = createTokenRefreshScheduler({ + refreshBufferMs: cfg.token_refresh_buffer_ms, + getAccessToken: async () => { + // Unconditionally refresh OAuth before calling /bridge — getAccessToken() + // returns expired tokens as non-null strings (doesn't check expiresAt), + // so truthiness doesn't mean valid. Pass the stale token to onAuth401 + // so handleOAuth401Error's keychain-comparison can detect parallel refresh. + const stale = getAccessToken() + if (onAuth401) await onAuth401(stale ?? '') + return getAccessToken() ?? stale + }, + onRefresh: (sid, oauthToken) => { + void (async () => { + // Laptop wake: overdue proactive timer + SSE 401 fire ~simultaneously. + // Claim the flag BEFORE the /bridge fetch so the other path skips + // entirely — prevents double epoch bump (each /bridge call bumps; if + // both fetch, the first rebuild gets a stale epoch and 409s). + if (authRecoveryInFlight || tornDown) { + logForDebugging( + '[remote-bridge] Recovery already in flight, skipping proactive refresh', + ) + return + } + authRecoveryInFlight = true + try { + const fresh = await withRetry( + () => + fetchRemoteCredentials( + sid, + baseUrl, + oauthToken, + cfg.http_timeout_ms, + ), + 'fetchRemoteCredentials (proactive)', + cfg, + ) + if (!fresh || tornDown) return + await rebuildTransport(fresh, 'proactive_refresh') + logForDebugging( + '[remote-bridge] Transport rebuilt (proactive refresh)', + ) + } catch (err) { + logForDebugging( + `[remote-bridge] Proactive refresh rebuild failed: ${errorMessage(err)}`, + { level: 'error' }, + ) + logForDiagnosticsNoPII( + 'error', + 'bridge_repl_v2_proactive_refresh_failed', + ) + if (!tornDown) { + onStateChange?.('failed', `Refresh failed: ${errorMessage(err)}`) + } + } finally { + authRecoveryInFlight = false + } + })() + }, + label: 'remote', + }) + refresh.scheduleFromExpiresIn(sessionId, credentials.expires_in) + + // ── 6. Wire callbacks (extracted so transport-rebuild can re-wire) ────── + function wireTransportCallbacks(): void { + transport.setOnConnect(() => { + clearTimeout(connectDeadline) + logForDebugging('[remote-bridge] v2 transport connected') + logForDiagnosticsNoPII('info', 'bridge_repl_v2_transport_connected') + logEvent('tengu_bridge_repl_ws_connected', { + v2: true, + cause: + connectCause as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + + if (!initialFlushDone && initialMessages && initialMessages.length > 0) { + initialFlushDone = true + // Capture current transport — if 401/teardown happens mid-flush, + // the stale .finally() must not drain the gate or signal connected. + // (Same guard pattern as replBridge.ts:1119.) + const flushTransport = transport + void flushHistory(initialMessages) + .catch(e => + logForDebugging(`[remote-bridge] flushHistory failed: ${e}`), + ) + .finally(() => { + // authRecoveryInFlight catches the v1-vs-v2 asymmetry: v1 nulls + // transport synchronously in setOnClose (replBridge.ts:1175), so + // transport !== flushTransport trips immediately. v2 doesn't null — + // transport reassigned only at rebuildTransport:346, 3 awaits deep. + // authRecoveryInFlight is set synchronously at rebuildTransport entry. + if ( + transport !== flushTransport || + tornDown || + authRecoveryInFlight + ) { + return + } + drainFlushGate() + onStateChange?.('connected') + }) + } else if (!flushGate.active) { + onStateChange?.('connected') + } + }) + + transport.setOnData((data: string) => { + handleIngressMessage( + data, + recentPostedUUIDs, + recentInboundUUIDs, + onInboundMessage, + // Remote client answered the permission prompt — the turn resumes. + // Without this the server stays on requires_action until the next + // user message or turn-end result. + onPermissionResponse + ? res => { + transport.reportState('running') + onPermissionResponse(res) + } + : undefined, + req => + handleServerControlRequest(req, { + transport, + sessionId, + onInterrupt, + onSetModel, + onSetMaxThinkingTokens, + onSetPermissionMode, + outboundOnly, + }), + ) + }) + + transport.setOnClose((code?: number) => { + clearTimeout(connectDeadline) + if (tornDown) return + logForDebugging(`[remote-bridge] v2 transport closed (code=${code})`) + logEvent('tengu_bridge_repl_ws_closed', { code, v2: true }) + // onClose fires only for TERMINAL failures: 401 (JWT invalid), + // 4090 (CCR epoch mismatch), 4091 (CCR init failed), or SSE 10-min + // reconnect budget exhausted. Transient disconnects are handled + // transparently inside SSETransport. 401 we can recover from (fetch + // fresh JWT, rebuild transport); all other codes are dead-ends. + if (code === 401 && !authRecoveryInFlight) { + void recoverFromAuthFailure() + return + } + onStateChange?.('failed', `Transport closed (code ${code})`) + }) + } + + // ── 7. Transport rebuild (shared by proactive refresh + 401 recovery) ── + // Every /bridge call bumps epoch server-side. Both refresh paths must + // rebuild the transport with the new epoch — a JWT-only swap leaves the + // old CCRClient heartbeating stale epoch → 409. SSE resumes from the old + // transport's high-water-mark seq-num so no server-side replay. + // Caller MUST set authRecoveryInFlight = true before calling (synchronously, + // before any await) and clear it in a finally. This function doesn't manage + // the flag — moving it here would be too late to prevent a double /bridge + // fetch, and each fetch bumps epoch. + async function rebuildTransport( + fresh: RemoteCredentials, + cause: Exclude, + ): Promise { + connectCause = cause + // Queue writes during rebuild — once /bridge returns, the old transport's + // epoch is stale and its next write/heartbeat 409s. Without this gate, + // writeMessages adds UUIDs to recentPostedUUIDs then writeBatch silently + // no-ops (closed uploader after 409) → permanent silent message loss. + flushGate.start() + try { + const seq = transport.getLastSequenceNum() + transport.close() + transport = await createV2ReplTransport({ + sessionUrl: buildCCRv2SdkUrl(fresh.api_base_url, sessionId), + ingressToken: fresh.worker_jwt, + sessionId, + epoch: fresh.worker_epoch, + heartbeatIntervalMs: cfg.heartbeat_interval_ms, + heartbeatJitterFraction: cfg.heartbeat_jitter_fraction, + initialSequenceNum: seq, + getAuthToken: () => fresh.worker_jwt, + outboundOnly, + }) + if (tornDown) { + // Teardown fired during the async createV2ReplTransport window. + // Don't wire/connect/schedule — we'd re-arm timers after cancelAll() + // and fire onInboundMessage into a torn-down bridge. + transport.close() + return + } + wireTransportCallbacks() + transport.connect() + connectDeadline = setTimeout( + onConnectTimeout, + cfg.connect_timeout_ms, + connectCause, + ) + refresh.scheduleFromExpiresIn(sessionId, fresh.expires_in) + // Drain queued writes into the new uploader. Runs before + // ccr.initialize() resolves (transport.connect() is fire-and-forget), + // but the uploader serializes behind the initial PUT /worker. If + // init fails (4091), events drop — but only recentPostedUUIDs + // (per-instance) is populated, so re-enabling the bridge re-flushes. + drainFlushGate() + } finally { + // End the gate on failure paths too — drainFlushGate already ended + // it on success. Queued messages are dropped (transport still dead). + flushGate.drop() + } + } + + // ── 8. 401 recovery (OAuth refresh + rebuild) ─────────────────────────── + async function recoverFromAuthFailure(): Promise { + // setOnClose already guards `!authRecoveryInFlight` but that check and + // this set must be atomic against onRefresh — claim synchronously before + // any await. Laptop wake fires both paths ~simultaneously. + if (authRecoveryInFlight) return + authRecoveryInFlight = true + onStateChange?.('reconnecting', 'JWT expired — refreshing') + logForDebugging('[remote-bridge] 401 on SSE — attempting JWT refresh') + try { + // Unconditionally try OAuth refresh — getAccessToken() returns expired + // tokens as non-null strings, so !oauthToken doesn't catch expiry. + // Pass the stale token so handleOAuth401Error's keychain-comparison + // can detect if another tab already refreshed. + const stale = getAccessToken() + if (onAuth401) await onAuth401(stale ?? '') + const oauthToken = getAccessToken() ?? stale + if (!oauthToken || tornDown) { + if (!tornDown) { + onStateChange?.('failed', 'JWT refresh failed: no OAuth token') + } + return + } + + const fresh = await withRetry( + () => + fetchRemoteCredentials( + sessionId, + baseUrl, + oauthToken, + cfg.http_timeout_ms, + ), + 'fetchRemoteCredentials (recovery)', + cfg, + ) + if (!fresh || tornDown) { + if (!tornDown) { + onStateChange?.('failed', 'JWT refresh failed after 401') + } + return + } + // If 401 interrupted the initial flush, writeBatch may have silently + // no-op'd on the closed uploader (ccr.close() ran in the SSE wrapper + // before our setOnClose callback). Reset so the new onConnect re-flushes. + // (v1 scopes initialFlushDone inside the per-transport closure at + // replBridge.ts:1027 so it resets naturally; v2 has it at outer scope.) + initialFlushDone = false + await rebuildTransport(fresh, 'auth_401_recovery') + logForDebugging('[remote-bridge] Transport rebuilt after 401') + } catch (err) { + logForDebugging( + `[remote-bridge] 401 recovery failed: ${errorMessage(err)}`, + { level: 'error' }, + ) + logForDiagnosticsNoPII('error', 'bridge_repl_v2_jwt_refresh_failed') + if (!tornDown) { + onStateChange?.('failed', `JWT refresh failed: ${errorMessage(err)}`) + } + } finally { + authRecoveryInFlight = false + } + } + + wireTransportCallbacks() + + // Start flushGate BEFORE connect so writeMessages() during handshake + // queues instead of racing the history POST. + if (initialMessages && initialMessages.length > 0) { + flushGate.start() + } + transport.connect() + connectDeadline = setTimeout( + onConnectTimeout, + cfg.connect_timeout_ms, + connectCause, + ) + + // ── 8. History flush + drain helpers ──────────────────────────────────── + function drainFlushGate(): void { + const msgs = flushGate.end() + if (msgs.length === 0) return + for (const msg of msgs) recentPostedUUIDs.add(msg.uuid) + const events = toSDKMessages(msgs).map(m => ({ + ...m, + session_id: sessionId, + })) + if (msgs.some(m => m.type === 'user')) { + transport.reportState('running') + } + logForDebugging( + `[remote-bridge] Drained ${msgs.length} queued message(s) after flush`, + ) + void transport.writeBatch(events) + } + + async function flushHistory(msgs: Message[]): Promise { + // v2 always creates a fresh server session (unconditional createCodeSession + // above) — no session reuse, no double-post risk. Unlike v1, we do NOT + // filter by previouslyFlushedUUIDs: that set persists across REPL enable/ + // disable cycles (useRef), so it would wrongly suppress history on re-enable. + const eligible = msgs.filter(isEligibleBridgeMessage) + const capped = + initialHistoryCap > 0 && eligible.length > initialHistoryCap + ? eligible.slice(-initialHistoryCap) + : eligible + if (capped.length < eligible.length) { + logForDebugging( + `[remote-bridge] Capped initial flush: ${eligible.length} -> ${capped.length} (cap=${initialHistoryCap})`, + ) + } + const events = toSDKMessages(capped).map(m => ({ + ...m, + session_id: sessionId, + })) + if (events.length === 0) return + // Mid-turn init: if Remote Control is enabled while a query is running, + // the last eligible message is a user prompt or tool_result (both 'user' + // type). Without this the init PUT's 'idle' sticks until the next user- + // type message forwards via writeMessages — which for a pure-text turn + // is never (only assistant chunks stream post-init). Check eligible (pre- + // cap), not capped: the cap may truncate to a user message even when the + // actual trailing message is assistant. + if (eligible.at(-1)?.type === 'user') { + transport.reportState('running') + } + logForDebugging(`[remote-bridge] Flushing ${events.length} history events`) + await transport.writeBatch(events) + } + + // ── 9. Teardown ─────────────────────────────────────────────────────────── + // On SIGINT/SIGTERM/⁠/exit, gracefulShutdown races runCleanupFunctions() + // against a 2s cap before forceExit kills the process. Budget accordingly: + // - archive: teardown_archive_timeout_ms (default 1500, cap 2000) + // - result write: fire-and-forget, archive latency covers the drain + // - 401 retry: only if first archive 401s, shares the same budget + async function teardown(): Promise { + if (tornDown) return + tornDown = true + refresh.cancelAll() + clearTimeout(connectDeadline) + flushGate.drop() + + // Fire the result message before archive — transport.write() only awaits + // enqueue (SerialBatchEventUploader resolves once buffered, drain is + // async). Archiving before close() gives the uploader's drain loop a + // window (typical archive ≈ 100-500ms) to POST the result without an + // explicit sleep. close() sets closed=true which interrupts drain at the + // next while-check, so close-before-archive drops the result. + transport.reportState('idle') + void transport.write(makeResultMessage(sessionId)) + + let token = getAccessToken() + let status = await archiveSession( + sessionId, + baseUrl, + token, + orgUUID, + cfg.teardown_archive_timeout_ms, + ) + + // Token is usually fresh (refresh scheduler runs 5min before expiry) but + // laptop-wake past the refresh window leaves getAccessToken() returning a + // stale string. Retry once on 401 — onAuth401 (= handleOAuth401Error) + // clears keychain cache + force-refreshes. No proactive refresh on the + // happy path: handleOAuth401Error force-refreshes even valid tokens, + // which would waste budget 99% of the time. try/catch mirrors + // recoverFromAuthFailure: keychain reads can throw (macOS locked after + // wake); an uncaught throw here would skip transport.close + telemetry. + if (status === 401 && onAuth401) { + try { + await onAuth401(token ?? '') + token = getAccessToken() + status = await archiveSession( + sessionId, + baseUrl, + token, + orgUUID, + cfg.teardown_archive_timeout_ms, + ) + } catch (err) { + logForDebugging( + `[remote-bridge] Teardown 401 retry threw: ${errorMessage(err)}`, + { level: 'error' }, + ) + } + } + + transport.close() + + const archiveStatus: ArchiveTelemetryStatus = + status === 'no_token' + ? 'skipped_no_token' + : status === 'timeout' || status === 'error' + ? 'network_error' + : status >= 500 + ? 'server_5xx' + : status >= 400 + ? 'server_4xx' + : 'ok' + + logForDebugging(`[remote-bridge] Torn down (archive=${status})`) + logForDiagnosticsNoPII('info', 'bridge_repl_v2_teardown') + logEvent( + feature('CCR_MIRROR') && outboundOnly + ? 'tengu_ccr_mirror_teardown' + : 'tengu_bridge_repl_teardown', + { + v2: true, + archive_status: + archiveStatus as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + archive_ok: typeof status === 'number' && status < 400, + archive_http_status: typeof status === 'number' ? status : undefined, + archive_timeout: status === 'timeout', + archive_no_token: status === 'no_token', + }, + ) + } + const unregister = registerCleanup(teardown) + + if (feature('CCR_MIRROR') && outboundOnly) { + logEvent('tengu_ccr_mirror_started', { + v2: true, + expires_in_s: credentials.expires_in, + }) + } else { + logEvent('tengu_bridge_repl_started', { + has_initial_messages: !!(initialMessages && initialMessages.length > 0), + v2: true, + expires_in_s: credentials.expires_in, + inProtectedNamespace: isInProtectedNamespace(), + }) + } + + // ── 10. Handle ────────────────────────────────────────────────────────── + return { + bridgeSessionId: sessionId, + environmentId: '', + sessionIngressUrl: credentials.api_base_url, + writeMessages(messages) { + const filtered = messages.filter( + m => + isEligibleBridgeMessage(m) && + !initialMessageUUIDs.has(m.uuid) && + !recentPostedUUIDs.has(m.uuid), + ) + if (filtered.length === 0) return + + // Fire onUserMessage for title derivation. Scan before the flushGate + // check — prompts are title-worthy even if they queue. Keeps calling + // on every title-worthy message until the callback returns true; the + // caller owns the policy (derive at 1st and 3rd, skip if explicit). + if (!userMessageCallbackDone) { + for (const m of filtered) { + const text = extractTitleText(m) + if (text !== undefined && onUserMessage?.(text, sessionId)) { + userMessageCallbackDone = true + break + } + } + } + + if (flushGate.enqueue(...filtered)) { + logForDebugging( + `[remote-bridge] Queued ${filtered.length} message(s) during flush`, + ) + return + } + + for (const msg of filtered) recentPostedUUIDs.add(msg.uuid) + const events = toSDKMessages(filtered).map(m => ({ + ...m, + session_id: sessionId, + })) + // v2 does not derive worker_status from events server-side (unlike v1 + // session-ingress session_status_updater.go). Push it from here so the + // CCR web session list shows Running instead of stuck on Idle. A user + // message in the batch marks turn start. CCRClient.reportState dedupes + // consecutive same-state pushes. + if (filtered.some(m => m.type === 'user')) { + transport.reportState('running') + } + logForDebugging(`[remote-bridge] Sending ${filtered.length} message(s)`) + void transport.writeBatch(events) + }, + writeSdkMessages(messages: SDKMessage[]) { + const filtered = messages.filter( + m => !m.uuid || !recentPostedUUIDs.has(m.uuid), + ) + if (filtered.length === 0) return + for (const msg of filtered) { + if (msg.uuid) recentPostedUUIDs.add(msg.uuid) + } + const events = filtered.map(m => ({ ...m, session_id: sessionId })) + void transport.writeBatch(events) + }, + sendControlRequest(request: SDKControlRequest) { + if (authRecoveryInFlight) { + logForDebugging( + `[remote-bridge] Dropping control_request during 401 recovery: ${request.request_id}`, + ) + return + } + const event = { ...request, session_id: sessionId } + if (request.request.subtype === 'can_use_tool') { + transport.reportState('requires_action') + } + void transport.write(event) + logForDebugging( + `[remote-bridge] Sent control_request request_id=${request.request_id}`, + ) + }, + sendControlResponse(response: SDKControlResponse) { + if (authRecoveryInFlight) { + logForDebugging( + '[remote-bridge] Dropping control_response during 401 recovery', + ) + return + } + const event = { ...response, session_id: sessionId } + transport.reportState('running') + void transport.write(event) + logForDebugging('[remote-bridge] Sent control_response') + }, + sendControlCancelRequest(requestId: string) { + if (authRecoveryInFlight) { + logForDebugging( + `[remote-bridge] Dropping control_cancel_request during 401 recovery: ${requestId}`, + ) + return + } + const event = { + type: 'control_cancel_request' as const, + request_id: requestId, + session_id: sessionId, + } + // Hook/classifier/channel/recheck resolved the permission locally — + // interactiveHandler calls only cancelRequest (no sendResponse) on + // those paths, so without this the server stays on requires_action. + transport.reportState('running') + void transport.write(event) + logForDebugging( + `[remote-bridge] Sent control_cancel_request request_id=${requestId}`, + ) + }, + sendResult() { + if (authRecoveryInFlight) { + logForDebugging('[remote-bridge] Dropping result during 401 recovery') + return + } + transport.reportState('idle') + void transport.write(makeResultMessage(sessionId)) + logForDebugging(`[remote-bridge] Sent result`) + }, + async teardown() { + unregister() + await teardown() + }, + } +} + +// ─── Session API (v2 /code/sessions, no env) ───────────────────────────────── + +/** Retry an async init call with exponential backoff + jitter. */ +async function withRetry( + fn: () => Promise, + label: string, + cfg: EnvLessBridgeConfig, +): Promise { + const max = cfg.init_retry_max_attempts + for (let attempt = 1; attempt <= max; attempt++) { + const result = await fn() + if (result !== null) return result + if (attempt < max) { + const base = cfg.init_retry_base_delay_ms * 2 ** (attempt - 1) + const jitter = + base * cfg.init_retry_jitter_fraction * (2 * Math.random() - 1) + const delay = Math.min(base + jitter, cfg.init_retry_max_delay_ms) + logForDebugging( + `[remote-bridge] ${label} failed (attempt ${attempt}/${max}), retrying in ${Math.round(delay)}ms`, + ) + await sleep(delay) + } + } + return null +} + +// Moved to codeSessionApi.ts so the SDK /bridge subpath can bundle them +// without pulling in this file's heavy CLI tree (analytics, transport). +export { + createCodeSession, + type RemoteCredentials, +} from './codeSessionApi.js' +import { + createCodeSession, + fetchRemoteCredentials as fetchRemoteCredentialsRaw, + type RemoteCredentials, +} from './codeSessionApi.js' +import { getBridgeBaseUrlOverride } from './bridgeConfig.js' + +// CLI-side wrapper that applies the CLAUDE_BRIDGE_BASE_URL dev override and +// injects the trusted-device token (both are env/GrowthBook reads that the +// SDK-facing codeSessionApi.ts export must stay free of). +export async function fetchRemoteCredentials( + sessionId: string, + baseUrl: string, + accessToken: string, + timeoutMs: number, +): Promise { + const creds = await fetchRemoteCredentialsRaw( + sessionId, + baseUrl, + accessToken, + timeoutMs, + getTrustedDeviceToken(), + ) + if (!creds) return null + return getBridgeBaseUrlOverride() + ? { ...creds, api_base_url: baseUrl } + : creds +} + +type ArchiveStatus = number | 'timeout' | 'error' | 'no_token' + +// Single categorical for BQ `GROUP BY archive_status`. The booleans on +// _teardown predate this and are redundant with it (except archive_timeout, +// which distinguishes ECONNABORTED from other network errors — both map to +// 'network_error' here since the dominant cause in a 1.5s window is timeout). +type ArchiveTelemetryStatus = + | 'ok' + | 'skipped_no_token' + | 'network_error' + | 'server_4xx' + | 'server_5xx' + +async function archiveSession( + sessionId: string, + baseUrl: string, + accessToken: string | undefined, + orgUUID: string, + timeoutMs: number, +): Promise { + if (!accessToken) return 'no_token' + // Archive lives at the compat layer (/v1/sessions/*, not /v1/code/sessions). + // compat.parseSessionID only accepts TagSession (session_*), so retag cse_*. + // anthropic-beta + x-organization-uuid are required — without them the + // compat gateway 404s before reaching the handler. + // + // Unlike bridgeMain.ts (which caches compatId in sessionCompatIds to keep + // in-memory titledSessions/logger keys consistent across a mid-session + // gate flip), this compatId is only a server URL path segment — no + // in-memory state. Fresh compute matches whatever the server currently + // validates: if the gate is OFF, the server has been updated to accept + // cse_* and we correctly send it. + const compatId = toCompatSessionId(sessionId) + try { + const response = await axios.post( + `${baseUrl}/v1/sessions/${compatId}/archive`, + {}, + { + headers: { + ...oauthHeaders(accessToken), + 'anthropic-beta': 'ccr-byoc-2025-07-29', + 'x-organization-uuid': orgUUID, + }, + timeout: timeoutMs, + validateStatus: () => true, + }, + ) + logForDebugging( + `[remote-bridge] Archive ${compatId} status=${response.status}`, + ) + return response.status + } catch (err) { + const msg = errorMessage(err) + logForDebugging(`[remote-bridge] Archive failed: ${msg}`) + return axios.isAxiosError(err) && err.code === 'ECONNABORTED' + ? 'timeout' + : 'error' + } +} diff --git a/src/bridge/replBridge.ts b/src/bridge/replBridge.ts new file mode 100644 index 0000000..7d7ac6a --- /dev/null +++ b/src/bridge/replBridge.ts @@ -0,0 +1,2406 @@ +// biome-ignore-all assist/source/organizeImports: ANT-ONLY import markers must not be reordered +import { randomUUID } from 'crypto' +import { + createBridgeApiClient, + BridgeFatalError, + isExpiredErrorType, + isSuppressible403, +} from './bridgeApi.js' +import type { BridgeConfig, BridgeApiClient } from './types.js' +import { logForDebugging } from '../utils/debug.js' +import { logForDiagnosticsNoPII } from '../utils/diagLogs.js' +import { + type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + logEvent, +} from '../services/analytics/index.js' +import { registerCleanup } from '../utils/cleanupRegistry.js' +import { + handleIngressMessage, + handleServerControlRequest, + makeResultMessage, + isEligibleBridgeMessage, + extractTitleText, + BoundedUUIDSet, +} from './bridgeMessaging.js' +import { + decodeWorkSecret, + buildSdkUrl, + buildCCRv2SdkUrl, + sameSessionId, +} from './workSecret.js' +import { toCompatSessionId, toInfraSessionId } from './sessionIdCompat.js' +import { updateSessionBridgeId } from '../utils/concurrentSessions.js' +import { getTrustedDeviceToken } from './trustedDevice.js' +import { HybridTransport } from '../cli/transports/HybridTransport.js' +import { + type ReplBridgeTransport, + createV1ReplTransport, + createV2ReplTransport, +} from './replBridgeTransport.js' +import { updateSessionIngressAuthToken } from '../utils/sessionIngressAuth.js' +import { isEnvTruthy, isInProtectedNamespace } from '../utils/envUtils.js' +import { validateBridgeId } from './bridgeApi.js' +import { + describeAxiosError, + extractHttpStatus, + logBridgeSkip, +} from './debugUtils.js' +import type { Message } from '../types/message.js' +import type { SDKMessage } from '../entrypoints/agentSdkTypes.js' +import type { PermissionMode } from '../utils/permissions/PermissionMode.js' +import type { + SDKControlRequest, + SDKControlResponse, +} from '../entrypoints/sdk/controlTypes.js' +import { createCapacityWake, type CapacitySignal } from './capacityWake.js' +import { FlushGate } from './flushGate.js' +import { + DEFAULT_POLL_CONFIG, + type PollIntervalConfig, +} from './pollConfigDefaults.js' +import { errorMessage } from '../utils/errors.js' +import { sleep } from '../utils/sleep.js' +import { + wrapApiForFaultInjection, + registerBridgeDebugHandle, + clearBridgeDebugHandle, + injectBridgeFault, +} from './bridgeDebug.js' + +export type ReplBridgeHandle = { + bridgeSessionId: string + environmentId: string + sessionIngressUrl: string + writeMessages(messages: Message[]): void + writeSdkMessages(messages: SDKMessage[]): void + sendControlRequest(request: SDKControlRequest): void + sendControlResponse(response: SDKControlResponse): void + sendControlCancelRequest(requestId: string): void + sendResult(): void + teardown(): Promise +} + +export type BridgeState = 'ready' | 'connected' | 'reconnecting' | 'failed' + +/** + * Explicit-param input to initBridgeCore. Everything initReplBridge reads + * from bootstrap state (cwd, session ID, git, OAuth) becomes a field here. + * A daemon caller (Agent SDK, PR 4) that never runs main.tsx fills these + * in itself. + */ +export type BridgeCoreParams = { + dir: string + machineName: string + branch: string + gitRepoUrl: string | null + title: string + baseUrl: string + sessionIngressUrl: string + /** + * Opaque string sent as metadata.worker_type. Use BridgeWorkerType for + * the two CLI-originated values; daemon callers may send any string the + * backend recognizes (it's just a filter key on the web side). + */ + workerType: string + getAccessToken: () => string | undefined + /** + * POST /v1/sessions. Injected because `createSession.ts` lazy-loads + * `auth.ts`/`model.ts`/`oauth/client.ts` and `bun --outfile` inlines + * dynamic imports — the lazy-load doesn't help, the whole REPL tree ends + * up in the Agent SDK bundle. + * + * REPL wrapper passes `createBridgeSession` from `createSession.ts`. + * Daemon wrapper passes `createBridgeSessionLean` from `sessionApi.ts` + * (HTTP-only, orgUUID+model supplied by the daemon caller). + * + * Receives `gitRepoUrl`+`branch` so the REPL wrapper can build the git + * source/outcome for claude.ai's session card. Daemon ignores them. + */ + createSession: (opts: { + environmentId: string + title: string + gitRepoUrl: string | null + branch: string + signal: AbortSignal + }) => Promise + /** + * POST /v1/sessions/{id}/archive. Same injection rationale. Best-effort; + * the callback MUST NOT throw. + */ + archiveSession: (sessionId: string) => Promise + /** + * Invoked on reconnect-after-env-lost to refresh the title. REPL wrapper + * reads session storage (picks up /rename); daemon returns the static + * title. Defaults to () => title. + */ + getCurrentTitle?: () => string + /** + * Converts internal Message[] → SDKMessage[] for writeMessages() and the + * initial-flush/drain paths. REPL wrapper passes the real toSDKMessages + * from utils/messages/mappers.ts. Daemon callers that only use + * writeSdkMessages() and pass no initialMessages can omit this — those + * code paths are unreachable. + * + * Injected rather than imported because mappers.ts transitively pulls in + * src/commands.ts via messages.ts → api.ts → prompts.ts, dragging the + * entire command registry + React tree into the Agent SDK bundle. + */ + toSDKMessages?: (messages: Message[]) => SDKMessage[] + /** + * OAuth 401 refresh handler passed to createBridgeApiClient. REPL wrapper + * passes handleOAuth401Error; daemon passes its AuthManager's handler. + * Injected because utils/auth.ts transitively pulls in the command + * registry via config.ts → file.ts → permissions/filesystem.ts → + * sessionStorage.ts → commands.ts. + */ + onAuth401?: (staleAccessToken: string) => Promise + /** + * Poll interval config getter for the work-poll heartbeat loop. REPL + * wrapper passes the GrowthBook-backed getPollIntervalConfig (allows ops + * to live-tune poll rates fleet-wide). Daemon passes a static config + * with a 60s heartbeat (5× headroom under the 300s work-lease TTL). + * Injected because growthbook.ts transitively pulls in the command + * registry via the same config.ts chain. + */ + getPollIntervalConfig?: () => PollIntervalConfig + /** + * Max initial messages to replay on connect. REPL wrapper reads from the + * tengu_bridge_initial_history_cap GrowthBook flag. Daemon passes no + * initialMessages so this is never read. Default 200 matches the flag + * default. + */ + initialHistoryCap?: number + // Same REPL-flush machinery as InitBridgeOptions — daemon omits these. + initialMessages?: Message[] + previouslyFlushedUUIDs?: Set + onInboundMessage?: (msg: SDKMessage) => void + onPermissionResponse?: (response: SDKControlResponse) => void + onInterrupt?: () => void + onSetModel?: (model: string | undefined) => void + onSetMaxThinkingTokens?: (maxTokens: number | null) => void + /** + * Returns a policy verdict so this module can emit an error control_response + * without importing the policy checks itself (bootstrap-isolation constraint). + * The callback must guard `auto` (isAutoModeGateEnabled) and + * `bypassPermissions` (isBypassPermissionsModeDisabled AND + * isBypassPermissionsModeAvailable) BEFORE calling transitionPermissionMode — + * that function's internal auto-gate check is a defensive throw, not a + * graceful guard, and its side-effect order is setAutoModeActive(true) then + * throw, which corrupts the 3-way invariant documented in src/CLAUDE.md if + * the callback lets the throw escape here. + */ + onSetPermissionMode?: ( + mode: PermissionMode, + ) => { ok: true } | { ok: false; error: string } + onStateChange?: (state: BridgeState, detail?: string) => void + /** + * Fires on each real user message to flow through writeMessages() until + * the callback returns true (done). Mirrors remoteBridgeCore.ts's + * onUserMessage so the REPL bridge can derive a session title from early + * prompts when none was set at init time (e.g. user runs /remote-control + * on an empty conversation, then types). Tool-result wrappers, meta + * messages, and display-tag-only messages are skipped. Receives + * currentSessionId so the wrapper can PATCH the title without a closure + * dance to reach the not-yet-returned handle. The caller owns the + * derive-at-count-1-and-3 policy; the transport just keeps calling until + * told to stop. Not fired for the writeSdkMessages daemon path (daemon + * sets its own title at init). Distinct from SessionSpawnOpts's + * onFirstUserMessage (spawn-bridge, PR #21250), which stays fire-once. + */ + onUserMessage?: (text: string, sessionId: string) => boolean + /** See InitBridgeOptions.perpetual. */ + perpetual?: boolean + /** + * Seeds lastTransportSequenceNum — the SSE event-stream high-water mark + * that's carried across transport swaps within one process. Daemon callers + * pass the value they persisted at shutdown so the FIRST SSE connect of a + * fresh process sends from_sequence_num and the server doesn't replay full + * history. REPL callers omit (fresh session each run → 0 is correct). + */ + initialSSESequenceNum?: number +} + +/** + * Superset of ReplBridgeHandle. Adds getSSESequenceNum for daemon callers + * that persist the SSE seq-num across process restarts and pass it back as + * initialSSESequenceNum on the next start. + */ +export type BridgeCoreHandle = ReplBridgeHandle & { + /** + * Current SSE sequence-number high-water mark. Updates as transports + * swap. Daemon callers persist this on shutdown and pass it back as + * initialSSESequenceNum on next start. + */ + getSSESequenceNum(): number +} + +/** + * Poll error recovery constants. When the work poll starts failing (e.g. + * server 500s), we use exponential backoff and give up after this timeout. + * This is deliberately long — the server is the authority on when a session + * is truly dead. As long as the server accepts our poll, we keep waiting + * for it to re-dispatch the work item. + */ +const POLL_ERROR_INITIAL_DELAY_MS = 2_000 +const POLL_ERROR_MAX_DELAY_MS = 60_000 +const POLL_ERROR_GIVE_UP_MS = 15 * 60 * 1000 + +// Monotonically increasing counter for distinguishing init calls in logs +let initSequence = 0 + +/** + * Bootstrap-free core: env registration → session creation → poll loop → + * ingress WS → teardown. Reads nothing from bootstrap/state or + * sessionStorage — all context comes from params. Caller (initReplBridge + * below, or a daemon in PR 4) has already passed entitlement gates and + * gathered git/auth/title. + * + * Returns null on registration or session-creation failure. + */ +export async function initBridgeCore( + params: BridgeCoreParams, +): Promise { + const { + dir, + machineName, + branch, + gitRepoUrl, + title, + baseUrl, + sessionIngressUrl, + workerType, + getAccessToken, + createSession, + archiveSession, + getCurrentTitle = () => title, + toSDKMessages = () => { + throw new Error( + 'BridgeCoreParams.toSDKMessages not provided. Pass it if you use writeMessages() or initialMessages — daemon callers that only use writeSdkMessages() never hit this path.', + ) + }, + onAuth401, + getPollIntervalConfig = () => DEFAULT_POLL_CONFIG, + initialHistoryCap = 200, + initialMessages, + previouslyFlushedUUIDs, + onInboundMessage, + onPermissionResponse, + onInterrupt, + onSetModel, + onSetMaxThinkingTokens, + onSetPermissionMode, + onStateChange, + onUserMessage, + perpetual, + initialSSESequenceNum = 0, + } = params + + const seq = ++initSequence + + // bridgePointer import hoisted: perpetual mode reads it before register; + // non-perpetual writes it after session create; both use clear at teardown. + const { writeBridgePointer, clearBridgePointer, readBridgePointer } = + await import('./bridgePointer.js') + + // Perpetual mode: read the crash-recovery pointer and treat it as prior + // state. The pointer is written unconditionally after session create + // (crash-recovery for all sessions); perpetual mode just skips the + // teardown clear so it survives clean exits too. Only reuse 'repl' + // pointers — a crashed standalone bridge (`claude remote-control`) + // writes source:'standalone' with a different workerType. + const rawPrior = perpetual ? await readBridgePointer(dir) : null + const prior = rawPrior?.source === 'repl' ? rawPrior : null + + logForDebugging( + `[bridge:repl] initBridgeCore #${seq} starting (initialMessages=${initialMessages?.length ?? 0}${prior ? ` perpetual prior=env:${prior.environmentId}` : ''})`, + ) + + // 5. Register bridge environment + const rawApi = createBridgeApiClient({ + baseUrl, + getAccessToken, + runnerVersion: MACRO.VERSION, + onDebug: logForDebugging, + onAuth401, + getTrustedDeviceToken, + }) + // Ant-only: interpose so /bridge-kick can inject poll/register/heartbeat + // failures. Zero cost in external builds (rawApi passes through unchanged). + const api = + process.env.USER_TYPE === 'ant' ? wrapApiForFaultInjection(rawApi) : rawApi + + const bridgeConfig: BridgeConfig = { + dir, + machineName, + branch, + gitRepoUrl, + maxSessions: 1, + spawnMode: 'single-session', + verbose: false, + sandbox: false, + bridgeId: randomUUID(), + workerType, + environmentId: randomUUID(), + reuseEnvironmentId: prior?.environmentId, + apiBaseUrl: baseUrl, + sessionIngressUrl, + } + + let environmentId: string + let environmentSecret: string + try { + const reg = await api.registerBridgeEnvironment(bridgeConfig) + environmentId = reg.environment_id + environmentSecret = reg.environment_secret + } catch (err) { + logBridgeSkip( + 'registration_failed', + `[bridge:repl] Environment registration failed: ${errorMessage(err)}`, + ) + // Stale pointer may be the cause (expired/deleted env) — clear it so + // the next start doesn't retry the same dead ID. + if (prior) { + await clearBridgePointer(dir) + } + onStateChange?.('failed', errorMessage(err)) + return null + } + + logForDebugging(`[bridge:repl] Environment registered: ${environmentId}`) + logForDiagnosticsNoPII('info', 'bridge_repl_env_registered') + logEvent('tengu_bridge_repl_env_registered', {}) + + /** + * Reconnect-in-place: if the just-registered environmentId matches what + * was requested, call reconnectSession to force-stop stale workers and + * re-queue the session. Used at init (perpetual mode — env is alive but + * idle after clean teardown) and in doReconnect() Strategy 1 (env lost + * then resurrected). Returns true on success; caller falls back to + * fresh session creation on false. + */ + async function tryReconnectInPlace( + requestedEnvId: string, + sessionId: string, + ): Promise { + if (environmentId !== requestedEnvId) { + logForDebugging( + `[bridge:repl] Env mismatch (requested ${requestedEnvId}, got ${environmentId}) — cannot reconnect in place`, + ) + return false + } + // The pointer stores what createBridgeSession returned (session_*, + // compat/convert.go:41). /bridge/reconnect is an environments-layer + // endpoint — once the server's ccr_v2_compat_enabled gate is on it + // looks sessions up by their infra tag (cse_*) and returns "Session + // not found" for the session_* costume. We don't know the gate state + // pre-poll, so try both; the re-tag is a no-op if the ID is already + // cse_* (doReconnect Strategy 1 path — currentSessionId never mutates + // to cse_* but future-proof the check). + const infraId = toInfraSessionId(sessionId) + const candidates = + infraId === sessionId ? [sessionId] : [sessionId, infraId] + for (const id of candidates) { + try { + await api.reconnectSession(environmentId, id) + logForDebugging( + `[bridge:repl] Reconnected session ${id} in place on env ${environmentId}`, + ) + return true + } catch (err) { + logForDebugging( + `[bridge:repl] reconnectSession(${id}) failed: ${errorMessage(err)}`, + ) + } + } + logForDebugging( + '[bridge:repl] reconnectSession exhausted — falling through to fresh session', + ) + return false + } + + // Perpetual init: env is alive but has no queued work after clean + // teardown. reconnectSession re-queues it. doReconnect() has the same + // call but only fires on poll 404 (env dead); + // here the env is alive but idle. + const reusedPriorSession = prior + ? await tryReconnectInPlace(prior.environmentId, prior.sessionId) + : false + if (prior && !reusedPriorSession) { + await clearBridgePointer(dir) + } + + // 6. Create session on the bridge. Initial messages are NOT included as + // session creation events because those use STREAM_ONLY persistence and + // are published before the CCR UI subscribes, so they get lost. Instead, + // initial messages are flushed via the ingress WebSocket once it connects. + + // Mutable session ID — updated when the environment+session pair is + // re-created after a connection loss. + let currentSessionId: string + + + if (reusedPriorSession && prior) { + currentSessionId = prior.sessionId + logForDebugging( + `[bridge:repl] Perpetual session reused: ${currentSessionId}`, + ) + // Server already has all initialMessages from the prior CLI run. Mark + // them as previously-flushed so the initial flush filter excludes them + // (previouslyFlushedUUIDs is a fresh Set on every CLI start). Duplicate + // UUIDs cause the server to kill the WebSocket. + if (initialMessages && previouslyFlushedUUIDs) { + for (const msg of initialMessages) { + previouslyFlushedUUIDs.add(msg.uuid) + } + } + } else { + const createdSessionId = await createSession({ + environmentId, + title, + gitRepoUrl, + branch, + signal: AbortSignal.timeout(15_000), + }) + + if (!createdSessionId) { + logForDebugging( + '[bridge:repl] Session creation failed, deregistering environment', + ) + logEvent('tengu_bridge_repl_session_failed', {}) + await api.deregisterEnvironment(environmentId).catch(() => {}) + onStateChange?.('failed', 'Session creation failed') + return null + } + + currentSessionId = createdSessionId + logForDebugging(`[bridge:repl] Session created: ${currentSessionId}`) + } + + // Crash-recovery pointer: written now so a kill -9 at any point after + // this leaves a recoverable trail. Cleared in teardown (non-perpetual) + // or left alone (perpetual mode — pointer survives clean exit too). + // `claude remote-control --continue` from the same directory will detect + // it and offer to resume. + await writeBridgePointer(dir, { + sessionId: currentSessionId, + environmentId, + source: 'repl', + }) + logForDiagnosticsNoPII('info', 'bridge_repl_session_created') + logEvent('tengu_bridge_repl_started', { + has_initial_messages: !!(initialMessages && initialMessages.length > 0), + inProtectedNamespace: isInProtectedNamespace(), + }) + + // UUIDs of initial messages. Used for dedup in writeMessages to avoid + // re-sending messages that were already flushed on WebSocket open. + const initialMessageUUIDs = new Set() + if (initialMessages) { + for (const msg of initialMessages) { + initialMessageUUIDs.add(msg.uuid) + } + } + + // Bounded ring buffer of UUIDs for messages we've already sent to the + // server via the ingress WebSocket. Serves two purposes: + // 1. Echo filtering — ignore our own messages bouncing back on the WS. + // 2. Secondary dedup in writeMessages — catch race conditions where + // the hook's index-based tracking isn't sufficient. + // + // Seeded with initialMessageUUIDs so that when the server echoes back + // the initial conversation context over the ingress WebSocket, those + // messages are recognized as echoes and not re-injected into the REPL. + // + // Capacity of 2000 covers well over any realistic echo window (echoes + // arrive within milliseconds) and any messages that might be re-encountered + // after compaction. The hook's lastWrittenIndexRef is the primary dedup; + // this is a safety net. + const recentPostedUUIDs = new BoundedUUIDSet(2000) + for (const uuid of initialMessageUUIDs) { + recentPostedUUIDs.add(uuid) + } + + // Bounded set of INBOUND prompt UUIDs we've already forwarded to the REPL. + // Defensive dedup for when the server re-delivers prompts (seq-num + // negotiation failure, server edge cases, transport swap races). The + // seq-num carryover below is the primary fix; this is the safety net. + const recentInboundUUIDs = new BoundedUUIDSet(2000) + + // 7. Start poll loop for work items — this is what makes the session + // "live" on claude.ai. When a user types there, the backend dispatches + // a work item to our environment. We poll for it, get the ingress token, + // and connect the ingress WebSocket. + // + // The poll loop keeps running: when work arrives it connects the ingress + // WebSocket, and if the WebSocket drops unexpectedly (code != 1000) it + // resumes polling to get a fresh ingress token and reconnect. + const pollController = new AbortController() + // Adapter over either HybridTransport (v1: WS reads + POST writes to + // Session-Ingress) or SSETransport+CCRClient (v2: SSE reads + POST + // writes to CCR /worker/*). The v1/v2 choice is made in onWorkReceived: + // server-driven via secret.use_code_sessions, with CLAUDE_BRIDGE_USE_CCR_V2 + // as an ant-dev override. + let transport: ReplBridgeTransport | null = null + // Bumped on every onWorkReceived. Captured in createV2ReplTransport's .then() + // closure to detect stale resolutions: if two calls race while transport is + // null, both registerWorker() (bumping server epoch), and whichever resolves + // SECOND is the correct one — but the transport !== null check gets this + // backwards (first-to-resolve installs, second discards). The generation + // counter catches it independent of transport state. + let v2Generation = 0 + // SSE sequence-number high-water mark carried across transport swaps. + // Without this, each new SSETransport starts at 0, sends no + // from_sequence_num / Last-Event-ID on its first connect, and the server + // replays the entire session event history — every prompt ever sent + // re-delivered as fresh inbound messages on every onWorkReceived. + // + // Seed only when we actually reconnected the prior session. If + // `reusedPriorSession` is false we fell through to `createSession()` — + // the caller's persisted seq-num belongs to a dead session and applying + // it to the fresh stream (starting at 1) silently drops events. Same + // hazard as doReconnect Strategy 2; same fix as the reset there. + let lastTransportSequenceNum = reusedPriorSession ? initialSSESequenceNum : 0 + // Track the current work ID so teardown can call stopWork + let currentWorkId: string | null = null + // Session ingress JWT for the current work item — used for heartbeat auth. + let currentIngressToken: string | null = null + // Signal to wake the at-capacity sleep early when the transport is lost, + // so the poll loop immediately switches back to fast polling for new work. + const capacityWake = createCapacityWake(pollController.signal) + const wakePollLoop = capacityWake.wake + const capacitySignal = capacityWake.signal + // Gates message writes during the initial flush to prevent ordering + // races where new messages arrive at the server interleaved with history. + const flushGate = new FlushGate() + + // Latch for onUserMessage — flips true when the callback returns true + // (policy says "done deriving"). If no callback, skip scanning entirely + // (daemon path — no title derivation needed). + let userMessageCallbackDone = !onUserMessage + + // Shared counter for environment re-creations, used by both + // onEnvironmentLost and the abnormal-close handler. + const MAX_ENVIRONMENT_RECREATIONS = 3 + let environmentRecreations = 0 + let reconnectPromise: Promise | null = null + + /** + * Recover from onEnvironmentLost (poll returned 404 — env was reaped + * server-side). Tries two strategies in order: + * + * 1. Reconnect-in-place: idempotent re-register with reuseEnvironmentId + * → if the backend returns the same env ID, call reconnectSession() + * to re-queue the existing session. currentSessionId stays the same; + * the URL on the user's phone stays valid; previouslyFlushedUUIDs is + * preserved so history isn't re-sent. + * + * 2. Fresh session fallback: if the backend returns a different env ID + * (original TTL-expired, e.g. laptop slept >4h) or reconnectSession() + * throws, archive the old session and create a new one on the + * now-registered env. Old behavior before #20460 primitives landed. + * + * Uses a promise-based reentrancy guard so concurrent callers share the + * same reconnection attempt. + */ + async function reconnectEnvironmentWithSession(): Promise { + if (reconnectPromise) { + return reconnectPromise + } + reconnectPromise = doReconnect() + try { + return await reconnectPromise + } finally { + reconnectPromise = null + } + } + + async function doReconnect(): Promise { + environmentRecreations++ + // Invalidate any in-flight v2 handshake — the environment is being + // recreated, so a stale transport arriving post-reconnect would be + // pointed at a dead session. + v2Generation++ + logForDebugging( + `[bridge:repl] Reconnecting after env lost (attempt ${environmentRecreations}/${MAX_ENVIRONMENT_RECREATIONS})`, + ) + + if (environmentRecreations > MAX_ENVIRONMENT_RECREATIONS) { + logForDebugging( + `[bridge:repl] Environment reconnect limit reached (${MAX_ENVIRONMENT_RECREATIONS}), giving up`, + ) + return false + } + + // Close the stale transport. Capture seq BEFORE close — if Strategy 1 + // (tryReconnectInPlace) succeeds we keep the SAME session, and the + // next transport must resume where this one left off, not replay from + // the last transport-swap checkpoint. + if (transport) { + const seq = transport.getLastSequenceNum() + if (seq > lastTransportSequenceNum) { + lastTransportSequenceNum = seq + } + transport.close() + transport = null + } + // Transport is gone — wake the poll loop out of its at-capacity + // heartbeat sleep so it can fast-poll for re-dispatched work. + wakePollLoop() + // Reset flush gate so writeMessages() hits the !transport guard + // instead of silently queuing into a dead buffer. + flushGate.drop() + + // Release the current work item (force=false — we may want the session + // back). Best-effort: the env is probably gone, so this likely 404s. + if (currentWorkId) { + const workIdBeingCleared = currentWorkId + await api + .stopWork(environmentId, workIdBeingCleared, false) + .catch(() => {}) + // When doReconnect runs concurrently with the poll loop (ws_closed + // handler case — void-called, unlike the awaited onEnvironmentLost + // path), onWorkReceived can fire during the stopWork await and set + // a fresh currentWorkId. If it did, the poll loop has already + // recovered on its own — defer to it rather than proceeding to + // archiveSession, which would destroy the session its new + // transport is connected to. + if (currentWorkId !== workIdBeingCleared) { + logForDebugging( + '[bridge:repl] Poll loop recovered during stopWork await — deferring to it', + ) + environmentRecreations = 0 + return true + } + currentWorkId = null + currentIngressToken = null + } + + // Bail out if teardown started while we were awaiting + if (pollController.signal.aborted) { + logForDebugging('[bridge:repl] Reconnect aborted by teardown') + return false + } + + // Strategy 1: idempotent re-register with the server-issued env ID. + // If the backend resurrects the same env (fresh secret), we can + // reconnect the existing session. If it hands back a different ID, the + // original env is truly gone and we fall through to a fresh session. + const requestedEnvId = environmentId + bridgeConfig.reuseEnvironmentId = requestedEnvId + try { + const reg = await api.registerBridgeEnvironment(bridgeConfig) + environmentId = reg.environment_id + environmentSecret = reg.environment_secret + } catch (err) { + bridgeConfig.reuseEnvironmentId = undefined + logForDebugging( + `[bridge:repl] Environment re-registration failed: ${errorMessage(err)}`, + ) + return false + } + // Clear before any await — a stale value would poison the next fresh + // registration if doReconnect runs again. + bridgeConfig.reuseEnvironmentId = undefined + + logForDebugging( + `[bridge:repl] Re-registered: requested=${requestedEnvId} got=${environmentId}`, + ) + + // Bail out if teardown started while we were registering + if (pollController.signal.aborted) { + logForDebugging( + '[bridge:repl] Reconnect aborted after env registration, cleaning up', + ) + await api.deregisterEnvironment(environmentId).catch(() => {}) + return false + } + + // Same race as above, narrower window: poll loop may have set up a + // transport during the registerBridgeEnvironment await. Bail before + // tryReconnectInPlace/archiveSession kill it server-side. + if (transport !== null) { + logForDebugging( + '[bridge:repl] Poll loop recovered during registerBridgeEnvironment await — deferring to it', + ) + environmentRecreations = 0 + return true + } + + // Strategy 1: same helper as perpetual init. currentSessionId stays + // the same on success; URL on mobile/web stays valid; + // previouslyFlushedUUIDs preserved (no re-flush). + if (await tryReconnectInPlace(requestedEnvId, currentSessionId)) { + logEvent('tengu_bridge_repl_reconnected_in_place', {}) + environmentRecreations = 0 + return true + } + // Env differs → TTL-expired/reaped; or reconnect failed. + // Don't deregister — we have a fresh secret for this env either way. + if (environmentId !== requestedEnvId) { + logEvent('tengu_bridge_repl_env_expired_fresh_session', {}) + } + + // Strategy 2: fresh session on the now-registered environment. + // Archive the old session first — it's orphaned (bound to a dead env, + // or reconnectSession rejected it). Don't deregister the env — we just + // got a fresh secret for it and are about to use it. + await archiveSession(currentSessionId) + + // Bail out if teardown started while we were archiving + if (pollController.signal.aborted) { + logForDebugging( + '[bridge:repl] Reconnect aborted after archive, cleaning up', + ) + await api.deregisterEnvironment(environmentId).catch(() => {}) + return false + } + + // Re-read the current title in case the user renamed the session. + // REPL wrapper reads session storage; daemon wrapper returns the + // original title (nothing to refresh). + const currentTitle = getCurrentTitle() + + // Create a new session on the now-registered environment + const newSessionId = await createSession({ + environmentId, + title: currentTitle, + gitRepoUrl, + branch, + signal: AbortSignal.timeout(15_000), + }) + + if (!newSessionId) { + logForDebugging( + '[bridge:repl] Session creation failed during reconnection', + ) + return false + } + + // Bail out if teardown started during session creation (up to 15s) + if (pollController.signal.aborted) { + logForDebugging( + '[bridge:repl] Reconnect aborted after session creation, cleaning up', + ) + await archiveSession(newSessionId) + return false + } + + currentSessionId = newSessionId + // Re-publish to the PID file so peer dedup (peerRegistry.ts) picks up the + // new ID — setReplBridgeHandle only fires at init/teardown, not reconnect. + void updateSessionBridgeId(toCompatSessionId(newSessionId)).catch(() => {}) + // Reset per-session transport state IMMEDIATELY after the session swap, + // before any await. If this runs after `await writeBridgePointer` below, + // there's a window where handle.bridgeSessionId already returns session B + // but getSSESequenceNum() still returns session A's seq — a daemon + // persistState() in that window writes {bridgeSessionId: B, seq: OLD_A}, + // which PASSES the session-ID validation check and defeats it entirely. + // + // The SSE seq-num is scoped to the session's event stream — carrying it + // over leaves the transport's lastSequenceNum stuck high (seq only + // advances when received > last), and its next internal reconnect would + // send from_sequence_num=OLD_SEQ against a stream starting at 1 → all + // events in the gap silently dropped. Inbound UUID dedup is also + // session-scoped. + lastTransportSequenceNum = 0 + recentInboundUUIDs.clear() + // Title derivation is session-scoped too: if the user typed during the + // createSession await above, the callback fired against the OLD archived + // session ID (PATCH lost) and the new session got `currentTitle` captured + // BEFORE they typed. Reset so the next prompt can re-derive. Self- + // correcting: if the caller's policy is already done (explicit title or + // count ≥ 3), it returns true on the first post-reset call and re-latches. + userMessageCallbackDone = !onUserMessage + logForDebugging(`[bridge:repl] Re-created session: ${currentSessionId}`) + + // Rewrite the crash-recovery pointer with the new IDs so a crash after + // this point resumes the right session. (The reconnect-in-place path + // above doesn't touch the pointer — same session, same env.) + await writeBridgePointer(dir, { + sessionId: currentSessionId, + environmentId, + source: 'repl', + }) + + // Clear flushed UUIDs so initial messages are re-sent to the new session. + // UUIDs are scoped per-session on the server, so re-flushing is safe. + previouslyFlushedUUIDs?.clear() + + + // Reset the counter so independent reconnections hours apart don't + // exhaust the limit — it guards against rapid consecutive failures, + // not lifetime total. + environmentRecreations = 0 + + return true + } + + // Helper: get the current OAuth access token for session ingress auth. + // Unlike the JWT path, OAuth tokens are refreshed by the standard OAuth + // flow — no proactive scheduler needed. + function getOAuthToken(): string | undefined { + return getAccessToken() + } + + // Drain any messages that were queued during the initial flush. + // Called after writeBatch completes (or fails) so queued messages + // are sent in order after the historical messages. + function drainFlushGate(): void { + const msgs = flushGate.end() + if (msgs.length === 0) return + if (!transport) { + logForDebugging( + `[bridge:repl] Cannot drain ${msgs.length} pending message(s): no transport`, + ) + return + } + for (const msg of msgs) { + recentPostedUUIDs.add(msg.uuid) + } + const sdkMessages = toSDKMessages(msgs) + const events = sdkMessages.map(sdkMsg => ({ + ...sdkMsg, + session_id: currentSessionId, + })) + logForDebugging( + `[bridge:repl] Drained ${msgs.length} pending message(s) after flush`, + ) + void transport.writeBatch(events) + } + + // Teardown reference — set after definition below. All callers are async + // callbacks that run after assignment, so the reference is always valid. + let doTeardownImpl: (() => Promise) | null = null + function triggerTeardown(): void { + void doTeardownImpl?.() + } + + /** + * Body of the transport's setOnClose callback, hoisted to initBridgeCore + * scope so /bridge-kick can fire it directly. setOnClose wraps this with + * a stale-transport guard; debugFireClose calls it bare. + * + * With autoReconnect:true, this only fires on: clean close (1000), + * permanent server rejection (4001/1002/4003), or 10-min budget + * exhaustion. Transient drops are retried internally by the transport. + */ + function handleTransportPermanentClose(closeCode: number | undefined): void { + logForDebugging( + `[bridge:repl] Transport permanently closed: code=${closeCode}`, + ) + logEvent('tengu_bridge_repl_ws_closed', { + code: closeCode, + }) + // Capture SSE seq high-water mark before nulling. When called from + // setOnClose the guard guarantees transport !== null; when fired from + // /bridge-kick it may already be null (e.g. fired twice) — skip. + if (transport) { + const closedSeq = transport.getLastSequenceNum() + if (closedSeq > lastTransportSequenceNum) { + lastTransportSequenceNum = closedSeq + } + transport = null + } + // Transport is gone — wake the poll loop out of its at-capacity + // heartbeat sleep so it's fast-polling by the time the reconnect + // below completes and the server re-queues work. + wakePollLoop() + // Reset flush state so writeMessages() hits the !transport guard + // (with a warning log) instead of silently queuing into a buffer + // that will never be drained. Unlike onWorkReceived (which + // preserves pending messages for the new transport), onClose is + // a permanent close — no new transport will drain these. + const dropped = flushGate.drop() + if (dropped > 0) { + logForDebugging( + `[bridge:repl] Dropping ${dropped} pending message(s) on transport close (code=${closeCode})`, + { level: 'warn' }, + ) + } + + if (closeCode === 1000) { + // Clean close — session ended normally. Tear down the bridge. + onStateChange?.('failed', 'session ended') + pollController.abort() + triggerTeardown() + return + } + + // Transport reconnect budget exhausted or permanent server + // rejection. By this point the env has usually been reaped + // server-side (BQ 2026-03-12: ~98% of ws_closed never recover + // via poll alone). stopWork(force=false) can't re-dispatch work + // from an archived env; reconnectEnvironmentWithSession can + // re-activate it via POST /bridge/reconnect, or fall through + // to a fresh session if the env is truly gone. The poll loop + // (already woken above) picks up the re-queued work once + // doReconnect completes. + onStateChange?.( + 'reconnecting', + `Remote Control connection lost (code ${closeCode})`, + ) + logForDebugging( + `[bridge:repl] Transport reconnect budget exhausted (code=${closeCode}), attempting env reconnect`, + ) + void reconnectEnvironmentWithSession().then(success => { + if (success) return + // doReconnect has four abort-check return-false sites for + // teardown-in-progress. Don't pollute the BQ failure signal + // or double-teardown when the user just quit. + if (pollController.signal.aborted) return + // doReconnect returns false (never throws) on genuine failure. + // The dangerous case: registerBridgeEnvironment succeeded (so + // environmentId now points at a fresh valid env) but + // createSession failed — poll loop would poll a sessionless + // env getting null work with no errors, never hitting any + // give-up path. Tear down explicitly. + logForDebugging( + '[bridge:repl] reconnectEnvironmentWithSession resolved false — tearing down', + ) + logEvent('tengu_bridge_repl_reconnect_failed', { + close_code: closeCode, + }) + onStateChange?.('failed', 'reconnection failed') + triggerTeardown() + }) + } + + // Ant-only: SIGUSR2 → force doReconnect() for manual testing. Skips the + // ~30s poll wait — fire-and-observe in the debug log immediately. + // Windows has no USR signals; `process.on` would throw there. + let sigusr2Handler: (() => void) | undefined + if (process.env.USER_TYPE === 'ant' && process.platform !== 'win32') { + sigusr2Handler = () => { + logForDebugging( + '[bridge:repl] SIGUSR2 received — forcing doReconnect() for testing', + ) + void reconnectEnvironmentWithSession() + } + process.on('SIGUSR2', sigusr2Handler) + } + + // Ant-only: /bridge-kick fault injection. handleTransportPermanentClose + // is defined below and assigned into this slot so the slash command can + // invoke it directly — the real setOnClose callback is buried inside + // wireTransport which is itself inside onWorkReceived. + let debugFireClose: ((code: number) => void) | null = null + if (process.env.USER_TYPE === 'ant') { + registerBridgeDebugHandle({ + fireClose: code => { + if (!debugFireClose) { + logForDebugging('[bridge:debug] fireClose: no transport wired yet') + return + } + logForDebugging(`[bridge:debug] fireClose(${code}) — injecting`) + debugFireClose(code) + }, + forceReconnect: () => { + logForDebugging('[bridge:debug] forceReconnect — injecting') + void reconnectEnvironmentWithSession() + }, + injectFault: injectBridgeFault, + wakePollLoop, + describe: () => + `env=${environmentId} session=${currentSessionId} transport=${transport?.getStateLabel() ?? 'null'} workId=${currentWorkId ?? 'null'}`, + }) + } + + const pollOpts = { + api, + getCredentials: () => ({ environmentId, environmentSecret }), + signal: pollController.signal, + getPollIntervalConfig, + onStateChange, + getWsState: () => transport?.getStateLabel() ?? 'null', + // REPL bridge is single-session: having any transport == at capacity. + // No need to check isConnectedStatus() — even while the transport is + // auto-reconnecting internally (up to 10 min), poll is heartbeat-only. + isAtCapacity: () => transport !== null, + capacitySignal, + onFatalError: triggerTeardown, + getHeartbeatInfo: () => { + if (!currentWorkId || !currentIngressToken) { + return null + } + return { + environmentId, + workId: currentWorkId, + sessionToken: currentIngressToken, + } + }, + // Work-item JWT expired (or work gone). The transport is useless — + // SSE reconnects and CCR writes use the same stale token. Without + // this callback the poll loop would do a 10-min at-capacity backoff, + // during which the work lease (300s TTL) expires and the server stops + // forwarding prompts → ~25-min dead window observed in daemon logs. + // Kill the transport + work state so isAtCapacity()=false; the loop + // fast-polls and picks up the server's re-dispatched work in seconds. + onHeartbeatFatal: (err: BridgeFatalError) => { + logForDebugging( + `[bridge:repl] heartbeatWork fatal (status=${err.status}) — tearing down work item for fast re-dispatch`, + ) + if (transport) { + const seq = transport.getLastSequenceNum() + if (seq > lastTransportSequenceNum) { + lastTransportSequenceNum = seq + } + transport.close() + transport = null + } + flushGate.drop() + // force=false → server re-queues. Likely already expired, but + // idempotent and makes re-dispatch immediate if not. + if (currentWorkId) { + void api + .stopWork(environmentId, currentWorkId, false) + .catch((e: unknown) => { + logForDebugging( + `[bridge:repl] stopWork after heartbeat fatal: ${errorMessage(e)}`, + ) + }) + } + currentWorkId = null + currentIngressToken = null + wakePollLoop() + onStateChange?.( + 'reconnecting', + 'Work item lease expired, fetching fresh token', + ) + }, + async onEnvironmentLost() { + const success = await reconnectEnvironmentWithSession() + if (!success) { + return null + } + return { environmentId, environmentSecret } + }, + onWorkReceived: ( + workSessionId: string, + ingressToken: string, + workId: string, + serverUseCcrV2: boolean, + ) => { + // When new work arrives while a transport is already open, the + // server has decided to re-dispatch (e.g. token rotation, server + // restart). Close the existing transport and reconnect — discarding + // the work causes a stuck 'reconnecting' state if the old WS dies + // shortly after (the server won't re-dispatch a work item it + // already delivered). + // ingressToken (JWT) is stored for heartbeat auth (both v1 and v2). + // Transport auth diverges — see the v1/v2 split below. + if (transport?.isConnectedStatus()) { + logForDebugging( + `[bridge:repl] Work received while transport connected, replacing with fresh token (workId=${workId})`, + ) + } + + logForDebugging( + `[bridge:repl] Work received: workId=${workId} workSessionId=${workSessionId} currentSessionId=${currentSessionId} match=${sameSessionId(workSessionId, currentSessionId)}`, + ) + + // Refresh the crash-recovery pointer's mtime. Staleness checks file + // mtime (not embedded timestamp) so this re-write bumps the clock — + // a 5h+ session that crashes still has a fresh pointer. Fires once + // per work dispatch (infrequent — bounded by user message rate). + void writeBridgePointer(dir, { + sessionId: currentSessionId, + environmentId, + source: 'repl', + }) + + // Reject foreign session IDs — the server shouldn't assign sessions + // from other environments. Since we create env+session as a pair, + // a mismatch indicates an unexpected server-side reassignment. + // + // Compare by underlying UUID, not by tagged-ID prefix. When CCR + // v2's compat layer serves the session, createBridgeSession gets + // session_* from the v1-facing API (compat/convert.go:41) but the + // infrastructure layer delivers cse_* in the work queue + // (container_manager.go:129). Same UUID, different tag. + if (!sameSessionId(workSessionId, currentSessionId)) { + logForDebugging( + `[bridge:repl] Rejecting foreign session: expected=${currentSessionId} got=${workSessionId}`, + ) + return + } + + currentWorkId = workId + currentIngressToken = ingressToken + + // Server decides per-session (secret.use_code_sessions from the work + // secret, threaded through runWorkPollLoop). The env var is an ant-dev + // override for forcing v2 before the server flag is on for your user — + // requires ccr_v2_compat_enabled server-side or registerWorker 404s. + // + // Kept separate from CLAUDE_CODE_USE_CCR_V2 (the child-SDK transport + // selector set by sessionRunner/environment-manager) to avoid the + // inheritance hazard in spawn mode where the parent's orchestrator + // var would leak into a v1 child. + const useCcrV2 = + serverUseCcrV2 || isEnvTruthy(process.env.CLAUDE_BRIDGE_USE_CCR_V2) + + // Auth is the one place v1 and v2 diverge hard: + // + // - v1 (Session-Ingress): accepts OAuth OR JWT. We prefer OAuth + // because the standard OAuth refresh flow handles expiry — no + // separate JWT refresh scheduler needed. + // + // - v2 (CCR /worker/*): REQUIRES the JWT. register_worker.go:32 + // validates the session_id claim, which OAuth tokens don't carry. + // The JWT from the work secret has both that claim and the worker + // role (environment_auth.py:856). JWT refresh: when it expires the + // server re-dispatches work with a fresh one, and onWorkReceived + // fires again. createV2ReplTransport stores it via + // updateSessionIngressAuthToken() before touching the network. + let v1OauthToken: string | undefined + if (!useCcrV2) { + v1OauthToken = getOAuthToken() + if (!v1OauthToken) { + logForDebugging( + '[bridge:repl] No OAuth token available for session ingress, skipping work', + ) + return + } + updateSessionIngressAuthToken(v1OauthToken) + } + logEvent('tengu_bridge_repl_work_received', {}) + + // Close the previous transport. Nullify BEFORE calling close() so + // the close callback doesn't treat the programmatic close as + // "session ended normally" and trigger a full teardown. + if (transport) { + const oldTransport = transport + transport = null + // Capture the SSE sequence high-water mark so the next transport + // resumes the stream instead of replaying from seq 0. Use max() — + // a transport that died early (never received any frames) would + // otherwise reset a non-zero mark back to 0. + const oldSeq = oldTransport.getLastSequenceNum() + if (oldSeq > lastTransportSequenceNum) { + lastTransportSequenceNum = oldSeq + } + oldTransport.close() + } + // Reset flush state — the old flush (if any) is no longer relevant. + // Preserve pending messages so they're drained after the new + // transport's flush completes (the hook has already advanced its + // lastWrittenIndex and won't re-send them). + flushGate.deactivate() + + // Closure adapter over the shared handleServerControlRequest — + // captures transport/currentSessionId so the transport.setOnData + // callback below doesn't need to thread them through. + const onServerControlRequest = (request: SDKControlRequest): void => + handleServerControlRequest(request, { + transport, + sessionId: currentSessionId, + onInterrupt, + onSetModel, + onSetMaxThinkingTokens, + onSetPermissionMode, + }) + + let initialFlushDone = false + + // Wire callbacks onto a freshly constructed transport and connect. + // Extracted so the (sync) v1 and (async) v2 construction paths can + // share the identical callback + flush machinery. + const wireTransport = (newTransport: ReplBridgeTransport): void => { + transport = newTransport + + newTransport.setOnConnect(() => { + // Guard: if transport was replaced by a newer onWorkReceived call + // while the WS was connecting, ignore this stale callback. + if (transport !== newTransport) return + + logForDebugging('[bridge:repl] Ingress transport connected') + logEvent('tengu_bridge_repl_ws_connected', {}) + + // Update the env var with the latest OAuth token so POST writes + // (which read via getSessionIngressAuthToken()) use a fresh token. + // v2 skips this — createV2ReplTransport already stored the JWT, + // and overwriting it with OAuth would break subsequent /worker/* + // requests (session_id claim check). + if (!useCcrV2) { + const freshToken = getOAuthToken() + if (freshToken) { + updateSessionIngressAuthToken(freshToken) + } + } + + // Reset teardownStarted so future teardowns are not blocked. + teardownStarted = false + + // Flush initial messages only on first connect, not on every + // WS reconnection. Re-flushing would cause duplicate messages. + // IMPORTANT: onStateChange('connected') is deferred until the + // flush completes. This prevents writeMessages() from sending + // new messages that could arrive at the server interleaved with + // the historical messages, and delays the web UI from showing + // the session as active until history is persisted. + if ( + !initialFlushDone && + initialMessages && + initialMessages.length > 0 + ) { + initialFlushDone = true + + // Cap the initial flush to the most recent N messages. The full + // history is UI-only (model doesn't see it) and large replays cause + // slow session-ingress persistence (each event is a threadstore write) + // plus elevated Firestore pressure. A 0 or negative cap disables it. + const historyCap = initialHistoryCap + const eligibleMessages = initialMessages.filter( + m => + isEligibleBridgeMessage(m) && + !previouslyFlushedUUIDs?.has(m.uuid), + ) + const cappedMessages = + historyCap > 0 && eligibleMessages.length > historyCap + ? eligibleMessages.slice(-historyCap) + : eligibleMessages + if (cappedMessages.length < eligibleMessages.length) { + logForDebugging( + `[bridge:repl] Capped initial flush: ${eligibleMessages.length} -> ${cappedMessages.length} (cap=${historyCap})`, + ) + logEvent('tengu_bridge_repl_history_capped', { + eligible_count: eligibleMessages.length, + capped_count: cappedMessages.length, + }) + } + const sdkMessages = toSDKMessages(cappedMessages) + if (sdkMessages.length > 0) { + logForDebugging( + `[bridge:repl] Flushing ${sdkMessages.length} initial message(s) via transport`, + ) + const events = sdkMessages.map(sdkMsg => ({ + ...sdkMsg, + session_id: currentSessionId, + })) + const dropsBefore = newTransport.droppedBatchCount + void newTransport + .writeBatch(events) + .then(() => { + // If any batch was dropped during this flush (SI down for + // maxConsecutiveFailures attempts), flush() still resolved + // normally but the events were NOT delivered. Don't mark + // UUIDs as flushed — keep them eligible for re-send on the + // next onWorkReceived (JWT refresh re-dispatch, line ~1144). + if (newTransport.droppedBatchCount > dropsBefore) { + logForDebugging( + `[bridge:repl] Initial flush dropped ${newTransport.droppedBatchCount - dropsBefore} batch(es) — not marking ${sdkMessages.length} UUID(s) as flushed`, + ) + return + } + if (previouslyFlushedUUIDs) { + for (const sdkMsg of sdkMessages) { + if (sdkMsg.uuid) { + previouslyFlushedUUIDs.add(sdkMsg.uuid) + } + } + } + }) + .catch(e => + logForDebugging(`[bridge:repl] Initial flush failed: ${e}`), + ) + .finally(() => { + // Guard: if transport was replaced during the flush, + // don't signal connected or drain — the new transport + // owns the lifecycle now. + if (transport !== newTransport) return + drainFlushGate() + onStateChange?.('connected') + }) + } else { + // All initial messages were already flushed (filtered by + // previouslyFlushedUUIDs). No flush POST needed — clear + // the flag and signal connected immediately. This is the + // first connect for this transport (inside !initialFlushDone), + // so no flush POST is in-flight — the flag was set before + // connect() and must be cleared here. + drainFlushGate() + onStateChange?.('connected') + } + } else if (!flushGate.active) { + // No initial messages or already flushed on first connect. + // WS auto-reconnect path — only signal connected if no flush + // POST is in-flight. If one is, .finally() owns the lifecycle. + onStateChange?.('connected') + } + }) + + newTransport.setOnData(data => { + handleIngressMessage( + data, + recentPostedUUIDs, + recentInboundUUIDs, + onInboundMessage, + onPermissionResponse, + onServerControlRequest, + ) + }) + + // Body lives at initBridgeCore scope so /bridge-kick can call it + // directly via debugFireClose. All referenced closures (transport, + // wakePollLoop, flushGate, reconnectEnvironmentWithSession, etc.) + // are already at that scope. The only lexical dependency on + // wireTransport was `newTransport.getLastSequenceNum()` — but after + // the guard below passes we know transport === newTransport. + debugFireClose = handleTransportPermanentClose + newTransport.setOnClose(closeCode => { + // Guard: if transport was replaced, ignore stale close. + if (transport !== newTransport) return + handleTransportPermanentClose(closeCode) + }) + + // Start the flush gate before connect() to cover the WS handshake + // window. Between transport assignment and setOnConnect firing, + // writeMessages() could send messages via HTTP POST before the + // initial flush starts. Starting the gate here ensures those + // calls are queued. If there are no initial messages, the gate + // stays inactive. + if ( + !initialFlushDone && + initialMessages && + initialMessages.length > 0 + ) { + flushGate.start() + } + + newTransport.connect() + } // end wireTransport + + // Bump unconditionally — ANY new transport (v1 or v2) invalidates an + // in-flight v2 handshake. Also bumped in doReconnect(). + v2Generation++ + + if (useCcrV2) { + // workSessionId is the cse_* form (infrastructure-layer ID from the + // work queue), which is what /v1/code/sessions/{id}/worker/* wants. + // The session_* form (currentSessionId) is NOT usable here — + // handler/convert.go:30 validates TagCodeSession. + const sessionUrl = buildCCRv2SdkUrl(baseUrl, workSessionId) + const thisGen = v2Generation + logForDebugging( + `[bridge:repl] CCR v2: sessionUrl=${sessionUrl} session=${workSessionId} gen=${thisGen}`, + ) + void createV2ReplTransport({ + sessionUrl, + ingressToken, + sessionId: workSessionId, + initialSequenceNum: lastTransportSequenceNum, + }).then( + t => { + // Teardown started while registerWorker was in flight. Teardown + // saw transport === null and skipped close(); installing now + // would leak CCRClient heartbeat timers and reset + // teardownStarted via wireTransport's side effects. + if (pollController.signal.aborted) { + t.close() + return + } + // onWorkReceived may have fired again while registerWorker() + // was in flight (server re-dispatch with a fresh JWT). The + // transport !== null check alone gets the race wrong when BOTH + // attempts saw transport === null — it keeps the first resolver + // (stale epoch) and discards the second (correct epoch). The + // generation check catches it regardless of transport state. + if (thisGen !== v2Generation) { + logForDebugging( + `[bridge:repl] CCR v2: discarding stale handshake gen=${thisGen} current=${v2Generation}`, + ) + t.close() + return + } + wireTransport(t) + }, + (err: unknown) => { + logForDebugging( + `[bridge:repl] CCR v2: createV2ReplTransport failed: ${errorMessage(err)}`, + { level: 'error' }, + ) + logEvent('tengu_bridge_repl_ccr_v2_init_failed', {}) + // If a newer attempt is in flight or already succeeded, don't + // touch its work item — our failure is irrelevant. + if (thisGen !== v2Generation) return + // Release the work item so the server re-dispatches immediately + // instead of waiting for its own timeout. currentWorkId was set + // above; without this, the session looks stuck to the user. + if (currentWorkId) { + void api + .stopWork(environmentId, currentWorkId, false) + .catch((e: unknown) => { + logForDebugging( + `[bridge:repl] stopWork after v2 init failure: ${errorMessage(e)}`, + ) + }) + currentWorkId = null + currentIngressToken = null + } + wakePollLoop() + }, + ) + } else { + // v1: HybridTransport (WS reads + POST writes to Session-Ingress). + // autoReconnect is true (default) — when the WS dies, the transport + // reconnects automatically with exponential backoff. POST writes + // continue during reconnection (they use getSessionIngressAuthToken() + // independently of WS state). The poll loop remains as a secondary + // fallback if the reconnect budget is exhausted (10 min). + // + // Auth: uses OAuth tokens directly instead of the JWT from the work + // secret. refreshHeaders picks up the latest OAuth token on each + // WS reconnect attempt. + const wsUrl = buildSdkUrl(sessionIngressUrl, workSessionId) + logForDebugging(`[bridge:repl] Ingress URL: ${wsUrl}`) + logForDebugging( + `[bridge:repl] Creating HybridTransport: session=${workSessionId}`, + ) + // v1OauthToken was validated non-null above (we'd have returned early). + const oauthToken = v1OauthToken ?? '' + wireTransport( + createV1ReplTransport( + new HybridTransport( + new URL(wsUrl), + { + Authorization: `Bearer ${oauthToken}`, + 'anthropic-version': '2023-06-01', + }, + workSessionId, + () => ({ + Authorization: `Bearer ${getOAuthToken() ?? oauthToken}`, + 'anthropic-version': '2023-06-01', + }), + // Cap retries so a persistently-failing session-ingress can't + // pin the uploader drain loop for the lifetime of the bridge. + // 50 attempts ≈ 20 min (15s POST timeout + 8s backoff + jitter + // per cycle at steady state). Bridge-only — 1P keeps indefinite. + { + maxConsecutiveFailures: 50, + isBridge: true, + onBatchDropped: () => { + onStateChange?.( + 'reconnecting', + 'Lost sync with Remote Control — events could not be delivered', + ) + // SI has been down ~20 min. Wake the poll loop so that when + // SI recovers, next poll → onWorkReceived → fresh transport + // → initial flush succeeds → onStateChange('connected') at + // ~line 1420. Without this, state stays 'reconnecting' even + // after SI recovers — daemon.ts:437 denies all permissions, + // useReplBridge.ts:311 keeps replBridgeSessionActive=false. + // If the env was archived during the outage, poll 404 → + // onEnvironmentLost recovery path handles it. + wakePollLoop() + }, + }, + ), + ), + ) + } + }, + } + void startWorkPollLoop(pollOpts) + + // Perpetual mode: hourly mtime refresh of the crash-recovery pointer. + // The onWorkReceived refresh only fires per user prompt — a + // daemon idle for >4h would have a stale pointer, and the next restart + // would clear it (readBridgePointer TTL check) → fresh session. The + // standalone bridge (bridgeMain.ts) has an identical hourly timer. + const pointerRefreshTimer = perpetual + ? setInterval(() => { + // doReconnect() reassigns currentSessionId/environmentId non- + // atomically (env at ~:634, session at ~:719, awaits in between). + // If this timer fires in that window, its fire-and-forget write can + // race with (and overwrite) doReconnect's own pointer write at ~:740, + // leaving the pointer at the now-archived old session. doReconnect + // writes the pointer itself, so skipping here is free. + if (reconnectPromise) return + void writeBridgePointer(dir, { + sessionId: currentSessionId, + environmentId, + source: 'repl', + }) + }, 60 * 60_000) + : null + pointerRefreshTimer?.unref?.() + + // Push a silent keep_alive frame on a fixed interval so upstream proxies + // and the session-ingress layer don't GC an otherwise-idle remote control + // session. The keep_alive type is filtered before reaching any client UI + // (Query.ts drops it; web/iOS/Android never see it in their message loop). + // Interval comes from GrowthBook (tengu_bridge_poll_interval_config + // session_keepalive_interval_v2_ms, default 120s); 0 = disabled. + const keepAliveIntervalMs = + getPollIntervalConfig().session_keepalive_interval_v2_ms + const keepAliveTimer = + keepAliveIntervalMs > 0 + ? setInterval(() => { + if (!transport) return + logForDebugging('[bridge:repl] keep_alive sent') + void transport.write({ type: 'keep_alive' }).catch((err: unknown) => { + logForDebugging( + `[bridge:repl] keep_alive write failed: ${errorMessage(err)}`, + ) + }) + }, keepAliveIntervalMs) + : null + keepAliveTimer?.unref?.() + + // Shared teardown sequence used by both cleanup registration and + // the explicit teardown() method on the returned handle. + let teardownStarted = false + doTeardownImpl = async (): Promise => { + if (teardownStarted) { + logForDebugging( + `[bridge:repl] Teardown already in progress, skipping duplicate call env=${environmentId} session=${currentSessionId}`, + ) + return + } + teardownStarted = true + const teardownStart = Date.now() + logForDebugging( + `[bridge:repl] Teardown starting: env=${environmentId} session=${currentSessionId} workId=${currentWorkId ?? 'none'} transportState=${transport?.getStateLabel() ?? 'null'}`, + ) + + if (pointerRefreshTimer !== null) { + clearInterval(pointerRefreshTimer) + } + if (keepAliveTimer !== null) { + clearInterval(keepAliveTimer) + } + if (sigusr2Handler) { + process.off('SIGUSR2', sigusr2Handler) + } + if (process.env.USER_TYPE === 'ant') { + clearBridgeDebugHandle() + debugFireClose = null + } + pollController.abort() + logForDebugging('[bridge:repl] Teardown: poll loop aborted') + + // Capture the live transport's seq BEFORE close() — close() is sync + // (just aborts the SSE fetch) and does NOT invoke onClose, so the + // setOnClose capture path never runs for explicit teardown. + // Without this, getSSESequenceNum() after teardown returns the stale + // lastTransportSequenceNum (captured at the last transport swap), and + // daemon callers persisting that value lose all events since then. + if (transport) { + const finalSeq = transport.getLastSequenceNum() + if (finalSeq > lastTransportSequenceNum) { + lastTransportSequenceNum = finalSeq + } + } + + if (perpetual) { + // Perpetual teardown is LOCAL-ONLY — do not send result, do not call + // stopWork, do not close the transport. All of those signal the + // server (and any mobile/attach subscribers) that the session is + // ending. Instead: stop polling, let the socket die with the + // process; the backend times the work-item lease back to pending on + // its own (TTL 300s). Next daemon start reads the pointer and + // reconnectSession re-queues work. + transport = null + flushGate.drop() + // Refresh the pointer mtime so that sessions lasting longer than + // BRIDGE_POINTER_TTL_MS (4h) don't appear stale on next start. + await writeBridgePointer(dir, { + sessionId: currentSessionId, + environmentId, + source: 'repl', + }) + logForDebugging( + `[bridge:repl] Teardown (perpetual): leaving env=${environmentId} session=${currentSessionId} alive on server, duration=${Date.now() - teardownStart}ms`, + ) + return + } + + // Fire the result message, then archive, THEN close. transport.write() + // only enqueues (SerialBatchEventUploader resolves on buffer-add); the + // stopWork/archive latency (~200-500ms) is the drain window for the + // result POST. Closing BEFORE archive meant relying on HybridTransport's + // void-ed 3s grace period, which nothing awaits — forceExit can kill the + // socket mid-POST. Same reorder as remoteBridgeCore.ts teardown (#22803). + const teardownTransport = transport + transport = null + flushGate.drop() + if (teardownTransport) { + void teardownTransport.write(makeResultMessage(currentSessionId)) + } + + const stopWorkP = currentWorkId + ? api + .stopWork(environmentId, currentWorkId, true) + .then(() => { + logForDebugging('[bridge:repl] Teardown: stopWork completed') + }) + .catch((err: unknown) => { + logForDebugging( + `[bridge:repl] Teardown stopWork failed: ${errorMessage(err)}`, + ) + }) + : Promise.resolve() + + // Run stopWork and archiveSession in parallel. gracefulShutdown.ts:407 + // races runCleanupFunctions() against 2s (NOT the 5s outer failsafe), + // so archive is capped at 1.5s at the injection site to stay under budget. + // archiveSession is contractually no-throw; the injected implementations + // log their own success/failure internally. + await Promise.all([stopWorkP, archiveSession(currentSessionId)]) + + teardownTransport?.close() + logForDebugging('[bridge:repl] Teardown: transport closed') + + await api.deregisterEnvironment(environmentId).catch((err: unknown) => { + logForDebugging( + `[bridge:repl] Teardown deregister failed: ${errorMessage(err)}`, + ) + }) + + // Clear the crash-recovery pointer — explicit disconnect or clean REPL + // exit means the user is done with this session. Crash/kill-9 never + // reaches this line, leaving the pointer for next-launch recovery. + await clearBridgePointer(dir) + + logForDebugging( + `[bridge:repl] Teardown complete: env=${environmentId} duration=${Date.now() - teardownStart}ms`, + ) + } + + // 8. Register cleanup for graceful shutdown + const unregister = registerCleanup(() => doTeardownImpl?.()) + + logForDebugging( + `[bridge:repl] Ready: env=${environmentId} session=${currentSessionId}`, + ) + onStateChange?.('ready') + + return { + get bridgeSessionId() { + return currentSessionId + }, + get environmentId() { + return environmentId + }, + getSSESequenceNum() { + // lastTransportSequenceNum only updates when a transport is CLOSED + // (captured at swap/onClose). During normal operation the CURRENT + // transport's live seq isn't reflected there. Merge both so callers + // (e.g. daemon persistState()) get the actual high-water mark. + const live = transport?.getLastSequenceNum() ?? 0 + return Math.max(lastTransportSequenceNum, live) + }, + sessionIngressUrl, + writeMessages(messages) { + // Filter to user/assistant messages that haven't already been sent. + // Two layers of dedup: + // - initialMessageUUIDs: messages sent as session creation events + // - recentPostedUUIDs: messages recently sent via POST + const filtered = messages.filter( + m => + isEligibleBridgeMessage(m) && + !initialMessageUUIDs.has(m.uuid) && + !recentPostedUUIDs.has(m.uuid), + ) + if (filtered.length === 0) return + + // Fire onUserMessage for title derivation. Scan before the flushGate + // check — prompts are title-worthy even if they queue behind the + // initial history flush. Keeps calling on every title-worthy message + // until the callback returns true; the caller owns the policy. + if (!userMessageCallbackDone) { + for (const m of filtered) { + const text = extractTitleText(m) + if (text !== undefined && onUserMessage?.(text, currentSessionId)) { + userMessageCallbackDone = true + break + } + } + } + + // Queue messages while the initial flush is in progress to prevent + // them from arriving at the server interleaved with history. + if (flushGate.enqueue(...filtered)) { + logForDebugging( + `[bridge:repl] Queued ${filtered.length} message(s) during initial flush`, + ) + return + } + + if (!transport) { + const types = filtered.map(m => m.type).join(',') + logForDebugging( + `[bridge:repl] Transport not configured, dropping ${filtered.length} message(s) [${types}] for session=${currentSessionId}`, + { level: 'warn' }, + ) + return + } + + // Track in the bounded ring buffer for echo filtering and dedup. + for (const msg of filtered) { + recentPostedUUIDs.add(msg.uuid) + } + + logForDebugging( + `[bridge:repl] Sending ${filtered.length} message(s) via transport`, + ) + + // Convert to SDK format and send via HTTP POST (HybridTransport). + // The web UI receives them via the subscribe WebSocket. + const sdkMessages = toSDKMessages(filtered) + const events = sdkMessages.map(sdkMsg => ({ + ...sdkMsg, + session_id: currentSessionId, + })) + void transport.writeBatch(events) + }, + writeSdkMessages(messages) { + // Daemon path: query() already yields SDKMessage, skip conversion. + // Still run echo dedup (server bounces writes back on the WS). + // No initialMessageUUIDs filter — daemon has no initial messages. + // No flushGate — daemon never starts it (no initial flush). + const filtered = messages.filter( + m => !m.uuid || !recentPostedUUIDs.has(m.uuid), + ) + if (filtered.length === 0) return + if (!transport) { + logForDebugging( + `[bridge:repl] Transport not configured, dropping ${filtered.length} SDK message(s) for session=${currentSessionId}`, + { level: 'warn' }, + ) + return + } + for (const msg of filtered) { + if (msg.uuid) recentPostedUUIDs.add(msg.uuid) + } + const events = filtered.map(m => ({ ...m, session_id: currentSessionId })) + void transport.writeBatch(events) + }, + sendControlRequest(request: SDKControlRequest) { + if (!transport) { + logForDebugging( + '[bridge:repl] Transport not configured, skipping control_request', + ) + return + } + const event = { ...request, session_id: currentSessionId } + void transport.write(event) + logForDebugging( + `[bridge:repl] Sent control_request request_id=${request.request_id}`, + ) + }, + sendControlResponse(response: SDKControlResponse) { + if (!transport) { + logForDebugging( + '[bridge:repl] Transport not configured, skipping control_response', + ) + return + } + const event = { ...response, session_id: currentSessionId } + void transport.write(event) + logForDebugging('[bridge:repl] Sent control_response') + }, + sendControlCancelRequest(requestId: string) { + if (!transport) { + logForDebugging( + '[bridge:repl] Transport not configured, skipping control_cancel_request', + ) + return + } + const event = { + type: 'control_cancel_request' as const, + request_id: requestId, + session_id: currentSessionId, + } + void transport.write(event) + logForDebugging( + `[bridge:repl] Sent control_cancel_request request_id=${requestId}`, + ) + }, + sendResult() { + if (!transport) { + logForDebugging( + `[bridge:repl] sendResult: skipping, transport not configured session=${currentSessionId}`, + ) + return + } + void transport.write(makeResultMessage(currentSessionId)) + logForDebugging( + `[bridge:repl] Sent result for session=${currentSessionId}`, + ) + }, + async teardown() { + unregister() + await doTeardownImpl?.() + logForDebugging('[bridge:repl] Torn down') + logEvent('tengu_bridge_repl_teardown', {}) + }, + } +} + +/** + * Persistent poll loop for work items. Runs in the background for the + * lifetime of the bridge connection. + * + * When a work item arrives, acknowledges it and calls onWorkReceived + * with the session ID and ingress token (which connects the ingress + * WebSocket). Then continues polling — the server will dispatch a new + * work item if the ingress WebSocket drops, allowing automatic + * reconnection without tearing down the bridge. + */ +async function startWorkPollLoop({ + api, + getCredentials, + signal, + onStateChange, + onWorkReceived, + onEnvironmentLost, + getWsState, + isAtCapacity, + capacitySignal, + onFatalError, + getPollIntervalConfig = () => DEFAULT_POLL_CONFIG, + getHeartbeatInfo, + onHeartbeatFatal, +}: { + api: BridgeApiClient + getCredentials: () => { environmentId: string; environmentSecret: string } + signal: AbortSignal + onStateChange?: (state: BridgeState, detail?: string) => void + onWorkReceived: ( + sessionId: string, + ingressToken: string, + workId: string, + useCodeSessions: boolean, + ) => void + /** Called when the environment has been deleted. Returns new credentials or null. */ + onEnvironmentLost?: () => Promise<{ + environmentId: string + environmentSecret: string + } | null> + /** Returns the current WebSocket readyState label for diagnostic logging. */ + getWsState?: () => string + /** + * Returns true when the caller cannot accept new work (transport already + * connected). When true, the loop polls at the configured at-capacity + * interval as a heartbeat only. Server-side BRIDGE_LAST_POLL_TTL is + * 4 hours — anything shorter than that is sufficient for liveness. + */ + isAtCapacity?: () => boolean + /** + * Produces a signal that aborts when capacity frees up (transport lost), + * merged with the loop signal. Used to interrupt the at-capacity sleep + * so recovery polling starts immediately. + */ + capacitySignal?: () => CapacitySignal + /** Called on unrecoverable errors (e.g. server-side expiry) to trigger full teardown. */ + onFatalError?: () => void + /** Poll interval config getter — defaults to DEFAULT_POLL_CONFIG. */ + getPollIntervalConfig?: () => PollIntervalConfig + /** + * Returns the current work ID and session ingress token for heartbeat. + * When null, heartbeat is not possible (no active work item). + */ + getHeartbeatInfo?: () => { + environmentId: string + workId: string + sessionToken: string + } | null + /** + * Called when heartbeatWork throws BridgeFatalError (401/403/404/410 — + * JWT expired or work item gone). Caller should tear down the transport + * + work state so isAtCapacity() flips to false and the loop fast-polls + * for the server's re-dispatched work item. When provided, the loop + * SKIPS the at-capacity backoff sleep (which would otherwise cause a + * ~10-minute dead window before recovery). When omitted, falls back to + * the backoff sleep to avoid a tight poll+heartbeat loop. + */ + onHeartbeatFatal?: (err: BridgeFatalError) => void +}): Promise { + const MAX_ENVIRONMENT_RECREATIONS = 3 + + logForDebugging( + `[bridge:repl] Starting work poll loop for env=${getCredentials().environmentId}`, + ) + + let consecutiveErrors = 0 + let firstErrorTime: number | null = null + let lastPollErrorTime: number | null = null + let environmentRecreations = 0 + // Set when the at-capacity sleep overruns its deadline by a large margin + // (process suspension). Consumed at the top of the next iteration to + // force one fast-poll cycle — isAtCapacity() is `transport !== null`, + // which stays true while the transport auto-reconnects, so the poll + // loop would otherwise go straight back to a 10-minute sleep on a + // transport that may be pointed at a dead socket. + let suspensionDetected = false + + while (!signal.aborted) { + // Capture credentials outside try so the catch block can detect + // whether a concurrent reconnection replaced the environment. + const { environmentId: envId, environmentSecret: envSecret } = + getCredentials() + const pollConfig = getPollIntervalConfig() + try { + const work = await api.pollForWork( + envId, + envSecret, + signal, + pollConfig.reclaim_older_than_ms, + ) + + // A successful poll proves the env is genuinely healthy — reset the + // env-loss counter so events hours apart each start fresh. Outside + // the state-change guard below because onEnvLost's success path + // already emits 'ready'; emitting again here would be a duplicate. + // (onEnvLost returning creds does NOT reset this — that would break + // oscillation protection when the new env immediately dies.) + environmentRecreations = 0 + + // Reset error tracking on successful poll + if (consecutiveErrors > 0) { + logForDebugging( + `[bridge:repl] Poll recovered after ${consecutiveErrors} consecutive error(s)`, + ) + consecutiveErrors = 0 + firstErrorTime = null + lastPollErrorTime = null + onStateChange?.('ready') + } + + if (!work) { + // Read-and-clear: after a detected suspension, skip the at-capacity + // branch exactly once. The pollForWork above already refreshed the + // server's BRIDGE_LAST_POLL_TTL; this fast cycle gives any + // re-dispatched work item a chance to land before we go back under. + const skipAtCapacityOnce = suspensionDetected + suspensionDetected = false + if (isAtCapacity?.() && capacitySignal && !skipAtCapacityOnce) { + const atCapMs = pollConfig.poll_interval_ms_at_capacity + // Heartbeat loops WITHOUT polling. When at-capacity polling is also + // enabled (atCapMs > 0), the loop tracks a deadline and breaks out + // to poll at that interval — heartbeat and poll compose instead of + // one suppressing the other. Breaks out when: + // - Poll deadline reached (atCapMs > 0 only) + // - Auth fails (JWT expired → poll refreshes tokens) + // - Capacity wake fires (transport lost → poll for new work) + // - Heartbeat config disabled (GrowthBook update) + // - Loop aborted (shutdown) + if ( + pollConfig.non_exclusive_heartbeat_interval_ms > 0 && + getHeartbeatInfo + ) { + logEvent('tengu_bridge_heartbeat_mode_entered', { + heartbeat_interval_ms: + pollConfig.non_exclusive_heartbeat_interval_ms, + }) + // Deadline computed once at entry — GB updates to atCapMs don't + // shift an in-flight deadline (next entry picks up the new value). + const pollDeadline = atCapMs > 0 ? Date.now() + atCapMs : null + let needsBackoff = false + let hbCycles = 0 + while ( + !signal.aborted && + isAtCapacity() && + (pollDeadline === null || Date.now() < pollDeadline) + ) { + const hbConfig = getPollIntervalConfig() + if (hbConfig.non_exclusive_heartbeat_interval_ms <= 0) break + + const info = getHeartbeatInfo() + if (!info) break + + // Capture capacity signal BEFORE the async heartbeat call so + // a transport loss during the HTTP request is caught by the + // subsequent sleep. + const cap = capacitySignal() + + try { + await api.heartbeatWork( + info.environmentId, + info.workId, + info.sessionToken, + ) + } catch (err) { + logForDebugging( + `[bridge:repl:heartbeat] Failed: ${errorMessage(err)}`, + ) + if (err instanceof BridgeFatalError) { + cap.cleanup() + logEvent('tengu_bridge_heartbeat_error', { + status: + err.status as unknown as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + error_type: (err.status === 401 || err.status === 403 + ? 'auth_failed' + : 'fatal') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + // JWT expired (401/403) or work item gone (404/410). + // Either way the current transport is dead — SSE + // reconnects and CCR writes will fail on the same + // stale token. If the caller gave us a recovery hook, + // tear down work state and skip backoff: isAtCapacity() + // flips to false, next outer-loop iteration fast-polls + // for the server's re-dispatched work item. Without + // the hook, backoff to avoid tight poll+heartbeat loop. + if (onHeartbeatFatal) { + onHeartbeatFatal(err) + logForDebugging( + `[bridge:repl:heartbeat] Fatal (status=${err.status}), work state cleared — fast-polling for re-dispatch`, + ) + } else { + needsBackoff = true + } + break + } + } + + hbCycles++ + await sleep( + hbConfig.non_exclusive_heartbeat_interval_ms, + cap.signal, + ) + cap.cleanup() + } + + const exitReason = needsBackoff + ? 'error' + : signal.aborted + ? 'shutdown' + : !isAtCapacity() + ? 'capacity_changed' + : pollDeadline !== null && Date.now() >= pollDeadline + ? 'poll_due' + : 'config_disabled' + logEvent('tengu_bridge_heartbeat_mode_exited', { + reason: + exitReason as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + heartbeat_cycles: hbCycles, + }) + + // On auth_failed or fatal, backoff before polling to avoid a + // tight poll+heartbeat loop. Fall through to the shared sleep + // below — it's the same capacitySignal-wrapped sleep the legacy + // path uses, and both need the suspension-overrun check. + if (!needsBackoff) { + if (exitReason === 'poll_due') { + // bridgeApi throttles empty-poll logs (EMPTY_POLL_LOG_INTERVAL=100) + // so the once-per-10min poll_due poll is invisible at counter=2. + // Log it here so verification runs see both endpoints in the debug log. + logForDebugging( + `[bridge:repl] Heartbeat poll_due after ${hbCycles} cycles — falling through to pollForWork`, + ) + } + continue + } + } + // At-capacity sleep — reached by both the legacy path (heartbeat + // disabled) and the heartbeat-backoff path (needsBackoff=true). + // Merged so the suspension detector covers both; previously the + // backoff path had no overrun check and could go straight back + // under for 10 min after a laptop wake. Use atCapMs when enabled, + // else the heartbeat interval as a floor (guaranteed > 0 on the + // backoff path) so heartbeat-only configs don't tight-loop. + const sleepMs = + atCapMs > 0 + ? atCapMs + : pollConfig.non_exclusive_heartbeat_interval_ms + if (sleepMs > 0) { + const cap = capacitySignal() + const sleepStart = Date.now() + await sleep(sleepMs, cap.signal) + cap.cleanup() + // Process-suspension detector. A setTimeout overshooting its + // deadline by 60s means the process was suspended (laptop lid, + // SIGSTOP, VM pause) — even a pathological GC pause is seconds, + // not minutes. Early aborts (wakePollLoop → cap.signal) produce + // overrun < 0 and fall through. Note: this only catches sleeps + // that outlast their deadline; WebSocketTransport's ping + // interval (10s granularity) is the primary detector for shorter + // suspensions. This is the backstop for when that detector isn't + // running (transport mid-reconnect, interval stopped). + const overrun = Date.now() - sleepStart - sleepMs + if (overrun > 60_000) { + logForDebugging( + `[bridge:repl] At-capacity sleep overran by ${Math.round(overrun / 1000)}s — process suspension detected, forcing one fast-poll cycle`, + ) + logEvent('tengu_bridge_repl_suspension_detected', { + overrun_ms: overrun, + }) + suspensionDetected = true + } + } + } else { + await sleep(pollConfig.poll_interval_ms_not_at_capacity, signal) + } + continue + } + + // Decode before type dispatch — need the JWT for the explicit ack. + let secret + try { + secret = decodeWorkSecret(work.secret) + } catch (err) { + logForDebugging( + `[bridge:repl] Failed to decode work secret: ${errorMessage(err)}`, + ) + logEvent('tengu_bridge_repl_work_secret_failed', {}) + // Can't ack (needs the JWT we failed to decode). stopWork uses OAuth. + // Prevents XAUTOCLAIM re-delivering this poisoned item every cycle. + await api.stopWork(envId, work.id, false).catch(() => {}) + continue + } + + // Explicitly acknowledge to prevent redelivery. Non-fatal on failure: + // server re-delivers, and the onWorkReceived callback handles dedup. + logForDebugging(`[bridge:repl] Acknowledging workId=${work.id}`) + try { + await api.acknowledgeWork(envId, work.id, secret.session_ingress_token) + } catch (err) { + logForDebugging( + `[bridge:repl] Acknowledge failed workId=${work.id}: ${errorMessage(err)}`, + ) + } + + if (work.data.type === 'healthcheck') { + logForDebugging('[bridge:repl] Healthcheck received') + continue + } + + if (work.data.type === 'session') { + const workSessionId = work.data.id + try { + validateBridgeId(workSessionId, 'session_id') + } catch { + logForDebugging( + `[bridge:repl] Invalid session_id in work: ${workSessionId}`, + ) + continue + } + + onWorkReceived( + workSessionId, + secret.session_ingress_token, + work.id, + secret.use_code_sessions === true, + ) + logForDebugging('[bridge:repl] Work accepted, continuing poll loop') + } + } catch (err) { + if (signal.aborted) break + + // Detect permanent "environment deleted" error — no amount of + // retrying will recover. Re-register a new environment instead. + // Checked BEFORE the generic BridgeFatalError bail. pollForWork uses + // validateStatus: s => s < 500, so 404 is always wrapped into a + // BridgeFatalError by handleErrorStatus() — never an axios-shaped + // error. The poll endpoint's only path param is the env ID; 404 + // unambiguously means env-gone (no-work is a 200 with null body). + // The server sends error.type='not_found_error' (standard Anthropic + // API shape), not a bridge-specific string — but status===404 is + // the real signal and survives body-shape changes. + if ( + err instanceof BridgeFatalError && + err.status === 404 && + onEnvironmentLost + ) { + // If credentials have already been refreshed by a concurrent + // reconnection (e.g. WS close handler), the stale poll's error + // is expected — skip onEnvironmentLost and retry with fresh creds. + const currentEnvId = getCredentials().environmentId + if (envId !== currentEnvId) { + logForDebugging( + `[bridge:repl] Stale poll error for old env=${envId}, current env=${currentEnvId} — skipping onEnvironmentLost`, + ) + consecutiveErrors = 0 + firstErrorTime = null + continue + } + + environmentRecreations++ + logForDebugging( + `[bridge:repl] Environment deleted, attempting re-registration (attempt ${environmentRecreations}/${MAX_ENVIRONMENT_RECREATIONS})`, + ) + logEvent('tengu_bridge_repl_env_lost', { + attempt: environmentRecreations, + } as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS) + + if (environmentRecreations > MAX_ENVIRONMENT_RECREATIONS) { + logForDebugging( + `[bridge:repl] Environment re-registration limit reached (${MAX_ENVIRONMENT_RECREATIONS}), giving up`, + ) + onStateChange?.( + 'failed', + 'Environment deleted and re-registration limit reached', + ) + onFatalError?.() + break + } + + onStateChange?.('reconnecting', 'environment lost, recreating session') + const newCreds = await onEnvironmentLost() + // doReconnect() makes several sequential network calls (1-5s). + // If the user triggered teardown during that window, its internal + // abort checks return false — but we need to re-check here to + // avoid emitting a spurious 'failed' + onFatalError() during + // graceful shutdown. + if (signal.aborted) break + if (newCreds) { + // Credentials are updated in the outer scope via + // reconnectEnvironmentWithSession — getCredentials() will + // return the fresh values on the next poll iteration. + // Do NOT reset environmentRecreations here — onEnvLost returning + // creds only proves we tried to fix it, not that the env is + // healthy. A successful poll (above) is the reset point; if the + // new env immediately dies again we still want the limit to fire. + consecutiveErrors = 0 + firstErrorTime = null + onStateChange?.('ready') + logForDebugging( + `[bridge:repl] Re-registered environment: ${newCreds.environmentId}`, + ) + continue + } + + onStateChange?.( + 'failed', + 'Environment deleted and re-registration failed', + ) + onFatalError?.() + break + } + + // Fatal errors (401/403/404/410) — no point retrying + if (err instanceof BridgeFatalError) { + const isExpiry = isExpiredErrorType(err.errorType) + const isSuppressible = isSuppressible403(err) + logForDebugging( + `[bridge:repl] Fatal poll error: ${err.message} (status=${err.status}, type=${err.errorType ?? 'unknown'})${isSuppressible ? ' (suppressed)' : ''}`, + ) + logEvent('tengu_bridge_repl_fatal_error', { + status: err.status, + error_type: + err.errorType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + logForDiagnosticsNoPII( + isExpiry ? 'info' : 'error', + 'bridge_repl_fatal_error', + { status: err.status, error_type: err.errorType }, + ) + // Cosmetic 403 errors (e.g., external_poll_sessions scope, + // environments:manage permission) — suppress user-visible error + // but always trigger teardown so cleanup runs. + if (!isSuppressible) { + onStateChange?.( + 'failed', + isExpiry + ? 'session expired · /remote-control to reconnect' + : err.message, + ) + } + // Always trigger teardown — matches bridgeMain.ts where fatalExit=true + // is unconditional and post-loop cleanup always runs. + onFatalError?.() + break + } + + const now = Date.now() + + // Detect system sleep/wake: if the gap since the last poll error + // greatly exceeds the max backoff delay, the machine likely slept. + // Reset error tracking so we retry with a fresh budget instead of + // immediately giving up. + if ( + lastPollErrorTime !== null && + now - lastPollErrorTime > POLL_ERROR_MAX_DELAY_MS * 2 + ) { + logForDebugging( + `[bridge:repl] Detected system sleep (${Math.round((now - lastPollErrorTime) / 1000)}s gap), resetting poll error budget`, + ) + logForDiagnosticsNoPII('info', 'bridge_repl_poll_sleep_detected', { + gapMs: now - lastPollErrorTime, + }) + consecutiveErrors = 0 + firstErrorTime = null + } + lastPollErrorTime = now + + consecutiveErrors++ + if (firstErrorTime === null) { + firstErrorTime = now + } + const elapsed = now - firstErrorTime + const httpStatus = extractHttpStatus(err) + const errMsg = describeAxiosError(err) + const wsLabel = getWsState?.() ?? 'unknown' + + logForDebugging( + `[bridge:repl] Poll error (attempt ${consecutiveErrors}, elapsed ${Math.round(elapsed / 1000)}s, ws=${wsLabel}): ${errMsg}`, + ) + logEvent('tengu_bridge_repl_poll_error', { + status: httpStatus, + consecutiveErrors, + elapsedMs: elapsed, + } as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS) + + // Only transition to 'reconnecting' on the first error — stay + // there until a successful poll (avoid flickering the UI state). + if (consecutiveErrors === 1) { + onStateChange?.('reconnecting', errMsg) + } + + // Give up after continuous failures + if (elapsed >= POLL_ERROR_GIVE_UP_MS) { + logForDebugging( + `[bridge:repl] Poll failures exceeded ${POLL_ERROR_GIVE_UP_MS / 1000}s (${consecutiveErrors} errors), giving up`, + ) + logForDiagnosticsNoPII('info', 'bridge_repl_poll_give_up') + logEvent('tengu_bridge_repl_poll_give_up', { + consecutiveErrors, + elapsedMs: elapsed, + lastStatus: httpStatus, + } as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS) + onStateChange?.('failed', 'connection to server lost') + break + } + + // Exponential backoff: 2s → 4s → 8s → 16s → 32s → 60s (cap) + const backoff = Math.min( + POLL_ERROR_INITIAL_DELAY_MS * 2 ** (consecutiveErrors - 1), + POLL_ERROR_MAX_DELAY_MS, + ) + // The poll_due heartbeat-loop exit leaves a healthy lease exposed to + // this backoff path. Heartbeat before each sleep so /poll outages + // (the VerifyEnvironmentSecretAuth DB path heartbeat was introduced to + // avoid) don't kill the 300s lease TTL. + if (getPollIntervalConfig().non_exclusive_heartbeat_interval_ms > 0) { + const info = getHeartbeatInfo?.() + if (info) { + try { + await api.heartbeatWork( + info.environmentId, + info.workId, + info.sessionToken, + ) + } catch { + // Best-effort — if heartbeat also fails the lease dies, same as + // pre-poll_due behavior (where the only heartbeat-loop exits were + // ones where the lease was already dying). + } + } + } + await sleep(backoff, signal) + } + } + + logForDebugging( + `[bridge:repl] Work poll loop ended (aborted=${signal.aborted}) env=${getCredentials().environmentId}`, + ) +} + +// Exported for testing only +export { + startWorkPollLoop as _startWorkPollLoopForTesting, + POLL_ERROR_INITIAL_DELAY_MS as _POLL_ERROR_INITIAL_DELAY_MS_ForTesting, + POLL_ERROR_MAX_DELAY_MS as _POLL_ERROR_MAX_DELAY_MS_ForTesting, + POLL_ERROR_GIVE_UP_MS as _POLL_ERROR_GIVE_UP_MS_ForTesting, +} diff --git a/src/bridge/replBridgeHandle.ts b/src/bridge/replBridgeHandle.ts new file mode 100644 index 0000000..f04d745 --- /dev/null +++ b/src/bridge/replBridgeHandle.ts @@ -0,0 +1,36 @@ +import { updateSessionBridgeId } from '../utils/concurrentSessions.js' +import type { ReplBridgeHandle } from './replBridge.js' +import { toCompatSessionId } from './sessionIdCompat.js' + +/** + * Global pointer to the active REPL bridge handle, so callers outside + * useReplBridge's React tree (tools, slash commands) can invoke handle methods + * like subscribePR. Same one-bridge-per-process justification as bridgeDebug.ts + * — the handle's closure captures the sessionId and getAccessToken that created + * the session, and re-deriving those independently (BriefTool/upload.ts pattern) + * risks staging/prod token divergence. + * + * Set from useReplBridge.tsx when init completes; cleared on teardown. + */ + +let handle: ReplBridgeHandle | null = null + +export function setReplBridgeHandle(h: ReplBridgeHandle | null): void { + handle = h + // Publish (or clear) our bridge session ID in the session record so other + // local peers can dedup us out of their bridge list — local is preferred. + void updateSessionBridgeId(getSelfBridgeCompatId() ?? null).catch(() => {}) +} + +export function getReplBridgeHandle(): ReplBridgeHandle | null { + return handle +} + +/** + * Our own bridge session ID in the session_* compat format the API returns + * in /v1/sessions responses — or undefined if bridge isn't connected. + */ +export function getSelfBridgeCompatId(): string | undefined { + const h = getReplBridgeHandle() + return h ? toCompatSessionId(h.bridgeSessionId) : undefined +} diff --git a/src/bridge/replBridgeTransport.ts b/src/bridge/replBridgeTransport.ts new file mode 100644 index 0000000..2a844f9 --- /dev/null +++ b/src/bridge/replBridgeTransport.ts @@ -0,0 +1,370 @@ +import type { StdoutMessage } from 'src/entrypoints/sdk/controlTypes.js' +import { CCRClient } from '../cli/transports/ccrClient.js' +import type { HybridTransport } from '../cli/transports/HybridTransport.js' +import { SSETransport } from '../cli/transports/SSETransport.js' +import { logForDebugging } from '../utils/debug.js' +import { errorMessage } from '../utils/errors.js' +import { updateSessionIngressAuthToken } from '../utils/sessionIngressAuth.js' +import type { SessionState } from '../utils/sessionState.js' +import { registerWorker } from './workSecret.js' + +/** + * Transport abstraction for replBridge. Covers exactly the surface that + * replBridge.ts uses against HybridTransport so the v1/v2 choice is + * confined to the construction site. + * + * - v1: HybridTransport (WS reads + POST writes to Session-Ingress) + * - v2: SSETransport (reads) + CCRClient (writes to CCR v2 /worker/*) + * + * The v2 write path goes through CCRClient.writeEvent → SerialBatchEventUploader, + * NOT through SSETransport.write() — SSETransport.write() targets the + * Session-Ingress POST URL shape, which is wrong for CCR v2. + */ +export type ReplBridgeTransport = { + write(message: StdoutMessage): Promise + writeBatch(messages: StdoutMessage[]): Promise + close(): void + isConnectedStatus(): boolean + getStateLabel(): string + setOnData(callback: (data: string) => void): void + setOnClose(callback: (closeCode?: number) => void): void + setOnConnect(callback: () => void): void + connect(): void + /** + * High-water mark of the underlying read stream's event sequence numbers. + * replBridge reads this before swapping transports so the new one can + * resume from where the old one left off (otherwise the server replays + * the entire session history from seq 0). + * + * v1 returns 0 — Session-Ingress WS doesn't use SSE sequence numbers; + * replay-on-reconnect is handled by the server-side message cursor. + */ + getLastSequenceNum(): number + /** + * Monotonic count of batches dropped via maxConsecutiveFailures. + * Snapshot before writeBatch() and compare after to detect silent drops + * (writeBatch() resolves normally even when batches were dropped). + * v2 returns 0 — the v2 write path doesn't set maxConsecutiveFailures. + */ + readonly droppedBatchCount: number + /** + * PUT /worker state (v2 only; v1 is a no-op). `requires_action` tells + * the backend a permission prompt is pending — claude.ai shows the + * "waiting for input" indicator. REPL/daemon callers don't need this + * (user watches the REPL locally); multi-session worker callers do. + */ + reportState(state: SessionState): void + /** PUT /worker external_metadata (v2 only; v1 is a no-op). */ + reportMetadata(metadata: Record): void + /** + * POST /worker/events/{id}/delivery (v2 only; v1 is a no-op). Populates + * CCR's processing_at/processed_at columns. `received` is auto-fired by + * CCRClient on every SSE frame and is not exposed here. + */ + reportDelivery(eventId: string, status: 'processing' | 'processed'): void + /** + * Drain the write queue before close() (v2 only; v1 resolves + * immediately — HybridTransport POSTs are already awaited per-write). + */ + flush(): Promise +} + +/** + * v1 adapter: HybridTransport already has the full surface (it extends + * WebSocketTransport which has setOnConnect + getStateLabel). This is a + * no-op wrapper that exists only so replBridge's `transport` variable + * has a single type. + */ +export function createV1ReplTransport( + hybrid: HybridTransport, +): ReplBridgeTransport { + return { + write: msg => hybrid.write(msg), + writeBatch: msgs => hybrid.writeBatch(msgs), + close: () => hybrid.close(), + isConnectedStatus: () => hybrid.isConnectedStatus(), + getStateLabel: () => hybrid.getStateLabel(), + setOnData: cb => hybrid.setOnData(cb), + setOnClose: cb => hybrid.setOnClose(cb), + setOnConnect: cb => hybrid.setOnConnect(cb), + connect: () => void hybrid.connect(), + // v1 Session-Ingress WS doesn't use SSE sequence numbers; replay + // semantics are different. Always return 0 so the seq-num carryover + // logic in replBridge is a no-op for v1. + getLastSequenceNum: () => 0, + get droppedBatchCount() { + return hybrid.droppedBatchCount + }, + reportState: () => {}, + reportMetadata: () => {}, + reportDelivery: () => {}, + flush: () => Promise.resolve(), + } +} + +/** + * v2 adapter: wrap SSETransport (reads) + CCRClient (writes, heartbeat, + * state, delivery tracking). + * + * Auth: v2 endpoints validate the JWT's session_id claim (register_worker.go:32) + * and worker role (environment_auth.py:856). OAuth tokens have neither. + * This is the inverse of the v1 replBridge path, which deliberately uses OAuth. + * The JWT is refreshed when the poll loop re-dispatches work — the caller + * invokes createV2ReplTransport again with the fresh token. + * + * Registration happens here (not in the caller) so the entire v2 handshake + * is one async step. registerWorker failure propagates — replBridge will + * catch it and stay on the poll loop. + */ +export async function createV2ReplTransport(opts: { + sessionUrl: string + ingressToken: string + sessionId: string + /** + * SSE sequence-number high-water mark from the previous transport. + * Passed to the new SSETransport so its first connect() sends + * from_sequence_num / Last-Event-ID and the server resumes from where + * the old stream left off. Without this, every transport swap asks the + * server to replay the entire session history from seq 0. + */ + initialSequenceNum?: number + /** + * Worker epoch from POST /bridge response. When provided, the server + * already bumped epoch (the /bridge call IS the register — see server + * PR #293280). When omitted (v1 CCR-v2 path via replBridge.ts poll loop), + * call registerWorker as before. + */ + epoch?: number + /** CCRClient heartbeat interval. Defaults to 20s when omitted. */ + heartbeatIntervalMs?: number + /** ±fraction per-beat jitter. Defaults to 0 (no jitter) when omitted. */ + heartbeatJitterFraction?: number + /** + * When true, skip opening the SSE read stream — only the CCRClient write + * path is activated. Use for mirror-mode attachments that forward events + * but never receive inbound prompts or control requests. + */ + outboundOnly?: boolean + /** + * Per-instance auth header source. When provided, CCRClient + SSETransport + * read auth from this closure instead of the process-wide + * CLAUDE_CODE_SESSION_ACCESS_TOKEN env var. Required for callers managing + * multiple concurrent sessions — the env-var path stomps across sessions. + * When omitted, falls back to the env var (single-session callers). + */ + getAuthToken?: () => string | undefined +}): Promise { + const { + sessionUrl, + ingressToken, + sessionId, + initialSequenceNum, + getAuthToken, + } = opts + + // Auth header builder. If getAuthToken is provided, read from it + // (per-instance, multi-session safe). Otherwise write ingressToken to + // the process-wide env var (legacy single-session path — CCRClient's + // default getAuthHeaders reads it via getSessionIngressAuthHeaders). + let getAuthHeaders: (() => Record) | undefined + if (getAuthToken) { + getAuthHeaders = (): Record => { + const token = getAuthToken() + if (!token) return {} + return { Authorization: `Bearer ${token}` } + } + } else { + // CCRClient.request() and SSETransport.connect() both read auth via + // getSessionIngressAuthHeaders() → this env var. Set it before either + // touches the network. + updateSessionIngressAuthToken(ingressToken) + } + + const epoch = opts.epoch ?? (await registerWorker(sessionUrl, ingressToken)) + logForDebugging( + `[bridge:repl] CCR v2: worker sessionId=${sessionId} epoch=${epoch}${opts.epoch !== undefined ? ' (from /bridge)' : ' (via registerWorker)'}`, + ) + + // Derive SSE stream URL. Same logic as transportUtils.ts:26-33 but + // starting from an http(s) base instead of a --sdk-url that might be ws://. + const sseUrl = new URL(sessionUrl) + sseUrl.pathname = sseUrl.pathname.replace(/\/$/, '') + '/worker/events/stream' + + const sse = new SSETransport( + sseUrl, + {}, + sessionId, + undefined, + initialSequenceNum, + getAuthHeaders, + ) + let onCloseCb: ((closeCode?: number) => void) | undefined + const ccr = new CCRClient(sse, new URL(sessionUrl), { + getAuthHeaders, + heartbeatIntervalMs: opts.heartbeatIntervalMs, + heartbeatJitterFraction: opts.heartbeatJitterFraction, + // Default is process.exit(1) — correct for spawn-mode children. In-process, + // that kills the REPL. Close instead: replBridge's onClose wakes the poll + // loop, which picks up the server's re-dispatch (with fresh epoch). + onEpochMismatch: () => { + logForDebugging( + '[bridge:repl] CCR v2: epoch superseded (409) — closing for poll-loop recovery', + ) + // Close resources in a try block so the throw always executes. + // If ccr.close() or sse.close() throw, we still need to unwind + // the caller (request()) — otherwise handleEpochMismatch's `never` + // return type is violated at runtime and control falls through. + try { + ccr.close() + sse.close() + onCloseCb?.(4090) + } catch (closeErr: unknown) { + logForDebugging( + `[bridge:repl] CCR v2: error during epoch-mismatch cleanup: ${errorMessage(closeErr)}`, + { level: 'error' }, + ) + } + // Don't return — the calling request() code continues after the 409 + // branch, so callers see the logged warning and a false return. We + // throw to unwind; the uploaders catch it as a send failure. + throw new Error('epoch superseded') + }, + }) + + // CCRClient's constructor wired sse.setOnEvent → reportDelivery('received'). + // remoteIO.ts additionally sends 'processing'/'processed' via + // setCommandLifecycleListener, which the in-process query loop fires. This + // transport's only caller (replBridge/daemonBridge) has no such wiring — the + // daemon's agent child is a separate process (ProcessTransport), and its + // notifyCommandLifecycle calls fire with listener=null in its own module + // scope. So events stay at 'received' forever, and reconnectSession re-queues + // them on every daemon restart (observed: 21→24→25 phantom prompts as + // "user sent a new message while you were working" system-reminders). + // + // Fix: ACK 'processed' immediately alongside 'received'. The window between + // SSE receipt and transcript-write is narrow (queue → SDK → child stdin → + // model); a crash there loses one prompt vs. the observed N-prompt flood on + // every restart. Overwrite the constructor's wiring to do both — setOnEvent + // replaces, not appends (SSETransport.ts:658). + sse.setOnEvent(event => { + ccr.reportDelivery(event.event_id, 'received') + ccr.reportDelivery(event.event_id, 'processed') + }) + + // Both sse.connect() and ccr.initialize() are deferred to connect() below. + // replBridge's calling order is newTransport → setOnConnect → setOnData → + // setOnClose → connect(), and both calls need those callbacks wired first: + // sse.connect() opens the stream (events flow to onData/onClose immediately), + // and ccr.initialize().then() fires onConnectCb. + // + // onConnect fires once ccr.initialize() resolves. Writes go via + // CCRClient HTTP POST (SerialBatchEventUploader), not SSE, so the + // write path is ready the moment workerEpoch is set. SSE.connect() + // awaits its read loop and never resolves — don't gate on it. + // The SSE stream opens in parallel (~30ms) and starts delivering + // inbound events via setOnData; outbound doesn't need to wait for it. + let onConnectCb: (() => void) | undefined + let ccrInitialized = false + let closed = false + + return { + write(msg) { + return ccr.writeEvent(msg) + }, + async writeBatch(msgs) { + // SerialBatchEventUploader already batches internally (maxBatchSize=100); + // sequential enqueue preserves order and the uploader coalesces. + // Check closed between writes to avoid sending partial batches after + // transport teardown (epoch mismatch, SSE drop). + for (const m of msgs) { + if (closed) break + await ccr.writeEvent(m) + } + }, + close() { + closed = true + ccr.close() + sse.close() + }, + isConnectedStatus() { + // Write-readiness, not read-readiness — replBridge checks this + // before calling writeBatch. SSE open state is orthogonal. + return ccrInitialized + }, + getStateLabel() { + // SSETransport doesn't expose its state string; synthesize from + // what we can observe. replBridge only uses this for debug logging. + if (sse.isClosedStatus()) return 'closed' + if (sse.isConnectedStatus()) return ccrInitialized ? 'connected' : 'init' + return 'connecting' + }, + setOnData(cb) { + sse.setOnData(cb) + }, + setOnClose(cb) { + onCloseCb = cb + // SSE reconnect-budget exhaustion fires onClose(undefined) — map to + // 4092 so ws_closed telemetry can distinguish it from HTTP-status + // closes (SSETransport:280 passes response.status). Stop CCRClient's + // heartbeat timer before notifying replBridge. (sse.close() doesn't + // invoke this, so the epoch-mismatch path above isn't double-firing.) + sse.setOnClose(code => { + ccr.close() + cb(code ?? 4092) + }) + }, + setOnConnect(cb) { + onConnectCb = cb + }, + getLastSequenceNum() { + return sse.getLastSequenceNum() + }, + // v2 write path (CCRClient) doesn't set maxConsecutiveFailures — no drops. + droppedBatchCount: 0, + reportState(state) { + ccr.reportState(state) + }, + reportMetadata(metadata) { + ccr.reportMetadata(metadata) + }, + reportDelivery(eventId, status) { + ccr.reportDelivery(eventId, status) + }, + flush() { + return ccr.flush() + }, + connect() { + // Outbound-only: skip the SSE read stream entirely — no inbound + // events to receive, no delivery ACKs to send. Only the CCRClient + // write path (POST /worker/events) and heartbeat are needed. + if (!opts.outboundOnly) { + // Fire-and-forget — SSETransport.connect() awaits readStream() + // (the read loop) and only resolves on stream close/error. The + // spawn-mode path in remoteIO.ts does the same void discard. + void sse.connect() + } + void ccr.initialize(epoch).then( + () => { + ccrInitialized = true + logForDebugging( + `[bridge:repl] v2 transport ready for writes (epoch=${epoch}, sse=${sse.isConnectedStatus() ? 'open' : 'opening'})`, + ) + onConnectCb?.() + }, + (err: unknown) => { + logForDebugging( + `[bridge:repl] CCR v2 initialize failed: ${errorMessage(err)}`, + { level: 'error' }, + ) + // Close transport resources and notify replBridge via onClose + // so the poll loop can retry on the next work dispatch. + // Without this callback, replBridge never learns the transport + // failed to initialize and sits with transport === null forever. + ccr.close() + sse.close() + onCloseCb?.(4091) // 4091 = init failure, distinguishable from 4090 epoch mismatch + }, + ) + }, + } +} diff --git a/src/bridge/sessionIdCompat.ts b/src/bridge/sessionIdCompat.ts new file mode 100644 index 0000000..57d8d22 --- /dev/null +++ b/src/bridge/sessionIdCompat.ts @@ -0,0 +1,57 @@ +/** + * Session ID tag translation helpers for the CCR v2 compat layer. + * + * Lives in its own file (rather than workSecret.ts) so that sessionHandle.ts + * and replBridgeTransport.ts (bridge.mjs entry points) can import from + * workSecret.ts without pulling in these retag functions. + * + * The isCseShimEnabled kill switch is injected via setCseShimGate() to avoid + * a static import of bridgeEnabled.ts → growthbook.ts → config.ts — all + * banned from the sdk.mjs bundle (scripts/build-agent-sdk.sh). Callers that + * already import bridgeEnabled.ts register the gate; the SDK path never does, + * so the shim defaults to active (matching isCseShimEnabled()'s own default). + */ + +let _isCseShimEnabled: (() => boolean) | undefined + +/** + * Register the GrowthBook gate for the cse_ shim. Called from bridge + * init code that already imports bridgeEnabled.ts. + */ +export function setCseShimGate(gate: () => boolean): void { + _isCseShimEnabled = gate +} + +/** + * Re-tag a `cse_*` session ID to `session_*` for use with the v1 compat API. + * + * Worker endpoints (/v1/code/sessions/{id}/worker/*) want `cse_*`; that's + * what the work poll delivers. Client-facing compat endpoints + * (/v1/sessions/{id}, /v1/sessions/{id}/archive, /v1/sessions/{id}/events) + * want `session_*` — compat/convert.go:27 validates TagSession. Same UUID, + * different costume. No-op for IDs that aren't `cse_*`. + * + * bridgeMain holds one sessionId variable for both worker registration and + * session-management calls. It arrives as `cse_*` from the work poll under + * the compat gate, so archiveSession/fetchSessionTitle need this re-tag. + */ +export function toCompatSessionId(id: string): string { + if (!id.startsWith('cse_')) return id + if (_isCseShimEnabled && !_isCseShimEnabled()) return id + return 'session_' + id.slice('cse_'.length) +} + +/** + * Re-tag a `session_*` session ID to `cse_*` for infrastructure-layer calls. + * + * Inverse of toCompatSessionId. POST /v1/environments/{id}/bridge/reconnect + * lives below the compat layer: once ccr_v2_compat_enabled is on server-side, + * it looks sessions up by their infra tag (`cse_*`). createBridgeSession still + * returns `session_*` (compat/convert.go:41) and that's what bridge-pointer + * stores — so perpetual reconnect passes the wrong costume and gets "Session + * not found" back. Same UUID, wrong tag. No-op for IDs that aren't `session_*`. + */ +export function toInfraSessionId(id: string): string { + if (!id.startsWith('session_')) return id + return 'cse_' + id.slice('session_'.length) +} diff --git a/src/bridge/sessionRunner.ts b/src/bridge/sessionRunner.ts new file mode 100644 index 0000000..bc232bc --- /dev/null +++ b/src/bridge/sessionRunner.ts @@ -0,0 +1,550 @@ +import { type ChildProcess, spawn } from 'child_process' +import { createWriteStream, type WriteStream } from 'fs' +import { tmpdir } from 'os' +import { dirname, join } from 'path' +import { createInterface } from 'readline' +import { jsonParse, jsonStringify } from '../utils/slowOperations.js' +import { debugTruncate } from './debugUtils.js' +import type { + SessionActivity, + SessionDoneStatus, + SessionHandle, + SessionSpawner, + SessionSpawnOpts, +} from './types.js' + +const MAX_ACTIVITIES = 10 +const MAX_STDERR_LINES = 10 + +/** + * Sanitize a session ID for use in file names. + * Strips any characters that could cause path traversal (e.g. `../`, `/`) + * or other filesystem issues, replacing them with underscores. + */ +export function safeFilenameId(id: string): string { + return id.replace(/[^a-zA-Z0-9_-]/g, '_') +} + +/** + * A control_request emitted by the child CLI when it needs permission to + * execute a **specific** tool invocation (not a general capability check). + * The bridge forwards this to the server so the user can approve/deny. + */ +export type PermissionRequest = { + type: 'control_request' + request_id: string + request: { + /** Per-invocation permission check — "may I run this tool with these inputs?" */ + subtype: 'can_use_tool' + tool_name: string + input: Record + tool_use_id: string + } +} + +type SessionSpawnerDeps = { + execPath: string + /** + * Arguments that must precede the CLI flags when spawning. Empty for + * compiled binaries (where execPath is the claude binary itself); contains + * the script path (process.argv[1]) for npm installs where execPath is the + * node runtime. Without this, node sees --sdk-url as a node option and + * exits with "bad option: --sdk-url" (see anthropics/claude-code#28334). + */ + scriptArgs: string[] + env: NodeJS.ProcessEnv + verbose: boolean + sandbox: boolean + debugFile?: string + permissionMode?: string + onDebug: (msg: string) => void + onActivity?: (sessionId: string, activity: SessionActivity) => void + onPermissionRequest?: ( + sessionId: string, + request: PermissionRequest, + accessToken: string, + ) => void +} + +/** Map tool names to human-readable verbs for the status display. */ +const TOOL_VERBS: Record = { + Read: 'Reading', + Write: 'Writing', + Edit: 'Editing', + MultiEdit: 'Editing', + Bash: 'Running', + Glob: 'Searching', + Grep: 'Searching', + WebFetch: 'Fetching', + WebSearch: 'Searching', + Task: 'Running task', + FileReadTool: 'Reading', + FileWriteTool: 'Writing', + FileEditTool: 'Editing', + GlobTool: 'Searching', + GrepTool: 'Searching', + BashTool: 'Running', + NotebookEditTool: 'Editing notebook', + LSP: 'LSP', +} + +function toolSummary(name: string, input: Record): string { + const verb = TOOL_VERBS[name] ?? name + const target = + (input.file_path as string) ?? + (input.filePath as string) ?? + (input.pattern as string) ?? + (input.command as string | undefined)?.slice(0, 60) ?? + (input.url as string) ?? + (input.query as string) ?? + '' + if (target) { + return `${verb} ${target}` + } + return verb +} + +function extractActivities( + line: string, + sessionId: string, + onDebug: (msg: string) => void, +): SessionActivity[] { + let parsed: unknown + try { + parsed = jsonParse(line) + } catch { + return [] + } + + if (!parsed || typeof parsed !== 'object') { + return [] + } + + const msg = parsed as Record + const activities: SessionActivity[] = [] + const now = Date.now() + + switch (msg.type) { + case 'assistant': { + const message = msg.message as Record | undefined + if (!message) break + const content = message.content + if (!Array.isArray(content)) break + + for (const block of content) { + if (!block || typeof block !== 'object') continue + const b = block as Record + + if (b.type === 'tool_use') { + const name = (b.name as string) ?? 'Tool' + const input = (b.input as Record) ?? {} + const summary = toolSummary(name, input) + activities.push({ + type: 'tool_start', + summary, + timestamp: now, + }) + onDebug( + `[bridge:activity] sessionId=${sessionId} tool_use name=${name} ${inputPreview(input)}`, + ) + } else if (b.type === 'text') { + const text = (b.text as string) ?? '' + if (text.length > 0) { + activities.push({ + type: 'text', + summary: text.slice(0, 80), + timestamp: now, + }) + onDebug( + `[bridge:activity] sessionId=${sessionId} text "${text.slice(0, 100)}"`, + ) + } + } + } + break + } + case 'result': { + const subtype = msg.subtype as string | undefined + if (subtype === 'success') { + activities.push({ + type: 'result', + summary: 'Session completed', + timestamp: now, + }) + onDebug( + `[bridge:activity] sessionId=${sessionId} result subtype=success`, + ) + } else if (subtype) { + const errors = msg.errors as string[] | undefined + const errorSummary = errors?.[0] ?? `Error: ${subtype}` + activities.push({ + type: 'error', + summary: errorSummary, + timestamp: now, + }) + onDebug( + `[bridge:activity] sessionId=${sessionId} result subtype=${subtype} error="${errorSummary}"`, + ) + } else { + onDebug( + `[bridge:activity] sessionId=${sessionId} result subtype=undefined`, + ) + } + break + } + default: + break + } + + return activities +} + +/** + * Extract plain text from a replayed SDKUserMessage NDJSON line. Returns the + * trimmed text if this looks like a real human-authored message, otherwise + * undefined so the caller keeps waiting for the first real message. + */ +function extractUserMessageText( + msg: Record, +): string | undefined { + // Skip tool-result user messages (wrapped subagent results) and synthetic + // caveat messages — neither is human-authored. + if (msg.parent_tool_use_id != null || msg.isSynthetic || msg.isReplay) + return undefined + + const message = msg.message as Record | undefined + const content = message?.content + let text: string | undefined + if (typeof content === 'string') { + text = content + } else if (Array.isArray(content)) { + for (const block of content) { + if ( + block && + typeof block === 'object' && + (block as Record).type === 'text' + ) { + text = (block as Record).text as string | undefined + break + } + } + } + text = text?.trim() + return text ? text : undefined +} + +/** Build a short preview of tool input for debug logging. */ +function inputPreview(input: Record): string { + const parts: string[] = [] + for (const [key, val] of Object.entries(input)) { + if (typeof val === 'string') { + parts.push(`${key}="${val.slice(0, 100)}"`) + } + if (parts.length >= 3) break + } + return parts.join(' ') +} + +export function createSessionSpawner(deps: SessionSpawnerDeps): SessionSpawner { + return { + spawn(opts: SessionSpawnOpts, dir: string): SessionHandle { + // Debug file resolution: + // 1. If deps.debugFile is provided, use it with session ID suffix for uniqueness + // 2. If verbose or ant build, auto-generate a temp file path + // 3. Otherwise, no debug file + const safeId = safeFilenameId(opts.sessionId) + let debugFile: string | undefined + if (deps.debugFile) { + const ext = deps.debugFile.lastIndexOf('.') + if (ext > 0) { + debugFile = `${deps.debugFile.slice(0, ext)}-${safeId}${deps.debugFile.slice(ext)}` + } else { + debugFile = `${deps.debugFile}-${safeId}` + } + } else if (deps.verbose || process.env.USER_TYPE === 'ant') { + debugFile = join(tmpdir(), 'claude', `bridge-session-${safeId}.log`) + } + + // Transcript file: write raw NDJSON lines for post-hoc analysis. + // Placed alongside the debug file when one is configured. + let transcriptStream: WriteStream | null = null + let transcriptPath: string | undefined + if (deps.debugFile) { + transcriptPath = join( + dirname(deps.debugFile), + `bridge-transcript-${safeId}.jsonl`, + ) + transcriptStream = createWriteStream(transcriptPath, { flags: 'a' }) + transcriptStream.on('error', err => { + deps.onDebug( + `[bridge:session] Transcript write error: ${err.message}`, + ) + transcriptStream = null + }) + deps.onDebug(`[bridge:session] Transcript log: ${transcriptPath}`) + } + + const args = [ + ...deps.scriptArgs, + '--print', + '--sdk-url', + opts.sdkUrl, + '--session-id', + opts.sessionId, + '--input-format', + 'stream-json', + '--output-format', + 'stream-json', + '--replay-user-messages', + ...(deps.verbose ? ['--verbose'] : []), + ...(debugFile ? ['--debug-file', debugFile] : []), + ...(deps.permissionMode + ? ['--permission-mode', deps.permissionMode] + : []), + ] + + const env: NodeJS.ProcessEnv = { + ...deps.env, + // Strip the bridge's OAuth token so the child CC process uses + // the session access token for inference instead. + CLAUDE_CODE_OAUTH_TOKEN: undefined, + CLAUDE_CODE_ENVIRONMENT_KIND: 'bridge', + ...(deps.sandbox && { CLAUDE_CODE_FORCE_SANDBOX: '1' }), + CLAUDE_CODE_SESSION_ACCESS_TOKEN: opts.accessToken, + // v1: HybridTransport (WS reads + POST writes) to Session-Ingress. + // Harmless in v2 mode — transportUtils checks CLAUDE_CODE_USE_CCR_V2 first. + CLAUDE_CODE_POST_FOR_SESSION_INGRESS_V2: '1', + // v2: SSETransport + CCRClient to CCR's /v1/code/sessions/* endpoints. + // Same env vars environment-manager sets in the container path. + ...(opts.useCcrV2 && { + CLAUDE_CODE_USE_CCR_V2: '1', + CLAUDE_CODE_WORKER_EPOCH: String(opts.workerEpoch), + }), + } + + deps.onDebug( + `[bridge:session] Spawning sessionId=${opts.sessionId} sdkUrl=${opts.sdkUrl} accessToken=${opts.accessToken ? 'present' : 'MISSING'}`, + ) + deps.onDebug(`[bridge:session] Child args: ${args.join(' ')}`) + if (debugFile) { + deps.onDebug(`[bridge:session] Debug log: ${debugFile}`) + } + + // Pipe all three streams: stdin for control, stdout for NDJSON parsing, + // stderr for error capture and diagnostics. + const child: ChildProcess = spawn(deps.execPath, args, { + cwd: dir, + stdio: ['pipe', 'pipe', 'pipe'], + env, + windowsHide: true, + }) + + deps.onDebug( + `[bridge:session] sessionId=${opts.sessionId} pid=${child.pid}`, + ) + + const activities: SessionActivity[] = [] + let currentActivity: SessionActivity | null = null + const lastStderr: string[] = [] + let sigkillSent = false + let firstUserMessageSeen = false + + // Buffer stderr for error diagnostics + if (child.stderr) { + const stderrRl = createInterface({ input: child.stderr }) + stderrRl.on('line', line => { + // Forward stderr to bridge's stderr in verbose mode + if (deps.verbose) { + process.stderr.write(line + '\n') + } + // Ring buffer of last N lines + if (lastStderr.length >= MAX_STDERR_LINES) { + lastStderr.shift() + } + lastStderr.push(line) + }) + } + + // Parse NDJSON from child stdout + if (child.stdout) { + const rl = createInterface({ input: child.stdout }) + rl.on('line', line => { + // Write raw NDJSON to transcript file + if (transcriptStream) { + transcriptStream.write(line + '\n') + } + + // Log all messages flowing from the child CLI to the bridge + deps.onDebug( + `[bridge:ws] sessionId=${opts.sessionId} <<< ${debugTruncate(line)}`, + ) + + // In verbose mode, forward raw output to stderr + if (deps.verbose) { + process.stderr.write(line + '\n') + } + + const extracted = extractActivities( + line, + opts.sessionId, + deps.onDebug, + ) + for (const activity of extracted) { + // Maintain ring buffer + if (activities.length >= MAX_ACTIVITIES) { + activities.shift() + } + activities.push(activity) + currentActivity = activity + + deps.onActivity?.(opts.sessionId, activity) + } + + // Detect control_request and replayed user messages. + // extractActivities parses the same line but swallows parse errors + // and skips 'user' type — re-parse here is cheap (NDJSON lines are + // small) and keeps each path self-contained. + { + let parsed: unknown + try { + parsed = jsonParse(line) + } catch { + // Non-JSON line, skip detection + } + if (parsed && typeof parsed === 'object') { + const msg = parsed as Record + + if (msg.type === 'control_request') { + const request = msg.request as + | Record + | undefined + if ( + request?.subtype === 'can_use_tool' && + deps.onPermissionRequest + ) { + deps.onPermissionRequest( + opts.sessionId, + parsed as PermissionRequest, + opts.accessToken, + ) + } + // interrupt is turn-level; the child handles it internally (print.ts) + } else if ( + msg.type === 'user' && + !firstUserMessageSeen && + opts.onFirstUserMessage + ) { + const text = extractUserMessageText(msg) + if (text) { + firstUserMessageSeen = true + opts.onFirstUserMessage(text) + } + } + } + } + }) + } + + const done = new Promise(resolve => { + child.on('close', (code, signal) => { + // Close transcript stream on exit + if (transcriptStream) { + transcriptStream.end() + transcriptStream = null + } + + if (signal === 'SIGTERM' || signal === 'SIGINT') { + deps.onDebug( + `[bridge:session] sessionId=${opts.sessionId} interrupted signal=${signal} pid=${child.pid}`, + ) + resolve('interrupted') + } else if (code === 0) { + deps.onDebug( + `[bridge:session] sessionId=${opts.sessionId} completed exit_code=0 pid=${child.pid}`, + ) + resolve('completed') + } else { + deps.onDebug( + `[bridge:session] sessionId=${opts.sessionId} failed exit_code=${code} pid=${child.pid}`, + ) + resolve('failed') + } + }) + + child.on('error', err => { + deps.onDebug( + `[bridge:session] sessionId=${opts.sessionId} spawn error: ${err.message}`, + ) + resolve('failed') + }) + }) + + const handle: SessionHandle = { + sessionId: opts.sessionId, + done, + activities, + accessToken: opts.accessToken, + lastStderr, + get currentActivity(): SessionActivity | null { + return currentActivity + }, + kill(): void { + if (!child.killed) { + deps.onDebug( + `[bridge:session] Sending SIGTERM to sessionId=${opts.sessionId} pid=${child.pid}`, + ) + // On Windows, child.kill('SIGTERM') throws; use default signal. + if (process.platform === 'win32') { + child.kill() + } else { + child.kill('SIGTERM') + } + } + }, + forceKill(): void { + // Use separate flag because child.killed is set when kill() is called, + // not when the process exits. We need to send SIGKILL even after SIGTERM. + if (!sigkillSent && child.pid) { + sigkillSent = true + deps.onDebug( + `[bridge:session] Sending SIGKILL to sessionId=${opts.sessionId} pid=${child.pid}`, + ) + if (process.platform === 'win32') { + child.kill() + } else { + child.kill('SIGKILL') + } + } + }, + writeStdin(data: string): void { + if (child.stdin && !child.stdin.destroyed) { + deps.onDebug( + `[bridge:ws] sessionId=${opts.sessionId} >>> ${debugTruncate(data)}`, + ) + child.stdin.write(data) + } + }, + updateAccessToken(token: string): void { + handle.accessToken = token + // Send the fresh token to the child process via stdin. The child's + // StructuredIO handles update_environment_variables messages by + // setting process.env directly, so getSessionIngressAuthToken() + // picks up the new token on the next refreshHeaders call. + handle.writeStdin( + jsonStringify({ + type: 'update_environment_variables', + variables: { CLAUDE_CODE_SESSION_ACCESS_TOKEN: token }, + }) + '\n', + ) + deps.onDebug( + `[bridge:session] Sent token refresh via stdin for sessionId=${opts.sessionId}`, + ) + }, + } + + return handle + }, + } +} + +export { extractActivities as _extractActivitiesForTesting } diff --git a/src/bridge/trustedDevice.ts b/src/bridge/trustedDevice.ts new file mode 100644 index 0000000..a4bcf35 --- /dev/null +++ b/src/bridge/trustedDevice.ts @@ -0,0 +1,210 @@ +import axios from 'axios' +import memoize from 'lodash-es/memoize.js' +import { hostname } from 'os' +import { getOauthConfig } from '../constants/oauth.js' +import { + checkGate_CACHED_OR_BLOCKING, + getFeatureValue_CACHED_MAY_BE_STALE, +} from '../services/analytics/growthbook.js' +import { logForDebugging } from '../utils/debug.js' +import { errorMessage } from '../utils/errors.js' +import { isEssentialTrafficOnly } from '../utils/privacyLevel.js' +import { getSecureStorage } from '../utils/secureStorage/index.js' +import { jsonStringify } from '../utils/slowOperations.js' + +/** + * Trusted device token source for bridge (remote-control) sessions. + * + * Bridge sessions have SecurityTier=ELEVATED on the server (CCR v2). + * The server gates ConnectBridgeWorker on its own flag + * (sessions_elevated_auth_enforcement in Anthropic Main); this CLI-side + * flag controls whether the CLI sends X-Trusted-Device-Token at all. + * Two flags so rollout can be staged: flip CLI-side first (headers + * start flowing, server still no-ops), then flip server-side. + * + * Enrollment (POST /auth/trusted_devices) is gated server-side by + * account_session.created_at < 10min, so it must happen during /login. + * Token is persistent (90d rolling expiry) and stored in keychain. + * + * See anthropics/anthropic#274559 (spec), #310375 (B1b tenant RPCs), + * #295987 (B2 Python routes), #307150 (C1' CCR v2 gate). + */ + +const TRUSTED_DEVICE_GATE = 'tengu_sessions_elevated_auth_enforcement' + +function isGateEnabled(): boolean { + return getFeatureValue_CACHED_MAY_BE_STALE(TRUSTED_DEVICE_GATE, false) +} + +// Memoized — secureStorage.read() spawns a macOS `security` subprocess (~40ms). +// bridgeApi.ts calls this from getHeaders() on every poll/heartbeat/ack. +// Cache cleared after enrollment (below) and on logout (clearAuthRelatedCaches). +// +// Only the storage read is memoized — the GrowthBook gate is checked live so +// that a gate flip after GrowthBook refresh takes effect without a restart. +const readStoredToken = memoize((): string | undefined => { + // Env var takes precedence for testing/canary. + const envToken = process.env.CLAUDE_TRUSTED_DEVICE_TOKEN + if (envToken) { + return envToken + } + return getSecureStorage().read()?.trustedDeviceToken +}) + +export function getTrustedDeviceToken(): string | undefined { + if (!isGateEnabled()) { + return undefined + } + return readStoredToken() +} + +export function clearTrustedDeviceTokenCache(): void { + readStoredToken.cache?.clear?.() +} + +/** + * Clear the stored trusted device token from secure storage and the memo cache. + * Called before enrollTrustedDevice() during /login so a stale token from the + * previous account isn't sent as X-Trusted-Device-Token while enrollment is + * in-flight (enrollTrustedDevice is async — bridge API calls between login and + * enrollment completion would otherwise still read the old cached token). + */ +export function clearTrustedDeviceToken(): void { + if (!isGateEnabled()) { + return + } + const secureStorage = getSecureStorage() + try { + const data = secureStorage.read() + if (data?.trustedDeviceToken) { + delete data.trustedDeviceToken + secureStorage.update(data) + } + } catch { + // Best-effort — don't block login if storage is inaccessible + } + readStoredToken.cache?.clear?.() +} + +/** + * Enroll this device via POST /auth/trusted_devices and persist the token + * to keychain. Best-effort — logs and returns on failure so callers + * (post-login hooks) don't block the login flow. + * + * The server gates enrollment on account_session.created_at < 10min, so + * this must be called immediately after a fresh /login. Calling it later + * (e.g. lazy enrollment on /bridge 403) will fail with 403 stale_session. + */ +export async function enrollTrustedDevice(): Promise { + try { + // checkGate_CACHED_OR_BLOCKING awaits any in-flight GrowthBook re-init + // (triggered by refreshGrowthBookAfterAuthChange in login.tsx) before + // reading the gate, so we get the post-refresh value. + if (!(await checkGate_CACHED_OR_BLOCKING(TRUSTED_DEVICE_GATE))) { + logForDebugging( + `[trusted-device] Gate ${TRUSTED_DEVICE_GATE} is off, skipping enrollment`, + ) + return + } + // If CLAUDE_TRUSTED_DEVICE_TOKEN is set (e.g. by an enterprise wrapper), + // skip enrollment — the env var takes precedence in readStoredToken() so + // any enrolled token would be shadowed and never used. + if (process.env.CLAUDE_TRUSTED_DEVICE_TOKEN) { + logForDebugging( + '[trusted-device] CLAUDE_TRUSTED_DEVICE_TOKEN env var is set, skipping enrollment (env var takes precedence)', + ) + return + } + // Lazy require — utils/auth.ts transitively pulls ~1300 modules + // (config → file → permissions → sessionStorage → commands). Daemon callers + // of getTrustedDeviceToken() don't need this; only /login does. + /* eslint-disable @typescript-eslint/no-require-imports */ + const { getClaudeAIOAuthTokens } = + require('../utils/auth.js') as typeof import('../utils/auth.js') + /* eslint-enable @typescript-eslint/no-require-imports */ + const accessToken = getClaudeAIOAuthTokens()?.accessToken + if (!accessToken) { + logForDebugging('[trusted-device] No OAuth token, skipping enrollment') + return + } + // Always re-enroll on /login — the existing token may belong to a + // different account (account-switch without /logout). Skipping enrollment + // would send the old account's token on the new account's bridge calls. + const secureStorage = getSecureStorage() + + if (isEssentialTrafficOnly()) { + logForDebugging( + '[trusted-device] Essential traffic only, skipping enrollment', + ) + return + } + + const baseUrl = getOauthConfig().BASE_API_URL + let response + try { + response = await axios.post<{ + device_token?: string + device_id?: string + }>( + `${baseUrl}/api/auth/trusted_devices`, + { display_name: `Claude Code on ${hostname()} · ${process.platform}` }, + { + headers: { + Authorization: `Bearer ${accessToken}`, + 'Content-Type': 'application/json', + }, + timeout: 10_000, + validateStatus: s => s < 500, + }, + ) + } catch (err: unknown) { + logForDebugging( + `[trusted-device] Enrollment request failed: ${errorMessage(err)}`, + ) + return + } + + if (response.status !== 200 && response.status !== 201) { + logForDebugging( + `[trusted-device] Enrollment failed ${response.status}: ${jsonStringify(response.data).slice(0, 200)}`, + ) + return + } + + const token = response.data?.device_token + if (!token || typeof token !== 'string') { + logForDebugging( + '[trusted-device] Enrollment response missing device_token field', + ) + return + } + + try { + const storageData = secureStorage.read() + if (!storageData) { + logForDebugging( + '[trusted-device] Cannot read storage, skipping token persist', + ) + return + } + storageData.trustedDeviceToken = token + const result = secureStorage.update(storageData) + if (!result.success) { + logForDebugging( + `[trusted-device] Failed to persist token: ${result.warning ?? 'unknown'}`, + ) + return + } + readStoredToken.cache?.clear?.() + logForDebugging( + `[trusted-device] Enrolled device_id=${response.data.device_id ?? 'unknown'}`, + ) + } catch (err: unknown) { + logForDebugging( + `[trusted-device] Storage write failed: ${errorMessage(err)}`, + ) + } + } catch (err: unknown) { + logForDebugging(`[trusted-device] Enrollment error: ${errorMessage(err)}`) + } +} diff --git a/src/bridge/types.ts b/src/bridge/types.ts new file mode 100644 index 0000000..210a3bb --- /dev/null +++ b/src/bridge/types.ts @@ -0,0 +1,262 @@ +/** Default per-session timeout (24 hours). */ +export const DEFAULT_SESSION_TIMEOUT_MS = 24 * 60 * 60 * 1000 + +/** Reusable login guidance appended to bridge auth errors. */ +export const BRIDGE_LOGIN_INSTRUCTION = + 'Remote Control is only available with claude.ai subscriptions. Please use `/login` to sign in with your claude.ai account.' + +/** Full error printed when `claude remote-control` is run without auth. */ +export const BRIDGE_LOGIN_ERROR = + 'Error: You must be logged in to use Remote Control.\n\n' + + BRIDGE_LOGIN_INSTRUCTION + +/** Shown when the user disconnects Remote Control (via /remote-control or ultraplan launch). */ +export const REMOTE_CONTROL_DISCONNECTED_MSG = 'Remote Control disconnected.' + +// --- Protocol types for the environments API --- + +export type WorkData = { + type: 'session' | 'healthcheck' + id: string +} + +export type WorkResponse = { + id: string + type: 'work' + environment_id: string + state: string + data: WorkData + secret: string // base64url-encoded JSON + created_at: string +} + +export type WorkSecret = { + version: number + session_ingress_token: string + api_base_url: string + sources: Array<{ + type: string + git_info?: { type: string; repo: string; ref?: string; token?: string } + }> + auth: Array<{ type: string; token: string }> + claude_code_args?: Record | null + mcp_config?: unknown | null + environment_variables?: Record | null + /** + * Server-driven CCR v2 selector. Set by prepare_work_secret() when the + * session was created via the v2 compat layer (ccr_v2_compat_enabled). + * Same field the BYOC runner reads at environment-runner/sessionExecutor.ts. + */ + use_code_sessions?: boolean +} + +export type SessionDoneStatus = 'completed' | 'failed' | 'interrupted' + +export type SessionActivityType = 'tool_start' | 'text' | 'result' | 'error' + +export type SessionActivity = { + type: SessionActivityType + summary: string // e.g. "Editing src/foo.ts", "Reading package.json" + timestamp: number +} + +/** + * How `claude remote-control` chooses session working directories. + * - `single-session`: one session in cwd, bridge tears down when it ends + * - `worktree`: persistent server, every session gets an isolated git worktree + * - `same-dir`: persistent server, every session shares cwd (can stomp each other) + */ +export type SpawnMode = 'single-session' | 'worktree' | 'same-dir' + +/** + * Well-known worker_type values THIS codebase produces. Sent as + * `metadata.worker_type` at environment registration so claude.ai can filter + * the session picker by origin (e.g. assistant tab only shows assistant + * workers). The backend treats this as an opaque string — desktop cowork + * sends `"cowork"`, which isn't in this union. REPL code uses this narrow + * type for its own exhaustiveness; wire-level fields accept any string. + */ +export type BridgeWorkerType = 'claude_code' | 'claude_code_assistant' + +export type BridgeConfig = { + dir: string + machineName: string + branch: string + gitRepoUrl: string | null + maxSessions: number + spawnMode: SpawnMode + verbose: boolean + sandbox: boolean + /** Client-generated UUID identifying this bridge instance. */ + bridgeId: string + /** + * Sent as metadata.worker_type so web clients can filter by origin. + * Backend treats this as opaque — any string, not just BridgeWorkerType. + */ + workerType: string + /** Client-generated UUID for idempotent environment registration. */ + environmentId: string + /** + * Backend-issued environment_id to reuse on re-register. When set, the + * backend treats registration as a reconnect to the existing environment + * instead of creating a new one. Used by `claude remote-control + * --session-id` resume. Must be a backend-format ID — client UUIDs are + * rejected with 400. + */ + reuseEnvironmentId?: string + /** API base URL the bridge is connected to (used for polling). */ + apiBaseUrl: string + /** Session ingress base URL for WebSocket connections (may differ from apiBaseUrl locally). */ + sessionIngressUrl: string + /** Debug file path passed via --debug-file. */ + debugFile?: string + /** Per-session timeout in milliseconds. Sessions exceeding this are killed. */ + sessionTimeoutMs?: number +} + +// --- Dependency interfaces (for testability) --- + +/** + * A control_response event sent back to a session (e.g. a permission decision). + * The `subtype` is `'success'` per the SDK protocol; the inner `response` + * carries the permission decision payload (e.g. `{ behavior: 'allow' }`). + */ +export type PermissionResponseEvent = { + type: 'control_response' + response: { + subtype: 'success' + request_id: string + response: Record + } +} + +export type BridgeApiClient = { + registerBridgeEnvironment(config: BridgeConfig): Promise<{ + environment_id: string + environment_secret: string + }> + pollForWork( + environmentId: string, + environmentSecret: string, + signal?: AbortSignal, + reclaimOlderThanMs?: number, + ): Promise + acknowledgeWork( + environmentId: string, + workId: string, + sessionToken: string, + ): Promise + /** Stop a work item via the environments API. */ + stopWork(environmentId: string, workId: string, force: boolean): Promise + /** Deregister/delete the bridge environment on graceful shutdown. */ + deregisterEnvironment(environmentId: string): Promise + /** Send a permission response (control_response) to a session via the session events API. */ + sendPermissionResponseEvent( + sessionId: string, + event: PermissionResponseEvent, + sessionToken: string, + ): Promise + /** Archive a session so it no longer appears as active on the server. */ + archiveSession(sessionId: string): Promise + /** + * Force-stop stale worker instances and re-queue a session on an environment. + * Used by `--session-id` to resume a session after the original bridge died. + */ + reconnectSession(environmentId: string, sessionId: string): Promise + /** + * Send a lightweight heartbeat for an active work item, extending its lease. + * Uses SessionIngressAuth (JWT, no DB hit) instead of EnvironmentSecretAuth. + * Returns the server's response with lease status. + */ + heartbeatWork( + environmentId: string, + workId: string, + sessionToken: string, + ): Promise<{ lease_extended: boolean; state: string }> +} + +export type SessionHandle = { + sessionId: string + done: Promise + kill(): void + forceKill(): void + activities: SessionActivity[] // ring buffer of recent activities (last ~10) + currentActivity: SessionActivity | null // most recent + accessToken: string // session_ingress_token for API calls + lastStderr: string[] // ring buffer of last stderr lines + writeStdin(data: string): void // write directly to child stdin + /** Update the access token for a running session (e.g. after token refresh). */ + updateAccessToken(token: string): void +} + +export type SessionSpawnOpts = { + sessionId: string + sdkUrl: string + accessToken: string + /** When true, spawn the child with CCR v2 env vars (SSE transport + CCRClient). */ + useCcrV2?: boolean + /** Required when useCcrV2 is true. Obtained from POST /worker/register. */ + workerEpoch?: number + /** + * Fires once with the text of the first real user message seen on the + * child's stdout (via --replay-user-messages). Lets the caller derive a + * session title when none exists yet. Tool-result and synthetic user + * messages are skipped. + */ + onFirstUserMessage?: (text: string) => void +} + +export type SessionSpawner = { + spawn(opts: SessionSpawnOpts, dir: string): SessionHandle +} + +export type BridgeLogger = { + printBanner(config: BridgeConfig, environmentId: string): void + logSessionStart(sessionId: string, prompt: string): void + logSessionComplete(sessionId: string, durationMs: number): void + logSessionFailed(sessionId: string, error: string): void + logStatus(message: string): void + logVerbose(message: string): void + logError(message: string): void + /** Log a reconnection success event after recovering from connection errors. */ + logReconnected(disconnectedMs: number): void + /** Show idle status with repo/branch info and shimmer animation. */ + updateIdleStatus(): void + /** Show reconnecting status in the live display. */ + updateReconnectingStatus(delayStr: string, elapsedStr: string): void + updateSessionStatus( + sessionId: string, + elapsed: string, + activity: SessionActivity, + trail: string[], + ): void + clearStatus(): void + /** Set repository info for status line display. */ + setRepoInfo(repoName: string, branch: string): void + /** Set debug log glob shown above the status line (ant users). */ + setDebugLogPath(path: string): void + /** Transition to "Attached" state when a session starts. */ + setAttached(sessionId: string): void + /** Show failed status in the live display. */ + updateFailedStatus(error: string): void + /** Toggle QR code visibility. */ + toggleQr(): void + /** Update the " of sessions" indicator and spawn mode hint. */ + updateSessionCount(active: number, max: number, mode: SpawnMode): void + /** Update the spawn mode shown in the session-count line. Pass null to hide (single-session or toggle unavailable). */ + setSpawnModeDisplay(mode: 'same-dir' | 'worktree' | null): void + /** Register a new session for multi-session display (called after spawn succeeds). */ + addSession(sessionId: string, url: string): void + /** Update the per-session activity summary (tool being run) in the multi-session list. */ + updateSessionActivity(sessionId: string, activity: SessionActivity): void + /** + * Set a session's display title. In multi-session mode, updates the bullet list + * entry. In single-session mode, also shows the title in the main status line. + * Triggers a render (guarded against reconnecting/failed states). + */ + setSessionTitle(sessionId: string, title: string): void + /** Remove a session from the multi-session display when it ends. */ + removeSession(sessionId: string): void + /** Force a re-render of the status display (for multi-session activity refresh). */ + refreshDisplay(): void +} diff --git a/src/bridge/workSecret.ts b/src/bridge/workSecret.ts new file mode 100644 index 0000000..bbc9373 --- /dev/null +++ b/src/bridge/workSecret.ts @@ -0,0 +1,127 @@ +import axios from 'axios' +import { jsonParse, jsonStringify } from '../utils/slowOperations.js' +import type { WorkSecret } from './types.js' + +/** Decode a base64url-encoded work secret and validate its version. */ +export function decodeWorkSecret(secret: string): WorkSecret { + const json = Buffer.from(secret, 'base64url').toString('utf-8') + const parsed: unknown = jsonParse(json) + if ( + !parsed || + typeof parsed !== 'object' || + !('version' in parsed) || + parsed.version !== 1 + ) { + throw new Error( + `Unsupported work secret version: ${parsed && typeof parsed === 'object' && 'version' in parsed ? parsed.version : 'unknown'}`, + ) + } + const obj = parsed as Record + if ( + typeof obj.session_ingress_token !== 'string' || + obj.session_ingress_token.length === 0 + ) { + throw new Error( + 'Invalid work secret: missing or empty session_ingress_token', + ) + } + if (typeof obj.api_base_url !== 'string') { + throw new Error('Invalid work secret: missing api_base_url') + } + return parsed as WorkSecret +} + +/** + * Build a WebSocket SDK URL from the API base URL and session ID. + * Strips the HTTP(S) protocol and constructs a ws(s):// ingress URL. + * + * Uses /v2/ for localhost (direct to session-ingress, no Envoy rewrite) + * and /v1/ for production (Envoy rewrites /v1/ → /v2/). + */ +export function buildSdkUrl(apiBaseUrl: string, sessionId: string): string { + const isLocalhost = + apiBaseUrl.includes('localhost') || apiBaseUrl.includes('127.0.0.1') + const protocol = isLocalhost ? 'ws' : 'wss' + const version = isLocalhost ? 'v2' : 'v1' + const host = apiBaseUrl.replace(/^https?:\/\//, '').replace(/\/+$/, '') + return `${protocol}://${host}/${version}/session_ingress/ws/${sessionId}` +} + +/** + * Compare two session IDs regardless of their tagged-ID prefix. + * + * Tagged IDs have the form {tag}_{body} or {tag}_staging_{body}, where the + * body encodes a UUID. CCR v2's compat layer returns `session_*` to v1 API + * clients (compat/convert.go:41) but the infrastructure layer (sandbox-gateway + * work queue, work poll response) uses `cse_*` (compat/CLAUDE.md:13). Both + * have the same underlying UUID. + * + * Without this, replBridge rejects its own session as "foreign" at the + * work-received check when the ccr_v2_compat_enabled gate is on. + */ +export function sameSessionId(a: string, b: string): boolean { + if (a === b) return true + // The body is everything after the last underscore — this handles both + // `{tag}_{body}` and `{tag}_staging_{body}`. + const aBody = a.slice(a.lastIndexOf('_') + 1) + const bBody = b.slice(b.lastIndexOf('_') + 1) + // Guard against IDs with no underscore (bare UUIDs): lastIndexOf returns -1, + // slice(0) returns the whole string, and we already checked a === b above. + // Require a minimum length to avoid accidental matches on short suffixes + // (e.g. single-char tag remnants from malformed IDs). + return aBody.length >= 4 && aBody === bBody +} + +/** + * Build a CCR v2 session URL from the API base URL and session ID. + * Unlike buildSdkUrl, this returns an HTTP(S) URL (not ws://) and points at + * /v1/code/sessions/{id} — the child CC will derive the SSE stream path + * and worker endpoints from this base. + */ +export function buildCCRv2SdkUrl( + apiBaseUrl: string, + sessionId: string, +): string { + const base = apiBaseUrl.replace(/\/+$/, '') + return `${base}/v1/code/sessions/${sessionId}` +} + +/** + * Register this bridge as the worker for a CCR v2 session. + * Returns the worker_epoch, which must be passed to the child CC process + * so its CCRClient can include it in every heartbeat/state/event request. + * + * Mirrors what environment-manager does in the container path + * (api-go/environment-manager/cmd/cmd_task_run.go RegisterWorker). + */ +export async function registerWorker( + sessionUrl: string, + accessToken: string, +): Promise { + const response = await axios.post( + `${sessionUrl}/worker/register`, + {}, + { + headers: { + Authorization: `Bearer ${accessToken}`, + 'Content-Type': 'application/json', + 'anthropic-version': '2023-06-01', + }, + timeout: 10_000, + }, + ) + // protojson serializes int64 as a string to avoid JS number precision loss; + // the Go side may also return a number depending on encoder settings. + const raw = response.data?.worker_epoch + const epoch = typeof raw === 'string' ? Number(raw) : raw + if ( + typeof epoch !== 'number' || + !Number.isFinite(epoch) || + !Number.isSafeInteger(epoch) + ) { + throw new Error( + `registerWorker: invalid worker_epoch in response: ${jsonStringify(response.data)}`, + ) + } + return epoch +} diff --git a/src/buddy/CompanionSprite.tsx b/src/buddy/CompanionSprite.tsx new file mode 100644 index 0000000..f7f1f72 --- /dev/null +++ b/src/buddy/CompanionSprite.tsx @@ -0,0 +1,371 @@ +import { c as _c } from "react/compiler-runtime"; +import { feature } from 'bun:bundle'; +import figures from 'figures'; +import React, { useEffect, useRef, useState } from 'react'; +import { useTerminalSize } from '../hooks/useTerminalSize.js'; +import { stringWidth } from '../ink/stringWidth.js'; +import { Box, Text } from '../ink.js'; +import { useAppState, useSetAppState } from '../state/AppState.js'; +import type { AppState } from '../state/AppStateStore.js'; +import { getGlobalConfig } from '../utils/config.js'; +import { isFullscreenActive } from '../utils/fullscreen.js'; +import type { Theme } from '../utils/theme.js'; +import { getCompanion } from './companion.js'; +import { renderFace, renderSprite, spriteFrameCount } from './sprites.js'; +import { RARITY_COLORS } from './types.js'; +const TICK_MS = 500; +const BUBBLE_SHOW = 20; // ticks → ~10s at 500ms +const FADE_WINDOW = 6; // last ~3s the bubble dims so you know it's about to go +const PET_BURST_MS = 2500; // how long hearts float after /buddy pet + +// Idle sequence: mostly rest (frame 0), occasional fidget (frames 1-2), rare blink. +// Sequence indices map to sprite frames; -1 means "blink on frame 0". +const IDLE_SEQUENCE = [0, 0, 0, 0, 1, 0, 0, 0, -1, 0, 0, 2, 0, 0, 0]; + +// Hearts float up-and-out over 5 ticks (~2.5s). Prepended above the sprite. +const H = figures.heart; +const PET_HEARTS = [` ${H} ${H} `, ` ${H} ${H} ${H} `, ` ${H} ${H} ${H} `, `${H} ${H} ${H} `, '· · · ']; +function wrap(text: string, width: number): string[] { + const words = text.split(' '); + const lines: string[] = []; + let cur = ''; + for (const w of words) { + if (cur.length + w.length + 1 > width && cur) { + lines.push(cur); + cur = w; + } else { + cur = cur ? `${cur} ${w}` : w; + } + } + if (cur) lines.push(cur); + return lines; +} +function SpeechBubble(t0) { + const $ = _c(31); + const { + text, + color, + fading, + tail + } = t0; + let T0; + let borderColor; + let t1; + let t2; + let t3; + let t4; + let t5; + let t6; + if ($[0] !== color || $[1] !== fading || $[2] !== text) { + const lines = wrap(text, 30); + borderColor = fading ? "inactive" : color; + T0 = Box; + t1 = "column"; + t2 = "round"; + t3 = borderColor; + t4 = 1; + t5 = 34; + let t7; + if ($[11] !== fading) { + t7 = (l, i) => {l}; + $[11] = fading; + $[12] = t7; + } else { + t7 = $[12]; + } + t6 = lines.map(t7); + $[0] = color; + $[1] = fading; + $[2] = text; + $[3] = T0; + $[4] = borderColor; + $[5] = t1; + $[6] = t2; + $[7] = t3; + $[8] = t4; + $[9] = t5; + $[10] = t6; + } else { + T0 = $[3]; + borderColor = $[4]; + t1 = $[5]; + t2 = $[6]; + t3 = $[7]; + t4 = $[8]; + t5 = $[9]; + t6 = $[10]; + } + let t7; + if ($[13] !== T0 || $[14] !== t1 || $[15] !== t2 || $[16] !== t3 || $[17] !== t4 || $[18] !== t5 || $[19] !== t6) { + t7 = {t6}; + $[13] = T0; + $[14] = t1; + $[15] = t2; + $[16] = t3; + $[17] = t4; + $[18] = t5; + $[19] = t6; + $[20] = t7; + } else { + t7 = $[20]; + } + const bubble = t7; + if (tail === "right") { + let t8; + if ($[21] !== borderColor) { + t8 = ; + $[21] = borderColor; + $[22] = t8; + } else { + t8 = $[22]; + } + let t9; + if ($[23] !== bubble || $[24] !== t8) { + t9 = {bubble}{t8}; + $[23] = bubble; + $[24] = t8; + $[25] = t9; + } else { + t9 = $[25]; + } + return t9; + } + let t8; + if ($[26] !== borderColor) { + t8 = ; + $[26] = borderColor; + $[27] = t8; + } else { + t8 = $[27]; + } + let t9; + if ($[28] !== bubble || $[29] !== t8) { + t9 = {bubble}{t8}; + $[28] = bubble; + $[29] = t8; + $[30] = t9; + } else { + t9 = $[30]; + } + return t9; +} +export const MIN_COLS_FOR_FULL_SPRITE = 100; +const SPRITE_BODY_WIDTH = 12; +const NAME_ROW_PAD = 2; // focused state wraps name in spaces: ` name ` +const SPRITE_PADDING_X = 2; +const BUBBLE_WIDTH = 36; // SpeechBubble box (34) + tail column +const NARROW_QUIP_CAP = 24; +function spriteColWidth(nameWidth: number): number { + return Math.max(SPRITE_BODY_WIDTH, nameWidth + NAME_ROW_PAD); +} + +// Width the sprite area consumes. PromptInput subtracts this so text wraps +// correctly. In fullscreen the bubble floats over scrollback (no extra +// width); in non-fullscreen it sits inline and needs BUBBLE_WIDTH more. +// Narrow terminals: 0 — REPL.tsx stacks the one-liner on its own row +// (above input in fullscreen, below in scrollback), so no reservation. +export function companionReservedColumns(terminalColumns: number, speaking: boolean): number { + if (!feature('BUDDY')) return 0; + const companion = getCompanion(); + if (!companion || getGlobalConfig().companionMuted) return 0; + if (terminalColumns < MIN_COLS_FOR_FULL_SPRITE) return 0; + const nameWidth = stringWidth(companion.name); + const bubble = speaking && !isFullscreenActive() ? BUBBLE_WIDTH : 0; + return spriteColWidth(nameWidth) + SPRITE_PADDING_X + bubble; +} +export function CompanionSprite(): React.ReactNode { + const reaction = useAppState(s => s.companionReaction); + const petAt = useAppState(s => s.companionPetAt); + const focused = useAppState(s => s.footerSelection === 'companion'); + const setAppState = useSetAppState(); + const { + columns + } = useTerminalSize(); + const [tick, setTick] = useState(0); + const lastSpokeTick = useRef(0); + // Sync-during-render (not useEffect) so the first post-pet render already + // has petStartTick=tick and petAge=0 — otherwise frame 0 is skipped. + const [{ + petStartTick, + forPetAt + }, setPetStart] = useState({ + petStartTick: 0, + forPetAt: petAt + }); + if (petAt !== forPetAt) { + setPetStart({ + petStartTick: tick, + forPetAt: petAt + }); + } + useEffect(() => { + const timer = setInterval(setT => setT((t: number) => t + 1), TICK_MS, setTick); + return () => clearInterval(timer); + }, []); + useEffect(() => { + if (!reaction) return; + lastSpokeTick.current = tick; + const timer = setTimeout(setA => setA((prev: AppState) => prev.companionReaction === undefined ? prev : { + ...prev, + companionReaction: undefined + }), BUBBLE_SHOW * TICK_MS, setAppState); + return () => clearTimeout(timer); + // eslint-disable-next-line react-hooks/exhaustive-deps -- tick intentionally captured at reaction-change, not tracked + }, [reaction, setAppState]); + if (!feature('BUDDY')) return null; + const companion = getCompanion(); + if (!companion || getGlobalConfig().companionMuted) return null; + const color = RARITY_COLORS[companion.rarity]; + const colWidth = spriteColWidth(stringWidth(companion.name)); + const bubbleAge = reaction ? tick - lastSpokeTick.current : 0; + const fading = reaction !== undefined && bubbleAge >= BUBBLE_SHOW - FADE_WINDOW; + const petAge = petAt ? tick - petStartTick : Infinity; + const petting = petAge * TICK_MS < PET_BURST_MS; + + // Narrow terminals: collapse to one-line face. When speaking, the quip + // replaces the name beside the face (no room for a bubble). + if (columns < MIN_COLS_FOR_FULL_SPRITE) { + const quip = reaction && reaction.length > NARROW_QUIP_CAP ? reaction.slice(0, NARROW_QUIP_CAP - 1) + '…' : reaction; + const label = quip ? `"${quip}"` : focused ? ` ${companion.name} ` : companion.name; + return + + {petting && {figures.heart} } + + {renderFace(companion)} + {' '} + + {label} + + + ; + } + const frameCount = spriteFrameCount(companion.species); + const heartFrame = petting ? PET_HEARTS[petAge % PET_HEARTS.length] : null; + let spriteFrame: number; + let blink = false; + if (reaction || petting) { + // Excited: cycle all fidget frames fast + spriteFrame = tick % frameCount; + } else { + const step = IDLE_SEQUENCE[tick % IDLE_SEQUENCE.length]!; + if (step === -1) { + spriteFrame = 0; + blink = true; + } else { + spriteFrame = step % frameCount; + } + } + const body = renderSprite(companion, spriteFrame).map(line => blink ? line.replaceAll(companion.eye, '-') : line); + const sprite = heartFrame ? [heartFrame, ...body] : body; + + // Name row doubles as hint row — unfocused shows dim name + ↓ discovery, + // focused shows inverse name. The enter-to-open hint lives in + // PromptInputFooter's right column so this row stays one line and the + // sprite doesn't jump up when selected. flexShrink=0 stops the + // inline-bubble row wrapper from squeezing the sprite to fit. + const spriteColumn = + {sprite.map((line, i) => + {line} + )} + + {focused ? ` ${companion.name} ` : companion.name} + + ; + if (!reaction) { + return {spriteColumn}; + } + + // Fullscreen: bubble renders separately via CompanionFloatingBubble in + // FullscreenLayout's bottomFloat slot (the bottom slot's overflowY:hidden + // would clip a position:absolute overlay here). Sprite body only. + // Non-fullscreen: bubble sits inline beside the sprite (input shrinks) + // because floating into Static scrollback can't be cleared. + if (isFullscreenActive()) { + return {spriteColumn}; + } + return + + {spriteColumn} + ; +} + +// Floating bubble overlay for fullscreen mode. Mounted in FullscreenLayout's +// bottomFloat slot (outside the overflowY:hidden clip) so it can extend into +// the ScrollBox region. CompanionSprite owns the clear-after-10s timer; this +// just reads companionReaction and renders the fade. +export function CompanionFloatingBubble() { + const $ = _c(8); + const reaction = useAppState(_temp); + let t0; + if ($[0] !== reaction) { + t0 = { + tick: 0, + forReaction: reaction + }; + $[0] = reaction; + $[1] = t0; + } else { + t0 = $[1]; + } + const [t1, setTick] = useState(t0); + const { + tick, + forReaction + } = t1; + if (reaction !== forReaction) { + setTick({ + tick: 0, + forReaction: reaction + }); + } + let t2; + let t3; + if ($[2] !== reaction) { + t2 = () => { + if (!reaction) { + return; + } + const timer = setInterval(_temp3, TICK_MS, setTick); + return () => clearInterval(timer); + }; + t3 = [reaction]; + $[2] = reaction; + $[3] = t2; + $[4] = t3; + } else { + t2 = $[3]; + t3 = $[4]; + } + useEffect(t2, t3); + if (!feature("BUDDY") || !reaction) { + return null; + } + const companion = getCompanion(); + if (!companion || getGlobalConfig().companionMuted) { + return null; + } + const t4 = tick >= BUBBLE_SHOW - FADE_WINDOW; + let t5; + if ($[5] !== reaction || $[6] !== t4) { + t5 = ; + $[5] = reaction; + $[6] = t4; + $[7] = t5; + } else { + t5 = $[7]; + } + return t5; +} +function _temp3(set) { + return set(_temp2); +} +function _temp2(s_0) { + return { + ...s_0, + tick: s_0.tick + 1 + }; +} +function _temp(s) { + return s.companionReaction; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJmZWF0dXJlIiwiZmlndXJlcyIsIlJlYWN0IiwidXNlRWZmZWN0IiwidXNlUmVmIiwidXNlU3RhdGUiLCJ1c2VUZXJtaW5hbFNpemUiLCJzdHJpbmdXaWR0aCIsIkJveCIsIlRleHQiLCJ1c2VBcHBTdGF0ZSIsInVzZVNldEFwcFN0YXRlIiwiQXBwU3RhdGUiLCJnZXRHbG9iYWxDb25maWciLCJpc0Z1bGxzY3JlZW5BY3RpdmUiLCJUaGVtZSIsImdldENvbXBhbmlvbiIsInJlbmRlckZhY2UiLCJyZW5kZXJTcHJpdGUiLCJzcHJpdGVGcmFtZUNvdW50IiwiUkFSSVRZX0NPTE9SUyIsIlRJQ0tfTVMiLCJCVUJCTEVfU0hPVyIsIkZBREVfV0lORE9XIiwiUEVUX0JVUlNUX01TIiwiSURMRV9TRVFVRU5DRSIsIkgiLCJoZWFydCIsIlBFVF9IRUFSVFMiLCJ3cmFwIiwidGV4dCIsIndpZHRoIiwid29yZHMiLCJzcGxpdCIsImxpbmVzIiwiY3VyIiwidyIsImxlbmd0aCIsInB1c2giLCJTcGVlY2hCdWJibGUiLCJ0MCIsIiQiLCJfYyIsImNvbG9yIiwiZmFkaW5nIiwidGFpbCIsIlQwIiwiYm9yZGVyQ29sb3IiLCJ0MSIsInQyIiwidDMiLCJ0NCIsInQ1IiwidDYiLCJ0NyIsImwiLCJpIiwidW5kZWZpbmVkIiwibWFwIiwiYnViYmxlIiwidDgiLCJ0OSIsIk1JTl9DT0xTX0ZPUl9GVUxMX1NQUklURSIsIlNQUklURV9CT0RZX1dJRFRIIiwiTkFNRV9ST1dfUEFEIiwiU1BSSVRFX1BBRERJTkdfWCIsIkJVQkJMRV9XSURUSCIsIk5BUlJPV19RVUlQX0NBUCIsInNwcml0ZUNvbFdpZHRoIiwibmFtZVdpZHRoIiwiTWF0aCIsIm1heCIsImNvbXBhbmlvblJlc2VydmVkQ29sdW1ucyIsInRlcm1pbmFsQ29sdW1ucyIsInNwZWFraW5nIiwiY29tcGFuaW9uIiwiY29tcGFuaW9uTXV0ZWQiLCJuYW1lIiwiQ29tcGFuaW9uU3ByaXRlIiwiUmVhY3ROb2RlIiwicmVhY3Rpb24iLCJzIiwiY29tcGFuaW9uUmVhY3Rpb24iLCJwZXRBdCIsImNvbXBhbmlvblBldEF0IiwiZm9jdXNlZCIsImZvb3RlclNlbGVjdGlvbiIsInNldEFwcFN0YXRlIiwiY29sdW1ucyIsInRpY2siLCJzZXRUaWNrIiwibGFzdFNwb2tlVGljayIsInBldFN0YXJ0VGljayIsImZvclBldEF0Iiwic2V0UGV0U3RhcnQiLCJ0aW1lciIsInNldEludGVydmFsIiwic2V0VCIsInQiLCJjbGVhckludGVydmFsIiwiY3VycmVudCIsInNldFRpbWVvdXQiLCJzZXRBIiwicHJldiIsImNsZWFyVGltZW91dCIsInJhcml0eSIsImNvbFdpZHRoIiwiYnViYmxlQWdlIiwicGV0QWdlIiwiSW5maW5pdHkiLCJwZXR0aW5nIiwicXVpcCIsInNsaWNlIiwibGFiZWwiLCJmcmFtZUNvdW50Iiwic3BlY2llcyIsImhlYXJ0RnJhbWUiLCJzcHJpdGVGcmFtZSIsImJsaW5rIiwic3RlcCIsImJvZHkiLCJsaW5lIiwicmVwbGFjZUFsbCIsImV5ZSIsInNwcml0ZSIsInNwcml0ZUNvbHVtbiIsIkNvbXBhbmlvbkZsb2F0aW5nQnViYmxlIiwiX3RlbXAiLCJmb3JSZWFjdGlvbiIsIl90ZW1wMyIsInNldCIsIl90ZW1wMiIsInNfMCJdLCJzb3VyY2VzIjpbIkNvbXBhbmlvblNwcml0ZS50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgZmVhdHVyZSB9IGZyb20gJ2J1bjpidW5kbGUnXG5pbXBvcnQgZmlndXJlcyBmcm9tICdmaWd1cmVzJ1xuaW1wb3J0IFJlYWN0LCB7IHVzZUVmZmVjdCwgdXNlUmVmLCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgdXNlVGVybWluYWxTaXplIH0gZnJvbSAnLi4vaG9va3MvdXNlVGVybWluYWxTaXplLmpzJ1xuaW1wb3J0IHsgc3RyaW5nV2lkdGggfSBmcm9tICcuLi9pbmsvc3RyaW5nV2lkdGguanMnXG5pbXBvcnQgeyBCb3gsIFRleHQgfSBmcm9tICcuLi9pbmsuanMnXG5pbXBvcnQgeyB1c2VBcHBTdGF0ZSwgdXNlU2V0QXBwU3RhdGUgfSBmcm9tICcuLi9zdGF0ZS9BcHBTdGF0ZS5qcydcbmltcG9ydCB0eXBlIHsgQXBwU3RhdGUgfSBmcm9tICcuLi9zdGF0ZS9BcHBTdGF0ZVN0b3JlLmpzJ1xuaW1wb3J0IHsgZ2V0R2xvYmFsQ29uZmlnIH0gZnJvbSAnLi4vdXRpbHMvY29uZmlnLmpzJ1xuaW1wb3J0IHsgaXNGdWxsc2NyZWVuQWN0aXZlIH0gZnJvbSAnLi4vdXRpbHMvZnVsbHNjcmVlbi5qcydcbmltcG9ydCB0eXBlIHsgVGhlbWUgfSBmcm9tICcuLi91dGlscy90aGVtZS5qcydcbmltcG9ydCB7IGdldENvbXBhbmlvbiB9IGZyb20gJy4vY29tcGFuaW9uLmpzJ1xuaW1wb3J0IHsgcmVuZGVyRmFjZSwgcmVuZGVyU3ByaXRlLCBzcHJpdGVGcmFtZUNvdW50IH0gZnJvbSAnLi9zcHJpdGVzLmpzJ1xuaW1wb3J0IHsgUkFSSVRZX0NPTE9SUyB9IGZyb20gJy4vdHlwZXMuanMnXG5cbmNvbnN0IFRJQ0tfTVMgPSA1MDBcbmNvbnN0IEJVQkJMRV9TSE9XID0gMjAgLy8gdGlja3Mg4oaSIH4xMHMgYXQgNTAwbXNcbmNvbnN0IEZBREVfV0lORE9XID0gNiAvLyBsYXN0IH4zcyB0aGUgYnViYmxlIGRpbXMgc28geW91IGtub3cgaXQncyBhYm91dCB0byBnb1xuY29uc3QgUEVUX0JVUlNUX01TID0gMjUwMCAvLyBob3cgbG9uZyBoZWFydHMgZmxvYXQgYWZ0ZXIgL2J1ZGR5IHBldFxuXG4vLyBJZGxlIHNlcXVlbmNlOiBtb3N0bHkgcmVzdCAoZnJhbWUgMCksIG9jY2FzaW9uYWwgZmlkZ2V0IChmcmFtZXMgMS0yKSwgcmFyZSBibGluay5cbi8vIFNlcXVlbmNlIGluZGljZXMgbWFwIHRvIHNwcml0ZSBmcmFtZXM7IC0xIG1lYW5zIFwiYmxpbmsgb24gZnJhbWUgMFwiLlxuY29uc3QgSURMRV9TRVFVRU5DRSA9IFswLCAwLCAwLCAwLCAxLCAwLCAwLCAwLCAtMSwgMCwgMCwgMiwgMCwgMCwgMF1cblxuLy8gSGVhcnRzIGZsb2F0IHVwLWFuZC1vdXQgb3ZlciA1IHRpY2tzICh+Mi41cykuIFByZXBlbmRlZCBhYm92ZSB0aGUgc3ByaXRlLlxuY29uc3QgSCA9IGZpZ3VyZXMuaGVhcnRcbmNvbnN0IFBFVF9IRUFSVFMgPSBbXG4gIGAgICAke0h9ICAgICR7SH0gICBgLFxuICBgICAke0h9ICAke0h9ICAgJHtIfSAgYCxcbiAgYCAke0h9ICAgJHtIfSAgJHtIfSAgIGAsXG4gIGAke0h9ICAke0h9ICAgICAgJHtIfSBgLFxuICAnwrcgICAgwrcgICDCtyAgJyxcbl1cblxuZnVuY3Rpb24gd3JhcCh0ZXh0OiBzdHJpbmcsIHdpZHRoOiBudW1iZXIpOiBzdHJpbmdbXSB7XG4gIGNvbnN0IHdvcmRzID0gdGV4dC5zcGxpdCgnICcpXG4gIGNvbnN0IGxpbmVzOiBzdHJpbmdbXSA9IFtdXG4gIGxldCBjdXIgPSAnJ1xuICBmb3IgKGNvbnN0IHcgb2Ygd29yZHMpIHtcbiAgICBpZiAoY3VyLmxlbmd0aCArIHcubGVuZ3RoICsgMSA+IHdpZHRoICYmIGN1cikge1xuICAgICAgbGluZXMucHVzaChjdXIpXG4gICAgICBjdXIgPSB3XG4gICAgfSBlbHNlIHtcbiAgICAgIGN1ciA9IGN1ciA/IGAke2N1cn0gJHt3fWAgOiB3XG4gICAgfVxuICB9XG4gIGlmIChjdXIpIGxpbmVzLnB1c2goY3VyKVxuICByZXR1cm4gbGluZXNcbn1cblxuZnVuY3Rpb24gU3BlZWNoQnViYmxlKHtcbiAgdGV4dCxcbiAgY29sb3IsXG4gIGZhZGluZyxcbiAgdGFpbCxcbn06IHtcbiAgdGV4dDogc3RyaW5nXG4gIGNvbG9yOiBrZXlvZiBUaGVtZVxuICBmYWRpbmc6IGJvb2xlYW5cbiAgdGFpbDogJ2Rvd24nIHwgJ3JpZ2h0J1xufSk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IGxpbmVzID0gd3JhcCh0ZXh0LCAzMClcbiAgY29uc3QgYm9yZGVyQ29sb3IgPSBmYWRpbmcgPyAnaW5hY3RpdmUnIDogY29sb3JcbiAgY29uc3QgYnViYmxlID0gKFxuICAgIDxCb3hcbiAgICAgIGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIlxuICAgICAgYm9yZGVyU3R5bGU9XCJyb3VuZFwiXG4gICAgICBib3JkZXJDb2xvcj17Ym9yZGVyQ29sb3J9XG4gICAgICBwYWRkaW5nWD17MX1cbiAgICAgIHdpZHRoPXszNH1cbiAgICA+XG4gICAgICB7bGluZXMubWFwKChsLCBpKSA9PiAoXG4gICAgICAgIDxUZXh0XG4gICAgICAgICAga2V5PXtpfVxuICAgICAgICAgIGl0YWxpY1xuICAgICAgICAgIGRpbUNvbG9yPXshZmFkaW5nfVxuICAgICAgICAgIGNvbG9yPXtmYWRpbmcgPyAnaW5hY3RpdmUnIDogdW5kZWZpbmVkfVxuICAgICAgICA+XG4gICAgICAgICAge2x9XG4gICAgICAgIDwvVGV4dD5cbiAgICAgICkpfVxuICAgIDwvQm94PlxuICApXG4gIGlmICh0YWlsID09PSAncmlnaHQnKSB7XG4gICAgcmV0dXJuIChcbiAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cInJvd1wiIGFsaWduSXRlbXM9XCJjZW50ZXJcIj5cbiAgICAgICAge2J1YmJsZX1cbiAgICAgICAgPFRleHQgY29sb3I9e2JvcmRlckNvbG9yfT7ilIA8L1RleHQ+XG4gICAgICA8L0JveD5cbiAgICApXG4gIH1cbiAgcmV0dXJuIChcbiAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIiBhbGlnbkl0ZW1zPVwiZmxleC1lbmRcIiBtYXJnaW5SaWdodD17MX0+XG4gICAgICB7YnViYmxlfVxuICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCIgYWxpZ25JdGVtcz1cImZsZXgtZW5kXCIgcGFkZGluZ1JpZ2h0PXs2fT5cbiAgICAgICAgPFRleHQgY29sb3I9e2JvcmRlckNvbG9yfT7ilbIgPC9UZXh0PlxuICAgICAgICA8VGV4dCBjb2xvcj17Ym9yZGVyQ29sb3J9PuKVsjwvVGV4dD5cbiAgICAgIDwvQm94PlxuICAgIDwvQm94PlxuICApXG59XG5cbmV4cG9ydCBjb25zdCBNSU5fQ09MU19GT1JfRlVMTF9TUFJJVEUgPSAxMDBcbmNvbnN0IFNQUklURV9CT0RZX1dJRFRIID0gMTJcbmNvbnN0IE5BTUVfUk9XX1BBRCA9IDIgLy8gZm9jdXNlZCBzdGF0ZSB3cmFwcyBuYW1lIGluIHNwYWNlczogYCBuYW1lIGBcbmNvbnN0IFNQUklURV9QQURESU5HX1ggPSAyXG5jb25zdCBCVUJCTEVfV0lEVEggPSAzNiAvLyBTcGVlY2hCdWJibGUgYm94ICgzNCkgKyB0YWlsIGNvbHVtblxuY29uc3QgTkFSUk9XX1FVSVBfQ0FQID0gMjRcblxuZnVuY3Rpb24gc3ByaXRlQ29sV2lkdGgobmFtZVdpZHRoOiBudW1iZXIpOiBudW1iZXIge1xuICByZXR1cm4gTWF0aC5tYXgoU1BSSVRFX0JPRFlfV0lEVEgsIG5hbWVXaWR0aCArIE5BTUVfUk9XX1BBRClcbn1cblxuLy8gV2lkdGggdGhlIHNwcml0ZSBhcmVhIGNvbnN1bWVzLiBQcm9tcHRJbnB1dCBzdWJ0cmFjdHMgdGhpcyBzbyB0ZXh0IHdyYXBzXG4vLyBjb3JyZWN0bHkuIEluIGZ1bGxzY3JlZW4gdGhlIGJ1YmJsZSBmbG9hdHMgb3ZlciBzY3JvbGxiYWNrIChubyBleHRyYVxuLy8gd2lkdGgpOyBpbiBub24tZnVsbHNjcmVlbiBpdCBzaXRzIGlubGluZSBhbmQgbmVlZHMgQlVCQkxFX1dJRFRIIG1vcmUuXG4vLyBOYXJyb3cgdGVybWluYWxzOiAwIOKAlCBSRVBMLnRzeCBzdGFja3MgdGhlIG9uZS1saW5lciBvbiBpdHMgb3duIHJvd1xuLy8gKGFib3ZlIGlucHV0IGluIGZ1bGxzY3JlZW4sIGJlbG93IGluIHNjcm9sbGJhY2spLCBzbyBubyByZXNlcnZhdGlvbi5cbmV4cG9ydCBmdW5jdGlvbiBjb21wYW5pb25SZXNlcnZlZENvbHVtbnMoXG4gIHRlcm1pbmFsQ29sdW1uczogbnVtYmVyLFxuICBzcGVha2luZzogYm9vbGVhbixcbik6IG51bWJlciB7XG4gIGlmICghZmVhdHVyZSgnQlVERFknKSkgcmV0dXJuIDBcbiAgY29uc3QgY29tcGFuaW9uID0gZ2V0Q29tcGFuaW9uKClcbiAgaWYgKCFjb21wYW5pb24gfHwgZ2V0R2xvYmFsQ29uZmlnKCkuY29tcGFuaW9uTXV0ZWQpIHJldHVybiAwXG4gIGlmICh0ZXJtaW5hbENvbHVtbnMgPCBNSU5fQ09MU19GT1JfRlVMTF9TUFJJVEUpIHJldHVybiAwXG4gIGNvbnN0IG5hbWVXaWR0aCA9IHN0cmluZ1dpZHRoKGNvbXBhbmlvbi5uYW1lKVxuICBjb25zdCBidWJibGUgPSBzcGVha2luZyAmJiAhaXNGdWxsc2NyZWVuQWN0aXZlKCkgPyBCVUJCTEVfV0lEVEggOiAwXG4gIHJldHVybiBzcHJpdGVDb2xXaWR0aChuYW1lV2lkdGgpICsgU1BSSVRFX1BBRERJTkdfWCArIGJ1YmJsZVxufVxuXG5leHBvcnQgZnVuY3Rpb24gQ29tcGFuaW9uU3ByaXRlKCk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IHJlYWN0aW9uID0gdXNlQXBwU3RhdGUocyA9PiBzLmNvbXBhbmlvblJlYWN0aW9uKVxuICBjb25zdCBwZXRBdCA9IHVzZUFwcFN0YXRlKHMgPT4gcy5jb21wYW5pb25QZXRBdClcbiAgY29uc3QgZm9jdXNlZCA9IHVzZUFwcFN0YXRlKHMgPT4gcy5mb290ZXJTZWxlY3Rpb24gPT09ICdjb21wYW5pb24nKVxuICBjb25zdCBzZXRBcHBTdGF0ZSA9IHVzZVNldEFwcFN0YXRlKClcbiAgY29uc3QgeyBjb2x1bW5zIH0gPSB1c2VUZXJtaW5hbFNpemUoKVxuICBjb25zdCBbdGljaywgc2V0VGlja10gPSB1c2VTdGF0ZSgwKVxuICBjb25zdCBsYXN0U3Bva2VUaWNrID0gdXNlUmVmKDApXG4gIC8vIFN5bmMtZHVyaW5nLXJlbmRlciAobm90IHVzZUVmZmVjdCkgc28gdGhlIGZpcnN0IHBvc3QtcGV0IHJlbmRlciBhbHJlYWR5XG4gIC8vIGhhcyBwZXRTdGFydFRpY2s9dGljayBhbmQgcGV0QWdlPTAg4oCUIG90aGVyd2lzZSBmcmFtZSAwIGlzIHNraXBwZWQuXG4gIGNvbnN0IFt7IHBldFN0YXJ0VGljaywgZm9yUGV0QXQgfSwgc2V0UGV0U3RhcnRdID0gdXNlU3RhdGUoe1xuICAgIHBldFN0YXJ0VGljazogMCxcbiAgICBmb3JQZXRBdDogcGV0QXQsXG4gIH0pXG4gIGlmIChwZXRBdCAhPT0gZm9yUGV0QXQpIHtcbiAgICBzZXRQZXRTdGFydCh7IHBldFN0YXJ0VGljazogdGljaywgZm9yUGV0QXQ6IHBldEF0IH0pXG4gIH1cblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGNvbnN0IHRpbWVyID0gc2V0SW50ZXJ2YWwoXG4gICAgICBzZXRUID0+IHNldFQoKHQ6IG51bWJlcikgPT4gdCArIDEpLFxuICAgICAgVElDS19NUyxcbiAgICAgIHNldFRpY2ssXG4gICAgKVxuICAgIHJldHVybiAoKSA9PiBjbGVhckludGVydmFsKHRpbWVyKVxuICB9LCBbXSlcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmICghcmVhY3Rpb24pIHJldHVyblxuICAgIGxhc3RTcG9rZVRpY2suY3VycmVudCA9IHRpY2tcbiAgICBjb25zdCB0aW1lciA9IHNldFRpbWVvdXQoXG4gICAgICBzZXRBID0+XG4gICAgICAgIHNldEEoKHByZXY6IEFwcFN0YXRlKSA9PlxuICAgICAgICAgIHByZXYuY29tcGFuaW9uUmVhY3Rpb24gPT09IHVuZGVmaW5lZFxuICAgICAgICAgICAgPyBwcmV2XG4gICAgICAgICAgICA6IHsgLi4ucHJldiwgY29tcGFuaW9uUmVhY3Rpb246IHVuZGVmaW5lZCB9LFxuICAgICAgICApLFxuICAgICAgQlVCQkxFX1NIT1cgKiBUSUNLX01TLFxuICAgICAgc2V0QXBwU3RhdGUsXG4gICAgKVxuICAgIHJldHVybiAoKSA9PiBjbGVhclRpbWVvdXQodGltZXIpXG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIHJlYWN0LWhvb2tzL2V4aGF1c3RpdmUtZGVwcyAtLSB0aWNrIGludGVudGlvbmFsbHkgY2FwdHVyZWQgYXQgcmVhY3Rpb24tY2hhbmdlLCBub3QgdHJhY2tlZFxuICB9LCBbcmVhY3Rpb24sIHNldEFwcFN0YXRlXSlcblxuICBpZiAoIWZlYXR1cmUoJ0JVRERZJykpIHJldHVybiBudWxsXG4gIGNvbnN0IGNvbXBhbmlvbiA9IGdldENvbXBhbmlvbigpXG4gIGlmICghY29tcGFuaW9uIHx8IGdldEdsb2JhbENvbmZpZygpLmNvbXBhbmlvbk11dGVkKSByZXR1cm4gbnVsbFxuXG4gIGNvbnN0IGNvbG9yID0gUkFSSVRZX0NPTE9SU1tjb21wYW5pb24ucmFyaXR5XVxuICBjb25zdCBjb2xXaWR0aCA9IHNwcml0ZUNvbFdpZHRoKHN0cmluZ1dpZHRoKGNvbXBhbmlvbi5uYW1lKSlcblxuICBjb25zdCBidWJibGVBZ2UgPSByZWFjdGlvbiA/IHRpY2sgLSBsYXN0U3Bva2VUaWNrLmN1cnJlbnQgOiAwXG4gIGNvbnN0IGZhZGluZyA9XG4gICAgcmVhY3Rpb24gIT09IHVuZGVmaW5lZCAmJiBidWJibGVBZ2UgPj0gQlVCQkxFX1NIT1cgLSBGQURFX1dJTkRPV1xuXG4gIGNvbnN0IHBldEFnZSA9IHBldEF0ID8gdGljayAtIHBldFN0YXJ0VGljayA6IEluZmluaXR5XG4gIGNvbnN0IHBldHRpbmcgPSBwZXRBZ2UgKiBUSUNLX01TIDwgUEVUX0JVUlNUX01TXG5cbiAgLy8gTmFycm93IHRlcm1pbmFsczogY29sbGFwc2UgdG8gb25lLWxpbmUgZmFjZS4gV2hlbiBzcGVha2luZywgdGhlIHF1aXBcbiAgLy8gcmVwbGFjZXMgdGhlIG5hbWUgYmVzaWRlIHRoZSBmYWNlIChubyByb29tIGZvciBhIGJ1YmJsZSkuXG4gIGlmIChjb2x1bW5zIDwgTUlOX0NPTFNfRk9SX0ZVTExfU1BSSVRFKSB7XG4gICAgY29uc3QgcXVpcCA9XG4gICAgICByZWFjdGlvbiAmJiByZWFjdGlvbi5sZW5ndGggPiBOQVJST1dfUVVJUF9DQVBcbiAgICAgICAgPyByZWFjdGlvbi5zbGljZSgwLCBOQVJST1dfUVVJUF9DQVAgLSAxKSArICfigKYnXG4gICAgICAgIDogcmVhY3Rpb25cbiAgICBjb25zdCBsYWJlbCA9IHF1aXBcbiAgICAgID8gYFwiJHtxdWlwfVwiYFxuICAgICAgOiBmb2N1c2VkXG4gICAgICAgID8gYCAke2NvbXBhbmlvbi5uYW1lfSBgXG4gICAgICAgIDogY29tcGFuaW9uLm5hbWVcbiAgICByZXR1cm4gKFxuICAgICAgPEJveCBwYWRkaW5nWD17MX0gYWxpZ25TZWxmPVwiZmxleC1lbmRcIj5cbiAgICAgICAgPFRleHQ+XG4gICAgICAgICAge3BldHRpbmcgJiYgPFRleHQgY29sb3I9XCJhdXRvQWNjZXB0XCI+e2ZpZ3VyZXMuaGVhcnR9IDwvVGV4dD59XG4gICAgICAgICAgPFRleHQgYm9sZCBjb2xvcj17Y29sb3J9PlxuICAgICAgICAgICAge3JlbmRlckZhY2UoY29tcGFuaW9uKX1cbiAgICAgICAgICA8L1RleHQ+eycgJ31cbiAgICAgICAgICA8VGV4dFxuICAgICAgICAgICAgaXRhbGljXG4gICAgICAgICAgICBkaW1Db2xvcj17IWZvY3VzZWQgJiYgIXJlYWN0aW9ufVxuICAgICAgICAgICAgYm9sZD17Zm9jdXNlZH1cbiAgICAgICAgICAgIGludmVyc2U9e2ZvY3VzZWQgJiYgIXJlYWN0aW9ufVxuICAgICAgICAgICAgY29sb3I9e1xuICAgICAgICAgICAgICByZWFjdGlvblxuICAgICAgICAgICAgICAgID8gZmFkaW5nXG4gICAgICAgICAgICAgICAgICA/ICdpbmFjdGl2ZSdcbiAgICAgICAgICAgICAgICAgIDogY29sb3JcbiAgICAgICAgICAgICAgICA6IGZvY3VzZWRcbiAgICAgICAgICAgICAgICAgID8gY29sb3JcbiAgICAgICAgICAgICAgICAgIDogdW5kZWZpbmVkXG4gICAgICAgICAgICB9XG4gICAgICAgICAgPlxuICAgICAgICAgICAge2xhYmVsfVxuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgPC9UZXh0PlxuICAgICAgPC9Cb3g+XG4gICAgKVxuICB9XG4gIGNvbnN0IGZyYW1lQ291bnQgPSBzcHJpdGVGcmFtZUNvdW50KGNvbXBhbmlvbi5zcGVjaWVzKVxuICBjb25zdCBoZWFydEZyYW1lID0gcGV0dGluZyA/IFBFVF9IRUFSVFNbcGV0QWdlICUgUEVUX0hFQVJUUy5sZW5ndGhdIDogbnVsbFxuXG4gIGxldCBzcHJpdGVGcmFtZTogbnVtYmVyXG4gIGxldCBibGluayA9IGZhbHNlXG4gIGlmIChyZWFjdGlvbiB8fCBwZXR0aW5nKSB7XG4gICAgLy8gRXhjaXRlZDogY3ljbGUgYWxsIGZpZGdldCBmcmFtZXMgZmFzdFxuICAgIHNwcml0ZUZyYW1lID0gdGljayAlIGZyYW1lQ291bnRcbiAgfSBlbHNlIHtcbiAgICBjb25zdCBzdGVwID0gSURMRV9TRVFVRU5DRVt0aWNrICUgSURMRV9TRVFVRU5DRS5sZW5ndGhdIVxuICAgIGlmIChzdGVwID09PSAtMSkge1xuICAgICAgc3ByaXRlRnJhbWUgPSAwXG4gICAgICBibGluayA9IHRydWVcbiAgICB9IGVsc2Uge1xuICAgICAgc3ByaXRlRnJhbWUgPSBzdGVwICUgZnJhbWVDb3VudFxuICAgIH1cbiAgfVxuXG4gIGNvbnN0IGJvZHkgPSByZW5kZXJTcHJpdGUoY29tcGFuaW9uLCBzcHJpdGVGcmFtZSkubWFwKGxpbmUgPT5cbiAgICBibGluayA/IGxpbmUucmVwbGFjZUFsbChjb21wYW5pb24uZXllLCAnLScpIDogbGluZSxcbiAgKVxuICBjb25zdCBzcHJpdGUgPSBoZWFydEZyYW1lID8gW2hlYXJ0RnJhbWUsIC4uLmJvZHldIDogYm9keVxuXG4gIC8vIE5hbWUgcm93IGRvdWJsZXMgYXMgaGludCByb3cg4oCUIHVuZm9jdXNlZCBzaG93cyBkaW0gbmFtZSArIOKGkyBkaXNjb3ZlcnksXG4gIC8vIGZvY3VzZWQgc2hvd3MgaW52ZXJzZSBuYW1lLiBUaGUgZW50ZXItdG8tb3BlbiBoaW50IGxpdmVzIGluXG4gIC8vIFByb21wdElucHV0Rm9vdGVyJ3MgcmlnaHQgY29sdW1uIHNvIHRoaXMgcm93IHN0YXlzIG9uZSBsaW5lIGFuZCB0aGVcbiAgLy8gc3ByaXRlIGRvZXNuJ3QganVtcCB1cCB3aGVuIHNlbGVjdGVkLiBmbGV4U2hyaW5rPTAgc3RvcHMgdGhlXG4gIC8vIGlubGluZS1idWJibGUgcm93IHdyYXBwZXIgZnJvbSBzcXVlZXppbmcgdGhlIHNwcml0ZSB0byBmaXQuXG4gIGNvbnN0IHNwcml0ZUNvbHVtbiA9IChcbiAgICA8Qm94XG4gICAgICBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCJcbiAgICAgIGZsZXhTaHJpbms9ezB9XG4gICAgICBhbGlnbkl0ZW1zPVwiY2VudGVyXCJcbiAgICAgIHdpZHRoPXtjb2xXaWR0aH1cbiAgICA+XG4gICAgICB7c3ByaXRlLm1hcCgobGluZSwgaSkgPT4gKFxuICAgICAgICA8VGV4dCBrZXk9e2l9IGNvbG9yPXtpID09PSAwICYmIGhlYXJ0RnJhbWUgPyAnYXV0b0FjY2VwdCcgOiBjb2xvcn0+XG4gICAgICAgICAge2xpbmV9XG4gICAgICAgIDwvVGV4dD5cbiAgICAgICkpfVxuICAgICAgPFRleHRcbiAgICAgICAgaXRhbGljXG4gICAgICAgIGJvbGQ9e2ZvY3VzZWR9XG4gICAgICAgIGRpbUNvbG9yPXshZm9jdXNlZH1cbiAgICAgICAgY29sb3I9e2ZvY3VzZWQgPyBjb2xvciA6IHVuZGVmaW5lZH1cbiAgICAgICAgaW52ZXJzZT17Zm9jdXNlZH1cbiAgICAgID5cbiAgICAgICAge2ZvY3VzZWQgPyBgICR7Y29tcGFuaW9uLm5hbWV9IGAgOiBjb21wYW5pb24ubmFtZX1cbiAgICAgIDwvVGV4dD5cbiAgICA8L0JveD5cbiAgKVxuXG4gIGlmICghcmVhY3Rpb24pIHtcbiAgICByZXR1cm4gPEJveCBwYWRkaW5nWD17MX0+e3Nwcml0ZUNvbHVtbn08L0JveD5cbiAgfVxuXG4gIC8vIEZ1bGxzY3JlZW46IGJ1YmJsZSByZW5kZXJzIHNlcGFyYXRlbHkgdmlhIENvbXBhbmlvbkZsb2F0aW5nQnViYmxlIGluXG4gIC8vIEZ1bGxzY3JlZW5MYXlvdXQncyBib3R0b21GbG9hdCBzbG90ICh0aGUgYm90dG9tIHNsb3QncyBvdmVyZmxvd1k6aGlkZGVuXG4gIC8vIHdvdWxkIGNsaXAgYSBwb3NpdGlvbjphYnNvbHV0ZSBvdmVybGF5IGhlcmUpLiBTcHJpdGUgYm9keSBvbmx5LlxuICAvLyBOb24tZnVsbHNjcmVlbjogYnViYmxlIHNpdHMgaW5saW5lIGJlc2lkZSB0aGUgc3ByaXRlIChpbnB1dCBzaHJpbmtzKVxuICAvLyBiZWNhdXNlIGZsb2F0aW5nIGludG8gU3RhdGljIHNjcm9sbGJhY2sgY2FuJ3QgYmUgY2xlYXJlZC5cbiAgaWYgKGlzRnVsbHNjcmVlbkFjdGl2ZSgpKSB7XG4gICAgcmV0dXJuIDxCb3ggcGFkZGluZ1g9ezF9PntzcHJpdGVDb2x1bW59PC9Cb3g+XG4gIH1cbiAgcmV0dXJuIChcbiAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJyb3dcIiBhbGlnbkl0ZW1zPVwiZmxleC1lbmRcIiBwYWRkaW5nWD17MX0gZmxleFNocmluaz17MH0+XG4gICAgICA8U3BlZWNoQnViYmxlXG4gICAgICAgIHRleHQ9e3JlYWN0aW9ufVxuICAgICAgICBjb2xvcj17Y29sb3J9XG4gICAgICAgIGZhZGluZz17ZmFkaW5nfVxuICAgICAgICB0YWlsPVwicmlnaHRcIlxuICAgICAgLz5cbiAgICAgIHtzcHJpdGVDb2x1bW59XG4gICAgPC9Cb3g+XG4gIClcbn1cblxuLy8gRmxvYXRpbmcgYnViYmxlIG92ZXJsYXkgZm9yIGZ1bGxzY3JlZW4gbW9kZS4gTW91bnRlZCBpbiBGdWxsc2NyZWVuTGF5b3V0J3Ncbi8vIGJvdHRvbUZsb2F0IHNsb3QgKG91dHNpZGUgdGhlIG92ZXJmbG93WTpoaWRkZW4gY2xpcCkgc28gaXQgY2FuIGV4dGVuZCBpbnRvXG4vLyB0aGUgU2Nyb2xsQm94IHJlZ2lvbi4gQ29tcGFuaW9uU3ByaXRlIG93bnMgdGhlIGNsZWFyLWFmdGVyLTEwcyB0aW1lcjsgdGhpc1xuLy8ganVzdCByZWFkcyBjb21wYW5pb25SZWFjdGlvbiBhbmQgcmVuZGVycyB0aGUgZmFkZS5cbmV4cG9ydCBmdW5jdGlvbiBDb21wYW5pb25GbG9hdGluZ0J1YmJsZSgpOiBSZWFjdC5SZWFjdE5vZGUge1xuICBjb25zdCByZWFjdGlvbiA9IHVzZUFwcFN0YXRlKHMgPT4gcy5jb21wYW5pb25SZWFjdGlvbilcbiAgY29uc3QgW3sgdGljaywgZm9yUmVhY3Rpb24gfSwgc2V0VGlja10gPSB1c2VTdGF0ZSh7XG4gICAgdGljazogMCxcbiAgICBmb3JSZWFjdGlvbjogcmVhY3Rpb24sXG4gIH0pXG5cbiAgLy8gUmVzZXQgdGljayBzeW5jaHJvbm91c2x5IHdoZW4gcmVhY3Rpb24gY2hhbmdlcyAobm90IGluIHVzZUVmZmVjdCwgd2hpY2hcbiAgLy8gcnVucyBwb3N0LXJlbmRlciBhbmQgd291bGQgc2hvdyBvbmUgc3RhbGUtZmFkZWQgZnJhbWUpLiBTdG9yaW5nIHRoZVxuICAvLyByZWFjdGlvbiB0aGUgdGljayBpcyBjb3VudGluZyBGT1IgYWxvbmdzaWRlIHRoZSB0aWNrIGl0c2VsZiBtZWFucyB0aGVcbiAgLy8gZmFkZSBjb21wdXRhdGlvbiBuZXZlciBzZWVzIGEgdGljayBmcm9tIGEgcHJldmlvdXMgcmVhY3Rpb24uXG4gIGlmIChyZWFjdGlvbiAhPT0gZm9yUmVhY3Rpb24pIHtcbiAgICBzZXRUaWNrKHsgdGljazogMCwgZm9yUmVhY3Rpb246IHJlYWN0aW9uIH0pXG4gIH1cblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmICghcmVhY3Rpb24pIHJldHVyblxuICAgIGNvbnN0IHRpbWVyID0gc2V0SW50ZXJ2YWwoXG4gICAgICBzZXQgPT4gc2V0KHMgPT4gKHsgLi4ucywgdGljazogcy50aWNrICsgMSB9KSksXG4gICAgICBUSUNLX01TLFxuICAgICAgc2V0VGljayxcbiAgICApXG4gICAgcmV0dXJuICgpID0+IGNsZWFySW50ZXJ2YWwodGltZXIpXG4gIH0sIFtyZWFjdGlvbl0pXG5cbiAgaWYgKCFmZWF0dXJlKCdCVUREWScpIHx8ICFyZWFjdGlvbikgcmV0dXJuIG51bGxcbiAgY29uc3QgY29tcGFuaW9uID0gZ2V0Q29tcGFuaW9uKClcbiAgaWYgKCFjb21wYW5pb24gfHwgZ2V0R2xvYmFsQ29uZmlnKCkuY29tcGFuaW9uTXV0ZWQpIHJldHVybiBudWxsXG5cbiAgcmV0dXJuIChcbiAgICA8U3BlZWNoQnViYmxlXG4gICAgICB0ZXh0PXtyZWFjdGlvbn1cbiAgICAgIGNvbG9yPXtSQVJJVFlfQ09MT1JTW2NvbXBhbmlvbi5yYXJpdHldfVxuICAgICAgZmFkaW5nPXt0aWNrID49IEJVQkJMRV9TSE9XIC0gRkFERV9XSU5ET1d9XG4gICAgICB0YWlsPVwiZG93blwiXG4gICAgLz5cbiAgKVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsU0FBU0EsT0FBTyxRQUFRLFlBQVk7QUFDcEMsT0FBT0MsT0FBTyxNQUFNLFNBQVM7QUFDN0IsT0FBT0MsS0FBSyxJQUFJQyxTQUFTLEVBQUVDLE1BQU0sRUFBRUMsUUFBUSxRQUFRLE9BQU87QUFDMUQsU0FBU0MsZUFBZSxRQUFRLDZCQUE2QjtBQUM3RCxTQUFTQyxXQUFXLFFBQVEsdUJBQXVCO0FBQ25ELFNBQVNDLEdBQUcsRUFBRUMsSUFBSSxRQUFRLFdBQVc7QUFDckMsU0FBU0MsV0FBVyxFQUFFQyxjQUFjLFFBQVEsc0JBQXNCO0FBQ2xFLGNBQWNDLFFBQVEsUUFBUSwyQkFBMkI7QUFDekQsU0FBU0MsZUFBZSxRQUFRLG9CQUFvQjtBQUNwRCxTQUFTQyxrQkFBa0IsUUFBUSx3QkFBd0I7QUFDM0QsY0FBY0MsS0FBSyxRQUFRLG1CQUFtQjtBQUM5QyxTQUFTQyxZQUFZLFFBQVEsZ0JBQWdCO0FBQzdDLFNBQVNDLFVBQVUsRUFBRUMsWUFBWSxFQUFFQyxnQkFBZ0IsUUFBUSxjQUFjO0FBQ3pFLFNBQVNDLGFBQWEsUUFBUSxZQUFZO0FBRTFDLE1BQU1DLE9BQU8sR0FBRyxHQUFHO0FBQ25CLE1BQU1DLFdBQVcsR0FBRyxFQUFFLEVBQUM7QUFDdkIsTUFBTUMsV0FBVyxHQUFHLENBQUMsRUFBQztBQUN0QixNQUFNQyxZQUFZLEdBQUcsSUFBSSxFQUFDOztBQUUxQjtBQUNBO0FBQ0EsTUFBTUMsYUFBYSxHQUFHLENBQUMsQ0FBQyxF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 \ No newline at end of file diff --git a/src/buddy/companion.ts b/src/buddy/companion.ts new file mode 100644 index 0000000..09c3838 --- /dev/null +++ b/src/buddy/companion.ts @@ -0,0 +1,133 @@ +import { getGlobalConfig } from '../utils/config.js' +import { + type Companion, + type CompanionBones, + EYES, + HATS, + RARITIES, + RARITY_WEIGHTS, + type Rarity, + SPECIES, + STAT_NAMES, + type StatName, +} from './types.js' + +// Mulberry32 — tiny seeded PRNG, good enough for picking ducks +function mulberry32(seed: number): () => number { + let a = seed >>> 0 + return function () { + a |= 0 + a = (a + 0x6d2b79f5) | 0 + let t = Math.imul(a ^ (a >>> 15), 1 | a) + t = (t + Math.imul(t ^ (t >>> 7), 61 | t)) ^ t + return ((t ^ (t >>> 14)) >>> 0) / 4294967296 + } +} + +function hashString(s: string): number { + if (typeof Bun !== 'undefined') { + return Number(BigInt(Bun.hash(s)) & 0xffffffffn) + } + let h = 2166136261 + for (let i = 0; i < s.length; i++) { + h ^= s.charCodeAt(i) + h = Math.imul(h, 16777619) + } + return h >>> 0 +} + +function pick(rng: () => number, arr: readonly T[]): T { + return arr[Math.floor(rng() * arr.length)]! +} + +function rollRarity(rng: () => number): Rarity { + const total = Object.values(RARITY_WEIGHTS).reduce((a, b) => a + b, 0) + let roll = rng() * total + for (const rarity of RARITIES) { + roll -= RARITY_WEIGHTS[rarity] + if (roll < 0) return rarity + } + return 'common' +} + +const RARITY_FLOOR: Record = { + common: 5, + uncommon: 15, + rare: 25, + epic: 35, + legendary: 50, +} + +// One peak stat, one dump stat, rest scattered. Rarity bumps the floor. +function rollStats( + rng: () => number, + rarity: Rarity, +): Record { + const floor = RARITY_FLOOR[rarity] + const peak = pick(rng, STAT_NAMES) + let dump = pick(rng, STAT_NAMES) + while (dump === peak) dump = pick(rng, STAT_NAMES) + + const stats = {} as Record + for (const name of STAT_NAMES) { + if (name === peak) { + stats[name] = Math.min(100, floor + 50 + Math.floor(rng() * 30)) + } else if (name === dump) { + stats[name] = Math.max(1, floor - 10 + Math.floor(rng() * 15)) + } else { + stats[name] = floor + Math.floor(rng() * 40) + } + } + return stats +} + +const SALT = 'friend-2026-401' + +export type Roll = { + bones: CompanionBones + inspirationSeed: number +} + +function rollFrom(rng: () => number): Roll { + const rarity = rollRarity(rng) + const bones: CompanionBones = { + rarity, + species: pick(rng, SPECIES), + eye: pick(rng, EYES), + hat: rarity === 'common' ? 'none' : pick(rng, HATS), + shiny: rng() < 0.01, + stats: rollStats(rng, rarity), + } + return { bones, inspirationSeed: Math.floor(rng() * 1e9) } +} + +// Called from three hot paths (500ms sprite tick, per-keystroke PromptInput, +// per-turn observer) with the same userId → cache the deterministic result. +let rollCache: { key: string; value: Roll } | undefined +export function roll(userId: string): Roll { + const key = userId + SALT + if (rollCache?.key === key) return rollCache.value + const value = rollFrom(mulberry32(hashString(key))) + rollCache = { key, value } + return value +} + +export function rollWithSeed(seed: string): Roll { + return rollFrom(mulberry32(hashString(seed))) +} + +export function companionUserId(): string { + const config = getGlobalConfig() + return config.oauthAccount?.accountUuid ?? config.userID ?? 'anon' +} + +// Regenerate bones from userId, merge with stored soul. Bones never persist +// so species renames and SPECIES-array edits can't break stored companions, +// and editing config.companion can't fake a rarity. +export function getCompanion(): Companion | undefined { + const stored = getGlobalConfig().companion + if (!stored) return undefined + const { bones } = roll(companionUserId()) + // bones last so stale bones fields in old-format configs get overridden + return { ...stored, ...bones } +} diff --git a/src/buddy/prompt.ts b/src/buddy/prompt.ts new file mode 100644 index 0000000..c5782c0 --- /dev/null +++ b/src/buddy/prompt.ts @@ -0,0 +1,36 @@ +import { feature } from 'bun:bundle' +import type { Message } from '../types/message.js' +import type { Attachment } from '../utils/attachments.js' +import { getGlobalConfig } from '../utils/config.js' +import { getCompanion } from './companion.js' + +export function companionIntroText(name: string, species: string): string { + return `# Companion + +A small ${species} named ${name} sits beside the user's input box and occasionally comments in a speech bubble. You're not ${name} — it's a separate watcher. + +When the user addresses ${name} directly (by name), its bubble will answer. Your job in that moment is to stay out of the way: respond in ONE line or less, or just answer any part of the message meant for you. Don't explain that you're not ${name} — they know. Don't narrate what ${name} might say — the bubble handles that.` +} + +export function getCompanionIntroAttachment( + messages: Message[] | undefined, +): Attachment[] { + if (!feature('BUDDY')) return [] + const companion = getCompanion() + if (!companion || getGlobalConfig().companionMuted) return [] + + // Skip if already announced for this companion. + for (const msg of messages ?? []) { + if (msg.type !== 'attachment') continue + if (msg.attachment.type !== 'companion_intro') continue + if (msg.attachment.name === companion.name) return [] + } + + return [ + { + type: 'companion_intro', + name: companion.name, + species: companion.species, + }, + ] +} diff --git a/src/buddy/sprites.ts b/src/buddy/sprites.ts new file mode 100644 index 0000000..0150b8c --- /dev/null +++ b/src/buddy/sprites.ts @@ -0,0 +1,514 @@ +import type { CompanionBones, Eye, Hat, Species } from './types.js' +import { + axolotl, + blob, + cactus, + capybara, + cat, + chonk, + dragon, + duck, + ghost, + goose, + mushroom, + octopus, + owl, + penguin, + rabbit, + robot, + snail, + turtle, +} from './types.js' + +// Each sprite is 5 lines tall, 12 wide (after {E}→1char substitution). +// Multiple frames per species for idle fidget animation. +// Line 0 is the hat slot — must be blank in frames 0-1; frame 2 may use it. +const BODIES: Record = { + [duck]: [ + [ + ' ', + ' __ ', + ' <({E} )___ ', + ' ( ._> ', + ' `--´ ', + ], + [ + ' ', + ' __ ', + ' <({E} )___ ', + ' ( ._> ', + ' `--´~ ', + ], + [ + ' ', + ' __ ', + ' <({E} )___ ', + ' ( .__> ', + ' `--´ ', + ], + ], + [goose]: [ + [ + ' ', + ' ({E}> ', + ' || ', + ' _(__)_ ', + ' ^^^^ ', + ], + [ + ' ', + ' ({E}> ', + ' || ', + ' _(__)_ ', + ' ^^^^ ', + ], + [ + ' ', + ' ({E}>> ', + ' || ', + ' _(__)_ ', + ' ^^^^ ', + ], + ], + [blob]: [ + [ + ' ', + ' .----. ', + ' ( {E} {E} ) ', + ' ( ) ', + ' `----´ ', + ], + [ + ' ', + ' .------. ', + ' ( {E} {E} ) ', + ' ( ) ', + ' `------´ ', + ], + [ + ' ', + ' .--. ', + ' ({E} {E}) ', + ' ( ) ', + ' `--´ ', + ], + ], + [cat]: [ + [ + ' ', + ' /\\_/\\ ', + ' ( {E} {E}) ', + ' ( ω ) ', + ' (")_(") ', + ], + [ + ' ', + ' /\\_/\\ ', + ' ( {E} {E}) ', + ' ( ω ) ', + ' (")_(")~ ', + ], + [ + ' ', + ' /\\-/\\ ', + ' ( {E} {E}) ', + ' ( ω ) ', + ' (")_(") ', + ], + ], + [dragon]: [ + [ + ' ', + ' /^\\ /^\\ ', + ' < {E} {E} > ', + ' ( ~~ ) ', + ' `-vvvv-´ ', + ], + [ + ' ', + ' /^\\ /^\\ ', + ' < {E} {E} > ', + ' ( ) ', + ' `-vvvv-´ ', + ], + [ + ' ~ ~ ', + ' /^\\ /^\\ ', + ' < {E} {E} > ', + ' ( ~~ ) ', + ' `-vvvv-´ ', + ], + ], + [octopus]: [ + [ + ' ', + ' .----. ', + ' ( {E} {E} ) ', + ' (______) ', + ' /\\/\\/\\/\\ ', + ], + [ + ' ', + ' .----. ', + ' ( {E} {E} ) ', + ' (______) ', + ' \\/\\/\\/\\/ ', + ], + [ + ' o ', + ' .----. ', + ' ( {E} {E} ) ', + ' (______) ', + ' /\\/\\/\\/\\ ', + ], + ], + [owl]: [ + [ + ' ', + ' /\\ /\\ ', + ' (({E})({E})) ', + ' ( >< ) ', + ' `----´ ', + ], + [ + ' ', + ' /\\ /\\ ', + ' (({E})({E})) ', + ' ( >< ) ', + ' .----. ', + ], + [ + ' ', + ' /\\ /\\ ', + ' (({E})(-)) ', + ' ( >< ) ', + ' `----´ ', + ], + ], + [penguin]: [ + [ + ' ', + ' .---. ', + ' ({E}>{E}) ', + ' /( )\\ ', + ' `---´ ', + ], + [ + ' ', + ' .---. ', + ' ({E}>{E}) ', + ' |( )| ', + ' `---´ ', + ], + [ + ' .---. ', + ' ({E}>{E}) ', + ' /( )\\ ', + ' `---´ ', + ' ~ ~ ', + ], + ], + [turtle]: [ + [ + ' ', + ' _,--._ ', + ' ( {E} {E} ) ', + ' /[______]\\ ', + ' `` `` ', + ], + [ + ' ', + ' _,--._ ', + ' ( {E} {E} ) ', + ' /[______]\\ ', + ' `` `` ', + ], + [ + ' ', + ' _,--._ ', + ' ( {E} {E} ) ', + ' /[======]\\ ', + ' `` `` ', + ], + ], + [snail]: [ + [ + ' ', + ' {E} .--. ', + ' \\ ( @ ) ', + ' \\_`--´ ', + ' ~~~~~~~ ', + ], + [ + ' ', + ' {E} .--. ', + ' | ( @ ) ', + ' \\_`--´ ', + ' ~~~~~~~ ', + ], + [ + ' ', + ' {E} .--. ', + ' \\ ( @ ) ', + ' \\_`--´ ', + ' ~~~~~~ ', + ], + ], + [ghost]: [ + [ + ' ', + ' .----. ', + ' / {E} {E} \\ ', + ' | | ', + ' ~`~``~`~ ', + ], + [ + ' ', + ' .----. ', + ' / {E} {E} \\ ', + ' | | ', + ' `~`~~`~` ', + ], + [ + ' ~ ~ ', + ' .----. ', + ' / {E} {E} \\ ', + ' | | ', + ' ~~`~~`~~ ', + ], + ], + [axolotl]: [ + [ + ' ', + '}~(______)~{', + '}~({E} .. {E})~{', + ' ( .--. ) ', + ' (_/ \\_) ', + ], + [ + ' ', + '~}(______){~', + '~}({E} .. {E}){~', + ' ( .--. ) ', + ' (_/ \\_) ', + ], + [ + ' ', + '}~(______)~{', + '}~({E} .. {E})~{', + ' ( -- ) ', + ' ~_/ \\_~ ', + ], + ], + [capybara]: [ + [ + ' ', + ' n______n ', + ' ( {E} {E} ) ', + ' ( oo ) ', + ' `------´ ', + ], + [ + ' ', + ' n______n ', + ' ( {E} {E} ) ', + ' ( Oo ) ', + ' `------´ ', + ], + [ + ' ~ ~ ', + ' u______n ', + ' ( {E} {E} ) ', + ' ( oo ) ', + ' `------´ ', + ], + ], + [cactus]: [ + [ + ' ', + ' n ____ n ', + ' | |{E} {E}| | ', + ' |_| |_| ', + ' | | ', + ], + [ + ' ', + ' ____ ', + ' n |{E} {E}| n ', + ' |_| |_| ', + ' | | ', + ], + [ + ' n n ', + ' | ____ | ', + ' | |{E} {E}| | ', + ' |_| |_| ', + ' | | ', + ], + ], + [robot]: [ + [ + ' ', + ' .[||]. ', + ' [ {E} {E} ] ', + ' [ ==== ] ', + ' `------´ ', + ], + [ + ' ', + ' .[||]. ', + ' [ {E} {E} ] ', + ' [ -==- ] ', + ' `------´ ', + ], + [ + ' * ', + ' .[||]. ', + ' [ {E} {E} ] ', + ' [ ==== ] ', + ' `------´ ', + ], + ], + [rabbit]: [ + [ + ' ', + ' (\\__/) ', + ' ( {E} {E} ) ', + ' =( .. )= ', + ' (")__(") ', + ], + [ + ' ', + ' (|__/) ', + ' ( {E} {E} ) ', + ' =( .. )= ', + ' (")__(") ', + ], + [ + ' ', + ' (\\__/) ', + ' ( {E} {E} ) ', + ' =( . . )= ', + ' (")__(") ', + ], + ], + [mushroom]: [ + [ + ' ', + ' .-o-OO-o-. ', + '(__________)', + ' |{E} {E}| ', + ' |____| ', + ], + [ + ' ', + ' .-O-oo-O-. ', + '(__________)', + ' |{E} {E}| ', + ' |____| ', + ], + [ + ' . o . ', + ' .-o-OO-o-. ', + '(__________)', + ' |{E} {E}| ', + ' |____| ', + ], + ], + [chonk]: [ + [ + ' ', + ' /\\ /\\ ', + ' ( {E} {E} ) ', + ' ( .. ) ', + ' `------´ ', + ], + [ + ' ', + ' /\\ /| ', + ' ( {E} {E} ) ', + ' ( .. ) ', + ' `------´ ', + ], + [ + ' ', + ' /\\ /\\ ', + ' ( {E} {E} ) ', + ' ( .. ) ', + ' `------´~ ', + ], + ], +} + +const HAT_LINES: Record = { + none: '', + crown: ' \\^^^/ ', + tophat: ' [___] ', + propeller: ' -+- ', + halo: ' ( ) ', + wizard: ' /^\\ ', + beanie: ' (___) ', + tinyduck: ' ,> ', +} + +export function renderSprite(bones: CompanionBones, frame = 0): string[] { + const frames = BODIES[bones.species] + const body = frames[frame % frames.length]!.map(line => + line.replaceAll('{E}', bones.eye), + ) + const lines = [...body] + // Only replace with hat if line 0 is empty (some fidget frames use it for smoke etc) + if (bones.hat !== 'none' && !lines[0]!.trim()) { + lines[0] = HAT_LINES[bones.hat] + } + // Drop blank hat slot — wastes a row in the Card and ambient sprite when + // there's no hat and the frame isn't using it for smoke/antenna/etc. + // Only safe when ALL frames have blank line 0; otherwise heights oscillate. + if (!lines[0]!.trim() && frames.every(f => !f[0]!.trim())) lines.shift() + return lines +} + +export function spriteFrameCount(species: Species): number { + return BODIES[species].length +} + +export function renderFace(bones: CompanionBones): string { + const eye: Eye = bones.eye + switch (bones.species) { + case duck: + case goose: + return `(${eye}>` + case blob: + return `(${eye}${eye})` + case cat: + return `=${eye}ω${eye}=` + case dragon: + return `<${eye}~${eye}>` + case octopus: + return `~(${eye}${eye})~` + case owl: + return `(${eye})(${eye})` + case penguin: + return `(${eye}>)` + case turtle: + return `[${eye}_${eye}]` + case snail: + return `${eye}(@)` + case ghost: + return `/${eye}${eye}\\` + case axolotl: + return `}${eye}.${eye}{` + case capybara: + return `(${eye}oo${eye})` + case cactus: + return `|${eye} ${eye}|` + case robot: + return `[${eye}${eye}]` + case rabbit: + return `(${eye}..${eye})` + case mushroom: + return `|${eye} ${eye}|` + case chonk: + return `(${eye}.${eye})` + } +} diff --git a/src/buddy/types.ts b/src/buddy/types.ts new file mode 100644 index 0000000..8f1c82a --- /dev/null +++ b/src/buddy/types.ts @@ -0,0 +1,148 @@ +export const RARITIES = [ + 'common', + 'uncommon', + 'rare', + 'epic', + 'legendary', +] as const +export type Rarity = (typeof RARITIES)[number] + +// One species name collides with a model-codename canary in excluded-strings.txt. +// The check greps build output (not source), so runtime-constructing the value keeps +// the literal out of the bundle while the check stays armed for the actual codename. +// All species encoded uniformly; `as` casts are type-position only (erased pre-bundle). +const c = String.fromCharCode +// biome-ignore format: keep the species list compact + +export const duck = c(0x64,0x75,0x63,0x6b) as 'duck' +export const goose = c(0x67, 0x6f, 0x6f, 0x73, 0x65) as 'goose' +export const blob = c(0x62, 0x6c, 0x6f, 0x62) as 'blob' +export const cat = c(0x63, 0x61, 0x74) as 'cat' +export const dragon = c(0x64, 0x72, 0x61, 0x67, 0x6f, 0x6e) as 'dragon' +export const octopus = c(0x6f, 0x63, 0x74, 0x6f, 0x70, 0x75, 0x73) as 'octopus' +export const owl = c(0x6f, 0x77, 0x6c) as 'owl' +export const penguin = c(0x70, 0x65, 0x6e, 0x67, 0x75, 0x69, 0x6e) as 'penguin' +export const turtle = c(0x74, 0x75, 0x72, 0x74, 0x6c, 0x65) as 'turtle' +export const snail = c(0x73, 0x6e, 0x61, 0x69, 0x6c) as 'snail' +export const ghost = c(0x67, 0x68, 0x6f, 0x73, 0x74) as 'ghost' +export const axolotl = c(0x61, 0x78, 0x6f, 0x6c, 0x6f, 0x74, 0x6c) as 'axolotl' +export const capybara = c( + 0x63, + 0x61, + 0x70, + 0x79, + 0x62, + 0x61, + 0x72, + 0x61, +) as 'capybara' +export const cactus = c(0x63, 0x61, 0x63, 0x74, 0x75, 0x73) as 'cactus' +export const robot = c(0x72, 0x6f, 0x62, 0x6f, 0x74) as 'robot' +export const rabbit = c(0x72, 0x61, 0x62, 0x62, 0x69, 0x74) as 'rabbit' +export const mushroom = c( + 0x6d, + 0x75, + 0x73, + 0x68, + 0x72, + 0x6f, + 0x6f, + 0x6d, +) as 'mushroom' +export const chonk = c(0x63, 0x68, 0x6f, 0x6e, 0x6b) as 'chonk' + +export const SPECIES = [ + duck, + goose, + blob, + cat, + dragon, + octopus, + owl, + penguin, + turtle, + snail, + ghost, + axolotl, + capybara, + cactus, + robot, + rabbit, + mushroom, + chonk, +] as const +export type Species = (typeof SPECIES)[number] // biome-ignore format: keep compact + +export const EYES = ['·', '✦', '×', '◉', '@', '°'] as const +export type Eye = (typeof EYES)[number] + +export const HATS = [ + 'none', + 'crown', + 'tophat', + 'propeller', + 'halo', + 'wizard', + 'beanie', + 'tinyduck', +] as const +export type Hat = (typeof HATS)[number] + +export const STAT_NAMES = [ + 'DEBUGGING', + 'PATIENCE', + 'CHAOS', + 'WISDOM', + 'SNARK', +] as const +export type StatName = (typeof STAT_NAMES)[number] + +// Deterministic parts — derived from hash(userId) +export type CompanionBones = { + rarity: Rarity + species: Species + eye: Eye + hat: Hat + shiny: boolean + stats: Record +} + +// Model-generated soul — stored in config after first hatch +export type CompanionSoul = { + name: string + personality: string +} + +export type Companion = CompanionBones & + CompanionSoul & { + hatchedAt: number + } + +// What actually persists in config. Bones are regenerated from hash(userId) +// on every read so species renames don't break stored companions and users +// can't edit their way to a legendary. +export type StoredCompanion = CompanionSoul & { hatchedAt: number } + +export const RARITY_WEIGHTS = { + common: 60, + uncommon: 25, + rare: 10, + epic: 4, + legendary: 1, +} as const satisfies Record + +export const RARITY_STARS = { + common: '★', + uncommon: '★★', + rare: '★★★', + epic: '★★★★', + legendary: '★★★★★', +} as const satisfies Record + +export const RARITY_COLORS = { + common: 'inactive', + uncommon: 'success', + rare: 'permission', + epic: 'autoAccept', + legendary: 'warning', +} as const satisfies Record diff --git a/src/buddy/useBuddyNotification.tsx b/src/buddy/useBuddyNotification.tsx new file mode 100644 index 0000000..d6eed22 --- /dev/null +++ b/src/buddy/useBuddyNotification.tsx @@ -0,0 +1,98 @@ +import { c as _c } from "react/compiler-runtime"; +import { feature } from 'bun:bundle'; +import React, { useEffect } from 'react'; +import { useNotifications } from '../context/notifications.js'; +import { Text } from '../ink.js'; +import { getGlobalConfig } from '../utils/config.js'; +import { getRainbowColor } from '../utils/thinking.js'; + +// Local date, not UTC — 24h rolling wave across timezones. Sustained Twitter +// buzz instead of a single UTC-midnight spike, gentler on soul-gen load. +// Teaser window: April 1-7, 2026 only. Command stays live forever after. +export function isBuddyTeaserWindow(): boolean { + if ("external" === 'ant') return true; + const d = new Date(); + return d.getFullYear() === 2026 && d.getMonth() === 3 && d.getDate() <= 7; +} +export function isBuddyLive(): boolean { + if ("external" === 'ant') return true; + const d = new Date(); + return d.getFullYear() > 2026 || d.getFullYear() === 2026 && d.getMonth() >= 3; +} +function RainbowText(t0) { + const $ = _c(2); + const { + text + } = t0; + let t1; + if ($[0] !== text) { + t1 = <>{[...text].map(_temp)}; + $[0] = text; + $[1] = t1; + } else { + t1 = $[1]; + } + return t1; +} + +// Rainbow /buddy teaser shown on startup when no companion hatched yet. +// Idle presence and reactions are handled by CompanionSprite directly. +function _temp(ch, i) { + return {ch}; +} +export function useBuddyNotification() { + const $ = _c(4); + const { + addNotification, + removeNotification + } = useNotifications(); + let t0; + let t1; + if ($[0] !== addNotification || $[1] !== removeNotification) { + t0 = () => { + if (!feature("BUDDY")) { + return; + } + const config = getGlobalConfig(); + if (config.companion || !isBuddyTeaserWindow()) { + return; + } + addNotification({ + key: "buddy-teaser", + jsx: , + priority: "immediate", + timeoutMs: 15000 + }); + return () => removeNotification("buddy-teaser"); + }; + t1 = [addNotification, removeNotification]; + $[0] = addNotification; + $[1] = removeNotification; + $[2] = t0; + $[3] = t1; + } else { + t0 = $[2]; + t1 = $[3]; + } + useEffect(t0, t1); +} +export function findBuddyTriggerPositions(text: string): Array<{ + start: number; + end: number; +}> { + if (!feature('BUDDY')) return []; + const triggers: Array<{ + start: number; + end: number; + }> = []; + const re = /\/buddy\b/g; + let m: RegExpExecArray | null; + while ((m = re.exec(text)) !== null) { + triggers.push({ + start: m.index, + end: m.index + m[0].length + }); + } + return triggers; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJmZWF0dXJlIiwiUmVhY3QiLCJ1c2VFZmZlY3QiLCJ1c2VOb3RpZmljYXRpb25zIiwiVGV4dCIsImdldEdsb2JhbENvbmZpZyIsImdldFJhaW5ib3dDb2xvciIsImlzQnVkZHlUZWFzZXJXaW5kb3ciLCJkIiwiRGF0ZSIsImdldEZ1bGxZZWFyIiwiZ2V0TW9udGgiLCJnZXREYXRlIiwiaXNCdWRkeUxpdmUiLCJSYWluYm93VGV4dCIsInQwIiwiJCIsIl9jIiwidGV4dCIsInQxIiwibWFwIiwiX3RlbXAiLCJjaCIsImkiLCJ1c2VCdWRkeU5vdGlmaWNhdGlvbiIsImFkZE5vdGlmaWNhdGlvbiIsInJlbW92ZU5vdGlmaWNhdGlvbiIsImNvbmZpZyIsImNvbXBhbmlvbiIsImtleSIsImpzeCIsInByaW9yaXR5IiwidGltZW91dE1zIiwiZmluZEJ1ZGR5VHJpZ2dlclBvc2l0aW9ucyIsIkFycmF5Iiwic3RhcnQiLCJlbmQiLCJ0cmlnZ2VycyIsInJlIiwibSIsIlJlZ0V4cEV4ZWNBcnJheSIsImV4ZWMiLCJwdXNoIiwiaW5kZXgiLCJsZW5ndGgiXSwic291cmNlcyI6WyJ1c2VCdWRkeU5vdGlmaWNhdGlvbi50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgZmVhdHVyZSB9IGZyb20gJ2J1bjpidW5kbGUnXG5pbXBvcnQgUmVhY3QsIHsgdXNlRWZmZWN0IH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyB1c2VOb3RpZmljYXRpb25zIH0gZnJvbSAnLi4vY29udGV4dC9ub3RpZmljYXRpb25zLmpzJ1xuaW1wb3J0IHsgVGV4dCB9IGZyb20gJy4uL2luay5qcydcbmltcG9ydCB7IGdldEdsb2JhbENvbmZpZyB9IGZyb20gJy4uL3V0aWxzL2NvbmZpZy5qcydcbmltcG9ydCB7IGdldFJhaW5ib3dDb2xvciB9IGZyb20gJy4uL3V0aWxzL3RoaW5raW5nLmpzJ1xuXG4vLyBMb2NhbCBkYXRlLCBub3QgVVRDIOKAlCAyNGggcm9sbGluZyB3YXZlIGFjcm9zcyB0aW1lem9uZXMuIFN1c3RhaW5lZCBUd2l0dGVyXG4vLyBidXp6IGluc3RlYWQgb2YgYSBzaW5nbGUgVVRDLW1pZG5pZ2h0IHNwaWtlLCBnZW50bGVyIG9uIHNvdWwtZ2VuIGxvYWQuXG4vLyBUZWFzZXIgd2luZG93OiBBcHJpbCAxLTcsIDIwMjYgb25seS4gQ29tbWFuZCBzdGF5cyBsaXZlIGZvcmV2ZXIgYWZ0ZXIuXG5leHBvcnQgZnVuY3Rpb24gaXNCdWRkeVRlYXNlcldpbmRvdygpOiBib29sZWFuIHtcbiAgaWYgKFwiZXh0ZXJuYWxcIiA9PT0gJ2FudCcpIHJldHVybiB0cnVlXG4gIGNvbnN0IGQgPSBuZXcgRGF0ZSgpXG4gIHJldHVybiBkLmdldEZ1bGxZZWFyKCkgPT09IDIwMjYgJiYgZC5nZXRNb250aCgpID09PSAzICYmIGQuZ2V0RGF0ZSgpIDw9IDdcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGlzQnVkZHlMaXZlKCk6IGJvb2xlYW4ge1xuICBpZiAoXCJleHRlcm5hbFwiID09PSAnYW50JykgcmV0dXJuIHRydWVcbiAgY29uc3QgZCA9IG5ldyBEYXRlKClcbiAgcmV0dXJuIChcbiAgICBkLmdldEZ1bGxZZWFyKCkgPiAyMDI2IHx8IChkLmdldEZ1bGxZZWFyKCkgPT09IDIwMjYgJiYgZC5nZXRNb250aCgpID49IDMpXG4gIClcbn1cblxuZnVuY3Rpb24gUmFpbmJvd1RleHQoeyB0ZXh0IH06IHsgdGV4dDogc3RyaW5nIH0pOiBSZWFjdC5SZWFjdE5vZGUge1xuICByZXR1cm4gKFxuICAgIDw+XG4gICAgICB7Wy4uLnRleHRdLm1hcCgoY2gsIGkpID0+IChcbiAgICAgICAgPFRleHQga2V5PXtpfSBjb2xvcj17Z2V0UmFpbmJvd0NvbG9yKGkpfT5cbiAgICAgICAgICB7Y2h9XG4gICAgICAgIDwvVGV4dD5cbiAgICAgICkpfVxuICAgIDwvPlxuICApXG59XG5cbi8vIFJhaW5ib3cgL2J1ZGR5IHRlYXNlciBzaG93biBvbiBzdGFydHVwIHdoZW4gbm8gY29tcGFuaW9uIGhhdGNoZWQgeWV0LlxuLy8gSWRsZSBwcmVzZW5jZSBhbmQgcmVhY3Rpb25zIGFyZSBoYW5kbGVkIGJ5IENvbXBhbmlvblNwcml0ZSBkaXJlY3RseS5cbmV4cG9ydCBmdW5jdGlvbiB1c2VCdWRkeU5vdGlmaWNhdGlvbigpOiB2b2lkIHtcbiAgY29uc3QgeyBhZGROb3RpZmljYXRpb24sIHJlbW92ZU5vdGlmaWNhdGlvbiB9ID0gdXNlTm90aWZpY2F0aW9ucygpXG5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAoIWZlYXR1cmUoJ0JVRERZJykpIHJldHVyblxuICAgIGNvbnN0IGNvbmZpZyA9IGdldEdsb2JhbENvbmZpZygpXG4gICAgaWYgKGNvbmZpZy5jb21wYW5pb24gfHwgIWlzQnVkZHlUZWFzZXJXaW5kb3coKSkgcmV0dXJuXG4gICAgYWRkTm90aWZpY2F0aW9uKHtcbiAgICAgIGtleTogJ2J1ZGR5LXRlYXNlcicsXG4gICAgICBqc3g6IDxSYWluYm93VGV4dCB0ZXh0PVwiL2J1ZGR5XCIgLz4sXG4gICAgICBwcmlvcml0eTogJ2ltbWVkaWF0ZScsXG4gICAgICB0aW1lb3V0TXM6IDE1XzAwMCxcbiAgICB9KVxuICAgIHJldHVybiAoKSA9PiByZW1vdmVOb3RpZmljYXRpb24oJ2J1ZGR5LXRlYXNlcicpXG4gIH0sIFthZGROb3RpZmljYXRpb24sIHJlbW92ZU5vdGlmaWNhdGlvbl0pXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBmaW5kQnVkZHlUcmlnZ2VyUG9zaXRpb25zKFxuICB0ZXh0OiBzdHJpbmcsXG4pOiBBcnJheTx7IHN0YXJ0OiBudW1iZXI7IGVuZDogbnVtYmVyIH0+IHtcbiAgaWYgKCFmZWF0dXJlKCdCVUREWScpKSByZXR1cm4gW11cbiAgY29uc3QgdHJpZ2dlcnM6IEFycmF5PHsgc3RhcnQ6IG51bWJlcjsgZW5kOiBudW1iZXIgfT4gPSBbXVxuICBjb25zdCByZSA9IC9cXC9idWRkeVxcYi9nXG4gIGxldCBtOiBSZWdFeHBFeGVjQXJyYXkgfCBudWxsXG4gIHdoaWxlICgobSA9IHJlLmV4ZWModGV4dCkpICE9PSBudWxsKSB7XG4gICAgdHJpZ2dlcnMucHVzaCh7IHN0YXJ0OiBtLmluZGV4LCBlbmQ6IG0uaW5kZXggKyBtWzBdLmxlbmd0aCB9KVxuICB9XG4gIHJldHVybiB0cmlnZ2Vyc1xufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsU0FBU0EsT0FBTyxRQUFRLFlBQVk7QUFDcEMsT0FBT0MsS0FBSyxJQUFJQyxTQUFTLFFBQVEsT0FBTztBQUN4QyxTQUFTQyxnQkFBZ0IsUUFBUSw2QkFBNkI7QUFDOUQsU0FBU0MsSUFBSSxRQUFRLFdBQVc7QUFDaEMsU0FBU0MsZUFBZSxRQUFRLG9CQUFvQjtBQUNwRCxTQUFTQyxlQUFlLFFBQVEsc0JBQXNCOztBQUV0RDtBQUNBO0FBQ0E7QUFDQSxPQUFPLFNBQVNDLG1CQUFtQkEsQ0FBQSxDQUFFLEVBQUUsT0FBTyxDQUFDO0VBQzdDLElBQUksVUFBVSxLQUFLLEtBQUssRUFBRSxPQUFPLElBQUk7RUFDckMsTUFBTUMsQ0FBQyxHQUFHLElBQUlDLElBQUksQ0FBQyxDQUFDO0VBQ3BCLE9BQU9ELENBQUMsQ0FBQ0UsV0FBVyxDQUFDLENBQUMsS0FBSyxJQUFJLElBQUlGLENBQUMsQ0FBQ0csUUFBUSxDQUFDLENBQUMsS0FBSyxDQUFDLElBQUlILENBQUMsQ0FBQ0ksT0FBTyxDQUFDLENBQUMsSUFBSSxDQUFDO0FBQzNFO0FBRUEsT0FBTyxTQUFTQyxXQUFXQSxDQUFBLENBQUUsRUFBRSxPQUFPLENBQUM7RUFDckMsSUFBSSxVQUFVLEtBQUssS0FBSyxFQUFFLE9BQU8sSUFBSTtFQUNyQyxNQUFNTCxDQUFDLEdBQUcsSUFBSUMsSUFBSSxDQUFDLENBQUM7RUFDcEIsT0FDRUQsQ0FBQyxDQUFDRSxXQUFXLENBQUMsQ0FBQyxHQUFHLElBQUksSUFBS0YsQ0FBQyxDQUFDRSxXQUFXLENBQUMsQ0FBQyxLQUFLLElBQUksSUFBSUYsQ0FBQyxDQUFDRyxRQUFRLENBQUMsQ0FBQyxJQUFJLENBQUU7QUFFN0U7QUFFQSxTQUFBRyxZQUFBQyxFQUFBO0VBQUEsTUFBQUMsQ0FBQSxHQUFBQyxFQUFBO0VBQXFCO0lBQUFDO0VBQUEsSUFBQUgsRUFBMEI7RUFBQSxJQUFBSSxFQUFBO0VBQUEsSUFBQUgsQ0FBQSxRQUFBRSxJQUFBO0lBRTNDQyxFQUFBLEtBQ0csS0FBSUQsSUFBSSxDQUFDLENBQUFFLEdBQUksQ0FBQ0MsS0FJZCxFQUFDLEdBQ0Q7SUFBQUwsQ0FBQSxNQUFBRSxJQUFBO0lBQUFGLENBQUEsTUFBQUcsRUFBQTtFQUFBO0lBQUFBLEVBQUEsR0FBQUgsQ0FBQTtFQUFBO0VBQUEsT0FOSEcsRUFNRztBQUFBOztBQUlQO0FBQ0E7QUFiQSxTQUFBRSxNQUFBQyxFQUFBLEVBQUFDLENBQUE7RUFBQSxPQUlRLENBQUMsSUFBSSxDQUFNQSxHQUFDLENBQURBLEVBQUEsQ0FBQyxDQUFTLEtBQWtCLENBQWxCLENBQUFqQixlQUFlLENBQUNpQixDQUFDLEVBQUMsQ0FDcENELEdBQUMsQ0FDSixFQUZDLElBQUksQ0FFRTtBQUFBO0FBUWYsT0FBTyxTQUFBRSxxQkFBQTtFQUFBLE1BQUFSLENBQUEsR0FBQUMsRUFBQTtFQUNMO0lBQUFRLGVBQUE7SUFBQUM7RUFBQSxJQUFnRHZCLGdCQUFnQixDQUFDLENBQUM7RUFBQSxJQUFBWSxFQUFBO0VBQUEsSUFBQUksRUFBQTtFQUFBLElBQUFILENBQUEsUUFBQVMsZUFBQSxJQUFBVCxDQUFBLFFBQUFVLGtCQUFBO0lBRXhEWCxFQUFBLEdBQUFBLENBQUE7TUFDUixJQUFJLENBQUNmLE9BQU8sQ0FBQyxPQUFPLENBQUM7UUFBQTtNQUFBO01BQ3JCLE1BQUEyQixNQUFBLEdBQWV0QixlQUFlLENBQUMsQ0FBQztNQUNoQyxJQUFJc0IsTUFBTSxDQUFBQyxTQUFvQyxJQUExQyxDQUFxQnJCLG1CQUFtQixDQUFDLENBQUM7UUFBQTtNQUFBO01BQzlDa0IsZUFBZSxDQUFDO1FBQUFJLEdBQUEsRUFDVCxjQUFjO1FBQUFDLEdBQUEsRUFDZCxDQUFDLFdBQVcsQ0FBTSxJQUFRLENBQVIsUUFBUSxHQUFHO1FBQUFDLFFBQUEsRUFDeEIsV0FBVztRQUFBQyxTQUFBLEVBQ1Y7TUFDYixDQUFDLENBQUM7TUFBQSxPQUNLLE1BQU1OLGtCQUFrQixDQUFDLGNBQWMsQ0FBQztJQUFBLENBQ2hEO0lBQUVQLEVBQUEsSUFBQ00sZUFBZSxFQUFFQyxrQkFBa0IsQ0FBQztJQUFBVixDQUFBLE1BQUFTLGVBQUE7SUFBQVQsQ0FBQSxNQUFBVSxrQkFBQTtJQUFBVixDQUFBLE1BQUFELEVBQUE7SUFBQUMsQ0FBQSxNQUFBRyxFQUFBO0VBQUE7SUFBQUosRUFBQSxHQUFBQyxDQUFBO0lBQUFHLEVBQUEsR0FBQUgsQ0FBQTtFQUFBO0VBWHhDZCxTQUFTLENBQUNhLEVBV1QsRUFBRUksRUFBcUMsQ0FBQztBQUFBO0FBRzNDLE9BQU8sU0FBU2MseUJBQXlCQSxDQUN2Q2YsSUFBSSxFQUFFLE1BQU0sQ0FDYixFQUFFZ0IsS0FBSyxDQUFDO0VBQUVDLEtBQUssRUFBRSxNQUFNO0VBQUVDLEdBQUcsRUFBRSxNQUFNO0FBQUMsQ0FBQyxDQUFDLENBQUM7RUFDdkMsSUFBSSxDQUFDcEMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxFQUFFLE9BQU8sRUFBRTtFQUNoQyxNQUFNcUMsUUFBUSxFQUFFSCxLQUFLLENBQUM7SUFBRUMsS0FBSyxFQUFFLE1BQU07SUFBRUMsR0FBRyxFQUFFLE1BQU07RUFBQyxDQUFDLENBQUMsR0FBRyxFQUFFO0VBQzFELE1BQU1FLEVBQUUsR0FBRyxZQUFZO0VBQ3ZCLElBQUlDLENBQUMsRUFBRUMsZUFBZSxHQUFHLElBQUk7RUFDN0IsT0FBTyxDQUFDRCxDQUFDLEdBQUdELEVBQUUsQ0FBQ0csSUFBSSxDQUFDdkIsSUFBSSxDQUFDLE1BQU0sSUFBSSxFQUFFO0lBQ25DbUIsUUFBUSxDQUFDSyxJQUFJLENBQUM7TUFBRVAsS0FBSyxFQUFFSSxDQUFDLENBQUNJLEtBQUs7TUFBRVAsR0FBRyxFQUFFRyxDQUFDLENBQUNJLEtBQUssR0FBR0osQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDSztJQUFPLENBQUMsQ0FBQztFQUMvRDtFQUNBLE9BQU9QLFFBQVE7QUFDakIiLCJpZ25vcmVMaXN0IjpbXX0= \ No newline at end of file diff --git a/src/cli/exit.ts b/src/cli/exit.ts new file mode 100644 index 0000000..99e56f9 --- /dev/null +++ b/src/cli/exit.ts @@ -0,0 +1,31 @@ +/** + * CLI exit helpers for subcommand handlers. + * + * Consolidates the 4-5 line "print + lint-suppress + exit" block that was + * copy-pasted ~60 times across `claude mcp *` / `claude plugin *` handlers. + * The `: never` return type lets TypeScript narrow control flow at call sites + * without a trailing `return`. + */ +/* eslint-disable custom-rules/no-process-exit -- centralized CLI exit point */ + +// `return undefined as never` (not a post-exit throw) — tests spy on +// process.exit and let it return. Call sites write `return cliError(...)` +// where subsequent code would dereference narrowed-away values under mock. +// cliError uses console.error (tests spy on console.error); cliOk uses +// process.stdout.write (tests spy on process.stdout.write — Bun's console.log +// doesn't route through a spied process.stdout.write). + +/** Write an error message to stderr (if given) and exit with code 1. */ +export function cliError(msg?: string): never { + // biome-ignore lint/suspicious/noConsole: centralized CLI error output + if (msg) console.error(msg) + process.exit(1) + return undefined as never +} + +/** Write a message to stdout (if given) and exit with code 0. */ +export function cliOk(msg?: string): never { + if (msg) process.stdout.write(msg + '\n') + process.exit(0) + return undefined as never +} diff --git a/src/cli/handlers/agents.ts b/src/cli/handlers/agents.ts new file mode 100644 index 0000000..c94723b --- /dev/null +++ b/src/cli/handlers/agents.ts @@ -0,0 +1,70 @@ +/** + * Agents subcommand handler — prints the list of configured agents. + * Dynamically imported only when `claude agents` runs. + */ + +import { + AGENT_SOURCE_GROUPS, + compareAgentsByName, + getOverrideSourceLabel, + type ResolvedAgent, + resolveAgentModelDisplay, + resolveAgentOverrides, +} from '../../tools/AgentTool/agentDisplay.js' +import { + getActiveAgentsFromList, + getAgentDefinitionsWithOverrides, +} from '../../tools/AgentTool/loadAgentsDir.js' +import { getCwd } from '../../utils/cwd.js' + +function formatAgent(agent: ResolvedAgent): string { + const model = resolveAgentModelDisplay(agent) + const parts = [agent.agentType] + if (model) { + parts.push(model) + } + if (agent.memory) { + parts.push(`${agent.memory} memory`) + } + return parts.join(' · ') +} + +export async function agentsHandler(): Promise { + const cwd = getCwd() + const { allAgents } = await getAgentDefinitionsWithOverrides(cwd) + const activeAgents = getActiveAgentsFromList(allAgents) + const resolvedAgents = resolveAgentOverrides(allAgents, activeAgents) + + const lines: string[] = [] + let totalActive = 0 + + for (const { label, source } of AGENT_SOURCE_GROUPS) { + const groupAgents = resolvedAgents + .filter(a => a.source === source) + .sort(compareAgentsByName) + + if (groupAgents.length === 0) continue + + lines.push(`${label}:`) + for (const agent of groupAgents) { + if (agent.overriddenBy) { + const winnerSource = getOverrideSourceLabel(agent.overriddenBy) + lines.push(` (shadowed by ${winnerSource}) ${formatAgent(agent)}`) + } else { + lines.push(` ${formatAgent(agent)}`) + totalActive++ + } + } + lines.push('') + } + + if (lines.length === 0) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log('No agents found.') + } else { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(`${totalActive} active agents\n`) + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(lines.join('\n').trimEnd()) + } +} diff --git a/src/cli/handlers/auth.ts b/src/cli/handlers/auth.ts new file mode 100644 index 0000000..c4cba5d --- /dev/null +++ b/src/cli/handlers/auth.ts @@ -0,0 +1,330 @@ +/* eslint-disable custom-rules/no-process-exit -- CLI subcommand handler intentionally exits */ + +import { + clearAuthRelatedCaches, + performLogout, +} from '../../commands/logout/logout.js' +import { + type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + logEvent, +} from '../../services/analytics/index.js' +import { getSSLErrorHint } from '../../services/api/errorUtils.js' +import { fetchAndStoreClaudeCodeFirstTokenDate } from '../../services/api/firstTokenDate.js' +import { + createAndStoreApiKey, + fetchAndStoreUserRoles, + refreshOAuthToken, + shouldUseClaudeAIAuth, + storeOAuthAccountInfo, +} from '../../services/oauth/client.js' +import { getOauthProfileFromOauthToken } from '../../services/oauth/getOauthProfile.js' +import { OAuthService } from '../../services/oauth/index.js' +import type { OAuthTokens } from '../../services/oauth/types.js' +import { + clearOAuthTokenCache, + getAnthropicApiKeyWithSource, + getAuthTokenSource, + getOauthAccountInfo, + getSubscriptionType, + isUsing3PServices, + saveOAuthTokensIfNeeded, + validateForceLoginOrg, +} from '../../utils/auth.js' +import { saveGlobalConfig } from '../../utils/config.js' +import { logForDebugging } from '../../utils/debug.js' +import { isRunningOnHomespace } from '../../utils/envUtils.js' +import { errorMessage } from '../../utils/errors.js' +import { logError } from '../../utils/log.js' +import { getAPIProvider } from '../../utils/model/providers.js' +import { getInitialSettings } from '../../utils/settings/settings.js' +import { jsonStringify } from '../../utils/slowOperations.js' +import { + buildAccountProperties, + buildAPIProviderProperties, +} from '../../utils/status.js' + +/** + * Shared post-token-acquisition logic. Saves tokens, fetches profile/roles, + * and sets up the local auth state. + */ +export async function installOAuthTokens(tokens: OAuthTokens): Promise { + // Clear old state before saving new credentials + await performLogout({ clearOnboarding: false }) + + // Reuse pre-fetched profile if available, otherwise fetch fresh + const profile = + tokens.profile ?? (await getOauthProfileFromOauthToken(tokens.accessToken)) + if (profile) { + storeOAuthAccountInfo({ + accountUuid: profile.account.uuid, + emailAddress: profile.account.email, + organizationUuid: profile.organization.uuid, + displayName: profile.account.display_name || undefined, + hasExtraUsageEnabled: + profile.organization.has_extra_usage_enabled ?? undefined, + billingType: profile.organization.billing_type ?? undefined, + subscriptionCreatedAt: + profile.organization.subscription_created_at ?? undefined, + accountCreatedAt: profile.account.created_at, + }) + } else if (tokens.tokenAccount) { + // Fallback to token exchange account data when profile endpoint fails + storeOAuthAccountInfo({ + accountUuid: tokens.tokenAccount.uuid, + emailAddress: tokens.tokenAccount.emailAddress, + organizationUuid: tokens.tokenAccount.organizationUuid, + }) + } + + const storageResult = saveOAuthTokensIfNeeded(tokens) + clearOAuthTokenCache() + + if (storageResult.warning) { + logEvent('tengu_oauth_storage_warning', { + warning: + storageResult.warning as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + } + + // Roles and first-token-date may fail for limited-scope tokens (e.g. + // inference-only from setup-token). They're not required for core auth. + await fetchAndStoreUserRoles(tokens.accessToken).catch(err => + logForDebugging(String(err), { level: 'error' }), + ) + + if (shouldUseClaudeAIAuth(tokens.scopes)) { + await fetchAndStoreClaudeCodeFirstTokenDate().catch(err => + logForDebugging(String(err), { level: 'error' }), + ) + } else { + // API key creation is critical for Console users — let it throw. + const apiKey = await createAndStoreApiKey(tokens.accessToken) + if (!apiKey) { + throw new Error( + 'Unable to create API key. The server accepted the request but did not return a key.', + ) + } + } + + await clearAuthRelatedCaches() +} + +export async function authLogin({ + email, + sso, + console: useConsole, + claudeai, +}: { + email?: string + sso?: boolean + console?: boolean + claudeai?: boolean +}): Promise { + if (useConsole && claudeai) { + process.stderr.write( + 'Error: --console and --claudeai cannot be used together.\n', + ) + process.exit(1) + } + + const settings = getInitialSettings() + // forceLoginMethod is a hard constraint (enterprise setting) — matches ConsoleOAuthFlow behavior. + // Without it, --console selects Console; --claudeai (or no flag) selects claude.ai. + const loginWithClaudeAi = settings.forceLoginMethod + ? settings.forceLoginMethod === 'claudeai' + : !useConsole + const orgUUID = settings.forceLoginOrgUUID + + // Fast path: if a refresh token is provided via env var, skip the browser + // OAuth flow and exchange it directly for tokens. + const envRefreshToken = process.env.CLAUDE_CODE_OAUTH_REFRESH_TOKEN + if (envRefreshToken) { + const envScopes = process.env.CLAUDE_CODE_OAUTH_SCOPES + if (!envScopes) { + process.stderr.write( + 'CLAUDE_CODE_OAUTH_SCOPES is required when using CLAUDE_CODE_OAUTH_REFRESH_TOKEN.\n' + + 'Set it to the space-separated scopes the refresh token was issued with\n' + + '(e.g. "user:inference" or "user:profile user:inference user:sessions:claude_code user:mcp_servers").\n', + ) + process.exit(1) + } + + const scopes = envScopes.split(/\s+/).filter(Boolean) + + try { + logEvent('tengu_login_from_refresh_token', {}) + + const tokens = await refreshOAuthToken(envRefreshToken, { scopes }) + await installOAuthTokens(tokens) + + const orgResult = await validateForceLoginOrg() + if (!orgResult.valid) { + process.stderr.write(orgResult.message + '\n') + process.exit(1) + } + + // Mark onboarding complete — interactive paths handle this via + // the Onboarding component, but the env var path skips it. + saveGlobalConfig(current => { + if (current.hasCompletedOnboarding) return current + return { ...current, hasCompletedOnboarding: true } + }) + + logEvent('tengu_oauth_success', { + loginWithClaudeAi: shouldUseClaudeAIAuth(tokens.scopes), + }) + process.stdout.write('Login successful.\n') + process.exit(0) + } catch (err) { + logError(err) + const sslHint = getSSLErrorHint(err) + process.stderr.write( + `Login failed: ${errorMessage(err)}\n${sslHint ? sslHint + '\n' : ''}`, + ) + process.exit(1) + } + } + + const resolvedLoginMethod = sso ? 'sso' : undefined + + const oauthService = new OAuthService() + + try { + logEvent('tengu_oauth_flow_start', { loginWithClaudeAi }) + + const result = await oauthService.startOAuthFlow( + async url => { + process.stdout.write('Opening browser to sign in…\n') + process.stdout.write(`If the browser didn't open, visit: ${url}\n`) + }, + { + loginWithClaudeAi, + loginHint: email, + loginMethod: resolvedLoginMethod, + orgUUID, + }, + ) + + await installOAuthTokens(result) + + const orgResult = await validateForceLoginOrg() + if (!orgResult.valid) { + process.stderr.write(orgResult.message + '\n') + process.exit(1) + } + + logEvent('tengu_oauth_success', { loginWithClaudeAi }) + + process.stdout.write('Login successful.\n') + process.exit(0) + } catch (err) { + logError(err) + const sslHint = getSSLErrorHint(err) + process.stderr.write( + `Login failed: ${errorMessage(err)}\n${sslHint ? sslHint + '\n' : ''}`, + ) + process.exit(1) + } finally { + oauthService.cleanup() + } +} + +export async function authStatus(opts: { + json?: boolean + text?: boolean +}): Promise { + const { source: authTokenSource, hasToken } = getAuthTokenSource() + const { source: apiKeySource } = getAnthropicApiKeyWithSource() + const hasApiKeyEnvVar = + !!process.env.ANTHROPIC_API_KEY && !isRunningOnHomespace() + const oauthAccount = getOauthAccountInfo() + const subscriptionType = getSubscriptionType() + const using3P = isUsing3PServices() + const loggedIn = + hasToken || apiKeySource !== 'none' || hasApiKeyEnvVar || using3P + + // Determine auth method + let authMethod: string = 'none' + if (using3P) { + authMethod = 'third_party' + } else if (authTokenSource === 'claude.ai') { + authMethod = 'claude.ai' + } else if (authTokenSource === 'apiKeyHelper') { + authMethod = 'api_key_helper' + } else if (authTokenSource !== 'none') { + authMethod = 'oauth_token' + } else if (apiKeySource === 'ANTHROPIC_API_KEY' || hasApiKeyEnvVar) { + authMethod = 'api_key' + } else if (apiKeySource === '/login managed key') { + authMethod = 'claude.ai' + } + + if (opts.text) { + const properties = [ + ...buildAccountProperties(), + ...buildAPIProviderProperties(), + ] + let hasAuthProperty = false + for (const prop of properties) { + const value = + typeof prop.value === 'string' + ? prop.value + : Array.isArray(prop.value) + ? prop.value.join(', ') + : null + if (value === null || value === 'none') { + continue + } + hasAuthProperty = true + if (prop.label) { + process.stdout.write(`${prop.label}: ${value}\n`) + } else { + process.stdout.write(`${value}\n`) + } + } + if (!hasAuthProperty && hasApiKeyEnvVar) { + process.stdout.write('API key: ANTHROPIC_API_KEY\n') + } + if (!loggedIn) { + process.stdout.write( + 'Not logged in. Run claude auth login to authenticate.\n', + ) + } + } else { + const apiProvider = getAPIProvider() + const resolvedApiKeySource = + apiKeySource !== 'none' + ? apiKeySource + : hasApiKeyEnvVar + ? 'ANTHROPIC_API_KEY' + : null + const output: Record = { + loggedIn, + authMethod, + apiProvider, + } + if (resolvedApiKeySource) { + output.apiKeySource = resolvedApiKeySource + } + if (authMethod === 'claude.ai') { + output.email = oauthAccount?.emailAddress ?? null + output.orgId = oauthAccount?.organizationUuid ?? null + output.orgName = oauthAccount?.organizationName ?? null + output.subscriptionType = subscriptionType ?? null + } + + process.stdout.write(jsonStringify(output, null, 2) + '\n') + } + process.exit(loggedIn ? 0 : 1) +} + +export async function authLogout(): Promise { + try { + await performLogout({ clearOnboarding: false }) + } catch { + process.stderr.write('Failed to log out.\n') + process.exit(1) + } + process.stdout.write('Successfully logged out from your Anthropic account.\n') + process.exit(0) +} diff --git a/src/cli/handlers/autoMode.ts b/src/cli/handlers/autoMode.ts new file mode 100644 index 0000000..fb2c3d2 --- /dev/null +++ b/src/cli/handlers/autoMode.ts @@ -0,0 +1,170 @@ +/** + * Auto mode subcommand handlers — dump default/merged classifier rules and + * critique user-written rules. Dynamically imported when `claude auto-mode ...` runs. + */ + +import { errorMessage } from '../../utils/errors.js' +import { + getMainLoopModel, + parseUserSpecifiedModel, +} from '../../utils/model/model.js' +import { + type AutoModeRules, + buildDefaultExternalSystemPrompt, + getDefaultExternalAutoModeRules, +} from '../../utils/permissions/yoloClassifier.js' +import { getAutoModeConfig } from '../../utils/settings/settings.js' +import { sideQuery } from '../../utils/sideQuery.js' +import { jsonStringify } from '../../utils/slowOperations.js' + +function writeRules(rules: AutoModeRules): void { + process.stdout.write(jsonStringify(rules, null, 2) + '\n') +} + +export function autoModeDefaultsHandler(): void { + writeRules(getDefaultExternalAutoModeRules()) +} + +/** + * Dump the effective auto mode config: user settings where provided, external + * defaults otherwise. Per-section REPLACE semantics — matches how + * buildYoloSystemPrompt resolves the external template (a non-empty user + * section replaces that section's defaults entirely; an empty/absent section + * falls through to defaults). + */ +export function autoModeConfigHandler(): void { + const config = getAutoModeConfig() + const defaults = getDefaultExternalAutoModeRules() + writeRules({ + allow: config?.allow?.length ? config.allow : defaults.allow, + soft_deny: config?.soft_deny?.length + ? config.soft_deny + : defaults.soft_deny, + environment: config?.environment?.length + ? config.environment + : defaults.environment, + }) +} + +const CRITIQUE_SYSTEM_PROMPT = + 'You are an expert reviewer of auto mode classifier rules for Claude Code.\n' + + '\n' + + 'Claude Code has an "auto mode" that uses an AI classifier to decide whether ' + + 'tool calls should be auto-approved or require user confirmation. Users can ' + + 'write custom rules in three categories:\n' + + '\n' + + '- **allow**: Actions the classifier should auto-approve\n' + + '- **soft_deny**: Actions the classifier should block (require user confirmation)\n' + + "- **environment**: Context about the user's setup that helps the classifier make decisions\n" + + '\n' + + "Your job is to critique the user's custom rules for clarity, completeness, " + + 'and potential issues. The classifier is an LLM that reads these rules as ' + + 'part of its system prompt.\n' + + '\n' + + 'For each rule, evaluate:\n' + + '1. **Clarity**: Is the rule unambiguous? Could the classifier misinterpret it?\n' + + "2. **Completeness**: Are there gaps or edge cases the rule doesn't cover?\n" + + '3. **Conflicts**: Do any of the rules conflict with each other?\n' + + '4. **Actionability**: Is the rule specific enough for the classifier to act on?\n' + + '\n' + + 'Be concise and constructive. Only comment on rules that could be improved. ' + + 'If all rules look good, say so.' + +export async function autoModeCritiqueHandler(options: { + model?: string +}): Promise { + const config = getAutoModeConfig() + const hasCustomRules = + (config?.allow?.length ?? 0) > 0 || + (config?.soft_deny?.length ?? 0) > 0 || + (config?.environment?.length ?? 0) > 0 + + if (!hasCustomRules) { + process.stdout.write( + 'No custom auto mode rules found.\n\n' + + 'Add rules to your settings file under autoMode.{allow, soft_deny, environment}.\n' + + 'Run `claude auto-mode defaults` to see the default rules for reference.\n', + ) + return + } + + const model = options.model + ? parseUserSpecifiedModel(options.model) + : getMainLoopModel() + + const defaults = getDefaultExternalAutoModeRules() + const classifierPrompt = buildDefaultExternalSystemPrompt() + + const userRulesSummary = + formatRulesForCritique('allow', config?.allow ?? [], defaults.allow) + + formatRulesForCritique( + 'soft_deny', + config?.soft_deny ?? [], + defaults.soft_deny, + ) + + formatRulesForCritique( + 'environment', + config?.environment ?? [], + defaults.environment, + ) + + process.stdout.write('Analyzing your auto mode rules…\n\n') + + let response + try { + response = await sideQuery({ + querySource: 'auto_mode_critique', + model, + system: CRITIQUE_SYSTEM_PROMPT, + skipSystemPromptPrefix: true, + max_tokens: 4096, + messages: [ + { + role: 'user', + content: + 'Here is the full classifier system prompt that the auto mode classifier receives:\n\n' + + '\n' + + classifierPrompt + + '\n\n\n' + + "Here are the user's custom rules that REPLACE the corresponding default sections:\n\n" + + userRulesSummary + + '\nPlease critique these custom rules.', + }, + ], + }) + } catch (error) { + process.stderr.write( + 'Failed to analyze rules: ' + errorMessage(error) + '\n', + ) + process.exitCode = 1 + return + } + + const textBlock = response.content.find(block => block.type === 'text') + if (textBlock?.type === 'text') { + process.stdout.write(textBlock.text + '\n') + } else { + process.stdout.write('No critique was generated. Please try again.\n') + } +} + +function formatRulesForCritique( + section: string, + userRules: string[], + defaultRules: string[], +): string { + if (userRules.length === 0) return '' + const customLines = userRules.map(r => '- ' + r).join('\n') + const defaultLines = defaultRules.map(r => '- ' + r).join('\n') + return ( + '## ' + + section + + ' (custom rules replacing defaults)\n' + + 'Custom:\n' + + customLines + + '\n\n' + + 'Defaults being replaced:\n' + + defaultLines + + '\n\n' + ) +} diff --git a/src/cli/handlers/mcp.tsx b/src/cli/handlers/mcp.tsx new file mode 100644 index 0000000..e530c26 --- /dev/null +++ b/src/cli/handlers/mcp.tsx @@ -0,0 +1,362 @@ +/** + * MCP subcommand handlers — extracted from main.tsx for lazy loading. + * These are dynamically imported only when the corresponding `claude mcp *` command runs. + */ + +import { stat } from 'fs/promises'; +import pMap from 'p-map'; +import { cwd } from 'process'; +import React from 'react'; +import { MCPServerDesktopImportDialog } from '../../components/MCPServerDesktopImportDialog.js'; +import { render } from '../../ink.js'; +import { KeybindingSetup } from '../../keybindings/KeybindingProviderSetup.js'; +import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from '../../services/analytics/index.js'; +import { clearMcpClientConfig, clearServerTokensFromLocalStorage, getMcpClientConfig, readClientSecret, saveMcpClientSecret } from '../../services/mcp/auth.js'; +import { connectToServer, getMcpServerConnectionBatchSize } from '../../services/mcp/client.js'; +import { addMcpConfig, getAllMcpConfigs, getMcpConfigByName, getMcpConfigsByScope, removeMcpConfig } from '../../services/mcp/config.js'; +import type { ConfigScope, ScopedMcpServerConfig } from '../../services/mcp/types.js'; +import { describeMcpConfigFilePath, ensureConfigScope, getScopeLabel } from '../../services/mcp/utils.js'; +import { AppStateProvider } from '../../state/AppState.js'; +import { getCurrentProjectConfig, getGlobalConfig, saveCurrentProjectConfig } from '../../utils/config.js'; +import { isFsInaccessible } from '../../utils/errors.js'; +import { gracefulShutdown } from '../../utils/gracefulShutdown.js'; +import { safeParseJSON } from '../../utils/json.js'; +import { getPlatform } from '../../utils/platform.js'; +import { cliError, cliOk } from '../exit.js'; +async function checkMcpServerHealth(name: string, server: ScopedMcpServerConfig): Promise { + try { + const result = await connectToServer(name, server); + if (result.type === 'connected') { + return '✓ Connected'; + } else if (result.type === 'needs-auth') { + return '! Needs authentication'; + } else { + return '✗ Failed to connect'; + } + } catch (_error) { + return '✗ Connection error'; + } +} + +// mcp serve (lines 4512–4532) +export async function mcpServeHandler({ + debug, + verbose +}: { + debug?: boolean; + verbose?: boolean; +}): Promise { + const providedCwd = cwd(); + logEvent('tengu_mcp_start', {}); + try { + await stat(providedCwd); + } catch (error) { + if (isFsInaccessible(error)) { + cliError(`Error: Directory ${providedCwd} does not exist`); + } + throw error; + } + try { + const { + setup + } = await import('../../setup.js'); + await setup(providedCwd, 'default', false, false, undefined, false); + const { + startMCPServer + } = await import('../../entrypoints/mcp.js'); + await startMCPServer(providedCwd, debug ?? false, verbose ?? false); + } catch (error) { + cliError(`Error: Failed to start MCP server: ${error}`); + } +} + +// mcp remove (lines 4545–4635) +export async function mcpRemoveHandler(name: string, options: { + scope?: string; +}): Promise { + // Look up config before removing so we can clean up secure storage + const serverBeforeRemoval = getMcpConfigByName(name); + const cleanupSecureStorage = () => { + if (serverBeforeRemoval && (serverBeforeRemoval.type === 'sse' || serverBeforeRemoval.type === 'http')) { + clearServerTokensFromLocalStorage(name, serverBeforeRemoval); + clearMcpClientConfig(name, serverBeforeRemoval); + } + }; + try { + if (options.scope) { + const scope = ensureConfigScope(options.scope); + logEvent('tengu_mcp_delete', { + name: name as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + scope: scope as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + await removeMcpConfig(name, scope); + cleanupSecureStorage(); + process.stdout.write(`Removed MCP server ${name} from ${scope} config\n`); + cliOk(`File modified: ${describeMcpConfigFilePath(scope)}`); + } + + // If no scope specified, check where the server exists + const projectConfig = getCurrentProjectConfig(); + const globalConfig = getGlobalConfig(); + + // Check if server exists in project scope (.mcp.json) + const { + servers: projectServers + } = getMcpConfigsByScope('project'); + const mcpJsonExists = !!projectServers[name]; + + // Count how many scopes contain this server + const scopes: Array> = []; + if (projectConfig.mcpServers?.[name]) scopes.push('local'); + if (mcpJsonExists) scopes.push('project'); + if (globalConfig.mcpServers?.[name]) scopes.push('user'); + if (scopes.length === 0) { + cliError(`No MCP server found with name: "${name}"`); + } else if (scopes.length === 1) { + // Server exists in only one scope, remove it + const scope = scopes[0]!; + logEvent('tengu_mcp_delete', { + name: name as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + scope: scope as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + await removeMcpConfig(name, scope); + cleanupSecureStorage(); + process.stdout.write(`Removed MCP server "${name}" from ${scope} config\n`); + cliOk(`File modified: ${describeMcpConfigFilePath(scope)}`); + } else { + // Server exists in multiple scopes + process.stderr.write(`MCP server "${name}" exists in multiple scopes:\n`); + scopes.forEach(scope => { + process.stderr.write(` - ${getScopeLabel(scope)} (${describeMcpConfigFilePath(scope)})\n`); + }); + process.stderr.write('\nTo remove from a specific scope, use:\n'); + scopes.forEach(scope => { + process.stderr.write(` claude mcp remove "${name}" -s ${scope}\n`); + }); + cliError(); + } + } catch (error) { + cliError((error as Error).message); + } +} + +// mcp list (lines 4641–4688) +export async function mcpListHandler(): Promise { + logEvent('tengu_mcp_list', {}); + const { + servers: configs + } = await getAllMcpConfigs(); + if (Object.keys(configs).length === 0) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log('No MCP servers configured. Use `claude mcp add` to add a server.'); + } else { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log('Checking MCP server health...\n'); + + // Check servers concurrently + const entries = Object.entries(configs); + const results = await pMap(entries, async ([name, server]) => ({ + name, + server, + status: await checkMcpServerHealth(name, server) + }), { + concurrency: getMcpServerConnectionBatchSize() + }); + for (const { + name, + server, + status + } of results) { + // Intentionally excluding sse-ide servers here since they're internal + if (server.type === 'sse') { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(`${name}: ${server.url} (SSE) - ${status}`); + } else if (server.type === 'http') { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(`${name}: ${server.url} (HTTP) - ${status}`); + } else if (server.type === 'claudeai-proxy') { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(`${name}: ${server.url} - ${status}`); + } else if (!server.type || server.type === 'stdio') { + const args = Array.isArray(server.args) ? server.args : []; + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(`${name}: ${server.command} ${args.join(' ')} - ${status}`); + } + } + } + // Use gracefulShutdown to properly clean up MCP server connections + // (process.exit bypasses cleanup handlers, leaving child processes orphaned) + await gracefulShutdown(0); +} + +// mcp get (lines 4694–4786) +export async function mcpGetHandler(name: string): Promise { + logEvent('tengu_mcp_get', { + name: name as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + const server = getMcpConfigByName(name); + if (!server) { + cliError(`No MCP server found with name: ${name}`); + } + + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(`${name}:`); + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Scope: ${getScopeLabel(server.scope)}`); + + // Check server health + const status = await checkMcpServerHealth(name, server); + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Status: ${status}`); + + // Intentionally excluding sse-ide servers here since they're internal + if (server.type === 'sse') { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Type: sse`); + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` URL: ${server.url}`); + if (server.headers) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(' Headers:'); + for (const [key, value] of Object.entries(server.headers)) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` ${key}: ${value}`); + } + } + if (server.oauth?.clientId || server.oauth?.callbackPort) { + const parts: string[] = []; + if (server.oauth.clientId) { + parts.push('client_id configured'); + const clientConfig = getMcpClientConfig(name, server); + if (clientConfig?.clientSecret) parts.push('client_secret configured'); + } + if (server.oauth.callbackPort) parts.push(`callback_port ${server.oauth.callbackPort}`); + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` OAuth: ${parts.join(', ')}`); + } + } else if (server.type === 'http') { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Type: http`); + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` URL: ${server.url}`); + if (server.headers) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(' Headers:'); + for (const [key, value] of Object.entries(server.headers)) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` ${key}: ${value}`); + } + } + if (server.oauth?.clientId || server.oauth?.callbackPort) { + const parts: string[] = []; + if (server.oauth.clientId) { + parts.push('client_id configured'); + const clientConfig = getMcpClientConfig(name, server); + if (clientConfig?.clientSecret) parts.push('client_secret configured'); + } + if (server.oauth.callbackPort) parts.push(`callback_port ${server.oauth.callbackPort}`); + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` OAuth: ${parts.join(', ')}`); + } + } else if (server.type === 'stdio') { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Type: stdio`); + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Command: ${server.command}`); + const args = Array.isArray(server.args) ? server.args : []; + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Args: ${args.join(' ')}`); + if (server.env) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(' Environment:'); + for (const [key, value] of Object.entries(server.env)) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` ${key}=${value}`); + } + } + } + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(`\nTo remove this server, run: claude mcp remove "${name}" -s ${server.scope}`); + // Use gracefulShutdown to properly clean up MCP server connections + // (process.exit bypasses cleanup handlers, leaving child processes orphaned) + await gracefulShutdown(0); +} + +// mcp add-json (lines 4801–4870) +export async function mcpAddJsonHandler(name: string, json: string, options: { + scope?: string; + clientSecret?: true; +}): Promise { + try { + const scope = ensureConfigScope(options.scope); + const parsedJson = safeParseJSON(json); + + // Read secret before writing config so cancellation doesn't leave partial state + const needsSecret = options.clientSecret && parsedJson && typeof parsedJson === 'object' && 'type' in parsedJson && (parsedJson.type === 'sse' || parsedJson.type === 'http') && 'url' in parsedJson && typeof parsedJson.url === 'string' && 'oauth' in parsedJson && parsedJson.oauth && typeof parsedJson.oauth === 'object' && 'clientId' in parsedJson.oauth; + const clientSecret = needsSecret ? await readClientSecret() : undefined; + await addMcpConfig(name, parsedJson, scope); + const transportType = parsedJson && typeof parsedJson === 'object' && 'type' in parsedJson ? String(parsedJson.type || 'stdio') : 'stdio'; + if (clientSecret && parsedJson && typeof parsedJson === 'object' && 'type' in parsedJson && (parsedJson.type === 'sse' || parsedJson.type === 'http') && 'url' in parsedJson && typeof parsedJson.url === 'string') { + saveMcpClientSecret(name, { + type: parsedJson.type, + url: parsedJson.url + }, clientSecret); + } + logEvent('tengu_mcp_add', { + scope: scope as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + source: 'json' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + type: transportType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + cliOk(`Added ${transportType} MCP server ${name} to ${scope} config`); + } catch (error) { + cliError((error as Error).message); + } +} + +// mcp add-from-claude-desktop (lines 4881–4927) +export async function mcpAddFromDesktopHandler(options: { + scope?: string; +}): Promise { + try { + const scope = ensureConfigScope(options.scope); + const platform = getPlatform(); + logEvent('tengu_mcp_add', { + scope: scope as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + platform: platform as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + source: 'desktop' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + const { + readClaudeDesktopMcpServers + } = await import('../../utils/claudeDesktop.js'); + const servers = await readClaudeDesktopMcpServers(); + if (Object.keys(servers).length === 0) { + cliOk('No MCP servers found in Claude Desktop configuration or configuration file does not exist.'); + } + const { + unmount + } = await render( + + { + unmount(); + }} /> + + , { + exitOnCtrlC: true + }); + } catch (error) { + cliError((error as Error).message); + } +} + +// mcp reset-project-choices (lines 4935–4952) +export async function mcpResetChoicesHandler(): Promise { + logEvent('tengu_mcp_reset_mcpjson_choices', {}); + saveCurrentProjectConfig(current => ({ + ...current, + enabledMcpjsonServers: [], + disabledMcpjsonServers: [], + enableAllProjectMcpServers: false + })); + cliOk('All project-scoped (.mcp.json) server approvals and rejections have been reset.\n' + 'You will be prompted for approval next time you start Claude Code.'); +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJzdGF0IiwicE1hcCIsImN3ZCIsIlJlYWN0IiwiTUNQU2VydmVyRGVza3RvcEltcG9ydERpYWxvZyIsInJlbmRlciIsIktleWJpbmRpbmdTZXR1cCIsIkFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMiLCJsb2dFdmVudCIsImNsZWFyTWNwQ2xpZW50Q29uZmlnIiwiY2xlYXJTZXJ2ZXJUb2tlbnNGcm9tTG9jYWxTdG9yYWdlIiwiZ2V0TWNwQ2xpZW50Q29uZmlnIiwicmVhZENsaWVudFNlY3JldCIsInNhdmVNY3BDbGllbnRTZWNyZXQiLCJjb25uZWN0VG9TZXJ2ZXIiLCJnZXRNY3BTZXJ2ZXJDb25uZWN0aW9uQmF0Y2hTaXplIiwiYWRkTWNwQ29uZmlnIiwiZ2V0QWxsTWNwQ29uZmlncyIsImdldE1jcENvbmZpZ0J5TmFtZSIsImdldE1jcENvbmZpZ3NCeVNjb3BlIiwicmVtb3ZlTWNwQ29uZmlnIiwiQ29uZmlnU2NvcGUiLCJTY29wZWRNY3BTZXJ2ZXJDb25maWciLCJkZXNjcmliZU1jcENvbmZpZ0ZpbGVQYXRoIiwiZW5zdXJlQ29uZmlnU2NvcGUiLCJnZXRTY29wZUxhYmVsIiwiQXBwU3RhdGVQcm92aWRlciIsImdldEN1cnJlbnRQcm9qZWN0Q29uZmlnIiwiZ2V0R2xvYmFsQ29uZmlnIiwic2F2ZUN1cnJlbnRQcm9qZWN0Q29uZmlnIiwiaXNGc0luYWNjZXNzaWJsZSIsImdyYWNlZnVsU2h1dGRvd24iLCJzYWZlUGFyc2VKU09OIiwiZ2V0UGxhdGZvcm0iLCJjbGlFcnJvciIsImNsaU9rIiwiY2hlY2tNY3BTZXJ2ZXJIZWFsdGgiLCJuYW1lIiwic2VydmVyIiwiUHJvbWlzZSIsInJlc3VsdCIsInR5cGUiLCJfZXJyb3IiLCJtY3BTZXJ2ZUhhbmRsZXIiLCJkZWJ1ZyIsInZlcmJvc2UiLCJwcm92aWRlZEN3ZCIsImVycm9yIiwic2V0dXAiLCJ1bmRlZmluZWQiLCJzdGFydE1DUFNlcnZlciIsIm1jcFJlbW92ZUhhbmRsZXIiLCJvcHRpb25zIiwic2NvcGUiLCJzZXJ2ZXJCZWZvcmVSZW1vdmFsIiwiY2xlYW51cFNlY3VyZVN0b3JhZ2UiLCJwcm9jZXNzIiwic3Rkb3V0Iiwid3JpdGUiLCJwcm9qZWN0Q29uZmlnIiwiZ2xvYmFsQ29uZmlnIiwic2VydmVycyIsInByb2plY3RTZXJ2ZXJzIiwibWNwSnNvbkV4aXN0cyIsInNjb3BlcyIsIkFycmF5IiwiRXhjbHVkZSIsIm1jcFNlcnZlcnMiLCJwdXNoIiwibGVuZ3RoIiwic3RkZXJyIiwiZm9yRWFjaCIsIkVycm9yIiwibWVzc2FnZSIsIm1jcExpc3RIYW5kbGVyIiwiY29uZmlncyIsIk9iamVjdCIsImtleXMiLCJjb25zb2xlIiwibG9nIiwiZW50cmllcyIsInJlc3VsdHMiLCJzdGF0dXMiLCJjb25jdXJyZW5jeSIsInVybCIsImFyZ3MiLCJpc0FycmF5IiwiY29tbWFuZCIsImpvaW4iLCJtY3BHZXRIYW5kbGVyIiwiaGVhZGVycyIsImtleSIsInZhbHVlIiwib2F1dGgiLCJjbGllbnRJZCIsImNhbGxiYWNrUG9ydCIsInBhcnRzIiwiY2xpZW50Q29uZmlnIiwiY2xpZW50U2VjcmV0IiwiZW52IiwibWNwQWRkSnNvbkhhbmRsZXIiLCJqc29uIiwicGFyc2VkSnNvbiIsIm5lZWRzU2VjcmV0IiwidHJhbnNwb3J0VHlwZSIsIlN0cmluZyIsInNvdXJjZSIsIm1jcEFkZEZyb21EZXNrdG9wSGFuZGxlciIsInBsYXRmb3JtIiwicmVhZENsYXVkZURlc2t0b3BNY3BTZXJ2ZXJzIiwidW5tb3VudCIsImV4aXRPbkN0cmxDIiwibWNwUmVzZXRDaG9pY2VzSGFuZGxlciIsImN1cnJlbnQiLCJlbmFibGVkTWNwanNvblNlcnZlcnMiLCJkaXNhYmxlZE1jcGpzb25TZXJ2ZXJzIiwiZW5hYmxlQWxsUHJvamVjdE1jcFNlcnZlcnMiXSwic291cmNlcyI6WyJtY3AudHN4Il0sInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogTUNQIHN1YmNvbW1hbmQgaGFuZGxlcnMg4oCUIGV4dHJhY3RlZCBmcm9tIG1haW4udHN4IGZvciBsYXp5IGxvYWRpbmcuXG4gKiBUaGVzZSBhcmUgZHluYW1pY2FsbHkgaW1wb3J0ZWQgb25seSB3aGVuIHRoZSBjb3JyZXNwb25kaW5nIGBjbGF1ZGUgbWNwICpgIGNvbW1hbmQgcnVucy5cbiAqL1xuXG5pbXBvcnQgeyBzdGF0IH0gZnJvbSAnZnMvcHJvbWlzZXMnXG5pbXBvcnQgcE1hcCBmcm9tICdwLW1hcCdcbmltcG9ydCB7IGN3ZCB9IGZyb20gJ3Byb2Nlc3MnXG5pbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBNQ1BTZXJ2ZXJEZXNrdG9wSW1wb3J0RGlhbG9nIH0gZnJvbSAnLi4vLi4vY29tcG9uZW50cy9NQ1BTZXJ2ZXJEZXNrdG9wSW1wb3J0RGlhbG9nLmpzJ1xuaW1wb3J0IHsgcmVuZGVyIH0gZnJvbSAnLi4vLi4vaW5rLmpzJ1xuaW1wb3J0IHsgS2V5YmluZGluZ1NldHVwIH0gZnJvbSAnLi4vLi4va2V5YmluZGluZ3MvS2V5YmluZGluZ1Byb3ZpZGVyU2V0dXAuanMnXG5pbXBvcnQge1xuICB0eXBlIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gIGxvZ0V2ZW50LFxufSBmcm9tICcuLi8uLi9zZXJ2aWNlcy9hbmFseXRpY3MvaW5kZXguanMnXG5pbXBvcnQge1xuICBjbGVhck1jcENsaWVudENvbmZpZyxcbiAgY2xlYXJTZXJ2ZXJUb2tlbnNGcm9tTG9jYWxTdG9yYWdlLFxuICBnZXRNY3BDbGllbnRDb25maWcsXG4gIHJlYWRDbGllbnRTZWNyZXQsXG4gIHNhdmVNY3BDbGllbnRTZWNyZXQsXG59IGZyb20gJy4uLy4uL3NlcnZpY2VzL21jcC9hdXRoLmpzJ1xuaW1wb3J0IHtcbiAgY29ubmVjdFRvU2VydmVyLFxuICBnZXRNY3BTZXJ2ZXJDb25uZWN0aW9uQmF0Y2hTaXplLFxufSBmcm9tICcuLi8uLi9zZXJ2aWNlcy9tY3AvY2xpZW50LmpzJ1xuaW1wb3J0IHtcbiAgYWRkTWNwQ29uZmlnLFxuICBnZXRBbGxNY3BDb25maWdzLFxuICBnZXRNY3BDb25maWdCeU5hbWUsXG4gIGdldE1jcENvbmZpZ3NCeVNjb3BlLFxuICByZW1vdmVNY3BDb25maWcsXG59IGZyb20gJy4uLy4uL3NlcnZpY2VzL21jcC9jb25maWcuanMnXG5pbXBvcnQgdHlwZSB7XG4gIENvbmZpZ1Njb3BlLFxuICBTY29wZWRNY3BTZXJ2ZXJDb25maWcsXG59IGZyb20gJy4uLy4uL3NlcnZpY2VzL21jcC90eXBlcy5qcydcbmltcG9ydCB7XG4gIGRlc2NyaWJlTWNwQ29uZmlnRmlsZVBhdGgsXG4gIGVuc3VyZUNvbmZpZ1Njb3BlLFxuICBnZXRTY29wZUxhYmVsLFxufSBmcm9tICcuLi8uLi9zZXJ2aWNlcy9tY3AvdXRpbHMuanMnXG5pbXBvcnQgeyBBcHBTdGF0ZVByb3ZpZGVyIH0gZnJvbSAnLi4vLi4vc3RhdGUvQXBwU3RhdGUuanMnXG5pbXBvcnQge1xuICBnZXRDdXJyZW50UHJvamVjdENvbmZpZyxcbiAgZ2V0R2xvYmFsQ29uZmlnLFxuICBzYXZlQ3VycmVudFByb2plY3RDb25maWcsXG59IGZyb20gJy4uLy4uL3V0aWxzL2NvbmZpZy5qcydcbmltcG9ydCB7IGlzRnNJbmFjY2Vzc2libGUgfSBmcm9tICcuLi8uLi91dGlscy9lcnJvcnMuanMnXG5pbXBvcnQgeyBncmFjZWZ1bFNodXRkb3duIH0gZnJvbSAnLi4vLi4vdXRpbHMvZ3JhY2VmdWxTaHV0ZG93bi5qcydcbmltcG9ydCB7IHNhZmVQYXJzZUpTT04gfSBmcm9tICcuLi8uLi91dGlscy9qc29uLmpzJ1xuaW1wb3J0IHsgZ2V0UGxhdGZvcm0gfSBmcm9tICcuLi8uLi91dGlscy9wbGF0Zm9ybS5qcydcbmltcG9ydCB7IGNsaUVycm9yLCBjbGlPayB9IGZyb20gJy4uL2V4aXQuanMnXG5cbmFzeW5jIGZ1bmN0aW9uIGNoZWNrTWNwU2VydmVySGVhbHRoKFxuICBuYW1lOiBzdHJpbmcsXG4gIHNlcnZlcjogU2NvcGVkTWNwU2VydmVyQ29uZmlnLFxuKTogUHJvbWlzZTxzdHJpbmc+IHtcbiAgdHJ5IHtcbiAgICBjb25zdCByZXN1bHQgPSBhd2FpdCBjb25uZWN0VG9TZXJ2ZXIobmFtZSwgc2VydmVyKVxuICAgIGlmIChyZXN1bHQudHlwZSA9PT0gJ2Nvbm5lY3RlZCcpIHtcbiAgICAgIHJldHVybiAn4pyTIENvbm5lY3RlZCdcbiAgICB9IGVsc2UgaWYgKHJlc3VsdC50eXBlID09PSAnbmVlZHMtYXV0aCcpIHtcbiAgICAgIHJldHVybiAnISBOZWVkcyBhdXRoZW50aWNhdGlvbidcbiAgICB9IGVsc2Uge1xuICAgICAgcmV0dXJuICfinJcgRmFpbGVkIHRvIGNvbm5lY3QnXG4gICAgfVxuICB9IGNhdGNoIChfZXJyb3IpIHtcbiAgICByZXR1cm4gJ+KclyBDb25uZWN0aW9uIGVycm9yJ1xuICB9XG59XG5cbi8vIG1jcCBzZXJ2ZSAobGluZXMgNDUxMuKAkzQ1MzIpXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gbWNwU2VydmVIYW5kbGVyKHtcbiAgZGVidWcsXG4gIHZlcmJvc2UsXG59OiB7XG4gIGRlYnVnPzogYm9vbGVhblxuICB2ZXJib3NlPzogYm9vbGVhblxufSk6IFByb21pc2U8dm9pZD4ge1xuICBjb25zdCBwcm92aWRlZEN3ZCA9IGN3ZCgpXG4gIGxvZ0V2ZW50KCd0ZW5ndV9tY3Bfc3RhcnQnLCB7fSlcblxuICB0cnkge1xuICAgIGF3YWl0IHN0YXQocHJvdmlkZWRDd2QpXG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgaWYgKGlzRnNJbmFjY2Vzc2libGUoZXJyb3IpKSB7XG4gICAgICBjbGlFcnJvcihgRXJyb3I6IERpcmVjdG9yeSAke3Byb3ZpZGVkQ3dkfSBkb2VzIG5vdCBleGlzdGApXG4gICAgfVxuICAgIHRocm93IGVycm9yXG4gIH1cblxuICB0cnkge1xuICAgIGNvbnN0IHsgc2V0dXAgfSA9IGF3YWl0IGltcG9ydCgnLi4vLi4vc2V0dXAuanMnKVxuICAgIGF3YWl0IHNldHVwKHByb3ZpZGVkQ3dkLCAnZGVmYXVsdCcsIGZhbHNlLCBmYWxzZSwgdW5kZWZpbmVkLCBmYWxzZSlcbiAgICBjb25zdCB7IHN0YXJ0TUNQU2VydmVyIH0gPSBhd2FpdCBpbXBvcnQoJy4uLy4uL2VudHJ5cG9pbnRzL21jcC5qcycpXG4gICAgYXdhaXQgc3RhcnRNQ1BTZXJ2ZXIocHJvdmlkZWRDd2QsIGRlYnVnID8/IGZhbHNlLCB2ZXJib3NlID8/IGZhbHNlKVxuICB9IGNhdGNoIChlcnJvcikge1xuICAgIGNsaUVycm9yKGBFcnJvcjogRmFpbGVkIHRvIHN0YXJ0IE1DUCBzZXJ2ZXI6ICR7ZXJyb3J9YClcbiAgfVxufVxuXG4vLyBtY3AgcmVtb3ZlIChsaW5lcyA0NTQ14oCTNDYzNSlcbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBtY3BSZW1vdmVIYW5kbGVyKFxuICBuYW1lOiBzdHJpbmcsXG4gIG9wdGlvbnM6IHsgc2NvcGU/OiBzdHJpbmcgfSxcbik6IFByb21pc2U8dm9pZD4ge1xuICAvLyBMb29rIHVwIGNvbmZpZyBiZWZvcmUgcmVtb3Zpbmcgc28gd2UgY2FuIGNsZWFuIHVwIHNlY3VyZSBzdG9yYWdlXG4gIGNvbnN0IHNlcnZlckJlZm9yZVJlbW92YWwgPSBnZXRNY3BDb25maWdCeU5hbWUobmFtZSlcblxuICBjb25zdCBjbGVhbnVwU2VjdXJlU3RvcmFnZSA9ICgpID0+IHtcbiAgICBpZiAoXG4gICAgICBzZXJ2ZXJCZWZvcmVSZW1vdmFsICYmXG4gICAgICAoc2VydmVyQmVmb3JlUmVtb3ZhbC50eXBlID09PSAnc3NlJyB8fFxuICAgICAgICBzZXJ2ZXJCZWZvcmVSZW1vdmFsLnR5cGUgPT09ICdodHRwJylcbiAgICApIHtcbiAgICAgIGNsZWFyU2VydmVyVG9rZW5zRnJvbUxvY2FsU3RvcmFnZShuYW1lLCBzZXJ2ZXJCZWZvcmVSZW1vdmFsKVxuICAgICAgY2xlYXJNY3BDbGllbnRDb25maWcobmFtZSwgc2VydmVyQmVmb3JlUmVtb3ZhbClcbiAgICB9XG4gIH1cblxuICB0cnkge1xuICAgIGlmIChvcHRpb25zLnNjb3BlKSB7XG4gICAgICBjb25zdCBzY29wZSA9IGVuc3VyZUNvbmZpZ1Njb3BlKG9wdGlvbnMuc2NvcGUpXG4gICAgICBsb2dFdmVudCgndGVuZ3VfbWNwX2RlbGV0ZScsIHtcbiAgICAgICAgbmFtZTogbmFtZSBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICBzY29wZTpcbiAgICAgICAgICBzY29wZSBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgfSlcblxuICAgICAgYXdhaXQgcmVtb3ZlTWNwQ29uZmlnKG5hbWUsIHNjb3BlKVxuICAgICAgY2xlYW51cFNlY3VyZVN0b3JhZ2UoKVxuICAgICAgcHJvY2Vzcy5zdGRvdXQud3JpdGUoYFJlbW92ZWQgTUNQIHNlcnZlciAke25hbWV9IGZyb20gJHtzY29wZX0gY29uZmlnXFxuYClcbiAgICAgIGNsaU9rKGBGaWxlIG1vZGlmaWVkOiAke2Rlc2NyaWJlTWNwQ29uZmlnRmlsZVBhdGgoc2NvcGUpfWApXG4gICAgfVxuXG4gICAgLy8gSWYgbm8gc2NvcGUgc3BlY2lmaWVkLCBjaGVjayB3aGVyZSB0aGUgc2VydmVyIGV4aXN0c1xuICAgIGNvbnN0IHByb2plY3RDb25maWcgPSBnZXRDdXJyZW50UHJvamVjdENvbmZpZygpXG4gICAgY29uc3QgZ2xvYmFsQ29uZmlnID0gZ2V0R2xvYmFsQ29uZmlnKClcblxuICAgIC8vIENoZWNrIGlmIHNlcnZlciBleGlzdHMgaW4gcHJvamVjdCBzY29wZSAoLm1jcC5qc29uKVxuICAgIGNvbnN0IHsgc2VydmVyczogcHJvamVjdFNlcnZlcnMgfSA9IGdldE1jcENvbmZpZ3NCeVNjb3BlKCdwcm9qZWN0JylcbiAgICBjb25zdCBtY3BKc29uRXhpc3RzID0gISFwcm9qZWN0U2VydmVyc1tuYW1lXVxuXG4gICAgLy8gQ291bnQgaG93IG1hbnkgc2NvcGVzIGNvbnRhaW4gdGhpcyBzZXJ2ZXJcbiAgICBjb25zdCBzY29wZXM6IEFycmF5PEV4Y2x1ZGU8Q29uZmlnU2NvcGUsICdkeW5hbWljJz4+ID0gW11cbiAgICBpZiAocHJvamVjdENvbmZpZy5tY3BTZXJ2ZXJzPy5bbmFtZV0pIHNjb3Blcy5wdXNoKCdsb2NhbCcpXG4gICAgaWYgKG1jcEpzb25FeGlzdHMpIHNjb3Blcy5wdXNoKCdwcm9qZWN0JylcbiAgICBpZiAoZ2xvYmFsQ29uZmlnLm1jcFNlcnZlcnM/LltuYW1lXSkgc2NvcGVzLnB1c2goJ3VzZXInKVxuXG4gICAgaWYgKHNjb3Blcy5sZW5ndGggPT09IDApIHtcbiAgICAgIGNsaUVycm9yKGBObyBNQ1Agc2VydmVyIGZvdW5kIHdpdGggbmFtZTogXCIke25hbWV9XCJgKVxuICAgIH0gZWxzZSBpZiAoc2NvcGVzLmxlbmd0aCA9PT0gMSkge1xuICAgICAgLy8gU2VydmVyIGV4aXN0cyBpbiBvbmx5IG9uZSBzY29wZSwgcmVtb3ZlIGl0XG4gICAgICBjb25zdCBzY29wZSA9IHNjb3Blc1swXSFcbiAgICAgIGxvZ0V2ZW50KCd0ZW5ndV9tY3BfZGVsZXRlJywge1xuICAgICAgICBuYW1lOiBuYW1lIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgIHNjb3BlOlxuICAgICAgICAgIHNjb3BlIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICB9KVxuXG4gICAgICBhd2FpdCByZW1vdmVNY3BDb25maWcobmFtZSwgc2NvcGUpXG4gICAgICBjbGVhbnVwU2VjdXJlU3RvcmFnZSgpXG4gICAgICBwcm9jZXNzLnN0ZG91dC53cml0ZShcbiAgICAgICAgYFJlbW92ZWQgTUNQIHNlcnZlciBcIiR7bmFtZX1cIiBmcm9tICR7c2NvcGV9IGNvbmZpZ1xcbmAsXG4gICAgICApXG4gICAgICBjbGlPayhgRmlsZSBtb2RpZmllZDogJHtkZXNjcmliZU1jcENvbmZpZ0ZpbGVQYXRoKHNjb3BlKX1gKVxuICAgIH0gZWxzZSB7XG4gICAgICAvLyBTZXJ2ZXIgZXhpc3RzIGluIG11bHRpcGxlIHNjb3Blc1xuICAgICAgcHJvY2Vzcy5zdGRlcnIud3JpdGUoYE1DUCBzZXJ2ZXIgXCIke25hbWV9XCIgZXhpc3RzIGluIG11bHRpcGxlIHNjb3BlczpcXG5gKVxuICAgICAgc2NvcGVzLmZvckVhY2goc2NvcGUgPT4ge1xuICAgICAgICBwcm9jZXNzLnN0ZGVyci53cml0ZShcbiAgICAgICAgICBgICAtICR7Z2V0U2NvcGVMYWJlbChzY29wZSl9ICgke2Rlc2NyaWJlTWNwQ29uZmlnRmlsZVBhdGgoc2NvcGUpfSlcXG5gLFxuICAgICAgICApXG4gICAgICB9KVxuICAgICAgcHJvY2Vzcy5zdGRlcnIud3JpdGUoJ1xcblRvIHJlbW92ZSBmcm9tIGEgc3BlY2lmaWMgc2NvcGUsIHVzZTpcXG4nKVxuICAgICAgc2NvcGVzLmZvckVhY2goc2NvcGUgPT4ge1xuICAgICAgICBwcm9jZXNzLnN0ZGVyci53cml0ZShgICBjbGF1ZGUgbWNwIHJlbW92ZSBcIiR7bmFtZX1cIiAtcyAke3Njb3BlfVxcbmApXG4gICAgICB9KVxuICAgICAgY2xpRXJyb3IoKVxuICAgIH1cbiAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICBjbGlFcnJvcigoZXJyb3IgYXMgRXJyb3IpLm1lc3NhZ2UpXG4gIH1cbn1cblxuLy8gbWNwIGxpc3QgKGxpbmVzIDQ2NDHigJM0Njg4KVxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIG1jcExpc3RIYW5kbGVyKCk6IFByb21pc2U8dm9pZD4ge1xuICBsb2dFdmVudCgndGVuZ3VfbWNwX2xpc3QnLCB7fSlcbiAgY29uc3QgeyBzZXJ2ZXJzOiBjb25maWdzIH0gPSBhd2FpdCBnZXRBbGxNY3BDb25maWdzKClcbiAgaWYgKE9iamVjdC5rZXlzKGNvbmZpZ3MpLmxlbmd0aCA9PT0gMCkge1xuICAgIC8vIGJpb21lLWlnbm9yZSBsaW50L3N1c3BpY2lvdXMvbm9Db25zb2xlOjogaW50ZW50aW9uYWwgY29uc29sZSBvdXRwdXRcbiAgICBjb25zb2xlLmxvZyhcbiAgICAgICdObyBNQ1Agc2VydmVycyBjb25maWd1cmVkLiBVc2UgYGNsYXVkZSBtY3AgYWRkYCB0byBhZGQgYSBzZXJ2ZXIuJyxcbiAgICApXG4gIH0gZWxzZSB7XG4gICAgLy8gYmlvbWUtaWdub3JlIGxpbnQvc3VzcGljaW91cy9ub0NvbnNvbGU6OiBpbnRlbnRpb25hbCBjb25zb2xlIG91dHB1dFxuICAgIGNvbnNvbGUubG9nKCdDaGVja2luZyBNQ1Agc2VydmVyIGhlYWx0aC4uLlxcbicpXG5cbiAgICAvLyBDaGVjayBzZXJ2ZXJzIGNvbmN1cnJlbnRseVxuICAgIGNvbnN0IGVudHJpZXMgPSBPYmplY3QuZW50cmllcyhjb25maWdzKVxuICAgIGNvbnN0IHJlc3VsdHMgPSBhd2FpdCBwTWFwKFxuICAgICAgZW50cmllcyxcbiAgICAgIGFzeW5jIChbbmFtZSwgc2VydmVyXSkgPT4gKHtcbiAgICAgICAgbmFtZSxcbiAgICAgICAgc2VydmVyLFxuICAgICAgICBzdGF0dXM6IGF3YWl0IGNoZWNrTWNwU2VydmVySGVhbHRoKG5hbWUsIHNlcnZlciksXG4gICAgICB9KSxcbiAgICAgIHsgY29uY3VycmVuY3k6IGdldE1jcFNlcnZlckNvbm5lY3Rpb25CYXRjaFNpemUoKSB9LFxuICAgIClcblxuICAgIGZvciAoY29uc3QgeyBuYW1lLCBzZXJ2ZXIsIHN0YXR1cyB9IG9mIHJlc3VsdHMpIHtcbiAgICAgIC8vIEludGVudGlvbmFsbHkgZXhjbHVkaW5nIHNzZS1pZGUgc2VydmVycyBoZXJlIHNpbmNlIHRoZXkncmUgaW50ZXJuYWxcbiAgICAgIGlmIChzZXJ2ZXIudHlwZSA9PT0gJ3NzZScpIHtcbiAgICAgICAgLy8gYmlvbWUtaWdub3JlIGxpbnQvc3VzcGljaW91cy9ub0NvbnNvbGU6OiBpbnRlbnRpb25hbCBjb25zb2xlIG91dHB1dFxuICAgICAgICBjb25zb2xlLmxvZyhgJHtuYW1lfTogJHtzZXJ2ZXIudXJsfSAoU1NFKSAtICR7c3RhdHVzfWApXG4gICAgICB9IGVsc2UgaWYgKHNlcnZlci50eXBlID09PSAnaHR0cCcpIHtcbiAgICAgICAgLy8gYmlvbWUtaWdub3JlIGxpbnQvc3VzcGljaW91cy9ub0NvbnNvbGU6OiBpbnRlbnRpb25hbCBjb25zb2xlIG91dHB1dFxuICAgICAgICBjb25zb2xlLmxvZyhgJHtuYW1lfTogJHtzZXJ2ZXIudXJsfSAoSFRUUCkgLSAke3N0YXR1c31gKVxuICAgICAgfSBlbHNlIGlmIChzZXJ2ZXIudHlwZSA9PT0gJ2NsYXVkZWFpLXByb3h5Jykge1xuICAgICAgICAvLyBiaW9tZS1pZ25vcmUgbGludC9zdXNwaWNpb3VzL25vQ29uc29sZTo6IGludGVudGlvbmFsIGNvbnNvbGUgb3V0cHV0XG4gICAgICAgIGNvbnNvbGUubG9nKGAke25hbWV9OiAke3NlcnZlci51cmx9IC0gJHtzdGF0dXN9YClcbiAgICAgIH0gZWxzZSBpZiAoIXNlcnZlci50eXBlIHx8IHNlcnZlci50eXBlID09PSAnc3RkaW8nKSB7XG4gICAgICAgIGNvbnN0IGFyZ3MgPSBBcnJheS5pc0FycmF5KHNlcnZlci5hcmdzKSA/IHNlcnZlci5hcmdzIDogW11cbiAgICAgICAgLy8gYmlvbWUtaWdub3JlIGxpbnQvc3VzcGljaW91cy9ub0NvbnNvbGU6OiBpbnRlbnRpb25hbCBjb25zb2xlIG91dHB1dFxuICAgICAgICBjb25zb2xlLmxvZyhgJHtuYW1lfTogJHtzZXJ2ZXIuY29tbWFuZH0gJHthcmdzLmpvaW4oJyAnKX0gLSAke3N0YXR1c31gKVxuICAgICAgfVxuICAgIH1cbiAgfVxuICAvLyBVc2UgZ3JhY2VmdWxTaHV0ZG93biB0byBwcm9wZXJseSBjbGVhbiB1cCBNQ1Agc2VydmVyIGNvbm5lY3Rpb25zXG4gIC8vIChwcm9jZXNzLmV4aXQgYnlwYXNzZXMgY2xlYW51cCBoYW5kbGVycywgbGVhdmluZyBjaGlsZCBwcm9jZXNzZXMgb3JwaGFuZWQpXG4gIGF3YWl0IGdyYWNlZnVsU2h1dGRvd24oMClcbn1cblxuLy8gbWNwIGdldCAobGluZXMgNDY5NOKAkzQ3ODYpXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gbWNwR2V0SGFuZGxlcihuYW1lOiBzdHJpbmcpOiBQcm9taXNlPHZvaWQ+IHtcbiAgbG9nRXZlbnQoJ3Rlbmd1X21jcF9nZXQnLCB7XG4gICAgbmFtZTogbmFtZSBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICB9KVxuICBjb25zdCBzZXJ2ZXIgPSBnZXRNY3BDb25maWdCeU5hbWUobmFtZSlcbiAgaWYgKCFzZXJ2ZXIpIHtcbiAgICBjbGlFcnJvcihgTm8gTUNQIHNlcnZlciBmb3VuZCB3aXRoIG5hbWU6ICR7bmFtZX1gKVxuICB9XG5cbiAgLy8gYmlvbWUtaWdub3JlIGxpbnQvc3VzcGljaW91cy9ub0NvbnNvbGU6OiBpbnRlbnRpb25hbCBjb25zb2xlIG91dHB1dFxuICBjb25zb2xlLmxvZyhgJHtuYW1lfTpgKVxuICAvLyBiaW9tZS1pZ25vcmUgbGludC9zdXNwaWNpb3VzL25vQ29uc29sZTo6IGludGVudGlvbmFsIGNvbnNvbGUgb3V0cHV0XG4gIGNvbnNvbGUubG9nKGAgIFNjb3BlOiAke2dldFNjb3BlTGFiZWwoc2VydmVyLnNjb3BlKX1gKVxuXG4gIC8vIENoZWNrIHNlcnZlciBoZWFsdGhcbiAgY29uc3Qgc3RhdHVzID0gYXdhaXQgY2hlY2tNY3BTZXJ2ZXJIZWFsdGgobmFtZSwgc2VydmVyKVxuICAvLyBiaW9tZS1pZ25vcmUgbGludC9zdXNwaWNpb3VzL25vQ29uc29sZTo6IGludGVudGlvbmFsIGNvbnNvbGUgb3V0cHV0XG4gIGNvbnNvbGUubG9nKGAgIFN0YXR1czogJHtzdGF0dXN9YClcblxuICAvLyBJbnRlbnRpb25hbGx5IGV4Y2x1ZGluZyBzc2UtaWRlIHNlcnZlcnMgaGVyZSBzaW5jZSB0aGV5J3JlIGludGVybmFsXG4gIGlmIChzZXJ2ZXIudHlwZSA9PT0gJ3NzZScpIHtcbiAgICAvLyBiaW9tZS1pZ25vcmUgbGludC9zdXNwaWNpb3VzL25vQ29uc29sZTo6IGludGVudGlvbmFsIGNvbnNvbGUgb3V0cHV0XG4gICAgY29uc29sZS5sb2coYCAgVHlwZTogc3NlYClcbiAgICAvLyBiaW9tZS1pZ25vcmUgbGludC9zdXNwaWNpb3VzL25vQ29uc29sZTo6IGludGVudGlvbmFsIGNvbnNvbGUgb3V0cHV0XG4gICAgY29uc29sZS5sb2coYCAgVVJMOiAke3NlcnZlci51cmx9YClcbiAgICBpZiAoc2VydmVyLmhlYWRlcnMpIHtcbiAgICAgIC8vIGJpb21lLWlnbm9yZSBsaW50L3N1c3BpY2lvdXMvbm9Db25zb2xlOjogaW50ZW50aW9uYWwgY29uc29sZSBvdXRwdXRcbiAgICAgIGNvbnNvbGUubG9nKCcgIEhlYWRlcnM6JylcbiAgICAgIGZvciAoY29uc3QgW2tleSwgdmFsdWVdIG9mIE9iamVjdC5lbnRyaWVzKHNlcnZlci5oZWFkZXJzKSkge1xuICAgICAgICAvLyBiaW9tZS1pZ25vcmUgbGludC9zdXNwaWNpb3VzL25vQ29uc29sZTo6IGludGVudGlvbmFsIGNvbnNvbGUgb3V0cHV0XG4gICAgICAgIGNvbnNvbGUubG9nKGAgICAgJHtrZXl9OiAke3ZhbHVlfWApXG4gICAgICB9XG4gICAgfVxuICAgIGlmIChzZXJ2ZXIub2F1dGg/LmNsaWVudElkIHx8IHNlcnZlci5vYXV0aD8uY2FsbGJhY2tQb3J0KSB7XG4gICAgICBjb25zdCBwYXJ0czogc3RyaW5nW10gPSBbXVxuICAgICAgaWYgKHNlcnZlci5vYXV0aC5jbGllbnRJZCkge1xuICAgICAgICBwYXJ0cy5wdXNoKCdjbGllbnRfaWQgY29uZmlndXJlZCcpXG4gICAgICAgIGNvbnN0IGNsaWVudENvbmZpZyA9IGdldE1jcENsaWVudENvbmZpZyhuYW1lLCBzZXJ2ZXIpXG4gICAgICAgIGlmIChjbGllbnRDb25maWc/LmNsaWVudFNlY3JldCkgcGFydHMucHVzaCgnY2xpZW50X3NlY3JldCBjb25maWd1cmVkJylcbiAgICAgIH1cbiAgICAgIGlmIChzZXJ2ZXIub2F1dGguY2FsbGJhY2tQb3J0KVxuICAgICAgICBwYXJ0cy5wdXNoKGBjYWxsYmFja19wb3J0ICR7c2VydmVyLm9hdXRoLmNhbGxiYWNrUG9ydH1gKVxuICAgICAgLy8gYmlvbWUtaWdub3JlIGxpbnQvc3VzcGljaW91cy9ub0NvbnNvbGU6OiBpbnRlbnRpb25hbCBjb25zb2xlIG91dHB1dFxuICAgICAgY29uc29sZS5sb2coYCAgT0F1dGg6ICR7cGFydHMuam9pbignLCAnKX1gKVxuICAgIH1cbiAgfSBlbHNlIGlmIChzZXJ2ZXIudHlwZSA9PT0gJ2h0dHAnKSB7XG4gICAgLy8gYmlvbWUtaWdub3JlIGxpbnQvc3VzcGljaW91cy9ub0NvbnNvbGU6OiBpbnRlbnRpb25hbCBjb25zb2xlIG91dHB1dFxuICAgIGNvbnNvbGUubG9nKGAgIFR5cGU6IGh0dHBgKVxuICAgIC8vIGJpb21lLWlnbm9yZSBsaW50L3N1c3BpY2lvdXMvbm9Db25zb2xlOjogaW50ZW50aW9uYWwgY29uc29sZSBvdXRwdXRcbiAgICBjb25zb2xlLmxvZyhgICBVUkw6ICR7c2VydmVyLnVybH1gKVxuICAgIGlmIChzZXJ2ZXIuaGVhZGVycykge1xuICAgICAgLy8gYmlvbWUtaWdub3JlIGxpbnQvc3VzcGljaW91cy9ub0NvbnNvbGU6OiBpbnRlbnRpb25hbCBjb25zb2xlIG91dHB1dFxuICAgICAgY29uc29sZS5sb2coJyAgSGVhZGVyczonKVxuICAgICAgZm9yIChjb25zdCBba2V5LCB2YWx1ZV0gb2YgT2JqZWN0LmVudHJpZXMoc2VydmVyLmhlYWRlcnMpKSB7XG4gICAgICAgIC8vIGJpb21lLWlnbm9yZSBsaW50L3N1c3BpY2lvdXMvbm9Db25zb2xlOjogaW50ZW50aW9uYWwgY29uc29sZSBvdXRwdXRcbiAgICAgICAgY29uc29sZS5sb2coYCAgICAke2tleX06ICR7dmFsdWV9YClcbiAgICAgIH1cbiAgICB9XG4gICAgaWYgKHNlcnZlci5vYXV0aD8uY2xpZW50SWQgfHwgc2VydmVyLm9hdXRoPy5jYWxsYmFja1BvcnQpIHtcbiAgICAgIGNvbnN0IHBhcnRzOiBzdHJpbmdbXSA9IFtdXG4gICAgICBpZiAoc2VydmVyLm9hdXRoLmNsaWVudElkKSB7XG4gICAgICAgIHBhcnRzLnB1c2goJ2NsaWVudF9pZCBjb25maWd1cmVkJylcbiAgICAgICAgY29uc3QgY2xpZW50Q29uZmlnID0gZ2V0TWNwQ2xpZW50Q29uZmlnKG5hbWUsIHNlcnZlcilcbiAgICAgICAgaWYgKGNsaWVudENvbmZpZz8uY2xpZW50U2VjcmV0KSBwYXJ0cy5wdXNoKCdjbGllbnRfc2VjcmV0IGNvbmZpZ3VyZWQnKVxuICAgICAgfVxuICAgICAgaWYgKHNlcnZlci5vYXV0aC5jYWxsYmFja1BvcnQpXG4gICAgICAgIHBhcnRzLnB1c2goYGNhbGxiYWNrX3BvcnQgJHtzZXJ2ZXIub2F1dGguY2FsbGJhY2tQb3J0fWApXG4gICAgICAvLyBiaW9tZS1pZ25vcmUgbGludC9zdXNwaWNpb3VzL25vQ29uc29sZTo6IGludGVudGlvbmFsIGNvbnNvbGUgb3V0cHV0XG4gICAgICBjb25zb2xlLmxvZyhgICBPQXV0aDogJHtwYXJ0cy5qb2luKCcsICcpfWApXG4gICAgfVxuICB9IGVsc2UgaWYgKHNlcnZlci50eXBlID09PSAnc3RkaW8nKSB7XG4gICAgLy8gYmlvbWUtaWdub3JlIGxpbnQvc3VzcGljaW91cy9ub0NvbnNvbGU6OiBpbnRlbnRpb25hbCBjb25zb2xlIG91dHB1dFxuICAgIGNvbnNvbGUubG9nKGAgIFR5cGU6IHN0ZGlvYClcbiAgICAvLyBiaW9tZS1pZ25vcmUgbGludC9zdXNwaWNpb3VzL25vQ29uc29sZTo6IGludGVudGlvbmFsIGNvbnNvbGUgb3V0cHV0XG4gICAgY29uc29sZS5sb2coYCAgQ29tbWFuZDogJHtzZXJ2ZXIuY29tbWFuZH1gKVxuICAgIGNvbnN0IGFyZ3MgPSBBcnJheS5pc0FycmF5KHNlcnZlci5hcmdzKSA/IHNlcnZlci5hcmdzIDogW11cbiAgICAvLyBiaW9tZS1pZ25vcmUgbGludC9zdXNwaWNpb3VzL25vQ29uc29sZTo6IGludGVudGlvbmFsIGNvbnNvbGUgb3V0cHV0XG4gICAgY29uc29sZS5sb2coYCAgQXJnczogJHthcmdzLmpvaW4oJyAnKX1gKVxuICAgIGlmIChzZXJ2ZXIuZW52KSB7XG4gICAgICAvLyBiaW9tZS1pZ25vcmUgbGludC9zdXNwaWNpb3VzL25vQ29uc29sZTo6IGludGVudGlvbmFsIGNvbnNvbGUgb3V0cHV0XG4gICAgICBjb25zb2xlLmxvZygnICBFbnZpcm9ubWVudDonKVxuICAgICAgZm9yIChjb25zdCBba2V5LCB2YWx1ZV0gb2YgT2JqZWN0LmVudHJpZXMoc2VydmVyLmVudikpIHtcbiAgICAgICAgLy8gYmlvbWUtaWdub3JlIGxpbnQvc3VzcGljaW91cy9ub0NvbnNvbGU6OiBpbnRlbnRpb25hbCBjb25zb2xlIG91dHB1dFxuICAgICAgICBjb25zb2xlLmxvZyhgICAgICR7a2V5fT0ke3ZhbHVlfWApXG4gICAgICB9XG4gICAgfVxuICB9XG4gIC8vIGJpb21lLWlnbm9yZSBsaW50L3N1c3BpY2lvdXMvbm9Db25zb2xlOjogaW50ZW50aW9uYWwgY29uc29sZSBvdXRwdXRcbiAgY29uc29sZS5sb2coXG4gICAgYFxcblRvIHJlbW92ZSB0aGlzIHNlcnZlciwgcnVuOiBjbGF1ZGUgbWNwIHJlbW92ZSBcIiR7bmFtZX1cIiAtcyAke3NlcnZlci5zY29wZX1gLFxuICApXG4gIC8vIFVzZSBncmFjZWZ1bFNodXRkb3duIHRvIHByb3Blcmx5IGNsZWFuIHVwIE1DUCBzZXJ2ZXIgY29ubmVjdGlvbnNcbiAgLy8gKHByb2Nlc3MuZXhpdCBieXBhc3NlcyBjbGVhbnVwIGhhbmRsZXJzLCBsZWF2aW5nIGNoaWxkIHByb2Nlc3NlcyBvcnBoYW5lZClcbiAgYXdhaXQgZ3JhY2VmdWxTaHV0ZG93bigwKVxufVxuXG4vLyBtY3AgYWRkLWpzb24gKGxpbmVzIDQ4MDHigJM0ODcwKVxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIG1jcEFkZEpzb25IYW5kbGVyKFxuICBuYW1lOiBzdHJpbmcsXG4gIGpzb246IHN0cmluZyxcbiAgb3B0aW9uczogeyBzY29wZT86IHN0cmluZzsgY2xpZW50U2VjcmV0PzogdHJ1ZSB9LFxuKTogUHJvbWlzZTx2b2lkPiB7XG4gIHRyeSB7XG4gICAgY29uc3Qgc2NvcGUgPSBlbnN1cmVDb25maWdTY29wZShvcHRpb25zLnNjb3BlKVxuICAgIGNvbnN0IHBhcnNlZEpzb24gPSBzYWZlUGFyc2VKU09OKGpzb24pXG5cbiAgICAvLyBSZWFkIHNlY3JldCBiZWZvcmUgd3JpdGluZyBjb25maWcgc28gY2FuY2VsbGF0aW9uIGRvZXNuJ3QgbGVhdmUgcGFydGlhbCBzdGF0ZVxuICAgIGNvbnN0IG5lZWRzU2VjcmV0ID1cbiAgICAgIG9wdGlvbnMuY2xpZW50U2VjcmV0ICYmXG4gICAgICBwYXJzZWRKc29uICYmXG4gICAgICB0eXBlb2YgcGFyc2VkSnNvbiA9PT0gJ29iamVjdCcgJiZcbiAgICAgICd0eXBlJyBpbiBwYXJzZWRKc29uICYmXG4gICAgICAocGFyc2VkSnNvbi50eXBlID09PSAnc3NlJyB8fCBwYXJzZWRKc29uLnR5cGUgPT09ICdodHRwJykgJiZcbiAgICAgICd1cmwnIGluIHBhcnNlZEpzb24gJiZcbiAgICAgIHR5cGVvZiBwYXJzZWRKc29uLnVybCA9PT0gJ3N0cmluZycgJiZcbiAgICAgICdvYXV0aCcgaW4gcGFyc2VkSnNvbiAmJlxuICAgICAgcGFyc2VkSnNvbi5vYXV0aCAmJlxuICAgICAgdHlwZW9mIHBhcnNlZEpzb24ub2F1dGggPT09ICdvYmplY3QnICYmXG4gICAgICAnY2xpZW50SWQnIGluIHBhcnNlZEpzb24ub2F1dGhcbiAgICBjb25zdCBjbGllbnRTZWNyZXQgPSBuZWVkc1NlY3JldCA/IGF3YWl0IHJlYWRDbGllbnRTZWNyZXQoKSA6IHVuZGVmaW5lZFxuXG4gICAgYXdhaXQgYWRkTWNwQ29uZmlnKG5hbWUsIHBhcnNlZEpzb24sIHNjb3BlKVxuXG4gICAgY29uc3QgdHJhbnNwb3J0VHlwZSA9XG4gICAgICBwYXJzZWRKc29uICYmIHR5cGVvZiBwYXJzZWRKc29uID09PSAnb2JqZWN0JyAmJiAndHlwZScgaW4gcGFyc2VkSnNvblxuICAgICAgICA/IFN0cmluZyhwYXJzZWRKc29uLnR5cGUgfHwgJ3N0ZGlvJylcbiAgICAgICAgOiAnc3RkaW8nXG5cbiAgICBpZiAoXG4gICAgICBjbGllbnRTZWNyZXQgJiZcbiAgICAgIHBhcnNlZEpzb24gJiZcbiAgICAgIHR5cGVvZiBwYXJzZWRKc29uID09PSAnb2JqZWN0JyAmJlxuICAgICAgJ3R5cGUnIGluIHBhcnNlZEpzb24gJiZcbiAgICAgIChwYXJzZWRKc29uLnR5cGUgPT09ICdzc2UnIHx8IHBhcnNlZEpzb24udHlwZSA9PT0gJ2h0dHAnKSAmJlxuICAgICAgJ3VybCcgaW4gcGFyc2VkSnNvbiAmJlxuICAgICAgdHlwZW9mIHBhcnNlZEpzb24udXJsID09PSAnc3RyaW5nJ1xuICAgICkge1xuICAgICAgc2F2ZU1jcENsaWVudFNlY3JldChcbiAgICAgICAgbmFtZSxcbiAgICAgICAgeyB0eXBlOiBwYXJzZWRKc29uLnR5cGUsIHVybDogcGFyc2VkSnNvbi51cmwgfSxcbiAgICAgICAgY2xpZW50U2VjcmV0LFxuICAgICAgKVxuICAgIH1cblxuICAgIGxvZ0V2ZW50KCd0ZW5ndV9tY3BfYWRkJywge1xuICAgICAgc2NvcGU6XG4gICAgICAgIHNjb3BlIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICBzb3VyY2U6XG4gICAgICAgICdqc29uJyBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgdHlwZTogdHJhbnNwb3J0VHlwZSBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgIH0pXG5cbiAgICBjbGlPayhgQWRkZWQgJHt0cmFuc3BvcnRUeXBlfSBNQ1Agc2VydmVyICR7bmFtZX0gdG8gJHtzY29wZX0gY29uZmlnYClcbiAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICBjbGlFcnJvcigoZXJyb3IgYXMgRXJyb3IpLm1lc3NhZ2UpXG4gIH1cbn1cblxuLy8gbWNwIGFkZC1mcm9tLWNsYXVkZS1kZXNrdG9wIChsaW5lcyA0ODgx4oCTNDkyNylcbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBtY3BBZGRGcm9tRGVza3RvcEhhbmRsZXIob3B0aW9uczoge1xuICBzY29wZT86IHN0cmluZ1xufSk6IFByb21pc2U8dm9pZD4ge1xuICB0cnkge1xuICAgIGNvbnN0IHNjb3BlID0gZW5zdXJlQ29uZmlnU2NvcGUob3B0aW9ucy5zY29wZSlcbiAgICBjb25zdCBwbGF0Zm9ybSA9IGdldFBsYXRmb3JtKClcblxuICAgIGxvZ0V2ZW50KCd0ZW5ndV9tY3BfYWRkJywge1xuICAgICAgc2NvcGU6XG4gICAgICAgIHNjb3BlIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICBwbGF0Zm9ybTpcbiAgICAgICAgcGxhdGZvcm0gYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICAgIHNvdXJjZTpcbiAgICAgICAgJ2Rlc2t0b3AnIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgfSlcblxuICAgIGNvbnN0IHsgcmVhZENsYXVkZURlc2t0b3BNY3BTZXJ2ZXJzIH0gPSBhd2FpdCBpbXBvcnQoXG4gICAgICAnLi4vLi4vdXRpbHMvY2xhdWRlRGVza3RvcC5qcydcbiAgICApXG4gICAgY29uc3Qgc2VydmVycyA9IGF3YWl0IHJlYWRDbGF1ZGVEZXNrdG9wTWNwU2VydmVycygpXG5cbiAgICBpZiAoT2JqZWN0LmtleXMoc2VydmVycykubGVuZ3RoID09PSAwKSB7XG4gICAgICBjbGlPayhcbiAgICAgICAgJ05vIE1DUCBzZXJ2ZXJzIGZvdW5kIGluIENsYXVkZSBEZXNrdG9wIGNvbmZpZ3VyYXRpb24gb3IgY29uZmlndXJhdGlvbiBmaWxlIGRvZXMgbm90IGV4aXN0LicsXG4gICAgICApXG4gICAgfVxuXG4gICAgY29uc3QgeyB1bm1vdW50IH0gPSBhd2FpdCByZW5kZXIoXG4gICAgICA8QXBwU3RhdGVQcm92aWRlcj5cbiAgICAgICAgPEtleWJpbmRpbmdTZXR1cD5cbiAgICAgICAgICA8TUNQU2VydmVyRGVza3RvcEltcG9ydERpYWxvZ1xuICAgICAgICAgICAgc2VydmVycz17c2VydmVyc31cbiAgICAgICAgICAgIHNjb3BlPXtzY29wZX1cbiAgICAgICAgICAgIG9uRG9uZT17KCkgPT4ge1xuICAgICAgICAgICAgICB1bm1vdW50KClcbiAgICAgICAgICAgIH19XG4gICAgICAgICAgLz5cbiAgICAgICAgPC9LZXliaW5kaW5nU2V0dXA+XG4gICAgICA8L0FwcFN0YXRlUHJvdmlkZXI+LFxuICAgICAgeyBleGl0T25DdHJsQzogdHJ1ZSB9LFxuICAgIClcbiAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICBjbGlFcnJvcigoZXJyb3IgYXMgRXJyb3IpLm1lc3NhZ2UpXG4gIH1cbn1cblxuLy8gbWNwIHJlc2V0LXByb2plY3QtY2hvaWNlcyAobGluZXMgNDkzNeKAkzQ5NTIpXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gbWNwUmVzZXRDaG9pY2VzSGFuZGxlcigpOiBQcm9taXNlPHZvaWQ+IHtcbiAgbG9nRXZlbnQoJ3Rlbmd1X21jcF9yZXNldF9tY3Bqc29uX2Nob2ljZXMnLCB7fSlcbiAgc2F2ZUN1cnJlbnRQcm9qZWN0Q29uZmlnKGN1cnJlbnQgPT4gKHtcbiAgICAuLi5jdXJyZW50LFxuICAgIGVuYWJsZWRNY3Bqc29uU2VydmVyczogW10sXG4gICAgZGlzYWJsZWRNY3Bqc29uU2VydmVyczogW10sXG4gICAgZW5hYmxlQWxsUHJvamVjdE1jcFNlcnZlcnM6IGZhbHNlLFxuICB9KSlcbiAgY2xpT2soXG4gICAgJ0FsbCBwcm9qZWN0LXNjb3BlZCAoLm1jcC5qc29uKSBzZXJ2ZXIgYXBwcm92YWxzIGFuZCByZWplY3Rpb25zIGhhdmUgYmVlbiByZXNldC5cXG4nICtcbiAgICAgICdZb3Ugd2lsbCBiZSBwcm9tcHRlZCBmb3IgYXBwcm92YWwgbmV4dCB0aW1lIHlvdSBzdGFydCBDbGF1ZGUgQ29kZS4nLFxuICApXG59XG4iXSwibWFwcGluZ3MiOiJBQUFBO0FBQ0E7QUFDQTtBQUNBOztBQUVBLFNBQVNBLElBQUksUUFBUSxhQUFhO0FBQ2xDLE9BQU9DLElBQUksTUFBTSxPQUFPO0FBQ3hCLFNBQVNDLEdBQUcsUUFBUSxTQUFTO0FBQzdCLE9BQU9DLEtBQUssTUFBTSxPQUFPO0FBQ3pCLFNBQVNDLDRCQUE0QixRQUFRLGtEQUFrRDtBQUMvRixTQUFTQyxNQUFNLFFBQVEsY0FBYztBQUNyQyxTQUFTQyxlQUFlLFFBQVEsOENBQThDO0FBQzlFLFNBQ0UsS0FBS0MsMERBQTBELEVBQy9EQyxRQUFRLFFBQ0gsbUNBQW1DO0FBQzFDLFNBQ0VDLG9CQUFvQixFQUNwQkMsaUNBQWlDLEVBQ2pDQyxrQkFBa0IsRUFDbEJDLGdCQUFnQixFQUNoQkMsbUJBQW1CLFFBQ2QsNEJBQTRCO0FBQ25DLFNBQ0VDLGVBQWUsRUFDZkMsK0JBQStCLFFBQzFCLDhCQUE4QjtBQUNyQyxTQUNFQyxZQUFZLEVBQ1pDLGdCQUFnQixFQUNoQkMsa0JBQWtCLEVBQ2xCQyxvQkFBb0IsRUFDcEJDLGVBQWUsUUFDViw4QkFBOEI7QUFDckMsY0FDRUMsV0FBVyxFQUNYQyxxQkFBcUIsUUFDaEIsNkJBQTZCO0FBQ3BDLFNBQ0VDLHlCQUF5QixFQUN6QkMsaUJBQWlCLEVBQ2pCQyxhQUFhLFFBQ1IsNkJBQTZCO0FBQ3BDLFNBQVNDLGdCQUFnQixRQUFRLHlCQUF5QjtBQUMxRCxTQUNFQyx1QkFBdUIsRUFDdkJDLGVBQWUsRUFDZkMsd0JBQXdCLFFBQ25CLHVCQUF1QjtBQUM5QixTQUFTQyxnQkFBZ0IsUUFBUSx1QkFBdUI7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 \ No newline at end of file diff --git a/src/cli/handlers/plugins.ts b/src/cli/handlers/plugins.ts new file mode 100644 index 0000000..9236abe --- /dev/null +++ b/src/cli/handlers/plugins.ts @@ -0,0 +1,878 @@ +/** + * Plugin and marketplace subcommand handlers — extracted from main.tsx for lazy loading. + * These are dynamically imported only when `claude plugin *` or `claude plugin marketplace *` runs. + */ +/* eslint-disable custom-rules/no-process-exit -- CLI subcommand handlers intentionally exit */ +import figures from 'figures' +import { basename, dirname } from 'path' +import { setUseCoworkPlugins } from '../../bootstrap/state.js' +import { + type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + type AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED, + logEvent, +} from '../../services/analytics/index.js' +import { + disableAllPlugins, + disablePlugin, + enablePlugin, + installPlugin, + uninstallPlugin, + updatePluginCli, + VALID_INSTALLABLE_SCOPES, + VALID_UPDATE_SCOPES, +} from '../../services/plugins/pluginCliCommands.js' +import { getPluginErrorMessage } from '../../types/plugin.js' +import { errorMessage } from '../../utils/errors.js' +import { logError } from '../../utils/log.js' +import { clearAllCaches } from '../../utils/plugins/cacheUtils.js' +import { getInstallCounts } from '../../utils/plugins/installCounts.js' +import { + isPluginInstalled, + loadInstalledPluginsV2, +} from '../../utils/plugins/installedPluginsManager.js' +import { + createPluginId, + loadMarketplacesWithGracefulDegradation, +} from '../../utils/plugins/marketplaceHelpers.js' +import { + addMarketplaceSource, + loadKnownMarketplacesConfig, + refreshAllMarketplaces, + refreshMarketplace, + removeMarketplaceSource, + saveMarketplaceToSettings, +} from '../../utils/plugins/marketplaceManager.js' +import { loadPluginMcpServers } from '../../utils/plugins/mcpPluginIntegration.js' +import { parseMarketplaceInput } from '../../utils/plugins/parseMarketplaceInput.js' +import { + parsePluginIdentifier, + scopeToSettingSource, +} from '../../utils/plugins/pluginIdentifier.js' +import { loadAllPlugins } from '../../utils/plugins/pluginLoader.js' +import type { PluginSource } from '../../utils/plugins/schemas.js' +import { + type ValidationResult, + validateManifest, + validatePluginContents, +} from '../../utils/plugins/validatePlugin.js' +import { jsonStringify } from '../../utils/slowOperations.js' +import { plural } from '../../utils/stringUtils.js' +import { cliError, cliOk } from '../exit.js' + +// Re-export for main.tsx to reference in option definitions +export { VALID_INSTALLABLE_SCOPES, VALID_UPDATE_SCOPES } + +/** + * Helper function to handle marketplace command errors consistently. + */ +export function handleMarketplaceError(error: unknown, action: string): never { + logError(error) + cliError(`${figures.cross} Failed to ${action}: ${errorMessage(error)}`) +} + +function printValidationResult(result: ValidationResult): void { + if (result.errors.length > 0) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log( + `${figures.cross} Found ${result.errors.length} ${plural(result.errors.length, 'error')}:\n`, + ) + result.errors.forEach(error => { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` ${figures.pointer} ${error.path}: ${error.message}`) + }) + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log('') + } + if (result.warnings.length > 0) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log( + `${figures.warning} Found ${result.warnings.length} ${plural(result.warnings.length, 'warning')}:\n`, + ) + result.warnings.forEach(warning => { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` ${figures.pointer} ${warning.path}: ${warning.message}`) + }) + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log('') + } +} + +// plugin validate +export async function pluginValidateHandler( + manifestPath: string, + options: { cowork?: boolean }, +): Promise { + if (options.cowork) setUseCoworkPlugins(true) + try { + const result = await validateManifest(manifestPath) + + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(`Validating ${result.fileType} manifest: ${result.filePath}\n`) + printValidationResult(result) + + // If this is a plugin manifest located inside a .claude-plugin directory, + // also validate the plugin's content files (skills, agents, commands, + // hooks). Works whether the user passed a directory or the plugin.json + // path directly. + let contentResults: ValidationResult[] = [] + if (result.fileType === 'plugin') { + const manifestDir = dirname(result.filePath) + if (basename(manifestDir) === '.claude-plugin') { + contentResults = await validatePluginContents(dirname(manifestDir)) + for (const r of contentResults) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(`Validating ${r.fileType}: ${r.filePath}\n`) + printValidationResult(r) + } + } + } + + const allSuccess = result.success && contentResults.every(r => r.success) + const hasWarnings = + result.warnings.length > 0 || + contentResults.some(r => r.warnings.length > 0) + + if (allSuccess) { + cliOk( + hasWarnings + ? `${figures.tick} Validation passed with warnings` + : `${figures.tick} Validation passed`, + ) + } else { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(`${figures.cross} Validation failed`) + process.exit(1) + } + } catch (error) { + logError(error) + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.error( + `${figures.cross} Unexpected error during validation: ${errorMessage(error)}`, + ) + process.exit(2) + } +} + +// plugin list (lines 5217–5416) +export async function pluginListHandler(options: { + json?: boolean + available?: boolean + cowork?: boolean +}): Promise { + if (options.cowork) setUseCoworkPlugins(true) + logEvent('tengu_plugin_list_command', {}) + + const installedData = loadInstalledPluginsV2() + const { getPluginEditableScopes } = await import( + '../../utils/plugins/pluginStartupCheck.js' + ) + const enabledPlugins = getPluginEditableScopes() + + const pluginIds = Object.keys(installedData.plugins) + + // Load all plugins once. The JSON and human paths both need: + // - loadErrors (to show load failures per plugin) + // - inline plugins (session-only via --plugin-dir, source='name@inline') + // which are NOT in installedData.plugins (V2 bookkeeping) — they must + // be surfaced separately or `plugin list` silently ignores --plugin-dir. + const { + enabled: loadedEnabled, + disabled: loadedDisabled, + errors: loadErrors, + } = await loadAllPlugins() + const allLoadedPlugins = [...loadedEnabled, ...loadedDisabled] + const inlinePlugins = allLoadedPlugins.filter(p => + p.source.endsWith('@inline'), + ) + // Path-level inline failures (dir doesn't exist, parse error before + // manifest is read) use source='inline[N]'. Plugin-level errors after + // manifest read use source='name@inline'. Collect both for the session + // section — these are otherwise invisible since they have no pluginId. + const inlineLoadErrors = loadErrors.filter( + e => e.source.endsWith('@inline') || e.source.startsWith('inline['), + ) + + if (options.json) { + // Create a map of plugin source to loaded plugin for quick lookup + const loadedPluginMap = new Map(allLoadedPlugins.map(p => [p.source, p])) + + const plugins: Array<{ + id: string + version: string + scope: string + enabled: boolean + installPath: string + installedAt?: string + lastUpdated?: string + projectPath?: string + mcpServers?: Record + errors?: string[] + }> = [] + + for (const pluginId of pluginIds.sort()) { + const installations = installedData.plugins[pluginId] + if (!installations || installations.length === 0) continue + + // Find loading errors for this plugin + const pluginName = parsePluginIdentifier(pluginId).name + const pluginErrors = loadErrors + .filter( + e => + e.source === pluginId || ('plugin' in e && e.plugin === pluginName), + ) + .map(getPluginErrorMessage) + + for (const installation of installations) { + // Try to find the loaded plugin to get MCP servers + const loadedPlugin = loadedPluginMap.get(pluginId) + let mcpServers: Record | undefined + + if (loadedPlugin) { + // Load MCP servers if not already cached + const servers = + loadedPlugin.mcpServers || + (await loadPluginMcpServers(loadedPlugin)) + if (servers && Object.keys(servers).length > 0) { + mcpServers = servers + } + } + + plugins.push({ + id: pluginId, + version: installation.version || 'unknown', + scope: installation.scope, + enabled: enabledPlugins.has(pluginId), + installPath: installation.installPath, + installedAt: installation.installedAt, + lastUpdated: installation.lastUpdated, + projectPath: installation.projectPath, + mcpServers, + errors: pluginErrors.length > 0 ? pluginErrors : undefined, + }) + } + } + + // Session-only plugins: scope='session', no install metadata. + // Filter from inlineLoadErrors (not loadErrors) so an installed plugin + // with the same manifest name doesn't cross-contaminate via e.plugin. + // The e.plugin fallback catches the dirName≠manifestName case: + // createPluginFromPath tags errors with `${dirName}@inline` but + // plugin.source is reassigned to `${manifest.name}@inline` afterward + // (pluginLoader.ts loadInlinePlugins), so e.source !== p.source when + // a dev checkout dir like ~/code/my-fork/ has manifest name 'cool-plugin'. + for (const p of inlinePlugins) { + const servers = p.mcpServers || (await loadPluginMcpServers(p)) + const pErrors = inlineLoadErrors + .filter( + e => e.source === p.source || ('plugin' in e && e.plugin === p.name), + ) + .map(getPluginErrorMessage) + plugins.push({ + id: p.source, + version: p.manifest.version ?? 'unknown', + scope: 'session', + enabled: p.enabled !== false, + installPath: p.path, + mcpServers: + servers && Object.keys(servers).length > 0 ? servers : undefined, + errors: pErrors.length > 0 ? pErrors : undefined, + }) + } + // Path-level inline failures (--plugin-dir /nonexistent): no LoadedPlugin + // exists so the loop above can't surface them. Mirror the human-path + // handling so JSON consumers see the failure instead of silent omission. + for (const e of inlineLoadErrors.filter(e => + e.source.startsWith('inline['), + )) { + plugins.push({ + id: e.source, + version: 'unknown', + scope: 'session', + enabled: false, + installPath: 'path' in e ? e.path : '', + errors: [getPluginErrorMessage(e)], + }) + } + + // If --available is set, also load available plugins from marketplaces + if (options.available) { + const available: Array<{ + pluginId: string + name: string + description?: string + marketplaceName: string + version?: string + source: PluginSource + installCount?: number + }> = [] + + try { + const [config, installCounts] = await Promise.all([ + loadKnownMarketplacesConfig(), + getInstallCounts(), + ]) + const { marketplaces } = + await loadMarketplacesWithGracefulDegradation(config) + + for (const { + name: marketplaceName, + data: marketplace, + } of marketplaces) { + if (marketplace) { + for (const entry of marketplace.plugins) { + const pluginId = createPluginId(entry.name, marketplaceName) + // Only include plugins that are not already installed + if (!isPluginInstalled(pluginId)) { + available.push({ + pluginId, + name: entry.name, + description: entry.description, + marketplaceName, + version: entry.version, + source: entry.source, + installCount: installCounts?.get(pluginId), + }) + } + } + } + } + } catch { + // Silently ignore marketplace loading errors + } + + cliOk(jsonStringify({ installed: plugins, available }, null, 2)) + } else { + cliOk(jsonStringify(plugins, null, 2)) + } + } + + if (pluginIds.length === 0 && inlinePlugins.length === 0) { + // inlineLoadErrors can exist with zero inline plugins (e.g. --plugin-dir + // points at a nonexistent path). Don't early-exit over them — fall + // through to the session section so the failure is visible. + if (inlineLoadErrors.length === 0) { + cliOk( + 'No plugins installed. Use `claude plugin install` to install a plugin.', + ) + } + } + + if (pluginIds.length > 0) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log('Installed plugins:\n') + } + + for (const pluginId of pluginIds.sort()) { + const installations = installedData.plugins[pluginId] + if (!installations || installations.length === 0) continue + + // Find loading errors for this plugin + const pluginName = parsePluginIdentifier(pluginId).name + const pluginErrors = loadErrors.filter( + e => e.source === pluginId || ('plugin' in e && e.plugin === pluginName), + ) + + for (const installation of installations) { + const isEnabled = enabledPlugins.has(pluginId) + const status = + pluginErrors.length > 0 + ? `${figures.cross} failed to load` + : isEnabled + ? `${figures.tick} enabled` + : `${figures.cross} disabled` + const version = installation.version || 'unknown' + const scope = installation.scope + + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` ${figures.pointer} ${pluginId}`) + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Version: ${version}`) + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Scope: ${scope}`) + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Status: ${status}`) + for (const error of pluginErrors) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Error: ${getPluginErrorMessage(error)}`) + } + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log('') + } + } + + if (inlinePlugins.length > 0 || inlineLoadErrors.length > 0) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log('Session-only plugins (--plugin-dir):\n') + for (const p of inlinePlugins) { + // Same dirName≠manifestName fallback as the JSON path above — error + // sources use the dir basename but p.source uses the manifest name. + const pErrors = inlineLoadErrors.filter( + e => e.source === p.source || ('plugin' in e && e.plugin === p.name), + ) + const status = + pErrors.length > 0 + ? `${figures.cross} loaded with errors` + : `${figures.tick} loaded` + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` ${figures.pointer} ${p.source}`) + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Version: ${p.manifest.version ?? 'unknown'}`) + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Path: ${p.path}`) + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Status: ${status}`) + for (const e of pErrors) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Error: ${getPluginErrorMessage(e)}`) + } + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log('') + } + // Path-level failures: no LoadedPlugin object exists. Show them so + // `--plugin-dir /typo` doesn't just silently produce nothing. + for (const e of inlineLoadErrors.filter(e => + e.source.startsWith('inline['), + )) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log( + ` ${figures.pointer} ${e.source}: ${figures.cross} ${getPluginErrorMessage(e)}\n`, + ) + } + } + + cliOk() +} + +// marketplace add (lines 5433–5487) +export async function marketplaceAddHandler( + source: string, + options: { cowork?: boolean; sparse?: string[]; scope?: string }, +): Promise { + if (options.cowork) setUseCoworkPlugins(true) + try { + const parsed = await parseMarketplaceInput(source) + + if (!parsed) { + cliError( + `${figures.cross} Invalid marketplace source format. Try: owner/repo, https://..., or ./path`, + ) + } + + if ('error' in parsed) { + cliError(`${figures.cross} ${parsed.error}`) + } + + // Validate scope + const scope = options.scope ?? 'user' + if (scope !== 'user' && scope !== 'project' && scope !== 'local') { + cliError( + `${figures.cross} Invalid scope '${scope}'. Use: user, project, or local`, + ) + } + const settingSource = scopeToSettingSource(scope) + + let marketplaceSource = parsed + + if (options.sparse && options.sparse.length > 0) { + if ( + marketplaceSource.source === 'github' || + marketplaceSource.source === 'git' + ) { + marketplaceSource = { + ...marketplaceSource, + sparsePaths: options.sparse, + } + } else { + cliError( + `${figures.cross} --sparse is only supported for github and git marketplace sources (got: ${marketplaceSource.source})`, + ) + } + } + + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log('Adding marketplace...') + + const { name, alreadyMaterialized, resolvedSource } = + await addMarketplaceSource(marketplaceSource, message => { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(message) + }) + + // Write intent to settings at the requested scope + saveMarketplaceToSettings(name, { source: resolvedSource }, settingSource) + + clearAllCaches() + + let sourceType = marketplaceSource.source + if (marketplaceSource.source === 'github') { + sourceType = + marketplaceSource.repo as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + } + logEvent('tengu_marketplace_added', { + source_type: + sourceType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + + cliOk( + alreadyMaterialized + ? `${figures.tick} Marketplace '${name}' already on disk — declared in ${scope} settings` + : `${figures.tick} Successfully added marketplace: ${name} (declared in ${scope} settings)`, + ) + } catch (error) { + handleMarketplaceError(error, 'add marketplace') + } +} + +// marketplace list (lines 5497–5565) +export async function marketplaceListHandler(options: { + json?: boolean + cowork?: boolean +}): Promise { + if (options.cowork) setUseCoworkPlugins(true) + try { + const config = await loadKnownMarketplacesConfig() + const names = Object.keys(config) + + if (options.json) { + const marketplaces = names.sort().map(name => { + const marketplace = config[name] + const source = marketplace?.source + return { + name, + source: source?.source, + ...(source?.source === 'github' && { repo: source.repo }), + ...(source?.source === 'git' && { url: source.url }), + ...(source?.source === 'url' && { url: source.url }), + ...(source?.source === 'directory' && { path: source.path }), + ...(source?.source === 'file' && { path: source.path }), + installLocation: marketplace?.installLocation, + } + }) + cliOk(jsonStringify(marketplaces, null, 2)) + } + + if (names.length === 0) { + cliOk('No marketplaces configured') + } + + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log('Configured marketplaces:\n') + names.forEach(name => { + const marketplace = config[name] + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` ${figures.pointer} ${name}`) + + if (marketplace?.source) { + const src = marketplace.source + if (src.source === 'github') { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Source: GitHub (${src.repo})`) + } else if (src.source === 'git') { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Source: Git (${src.url})`) + } else if (src.source === 'url') { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Source: URL (${src.url})`) + } else if (src.source === 'directory') { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Source: Directory (${src.path})`) + } else if (src.source === 'file') { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(` Source: File (${src.path})`) + } + } + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log('') + }) + + cliOk() + } catch (error) { + handleMarketplaceError(error, 'list marketplaces') + } +} + +// marketplace remove (lines 5576–5598) +export async function marketplaceRemoveHandler( + name: string, + options: { cowork?: boolean }, +): Promise { + if (options.cowork) setUseCoworkPlugins(true) + try { + await removeMarketplaceSource(name) + clearAllCaches() + + logEvent('tengu_marketplace_removed', { + marketplace_name: + name as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + + cliOk(`${figures.tick} Successfully removed marketplace: ${name}`) + } catch (error) { + handleMarketplaceError(error, 'remove marketplace') + } +} + +// marketplace update (lines 5609–5672) +export async function marketplaceUpdateHandler( + name: string | undefined, + options: { cowork?: boolean }, +): Promise { + if (options.cowork) setUseCoworkPlugins(true) + try { + if (name) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(`Updating marketplace: ${name}...`) + + await refreshMarketplace(name, message => { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(message) + }) + + clearAllCaches() + + logEvent('tengu_marketplace_updated', { + marketplace_name: + name as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + + cliOk(`${figures.tick} Successfully updated marketplace: ${name}`) + } else { + const config = await loadKnownMarketplacesConfig() + const marketplaceNames = Object.keys(config) + + if (marketplaceNames.length === 0) { + cliOk('No marketplaces configured') + } + + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.log(`Updating ${marketplaceNames.length} marketplace(s)...`) + + await refreshAllMarketplaces() + clearAllCaches() + + logEvent('tengu_marketplace_updated_all', { + count: + marketplaceNames.length as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + + cliOk( + `${figures.tick} Successfully updated ${marketplaceNames.length} marketplace(s)`, + ) + } + } catch (error) { + handleMarketplaceError(error, 'update marketplace(s)') + } +} + +// plugin install (lines 5690–5721) +export async function pluginInstallHandler( + plugin: string, + options: { scope?: string; cowork?: boolean }, +): Promise { + if (options.cowork) setUseCoworkPlugins(true) + const scope = options.scope || 'user' + if (options.cowork && scope !== 'user') { + cliError('--cowork can only be used with user scope') + } + if ( + !VALID_INSTALLABLE_SCOPES.includes( + scope as (typeof VALID_INSTALLABLE_SCOPES)[number], + ) + ) { + cliError( + `Invalid scope: ${scope}. Must be one of: ${VALID_INSTALLABLE_SCOPES.join(', ')}.`, + ) + } + // _PROTO_* routes to PII-tagged plugin_name/marketplace_name BQ columns. + // Unredacted plugin arg was previously logged to general-access + // additional_metadata for all users — dropped in favor of the privileged + // column route. marketplace may be undefined (fires before resolution). + const { name, marketplace } = parsePluginIdentifier(plugin) + logEvent('tengu_plugin_install_command', { + _PROTO_plugin_name: name as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED, + ...(marketplace && { + _PROTO_marketplace_name: + marketplace as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED, + }), + scope: scope as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + + await installPlugin(plugin, scope as 'user' | 'project' | 'local') +} + +// plugin uninstall (lines 5738–5769) +export async function pluginUninstallHandler( + plugin: string, + options: { scope?: string; cowork?: boolean; keepData?: boolean }, +): Promise { + if (options.cowork) setUseCoworkPlugins(true) + const scope = options.scope || 'user' + if (options.cowork && scope !== 'user') { + cliError('--cowork can only be used with user scope') + } + if ( + !VALID_INSTALLABLE_SCOPES.includes( + scope as (typeof VALID_INSTALLABLE_SCOPES)[number], + ) + ) { + cliError( + `Invalid scope: ${scope}. Must be one of: ${VALID_INSTALLABLE_SCOPES.join(', ')}.`, + ) + } + const { name, marketplace } = parsePluginIdentifier(plugin) + logEvent('tengu_plugin_uninstall_command', { + _PROTO_plugin_name: name as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED, + ...(marketplace && { + _PROTO_marketplace_name: + marketplace as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED, + }), + scope: scope as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + + await uninstallPlugin( + plugin, + scope as 'user' | 'project' | 'local', + options.keepData, + ) +} + +// plugin enable (lines 5783–5818) +export async function pluginEnableHandler( + plugin: string, + options: { scope?: string; cowork?: boolean }, +): Promise { + if (options.cowork) setUseCoworkPlugins(true) + let scope: (typeof VALID_INSTALLABLE_SCOPES)[number] | undefined + if (options.scope) { + if ( + !VALID_INSTALLABLE_SCOPES.includes( + options.scope as (typeof VALID_INSTALLABLE_SCOPES)[number], + ) + ) { + cliError( + `Invalid scope "${options.scope}". Valid scopes: ${VALID_INSTALLABLE_SCOPES.join(', ')}`, + ) + } + scope = options.scope as (typeof VALID_INSTALLABLE_SCOPES)[number] + } + if (options.cowork && scope !== undefined && scope !== 'user') { + cliError('--cowork can only be used with user scope') + } + + // --cowork always operates at user scope + if (options.cowork && scope === undefined) { + scope = 'user' + } + + const { name, marketplace } = parsePluginIdentifier(plugin) + logEvent('tengu_plugin_enable_command', { + _PROTO_plugin_name: name as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED, + ...(marketplace && { + _PROTO_marketplace_name: + marketplace as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED, + }), + scope: (scope ?? + 'auto') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + + await enablePlugin(plugin, scope) +} + +// plugin disable (lines 5833–5902) +export async function pluginDisableHandler( + plugin: string | undefined, + options: { scope?: string; cowork?: boolean; all?: boolean }, +): Promise { + if (options.all && plugin) { + cliError('Cannot use --all with a specific plugin') + } + + if (!options.all && !plugin) { + cliError('Please specify a plugin name or use --all to disable all plugins') + } + + if (options.cowork) setUseCoworkPlugins(true) + + if (options.all) { + if (options.scope) { + cliError('Cannot use --scope with --all') + } + + // No _PROTO_plugin_name here — --all disables all plugins. + // Distinguishable from the specific-plugin branch by plugin_name IS NULL. + logEvent('tengu_plugin_disable_command', {}) + + await disableAllPlugins() + return + } + + let scope: (typeof VALID_INSTALLABLE_SCOPES)[number] | undefined + if (options.scope) { + if ( + !VALID_INSTALLABLE_SCOPES.includes( + options.scope as (typeof VALID_INSTALLABLE_SCOPES)[number], + ) + ) { + cliError( + `Invalid scope "${options.scope}". Valid scopes: ${VALID_INSTALLABLE_SCOPES.join(', ')}`, + ) + } + scope = options.scope as (typeof VALID_INSTALLABLE_SCOPES)[number] + } + if (options.cowork && scope !== undefined && scope !== 'user') { + cliError('--cowork can only be used with user scope') + } + + // --cowork always operates at user scope + if (options.cowork && scope === undefined) { + scope = 'user' + } + + const { name, marketplace } = parsePluginIdentifier(plugin!) + logEvent('tengu_plugin_disable_command', { + _PROTO_plugin_name: name as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED, + ...(marketplace && { + _PROTO_marketplace_name: + marketplace as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED, + }), + scope: (scope ?? + 'auto') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + + await disablePlugin(plugin!, scope) +} + +// plugin update (lines 5918–5948) +export async function pluginUpdateHandler( + plugin: string, + options: { scope?: string; cowork?: boolean }, +): Promise { + if (options.cowork) setUseCoworkPlugins(true) + const { name, marketplace } = parsePluginIdentifier(plugin) + logEvent('tengu_plugin_update_command', { + _PROTO_plugin_name: name as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED, + ...(marketplace && { + _PROTO_marketplace_name: + marketplace as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED, + }), + }) + + let scope: (typeof VALID_UPDATE_SCOPES)[number] = 'user' + if (options.scope) { + if ( + !VALID_UPDATE_SCOPES.includes( + options.scope as (typeof VALID_UPDATE_SCOPES)[number], + ) + ) { + cliError( + `Invalid scope "${options.scope}". Valid scopes: ${VALID_UPDATE_SCOPES.join(', ')}`, + ) + } + scope = options.scope as (typeof VALID_UPDATE_SCOPES)[number] + } + if (options.cowork && scope !== 'user') { + cliError('--cowork can only be used with user scope') + } + + await updatePluginCli(plugin, scope) +} diff --git a/src/cli/handlers/util.tsx b/src/cli/handlers/util.tsx new file mode 100644 index 0000000..03ff3cd --- /dev/null +++ b/src/cli/handlers/util.tsx @@ -0,0 +1,110 @@ +import { c as _c } from "react/compiler-runtime"; +/** + * Miscellaneous subcommand handlers — extracted from main.tsx for lazy loading. + * setup-token, doctor, install + */ +/* eslint-disable custom-rules/no-process-exit -- CLI subcommand handlers intentionally exit */ + +import { cwd } from 'process'; +import React from 'react'; +import { WelcomeV2 } from '../../components/LogoV2/WelcomeV2.js'; +import { useManagePlugins } from '../../hooks/useManagePlugins.js'; +import type { Root } from '../../ink.js'; +import { Box, Text } from '../../ink.js'; +import { KeybindingSetup } from '../../keybindings/KeybindingProviderSetup.js'; +import { logEvent } from '../../services/analytics/index.js'; +import { MCPConnectionManager } from '../../services/mcp/MCPConnectionManager.js'; +import { AppStateProvider } from '../../state/AppState.js'; +import { onChangeAppState } from '../../state/onChangeAppState.js'; +import { isAnthropicAuthEnabled } from '../../utils/auth.js'; +export async function setupTokenHandler(root: Root): Promise { + logEvent('tengu_setup_token_command', {}); + const showAuthWarning = !isAnthropicAuthEnabled(); + const { + ConsoleOAuthFlow + } = await import('../../components/ConsoleOAuthFlow.js'); + await new Promise(resolve => { + root.render( + + + + {showAuthWarning && + + Warning: You already have authentication configured via + environment variable or API key helper. + + + The setup-token command will create a new OAuth token which + you can use instead. + + } + { + void resolve(); + }} mode="setup-token" startingMessage="This will guide you through long-lived (1-year) auth token setup for your Claude account. Claude subscription required." /> + + + ); + }); + root.unmount(); + process.exit(0); +} + +// DoctorWithPlugins wrapper + doctor handler +const DoctorLazy = React.lazy(() => import('../../screens/Doctor.js').then(m => ({ + default: m.Doctor +}))); +function DoctorWithPlugins(t0) { + const $ = _c(2); + const { + onDone + } = t0; + useManagePlugins(); + let t1; + if ($[0] !== onDone) { + t1 = ; + $[0] = onDone; + $[1] = t1; + } else { + t1 = $[1]; + } + return t1; +} +export async function doctorHandler(root: Root): Promise { + logEvent('tengu_doctor_command', {}); + await new Promise(resolve => { + root.render( + + + { + void resolve(); + }} /> + + + ); + }); + root.unmount(); + process.exit(0); +} + +// install handler +export async function installHandler(target: string | undefined, options: { + force?: boolean; +}): Promise { + const { + setup + } = await import('../../setup.js'); + await setup(cwd(), 'default', false, false, undefined, false); + const { + install + } = await import('../../commands/install.js'); + await new Promise(resolve => { + const args: string[] = []; + if (target) args.push(target); + if (options.force) args.push('--force'); + void install.call(result => { + void resolve(); + process.exit(result.includes('failed') ? 1 : 0); + }, {}, args); + }); +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJjd2QiLCJSZWFjdCIsIldlbGNvbWVWMiIsInVzZU1hbmFnZVBsdWdpbnMiLCJSb290IiwiQm94IiwiVGV4dCIsIktleWJpbmRpbmdTZXR1cCIsImxvZ0V2ZW50IiwiTUNQQ29ubmVjdGlvbk1hbmFnZXIiLCJBcHBTdGF0ZVByb3ZpZGVyIiwib25DaGFuZ2VBcHBTdGF0ZSIsImlzQW50aHJvcGljQXV0aEVuYWJsZWQiLCJzZXR1cFRva2VuSGFuZGxlciIsInJvb3QiLCJQcm9taXNlIiwic2hvd0F1dGhXYXJuaW5nIiwiQ29uc29sZU9BdXRoRmxvdyIsInJlc29sdmUiLCJyZW5kZXIiLCJ1bm1vdW50IiwicHJvY2VzcyIsImV4aXQiLCJEb2N0b3JMYXp5IiwibGF6eSIsInRoZW4iLCJtIiwiZGVmYXVsdCIsIkRvY3RvciIsIkRvY3RvcldpdGhQbHVnaW5zIiwidDAiLCIkIiwiX2MiLCJvbkRvbmUiLCJ0MSIsImRvY3RvckhhbmRsZXIiLCJ1bmRlZmluZWQiLCJpbnN0YWxsSGFuZGxlciIsInRhcmdldCIsIm9wdGlvbnMiLCJmb3JjZSIsInNldHVwIiwiaW5zdGFsbCIsImFyZ3MiLCJwdXNoIiwiY2FsbCIsInJlc3VsdCIsImluY2x1ZGVzIl0sInNvdXJjZXMiOlsidXRpbC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBNaXNjZWxsYW5lb3VzIHN1YmNvbW1hbmQgaGFuZGxlcnMg4oCUIGV4dHJhY3RlZCBmcm9tIG1haW4udHN4IGZvciBsYXp5IGxvYWRpbmcuXG4gKiBzZXR1cC10b2tlbiwgZG9jdG9yLCBpbnN0YWxsXG4gKi9cbi8qIGVzbGludC1kaXNhYmxlIGN1c3RvbS1ydWxlcy9uby1wcm9jZXNzLWV4aXQgLS0gQ0xJIHN1YmNvbW1hbmQgaGFuZGxlcnMgaW50ZW50aW9uYWxseSBleGl0ICovXG5cbmltcG9ydCB7IGN3ZCB9IGZyb20gJ3Byb2Nlc3MnXG5pbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBXZWxjb21lVjIgfSBmcm9tICcuLi8uLi9jb21wb25lbnRzL0xvZ29WMi9XZWxjb21lVjIuanMnXG5pbXBvcnQgeyB1c2VNYW5hZ2VQbHVnaW5zIH0gZnJvbSAnLi4vLi4vaG9va3MvdXNlTWFuYWdlUGx1Z2lucy5qcydcbmltcG9ydCB0eXBlIHsgUm9vdCB9IGZyb20gJy4uLy4uL2luay5qcydcbmltcG9ydCB7IEJveCwgVGV4dCB9IGZyb20gJy4uLy4uL2luay5qcydcbmltcG9ydCB7IEtleWJpbmRpbmdTZXR1cCB9IGZyb20gJy4uLy4uL2tleWJpbmRpbmdzL0tleWJpbmRpbmdQcm92aWRlclNldHVwLmpzJ1xuaW1wb3J0IHsgbG9nRXZlbnQgfSBmcm9tICcuLi8uLi9zZXJ2aWNlcy9hbmFseXRpY3MvaW5kZXguanMnXG5pbXBvcnQgeyBNQ1BDb25uZWN0aW9uTWFuYWdlciB9IGZyb20gJy4uLy4uL3NlcnZpY2VzL21jcC9NQ1BDb25uZWN0aW9uTWFuYWdlci5qcydcbmltcG9ydCB7IEFwcFN0YXRlUHJvdmlkZXIgfSBmcm9tICcuLi8uLi9zdGF0ZS9BcHBTdGF0ZS5qcydcbmltcG9ydCB7IG9uQ2hhbmdlQXBwU3RhdGUgfSBmcm9tICcuLi8uLi9zdGF0ZS9vbkNoYW5nZUFwcFN0YXRlLmpzJ1xuaW1wb3J0IHsgaXNBbnRocm9waWNBdXRoRW5hYmxlZCB9IGZyb20gJy4uLy4uL3V0aWxzL2F1dGguanMnXG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBzZXR1cFRva2VuSGFuZGxlcihyb290OiBSb290KTogUHJvbWlzZTx2b2lkPiB7XG4gIGxvZ0V2ZW50KCd0ZW5ndV9zZXR1cF90b2tlbl9jb21tYW5kJywge30pXG5cbiAgY29uc3Qgc2hvd0F1dGhXYXJuaW5nID0gIWlzQW50aHJvcGljQXV0aEVuYWJsZWQoKVxuICBjb25zdCB7IENvbnNvbGVPQXV0aEZsb3cgfSA9IGF3YWl0IGltcG9ydChcbiAgICAnLi4vLi4vY29tcG9uZW50cy9Db25zb2xlT0F1dGhGbG93LmpzJ1xuICApXG4gIGF3YWl0IG5ldyBQcm9taXNlPHZvaWQ+KHJlc29sdmUgPT4ge1xuICAgIHJvb3QucmVuZGVyKFxuICAgICAgPEFwcFN0YXRlUHJvdmlkZXIgb25DaGFuZ2VBcHBTdGF0ZT17b25DaGFuZ2VBcHBTdGF0ZX0+XG4gICAgICAgIDxLZXliaW5kaW5nU2V0dXA+XG4gICAgICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCIgZ2FwPXsxfT5cbiAgICAgICAgICAgIDxXZWxjb21lVjIgLz5cbiAgICAgICAgICAgIHtzaG93QXV0aFdhcm5pbmcgJiYgKFxuICAgICAgICAgICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIj5cbiAgICAgICAgICAgICAgICA8VGV4dCBjb2xvcj1cIndhcm5pbmdcIj5cbiAgICAgICAgICAgICAgICAgIFdhcm5pbmc6IFlvdSBhbHJlYWR5IGhhdmUgYXV0aGVudGljYXRpb24gY29uZmlndXJlZCB2aWFcbiAgICAgICAgICAgICAgICAgIGVudmlyb25tZW50IHZhcmlhYmxlIG9yIEFQSSBrZXkgaGVscGVyLlxuICAgICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgICA8VGV4dCBjb2xvcj1cIndhcm5pbmdcIj5cbiAgICAgICAgICAgICAgICAgIFRoZSBzZXR1cC10b2tlbiBjb21tYW5kIHdpbGwgY3JlYXRlIGEgbmV3IE9BdXRoIHRva2VuIHdoaWNoXG4gICAgICAgICAgICAgICAgICB5b3UgY2FuIHVzZSBpbnN0ZWFkLlxuICAgICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgPC9Cb3g+XG4gICAgICAgICAgICApfVxuICAgICAgICAgICAgPENvbnNvbGVPQXV0aEZsb3dcbiAgICAgICAgICAgICAgb25Eb25lPXsoKSA9PiB7XG4gICAgICAgICAgICAgICAgdm9pZCByZXNvbHZlKClcbiAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgbW9kZT1cInNldHVwLXRva2VuXCJcbiAgICAgICAgICAgICAgc3RhcnRpbmdNZXNzYWdlPVwiVGhpcyB3aWxsIGd1aWRlIHlvdSB0aHJvdWdoIGxvbmctbGl2ZWQgKDEteWVhcikgYXV0aCB0b2tlbiBzZXR1cCBmb3IgeW91ciBDbGF1ZGUgYWNjb3VudC4gQ2xhdWRlIHN1YnNjcmlwdGlvbiByZXF1aXJlZC5cIlxuICAgICAgICAgICAgLz5cbiAgICAgICAgICA8L0JveD5cbiAgICAgICAgPC9LZXliaW5kaW5nU2V0dXA+XG4gICAgICA8L0FwcFN0YXRlUHJvdmlkZXI+LFxuICAgIClcbiAgfSlcbiAgcm9vdC51bm1vdW50KClcbiAgcHJvY2Vzcy5leGl0KDApXG59XG5cbi8vIERvY3RvcldpdGhQbHVnaW5zIHdyYXBwZXIgKyBkb2N0b3IgaGFuZGxlclxuY29uc3QgRG9jdG9yTGF6eSA9IFJlYWN0LmxhenkoKCkgPT5cbiAgaW1wb3J0KCcuLi8uLi9zY3JlZW5zL0RvY3Rvci5qcycpLnRoZW4obSA9PiAoeyBkZWZhdWx0OiBtLkRvY3RvciB9KSksXG4pXG5cbmZ1bmN0aW9uIERvY3RvcldpdGhQbHVnaW5zKHtcbiAgb25Eb25lLFxufToge1xuICBvbkRvbmU6ICgpID0+IHZvaWRcbn0pOiBSZWFjdC5SZWFjdE5vZGUge1xuICB1c2VNYW5hZ2VQbHVnaW5zKClcbiAgcmV0dXJuIChcbiAgICA8UmVhY3QuU3VzcGVuc2UgZmFsbGJhY2s9e251bGx9PlxuICAgICAgPERvY3Rvckxhenkgb25Eb25lPXtvbkRvbmV9IC8+XG4gICAgPC9SZWFjdC5TdXNwZW5zZT5cbiAgKVxufVxuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gZG9jdG9ySGFuZGxlcihyb290OiBSb290KTogUHJvbWlzZTx2b2lkPiB7XG4gIGxvZ0V2ZW50KCd0ZW5ndV9kb2N0b3JfY29tbWFuZCcsIHt9KVxuXG4gIGF3YWl0IG5ldyBQcm9taXNlPHZvaWQ+KHJlc29sdmUgPT4ge1xuICAgIHJvb3QucmVuZGVyKFxuICAgICAgPEFwcFN0YXRlUHJvdmlkZXI+XG4gICAgICAgIDxLZXliaW5kaW5nU2V0dXA+XG4gICAgICAgICAgPE1DUENvbm5lY3Rpb25NYW5hZ2VyXG4gICAgICAgICAgICBkeW5hbWljTWNwQ29uZmlnPXt1bmRlZmluZWR9XG4gICAgICAgICAgICBpc1N0cmljdE1jcENvbmZpZz17ZmFsc2V9XG4gICAgICAgICAgPlxuICAgICAgICAgICAgPERvY3RvcldpdGhQbHVnaW5zXG4gICAgICAgICAgICAgIG9uRG9uZT17KCkgPT4ge1xuICAgICAgICAgICAgICAgIHZvaWQgcmVzb2x2ZSgpXG4gICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAvPlxuICAgICAgICAgIDwvTUNQQ29ubmVjdGlvbk1hbmFnZXI+XG4gICAgICAgIDwvS2V5YmluZGluZ1NldHVwPlxuICAgICAgPC9BcHBTdGF0ZVByb3ZpZGVyPixcbiAgICApXG4gIH0pXG4gIHJvb3QudW5tb3VudCgpXG4gIHByb2Nlc3MuZXhpdCgwKVxufVxuXG4vLyBpbnN0YWxsIGhhbmRsZXJcbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBpbnN0YWxsSGFuZGxlcihcbiAgdGFyZ2V0OiBzdHJpbmcgfCB1bmRlZmluZWQsXG4gIG9wdGlvbnM6IHsgZm9yY2U/OiBib29sZWFuIH0sXG4pOiBQcm9taXNlPHZvaWQ+IHtcbiAgY29uc3QgeyBzZXR1cCB9ID0gYXdhaXQgaW1wb3J0KCcuLi8uLi9zZXR1cC5qcycpXG4gIGF3YWl0IHNldHVwKGN3ZCgpLCAnZGVmYXVsdCcsIGZhbHNlLCBmYWxzZSwgdW5kZWZpbmVkLCBmYWxzZSlcbiAgY29uc3QgeyBpbnN0YWxsIH0gPSBhd2FpdCBpbXBvcnQoJy4uLy4uL2NvbW1hbmRzL2luc3RhbGwuanMnKVxuICBhd2FpdCBuZXcgUHJvbWlzZTx2b2lkPihyZXNvbHZlID0+IHtcbiAgICBjb25zdCBhcmdzOiBzdHJpbmdbXSA9IFtdXG4gICAgaWYgKHRhcmdldCkgYXJncy5wdXNoKHRhcmdldClcbiAgICBpZiAob3B0aW9ucy5mb3JjZSkgYXJncy5wdXNoKCctLWZvcmNlJylcblxuICAgIHZvaWQgaW5zdGFsbC5jYWxsKFxuICAgICAgcmVzdWx0ID0+IHtcbiAgICAgICAgdm9pZCByZXNvbHZlKClcbiAgICAgICAgcHJvY2Vzcy5leGl0KHJlc3VsdC5pbmNsdWRlcygnZmFpbGVkJykgPyAxIDogMClcbiAgICAgIH0sXG4gICAgICB7fSxcbiAgICAgIGFyZ3MsXG4gICAgKVxuICB9KVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUFFQSxTQUFTQSxHQUFHLFFBQVEsU0FBUztBQUM3QixPQUFPQyxLQUFLLE1BQU0sT0FBTztBQUN6QixTQUFTQyxTQUFTLFFBQVEsc0NBQXNDO0FBQ2hFLFNBQVNDLGdCQUFnQixRQUFRLGlDQUFpQztBQUNsRSxjQUFjQyxJQUFJLFFBQVEsY0FBYztBQUN4QyxTQUFTQyxHQUFHLEVBQUVDLElBQUksUUFBUSxjQUFjO0FBQ3hDLFNBQVNDLGVBQWUsUUFBUSw4Q0FBOEM7QUFDOUUsU0FBU0MsUUFBUSxRQUFRLG1DQUFtQztBQUM1RCxTQUFTQyxvQkFBb0IsUUFBUSw0Q0FBNEM7QUFDakYsU0FBU0MsZ0JBQWdCLFFBQVEseUJBQXlCO0FBQzFELFNBQVNDLGdCQUFnQixRQUFRLGlDQUFpQztBQUNsRSxTQUFTQyxzQkFBc0IsUUFBUSxxQkFBcUI7QUFFNUQsT0FBTyxlQUFlQyxpQkFBaUJBLENBQUNDLElBQUksRUFBRVYsSUFBSSxDQUFDLEVBQUVXLE9BQU8sQ0FBQyxJQUFJLENBQUMsQ0FBQztFQUNqRVAsUUFBUSxDQUFDLDJCQUEyQixFQUFFLENBQUMsQ0FBQyxDQUFDO0VBRXpDLE1BQU1RLGVBQWUsR0FBRyxDQUFDSixzQkFBc0IsQ0FBQyxDQUFDO0VBQ2pELE1BQU07SUFBRUs7RUFBaUIsQ0FBQyxHQUFHLE1BQU0sTUFBTSxDQUN2QyxzQ0FDRixDQUFDO0VBQ0QsTUFBTSxJQUFJRixPQUFPLENBQUMsSUFBSSxDQUFDLENBQUNHLE9BQU8sSUFBSTtJQUNqQ0osSUFBSSxDQUFDSyxNQUFNLENBQ1QsQ0FBQyxnQkFBZ0IsQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDUixnQkFBZ0IsQ0FBQztBQUMzRCxRQUFRLENB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 \ No newline at end of file diff --git a/src/cli/ndjsonSafeStringify.ts b/src/cli/ndjsonSafeStringify.ts new file mode 100644 index 0000000..af570ad --- /dev/null +++ b/src/cli/ndjsonSafeStringify.ts @@ -0,0 +1,32 @@ +import { jsonStringify } from '../utils/slowOperations.js' + +// JSON.stringify emits U+2028/U+2029 raw (valid per ECMA-404). When the +// output is a single NDJSON line, any receiver that uses JavaScript +// line-terminator semantics (ECMA-262 §11.3 — \n \r U+2028 U+2029) to +// split the stream will cut the JSON mid-string. ProcessTransport now +// silently skips non-JSON lines rather than crashing (gh-28405), but +// the truncated fragment is still lost — the message is silently dropped. +// +// The \uXXXX form is equivalent JSON (parses to the same string) but +// can never be mistaken for a line terminator by ANY receiver. This is +// what ES2019's "Subsume JSON" proposal and Node's util.inspect do. +// +// Single regex with alternation: the callback's one dispatch per match +// is cheaper than two full-string scans. +const JS_LINE_TERMINATORS = /\u2028|\u2029/g + +function escapeJsLineTerminators(json: string): string { + return json.replace(JS_LINE_TERMINATORS, c => + c === '\u2028' ? '\\u2028' : '\\u2029', + ) +} + +/** + * JSON.stringify for one-message-per-line transports. Escapes U+2028 + * LINE SEPARATOR and U+2029 PARAGRAPH SEPARATOR so the serialized output + * cannot be broken by a line-splitting receiver. Output is still valid + * JSON and parses to the same value. + */ +export function ndjsonSafeStringify(value: unknown): string { + return escapeJsLineTerminators(jsonStringify(value)) +} diff --git a/src/cli/print.ts b/src/cli/print.ts new file mode 100644 index 0000000..6047257 --- /dev/null +++ b/src/cli/print.ts @@ -0,0 +1,5594 @@ +// biome-ignore-all assist/source/organizeImports: ANT-ONLY import markers must not be reordered +import { feature } from 'bun:bundle' +import { readFile, stat } from 'fs/promises' +import { dirname } from 'path' +import { + downloadUserSettings, + redownloadUserSettings, +} from 'src/services/settingsSync/index.js' +import { waitForRemoteManagedSettingsToLoad } from 'src/services/remoteManagedSettings/index.js' +import { StructuredIO } from 'src/cli/structuredIO.js' +import { RemoteIO } from 'src/cli/remoteIO.js' +import { + type Command, + formatDescriptionWithSource, + getCommandName, +} from 'src/commands.js' +import { createStreamlinedTransformer } from 'src/utils/streamlinedTransform.js' +import { installStreamJsonStdoutGuard } from 'src/utils/streamJsonStdoutGuard.js' +import type { ToolPermissionContext } from 'src/Tool.js' +import type { ThinkingConfig } from 'src/utils/thinking.js' +import { assembleToolPool, filterToolsByDenyRules } from 'src/tools.js' +import uniqBy from 'lodash-es/uniqBy.js' +import { uniq } from 'src/utils/array.js' +import { mergeAndFilterTools } from 'src/utils/toolPool.js' +import { + logEvent, + type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, +} from 'src/services/analytics/index.js' +import { getFeatureValue_CACHED_MAY_BE_STALE } from 'src/services/analytics/growthbook.js' +import { logForDebugging } from 'src/utils/debug.js' +import { + logForDiagnosticsNoPII, + withDiagnosticsTiming, +} from 'src/utils/diagLogs.js' +import { toolMatchesName, type Tool, type Tools } from 'src/Tool.js' +import { + type AgentDefinition, + isBuiltInAgent, + parseAgentsFromJson, +} from 'src/tools/AgentTool/loadAgentsDir.js' +import type { Message, NormalizedUserMessage } from 'src/types/message.js' +import type { QueuedCommand } from 'src/types/textInputTypes.js' +import { + dequeue, + dequeueAllMatching, + enqueue, + hasCommandsInQueue, + peek, + subscribeToCommandQueue, + getCommandsByMaxPriority, +} from 'src/utils/messageQueueManager.js' +import { notifyCommandLifecycle } from 'src/utils/commandLifecycle.js' +import { + getSessionState, + notifySessionStateChanged, + notifySessionMetadataChanged, + setPermissionModeChangedListener, + type RequiresActionDetails, + type SessionExternalMetadata, +} from 'src/utils/sessionState.js' +import { externalMetadataToAppState } from 'src/state/onChangeAppState.js' +import { getInMemoryErrors, logError, logMCPDebug } from 'src/utils/log.js' +import { + writeToStdout, + registerProcessOutputErrorHandlers, +} from 'src/utils/process.js' +import type { Stream } from 'src/utils/stream.js' +import { EMPTY_USAGE } from 'src/services/api/logging.js' +import { + loadConversationForResume, + type TurnInterruptionState, +} from 'src/utils/conversationRecovery.js' +import type { + MCPServerConnection, + McpSdkServerConfig, + ScopedMcpServerConfig, +} from 'src/services/mcp/types.js' +import { + ChannelMessageNotificationSchema, + gateChannelServer, + wrapChannelMessage, + findChannelEntry, +} from 'src/services/mcp/channelNotification.js' +import { + isChannelAllowlisted, + isChannelsEnabled, +} from 'src/services/mcp/channelAllowlist.js' +import { parsePluginIdentifier } from 'src/utils/plugins/pluginIdentifier.js' +import { validateUuid } from 'src/utils/uuid.js' +import { fromArray } from 'src/utils/generators.js' +import { ask } from 'src/QueryEngine.js' +import type { PermissionPromptTool } from 'src/utils/queryHelpers.js' +import { + createFileStateCacheWithSizeLimit, + mergeFileStateCaches, + READ_FILE_STATE_CACHE_SIZE, +} from 'src/utils/fileStateCache.js' +import { expandPath } from 'src/utils/path.js' +import { extractReadFilesFromMessages } from 'src/utils/queryHelpers.js' +import { registerHookEventHandler } from 'src/utils/hooks/hookEvents.js' +import { executeFilePersistence } from 'src/utils/filePersistence/filePersistence.js' +import { finalizePendingAsyncHooks } from 'src/utils/hooks/AsyncHookRegistry.js' +import { + gracefulShutdown, + gracefulShutdownSync, + isShuttingDown, +} from 'src/utils/gracefulShutdown.js' +import { registerCleanup } from 'src/utils/cleanupRegistry.js' +import { createIdleTimeoutManager } from 'src/utils/idleTimeout.js' +import type { + SDKStatus, + ModelInfo, + SDKMessage, + SDKUserMessage, + SDKUserMessageReplay, + PermissionResult, + McpServerConfigForProcessTransport, + McpServerStatus, + RewindFilesResult, +} from 'src/entrypoints/agentSdkTypes.js' +import type { + StdoutMessage, + SDKControlInitializeRequest, + SDKControlInitializeResponse, + SDKControlRequest, + SDKControlResponse, + SDKControlMcpSetServersResponse, + SDKControlReloadPluginsResponse, +} from 'src/entrypoints/sdk/controlTypes.js' +import type { PermissionMode } from '@anthropic-ai/claude-agent-sdk' +import type { PermissionMode as InternalPermissionMode } from 'src/types/permissions.js' +import { cwd } from 'process' +import { getCwd } from 'src/utils/cwd.js' +import omit from 'lodash-es/omit.js' +import reject from 'lodash-es/reject.js' +import { isPolicyAllowed } from 'src/services/policyLimits/index.js' +import type { ReplBridgeHandle } from 'src/bridge/replBridge.js' +import { getRemoteSessionUrl } from 'src/constants/product.js' +import { buildBridgeConnectUrl } from 'src/bridge/bridgeStatusUtil.js' +import { extractInboundMessageFields } from 'src/bridge/inboundMessages.js' +import { resolveAndPrepend } from 'src/bridge/inboundAttachments.js' +import type { CanUseToolFn } from 'src/hooks/useCanUseTool.js' +import { hasPermissionsToUseTool } from 'src/utils/permissions/permissions.js' +import { safeParseJSON } from 'src/utils/json.js' +import { + outputSchema as permissionToolOutputSchema, + permissionPromptToolResultToPermissionDecision, +} from 'src/utils/permissions/PermissionPromptToolResultSchema.js' +import { createAbortController } from 'src/utils/abortController.js' +import { createCombinedAbortSignal } from 'src/utils/combinedAbortSignal.js' +import { generateSessionTitle } from 'src/utils/sessionTitle.js' +import { buildSideQuestionFallbackParams } from 'src/utils/queryContext.js' +import { runSideQuestion } from 'src/utils/sideQuestion.js' +import { + processSessionStartHooks, + processSetupHooks, + takeInitialUserMessage, +} from 'src/utils/sessionStart.js' +import { + DEFAULT_OUTPUT_STYLE_NAME, + getAllOutputStyles, +} from 'src/constants/outputStyles.js' +import { TEAMMATE_MESSAGE_TAG, TICK_TAG } from 'src/constants/xml.js' +import { + getSettings_DEPRECATED, + getSettingsWithSources, +} from 'src/utils/settings/settings.js' +import { settingsChangeDetector } from 'src/utils/settings/changeDetector.js' +import { applySettingsChange } from 'src/utils/settings/applySettingsChange.js' +import { + isFastModeAvailable, + isFastModeEnabled, + isFastModeSupportedByModel, + getFastModeState, +} from 'src/utils/fastMode.js' +import { + isAutoModeGateEnabled, + getAutoModeUnavailableNotification, + getAutoModeUnavailableReason, + isBypassPermissionsModeDisabled, + transitionPermissionMode, +} from 'src/utils/permissions/permissionSetup.js' +import { + tryGenerateSuggestion, + logSuggestionOutcome, + logSuggestionSuppressed, + type PromptVariant, +} from 'src/services/PromptSuggestion/promptSuggestion.js' +import { getLastCacheSafeParams } from 'src/utils/forkedAgent.js' +import { getAccountInformation } from 'src/utils/auth.js' +import { OAuthService } from 'src/services/oauth/index.js' +import { installOAuthTokens } from 'src/cli/handlers/auth.js' +import { getAPIProvider } from 'src/utils/model/providers.js' +import type { HookCallbackMatcher } from 'src/types/hooks.js' +import { AwsAuthStatusManager } from 'src/utils/awsAuthStatusManager.js' +import type { HookEvent } from 'src/entrypoints/agentSdkTypes.js' +import { + registerHookCallbacks, + setInitJsonSchema, + getInitJsonSchema, + setSdkAgentProgressSummariesEnabled, +} from 'src/bootstrap/state.js' +import { createSyntheticOutputTool } from 'src/tools/SyntheticOutputTool/SyntheticOutputTool.js' +import { parseSessionIdentifier } from 'src/utils/sessionUrl.js' +import { + hydrateRemoteSession, + hydrateFromCCRv2InternalEvents, + resetSessionFilePointer, + doesMessageExistInSession, + findUnresolvedToolUse, + recordAttributionSnapshot, + saveAgentSetting, + saveMode, + saveAiGeneratedTitle, + restoreSessionMetadata, +} from 'src/utils/sessionStorage.js' +import { incrementPromptCount } from 'src/utils/commitAttribution.js' +import { + setupSdkMcpClients, + connectToServer, + clearServerCache, + fetchToolsForClient, + areMcpConfigsEqual, + reconnectMcpServerImpl, +} from 'src/services/mcp/client.js' +import { + filterMcpServersByPolicy, + getMcpConfigByName, + isMcpServerDisabled, + setMcpServerEnabled, +} from 'src/services/mcp/config.js' +import { + performMCPOAuthFlow, + revokeServerTokens, +} from 'src/services/mcp/auth.js' +import { + runElicitationHooks, + runElicitationResultHooks, +} from 'src/services/mcp/elicitationHandler.js' +import { executeNotificationHooks } from 'src/utils/hooks.js' +import { + ElicitRequestSchema, + ElicitationCompleteNotificationSchema, +} from '@modelcontextprotocol/sdk/types.js' +import { getMcpPrefix } from 'src/services/mcp/mcpStringUtils.js' +import { + commandBelongsToServer, + filterToolsByServer, +} from 'src/services/mcp/utils.js' +import { setupVscodeSdkMcp } from 'src/services/mcp/vscodeSdkMcp.js' +import { getAllMcpConfigs } from 'src/services/mcp/config.js' +import { + isQualifiedForGrove, + checkGroveForNonInteractive, +} from 'src/services/api/grove.js' +import { + toInternalMessages, + toSDKRateLimitInfo, +} from 'src/utils/messages/mappers.js' +import { createModelSwitchBreadcrumbs } from 'src/utils/messages.js' +import { collectContextData } from 'src/commands/context/context-noninteractive.js' +import { LOCAL_COMMAND_STDOUT_TAG } from 'src/constants/xml.js' +import { + statusListeners, + type ClaudeAILimits, +} from 'src/services/claudeAiLimits.js' +import { + getDefaultMainLoopModel, + getMainLoopModel, + modelDisplayString, + parseUserSpecifiedModel, +} from 'src/utils/model/model.js' +import { getModelOptions } from 'src/utils/model/modelOptions.js' +import { + modelSupportsEffort, + modelSupportsMaxEffort, + EFFORT_LEVELS, + resolveAppliedEffort, +} from 'src/utils/effort.js' +import { modelSupportsAdaptiveThinking } from 'src/utils/thinking.js' +import { modelSupportsAutoMode } from 'src/utils/betas.js' +import { ensureModelStringsInitialized } from 'src/utils/model/modelStrings.js' +import { + getSessionId, + setMainLoopModelOverride, + setMainThreadAgentType, + switchSession, + isSessionPersistenceDisabled, + getIsRemoteMode, + getFlagSettingsInline, + setFlagSettingsInline, + getMainThreadAgentType, + getAllowedChannels, + setAllowedChannels, + type ChannelEntry, +} from 'src/bootstrap/state.js' +import { runWithWorkload, WORKLOAD_CRON } from 'src/utils/workloadContext.js' +import type { UUID } from 'crypto' +import { randomUUID } from 'crypto' +import type { ContentBlockParam } from '@anthropic-ai/sdk/resources/messages.mjs' +import type { AppState } from 'src/state/AppStateStore.js' +import { + fileHistoryRewind, + fileHistoryCanRestore, + fileHistoryEnabled, + fileHistoryGetDiffStats, +} from 'src/utils/fileHistory.js' +import { + restoreAgentFromSession, + restoreSessionStateFromLog, +} from 'src/utils/sessionRestore.js' +import { SandboxManager } from 'src/utils/sandbox/sandbox-adapter.js' +import { + headlessProfilerStartTurn, + headlessProfilerCheckpoint, + logHeadlessProfilerTurn, +} from 'src/utils/headlessProfiler.js' +import { + startQueryProfile, + logQueryProfileReport, +} from 'src/utils/queryProfiler.js' +import { asSessionId } from 'src/types/ids.js' +import { jsonStringify } from '../utils/slowOperations.js' +import { skillChangeDetector } from '../utils/skills/skillChangeDetector.js' +import { getCommands, clearCommandsCache } from '../commands.js' +import { + isBareMode, + isEnvTruthy, + isEnvDefinedFalsy, +} from '../utils/envUtils.js' +import { installPluginsForHeadless } from '../utils/plugins/headlessPluginInstall.js' +import { refreshActivePlugins } from '../utils/plugins/refresh.js' +import { loadAllPluginsCacheOnly } from '../utils/plugins/pluginLoader.js' +import { + isTeamLead, + hasActiveInProcessTeammates, + hasWorkingInProcessTeammates, + waitForTeammatesToBecomeIdle, +} from '../utils/teammate.js' +import { + readUnreadMessages, + markMessagesAsRead, + isShutdownApproved, +} from '../utils/teammateMailbox.js' +import { removeTeammateFromTeamFile } from '../utils/swarm/teamHelpers.js' +import { unassignTeammateTasks } from '../utils/tasks.js' +import { getRunningTasks } from '../utils/task/framework.js' +import { isBackgroundTask } from '../tasks/types.js' +import { stopTask } from '../tasks/stopTask.js' +import { drainSdkEvents } from '../utils/sdkEventQueue.js' +import { initializeGrowthBook } from '../services/analytics/growthbook.js' +import { errorMessage, toError } from '../utils/errors.js' +import { sleep } from '../utils/sleep.js' +import { isExtractModeActive } from '../memdir/paths.js' + +// Dead code elimination: conditional imports +/* eslint-disable @typescript-eslint/no-require-imports */ +const coordinatorModeModule = feature('COORDINATOR_MODE') + ? (require('../coordinator/coordinatorMode.js') as typeof import('../coordinator/coordinatorMode.js')) + : null +const proactiveModule = + feature('PROACTIVE') || feature('KAIROS') + ? (require('../proactive/index.js') as typeof import('../proactive/index.js')) + : null +const cronSchedulerModule = feature('AGENT_TRIGGERS') + ? (require('../utils/cronScheduler.js') as typeof import('../utils/cronScheduler.js')) + : null +const cronJitterConfigModule = feature('AGENT_TRIGGERS') + ? (require('../utils/cronJitterConfig.js') as typeof import('../utils/cronJitterConfig.js')) + : null +const cronGate = feature('AGENT_TRIGGERS') + ? (require('../tools/ScheduleCronTool/prompt.js') as typeof import('../tools/ScheduleCronTool/prompt.js')) + : null +const extractMemoriesModule = feature('EXTRACT_MEMORIES') + ? (require('../services/extractMemories/extractMemories.js') as typeof import('../services/extractMemories/extractMemories.js')) + : null +/* eslint-enable @typescript-eslint/no-require-imports */ + +const SHUTDOWN_TEAM_PROMPT = ` +You are running in non-interactive mode and cannot return a response to the user until your team is shut down. + +You MUST shut down your team before preparing your final response: +1. Use requestShutdown to ask each team member to shut down gracefully +2. Wait for shutdown approvals +3. Use the cleanup operation to clean up the team +4. Only then provide your final response to the user + +The user cannot receive your response until the team is completely shut down. + + +Shut down your team and prepare your final response for the user.` + +// Track message UUIDs received during the current session runtime +const MAX_RECEIVED_UUIDS = 10_000 +const receivedMessageUuids = new Set() +const receivedMessageUuidsOrder: UUID[] = [] + +function trackReceivedMessageUuid(uuid: UUID): boolean { + if (receivedMessageUuids.has(uuid)) { + return false // duplicate + } + receivedMessageUuids.add(uuid) + receivedMessageUuidsOrder.push(uuid) + // Evict oldest entries when at capacity + if (receivedMessageUuidsOrder.length > MAX_RECEIVED_UUIDS) { + const toEvict = receivedMessageUuidsOrder.splice( + 0, + receivedMessageUuidsOrder.length - MAX_RECEIVED_UUIDS, + ) + for (const old of toEvict) { + receivedMessageUuids.delete(old) + } + } + return true // new UUID +} + +type PromptValue = string | ContentBlockParam[] + +function toBlocks(v: PromptValue): ContentBlockParam[] { + return typeof v === 'string' ? [{ type: 'text', text: v }] : v +} + +/** + * Join prompt values from multiple queued commands into one. Strings are + * newline-joined; if any value is a block array, all values are normalized + * to blocks and concatenated. + */ +export function joinPromptValues(values: PromptValue[]): PromptValue { + if (values.length === 1) return values[0]! + if (values.every(v => typeof v === 'string')) { + return values.join('\n') + } + return values.flatMap(toBlocks) +} + +/** + * Whether `next` can be batched into the same ask() call as `head`. Only + * prompt-mode commands batch, and only when the workload tag matches (so the + * combined turn is attributed correctly) and the isMeta flag matches (so a + * proactive tick can't merge into a user prompt and lose its hidden-in- + * transcript marking when the head is spread over the merged command). + */ +export function canBatchWith( + head: QueuedCommand, + next: QueuedCommand | undefined, +): boolean { + return ( + next !== undefined && + next.mode === 'prompt' && + next.workload === head.workload && + next.isMeta === head.isMeta + ) +} + +export async function runHeadless( + inputPrompt: string | AsyncIterable, + getAppState: () => AppState, + setAppState: (f: (prev: AppState) => AppState) => void, + commands: Command[], + tools: Tools, + sdkMcpConfigs: Record, + agents: AgentDefinition[], + options: { + continue: boolean | undefined + resume: string | boolean | undefined + resumeSessionAt: string | undefined + verbose: boolean | undefined + outputFormat: string | undefined + jsonSchema: Record | undefined + permissionPromptToolName: string | undefined + allowedTools: string[] | undefined + thinkingConfig: ThinkingConfig | undefined + maxTurns: number | undefined + maxBudgetUsd: number | undefined + taskBudget: { total: number } | undefined + systemPrompt: string | undefined + appendSystemPrompt: string | undefined + userSpecifiedModel: string | undefined + fallbackModel: string | undefined + teleport: string | true | null | undefined + sdkUrl: string | undefined + replayUserMessages: boolean | undefined + includePartialMessages: boolean | undefined + forkSession: boolean | undefined + rewindFiles: string | undefined + enableAuthStatus: boolean | undefined + agent: string | undefined + workload: string | undefined + setupTrigger?: 'init' | 'maintenance' | undefined + sessionStartHooksPromise?: ReturnType + setSDKStatus?: (status: SDKStatus) => void + }, +): Promise { + if ( + process.env.USER_TYPE === 'ant' && + isEnvTruthy(process.env.CLAUDE_CODE_EXIT_AFTER_FIRST_RENDER) + ) { + process.stderr.write( + `\nStartup time: ${Math.round(process.uptime() * 1000)}ms\n`, + ) + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(0) + } + + // Fire user settings download now so it overlaps with the MCP/tool setup + // below. Managed settings already started in main.tsx preAction; this gives + // user settings a similar head start. The cached promise is joined in + // installPluginsAndApplyMcpInBackground before plugin install reads + // enabledPlugins. + if ( + feature('DOWNLOAD_USER_SETTINGS') && + (isEnvTruthy(process.env.CLAUDE_CODE_REMOTE) || getIsRemoteMode()) + ) { + void downloadUserSettings() + } + + // In headless mode there is no React tree, so the useSettingsChange hook + // never runs. Subscribe directly so that settings changes (including + // managed-settings / policy updates) are fully applied. + settingsChangeDetector.subscribe(source => { + applySettingsChange(source, setAppState) + + // In headless mode, also sync the denormalized fastMode field from + // settings. The TUI manages fastMode via the UI so it skips this. + if (isFastModeEnabled()) { + setAppState(prev => { + const s = prev.settings as Record + const fastMode = s.fastMode === true && !s.fastModePerSessionOptIn + return { ...prev, fastMode } + }) + } + }) + + // Proactive activation is now handled in main.tsx before getTools() so + // SleepTool passes isEnabled() filtering. This fallback covers the case + // where CLAUDE_CODE_PROACTIVE is set but main.tsx's check didn't fire + // (e.g. env was injected by the SDK transport after argv parsing). + if ( + (feature('PROACTIVE') || feature('KAIROS')) && + proactiveModule && + !proactiveModule.isProactiveActive() && + isEnvTruthy(process.env.CLAUDE_CODE_PROACTIVE) + ) { + proactiveModule.activateProactive('command') + } + + // Periodically force a full GC to keep memory usage in check + if (typeof Bun !== 'undefined') { + const gcTimer = setInterval(Bun.gc, 1000) + gcTimer.unref() + } + + // Start headless profiler for first turn + headlessProfilerStartTurn() + headlessProfilerCheckpoint('runHeadless_entry') + + // Check Grove requirements for non-interactive consumer subscribers + if (await isQualifiedForGrove()) { + await checkGroveForNonInteractive() + } + headlessProfilerCheckpoint('after_grove_check') + + // Initialize GrowthBook so feature flags take effect in headless mode. + // Without this, the disk cache is empty and all flags fall back to defaults. + void initializeGrowthBook() + + if (options.resumeSessionAt && !options.resume) { + process.stderr.write(`Error: --resume-session-at requires --resume\n`) + gracefulShutdownSync(1) + return + } + + if (options.rewindFiles && !options.resume) { + process.stderr.write(`Error: --rewind-files requires --resume\n`) + gracefulShutdownSync(1) + return + } + + if (options.rewindFiles && inputPrompt) { + process.stderr.write( + `Error: --rewind-files is a standalone operation and cannot be used with a prompt\n`, + ) + gracefulShutdownSync(1) + return + } + + const structuredIO = getStructuredIO(inputPrompt, options) + + // When emitting NDJSON for SDK clients, any stray write to stdout (debug + // prints, dependency console.log, library banners) breaks the client's + // line-by-line JSON parser. Install a guard that diverts non-JSON lines to + // stderr so the stream stays clean. Must run before the first + // structuredIO.write below. + if (options.outputFormat === 'stream-json') { + installStreamJsonStdoutGuard() + } + + // #34044: if user explicitly set sandbox.enabled=true but deps are missing, + // isSandboxingEnabled() returns false silently. Surface the reason so users + // know their security config isn't being enforced. + const sandboxUnavailableReason = SandboxManager.getSandboxUnavailableReason() + if (sandboxUnavailableReason) { + if (SandboxManager.isSandboxRequired()) { + process.stderr.write( + `\nError: sandbox required but unavailable: ${sandboxUnavailableReason}\n` + + ` sandbox.failIfUnavailable is set — refusing to start without a working sandbox.\n\n`, + ) + gracefulShutdownSync(1) + return + } + process.stderr.write( + `\n⚠ Sandbox disabled: ${sandboxUnavailableReason}\n` + + ` Commands will run WITHOUT sandboxing. Network and filesystem restrictions will NOT be enforced.\n\n`, + ) + } else if (SandboxManager.isSandboxingEnabled()) { + // Initialize sandbox with a callback that forwards network permission + // requests to the SDK host via the can_use_tool control_request protocol. + // This must happen after structuredIO is created so we can send requests. + try { + await SandboxManager.initialize(structuredIO.createSandboxAskCallback()) + } catch (err) { + process.stderr.write(`\n❌ Sandbox Error: ${errorMessage(err)}\n`) + gracefulShutdownSync(1, 'other') + return + } + } + + if (options.outputFormat === 'stream-json' && options.verbose) { + registerHookEventHandler(event => { + const message: StdoutMessage = (() => { + switch (event.type) { + case 'started': + return { + type: 'system' as const, + subtype: 'hook_started' as const, + hook_id: event.hookId, + hook_name: event.hookName, + hook_event: event.hookEvent, + uuid: randomUUID(), + session_id: getSessionId(), + } + case 'progress': + return { + type: 'system' as const, + subtype: 'hook_progress' as const, + hook_id: event.hookId, + hook_name: event.hookName, + hook_event: event.hookEvent, + stdout: event.stdout, + stderr: event.stderr, + output: event.output, + uuid: randomUUID(), + session_id: getSessionId(), + } + case 'response': + return { + type: 'system' as const, + subtype: 'hook_response' as const, + hook_id: event.hookId, + hook_name: event.hookName, + hook_event: event.hookEvent, + output: event.output, + stdout: event.stdout, + stderr: event.stderr, + exit_code: event.exitCode, + outcome: event.outcome, + uuid: randomUUID(), + session_id: getSessionId(), + } + } + })() + void structuredIO.write(message) + }) + } + + if (options.setupTrigger) { + await processSetupHooks(options.setupTrigger) + } + + headlessProfilerCheckpoint('before_loadInitialMessages') + const appState = getAppState() + const { + messages: initialMessages, + turnInterruptionState, + agentSetting: resumedAgentSetting, + } = await loadInitialMessages(setAppState, { + continue: options.continue, + teleport: options.teleport, + resume: options.resume, + resumeSessionAt: options.resumeSessionAt, + forkSession: options.forkSession, + outputFormat: options.outputFormat, + sessionStartHooksPromise: options.sessionStartHooksPromise, + restoredWorkerState: structuredIO.restoredWorkerState, + }) + + // SessionStart hooks can emit initialUserMessage — the first user turn for + // headless orchestrator sessions where stdin is empty and additionalContext + // alone (an attachment, not a turn) would leave the REPL with nothing to + // respond to. The hook promise is awaited inside loadInitialMessages, so the + // module-level pending value is set by the time we get here. + const hookInitialUserMessage = takeInitialUserMessage() + if (hookInitialUserMessage) { + structuredIO.prependUserMessage(hookInitialUserMessage) + } + + // Restore agent setting from the resumed session (if not overridden by current --agent flag + // or settings-based agent, which would already have set mainThreadAgentType in main.tsx) + if (!options.agent && !getMainThreadAgentType() && resumedAgentSetting) { + const { agentDefinition: restoredAgent } = restoreAgentFromSession( + resumedAgentSetting, + undefined, + { activeAgents: agents, allAgents: agents }, + ) + if (restoredAgent) { + setAppState(prev => ({ ...prev, agent: restoredAgent.agentType })) + // Apply the agent's system prompt for non-built-in agents (mirrors main.tsx initial --agent path) + if (!options.systemPrompt && !isBuiltInAgent(restoredAgent)) { + const agentSystemPrompt = restoredAgent.getSystemPrompt() + if (agentSystemPrompt) { + options.systemPrompt = agentSystemPrompt + } + } + // Re-persist agent setting so future resumes maintain the agent + saveAgentSetting(restoredAgent.agentType) + } + } + + // gracefulShutdownSync schedules an async shutdown and sets process.exitCode. + // If a loadInitialMessages error path triggered it, bail early to avoid + // unnecessary work while the process winds down. + if (initialMessages.length === 0 && process.exitCode !== undefined) { + return + } + + // Handle --rewind-files: restore filesystem and exit immediately + if (options.rewindFiles) { + // File history snapshots are only created for user messages, + // so we require the target to be a user message + const targetMessage = initialMessages.find( + m => m.uuid === options.rewindFiles, + ) + + if (!targetMessage || targetMessage.type !== 'user') { + process.stderr.write( + `Error: --rewind-files requires a user message UUID, but ${options.rewindFiles} is not a user message in this session\n`, + ) + gracefulShutdownSync(1) + return + } + + const currentAppState = getAppState() + const result = await handleRewindFiles( + options.rewindFiles as UUID, + currentAppState, + setAppState, + false, + ) + if (!result.canRewind) { + process.stderr.write(`Error: ${result.error || 'Unexpected error'}\n`) + gracefulShutdownSync(1) + return + } + + // Rewind complete - exit successfully + process.stdout.write( + `Files rewound to state at message ${options.rewindFiles}\n`, + ) + gracefulShutdownSync(0) + return + } + + // Check if we need input prompt - skip if we're resuming with a valid session ID/JSONL file or using SDK URL + const hasValidResumeSessionId = + typeof options.resume === 'string' && + (Boolean(validateUuid(options.resume)) || options.resume.endsWith('.jsonl')) + const isUsingSdkUrl = Boolean(options.sdkUrl) + + if (!inputPrompt && !hasValidResumeSessionId && !isUsingSdkUrl) { + process.stderr.write( + `Error: Input must be provided either through stdin or as a prompt argument when using --print\n`, + ) + gracefulShutdownSync(1) + return + } + + if (options.outputFormat === 'stream-json' && !options.verbose) { + process.stderr.write( + 'Error: When using --print, --output-format=stream-json requires --verbose\n', + ) + gracefulShutdownSync(1) + return + } + + // Filter out MCP tools that are in the deny list + const allowedMcpTools = filterToolsByDenyRules( + appState.mcp.tools, + appState.toolPermissionContext, + ) + let filteredTools = [...tools, ...allowedMcpTools] + + // When using SDK URL, always use stdio permission prompting to delegate to the SDK + const effectivePermissionPromptToolName = options.sdkUrl + ? 'stdio' + : options.permissionPromptToolName + + // Callback for when a permission prompt is shown + const onPermissionPrompt = (details: RequiresActionDetails) => { + if (feature('COMMIT_ATTRIBUTION')) { + setAppState(prev => ({ + ...prev, + attribution: { + ...prev.attribution, + permissionPromptCount: prev.attribution.permissionPromptCount + 1, + }, + })) + } + notifySessionStateChanged('requires_action', details) + } + + const canUseTool = getCanUseToolFn( + effectivePermissionPromptToolName, + structuredIO, + () => getAppState().mcp.tools, + onPermissionPrompt, + ) + if (options.permissionPromptToolName) { + // Remove the permission prompt tool from the list of available tools. + filteredTools = filteredTools.filter( + tool => !toolMatchesName(tool, options.permissionPromptToolName!), + ) + } + + // Install errors handlers to gracefully handle broken pipes (e.g., when parent process dies) + registerProcessOutputErrorHandlers() + + headlessProfilerCheckpoint('after_loadInitialMessages') + + // Ensure model strings are initialized before generating model options. + // For Bedrock users, this waits for the profile fetch to get correct region strings. + await ensureModelStringsInitialized() + headlessProfilerCheckpoint('after_modelStrings') + + // UDS inbox store registration is deferred until after `run` is defined + // so we can pass `run` as the onEnqueue callback (see below). + + // Only `json` + `verbose` needs the full array (jsonStringify(messages) below). + // For stream-json (SDK/CCR) and default text output, only the last message is + // read for the exit code / final result. Avoid accumulating every message in + // memory for the entire session. + const needsFullArray = options.outputFormat === 'json' && options.verbose + const messages: SDKMessage[] = [] + let lastMessage: SDKMessage | undefined + // Streamlined mode transforms messages when CLAUDE_CODE_STREAMLINED_OUTPUT=true and using stream-json + // Build flag gates this out of external builds; env var is the runtime opt-in for ant builds + const transformToStreamlined = + feature('STREAMLINED_OUTPUT') && + isEnvTruthy(process.env.CLAUDE_CODE_STREAMLINED_OUTPUT) && + options.outputFormat === 'stream-json' + ? createStreamlinedTransformer() + : null + + headlessProfilerCheckpoint('before_runHeadlessStreaming') + for await (const message of runHeadlessStreaming( + structuredIO, + appState.mcp.clients, + [...commands, ...appState.mcp.commands], + filteredTools, + initialMessages, + canUseTool, + sdkMcpConfigs, + getAppState, + setAppState, + agents, + options, + turnInterruptionState, + )) { + if (transformToStreamlined) { + // Streamlined mode: transform messages and stream immediately + const transformed = transformToStreamlined(message) + if (transformed) { + await structuredIO.write(transformed) + } + } else if (options.outputFormat === 'stream-json' && options.verbose) { + await structuredIO.write(message) + } + // Should not be getting control messages or stream events in non-stream mode. + // Also filter out streamlined types since they're only produced by the transformer. + // SDK-only system events are excluded so lastMessage stays at the result + // (session_state_changed(idle) and any late task_notification drain after + // result in the finally block). + if ( + message.type !== 'control_response' && + message.type !== 'control_request' && + message.type !== 'control_cancel_request' && + !( + message.type === 'system' && + (message.subtype === 'session_state_changed' || + message.subtype === 'task_notification' || + message.subtype === 'task_started' || + message.subtype === 'task_progress' || + message.subtype === 'post_turn_summary') + ) && + message.type !== 'stream_event' && + message.type !== 'keep_alive' && + message.type !== 'streamlined_text' && + message.type !== 'streamlined_tool_use_summary' && + message.type !== 'prompt_suggestion' + ) { + if (needsFullArray) { + messages.push(message) + } + lastMessage = message + } + } + + switch (options.outputFormat) { + case 'json': + if (!lastMessage || lastMessage.type !== 'result') { + throw new Error('No messages returned') + } + if (options.verbose) { + writeToStdout(jsonStringify(messages) + '\n') + break + } + writeToStdout(jsonStringify(lastMessage) + '\n') + break + case 'stream-json': + // already logged above + break + default: + if (!lastMessage || lastMessage.type !== 'result') { + throw new Error('No messages returned') + } + switch (lastMessage.subtype) { + case 'success': + writeToStdout( + lastMessage.result.endsWith('\n') + ? lastMessage.result + : lastMessage.result + '\n', + ) + break + case 'error_during_execution': + writeToStdout(`Execution error`) + break + case 'error_max_turns': + writeToStdout(`Error: Reached max turns (${options.maxTurns})`) + break + case 'error_max_budget_usd': + writeToStdout(`Error: Exceeded USD budget (${options.maxBudgetUsd})`) + break + case 'error_max_structured_output_retries': + writeToStdout( + `Error: Failed to provide valid structured output after maximum retries`, + ) + } + } + + // Log headless latency metrics for the final turn + logHeadlessProfilerTurn() + + // Drain any in-flight memory extraction before shutdown. The response is + // already flushed above, so this adds no user-visible latency — it just + // delays process exit so gracefulShutdownSync's 5s failsafe doesn't kill + // the forked agent mid-flight. Gated by isExtractModeActive so the + // tengu_slate_thimble flag controls non-interactive extraction end-to-end. + if (feature('EXTRACT_MEMORIES') && isExtractModeActive()) { + await extractMemoriesModule!.drainPendingExtraction() + } + + gracefulShutdownSync( + lastMessage?.type === 'result' && lastMessage?.is_error ? 1 : 0, + ) +} + +function runHeadlessStreaming( + structuredIO: StructuredIO, + mcpClients: MCPServerConnection[], + commands: Command[], + tools: Tools, + initialMessages: Message[], + canUseTool: CanUseToolFn, + sdkMcpConfigs: Record, + getAppState: () => AppState, + setAppState: (f: (prev: AppState) => AppState) => void, + agents: AgentDefinition[], + options: { + verbose: boolean | undefined + jsonSchema: Record | undefined + permissionPromptToolName: string | undefined + allowedTools: string[] | undefined + thinkingConfig: ThinkingConfig | undefined + maxTurns: number | undefined + maxBudgetUsd: number | undefined + taskBudget: { total: number } | undefined + systemPrompt: string | undefined + appendSystemPrompt: string | undefined + userSpecifiedModel: string | undefined + fallbackModel: string | undefined + replayUserMessages?: boolean | undefined + includePartialMessages?: boolean | undefined + enableAuthStatus?: boolean | undefined + agent?: string | undefined + setSDKStatus?: (status: SDKStatus) => void + promptSuggestions?: boolean | undefined + workload?: string | undefined + }, + turnInterruptionState?: TurnInterruptionState, +): AsyncIterable { + let running = false + let runPhase: + | 'draining_commands' + | 'waiting_for_agents' + | 'finally_flush' + | 'finally_post_flush' + | undefined + let inputClosed = false + let shutdownPromptInjected = false + let heldBackResult: StdoutMessage | null = null + let abortController: AbortController | undefined + // Same queue sendRequest() enqueues to — one FIFO for everything. + const output = structuredIO.outbound + + // Ctrl+C in -p mode: abort the in-flight query, then shut down gracefully. + // gracefulShutdown persists session state and flushes analytics, with a + // failsafe timer that force-exits if cleanup hangs. + const sigintHandler = () => { + logForDiagnosticsNoPII('info', 'shutdown_signal', { signal: 'SIGINT' }) + if (abortController && !abortController.signal.aborted) { + abortController.abort() + } + void gracefulShutdown(0) + } + process.on('SIGINT', sigintHandler) + + // Dump run()'s state at SIGTERM so a stuck session's healthsweep can name + // the do/while(waitingForAgents) poll without reading the transcript. + registerCleanup(async () => { + const bg: Record = {} + for (const t of getRunningTasks(getAppState())) { + if (isBackgroundTask(t)) bg[t.type] = (bg[t.type] ?? 0) + 1 + } + logForDiagnosticsNoPII('info', 'run_state_at_shutdown', { + run_active: running, + run_phase: runPhase, + worker_status: getSessionState(), + internal_events_pending: structuredIO.internalEventsPending, + bg_tasks: bg, + }) + }) + + // Wire the central onChangeAppState mode-diff hook to the SDK output stream. + // This fires whenever ANY code path mutates toolPermissionContext.mode — + // Shift+Tab, ExitPlanMode dialog, /plan slash command, rewind, bridge + // set_permission_mode, the query loop, stop_task — rather than the two + // paths that previously went through a bespoke wrapper. + // The wrapper's body was fully redundant (it enqueued here AND called + // notifySessionMetadataChanged, both of which onChangeAppState now covers); + // keeping it would double-emit status messages. + setPermissionModeChangedListener(newMode => { + // Only emit for SDK-exposed modes. + if ( + newMode === 'default' || + newMode === 'acceptEdits' || + newMode === 'bypassPermissions' || + newMode === 'plan' || + newMode === (feature('TRANSCRIPT_CLASSIFIER') && 'auto') || + newMode === 'dontAsk' + ) { + output.enqueue({ + type: 'system', + subtype: 'status', + status: null, + permissionMode: newMode as PermissionMode, + uuid: randomUUID(), + session_id: getSessionId(), + }) + } + }) + + // Prompt suggestion tracking (push model) + const suggestionState: { + abortController: AbortController | null + inflightPromise: Promise | null + lastEmitted: { + text: string + emittedAt: number + promptId: PromptVariant + generationRequestId: string | null + } | null + pendingSuggestion: { + type: 'prompt_suggestion' + suggestion: string + uuid: UUID + session_id: string + } | null + pendingLastEmittedEntry: { + text: string + promptId: PromptVariant + generationRequestId: string | null + } | null + } = { + abortController: null, + inflightPromise: null, + lastEmitted: null, + pendingSuggestion: null, + pendingLastEmittedEntry: null, + } + + // Set up AWS auth status listener if enabled + let unsubscribeAuthStatus: (() => void) | undefined + if (options.enableAuthStatus) { + const authStatusManager = AwsAuthStatusManager.getInstance() + unsubscribeAuthStatus = authStatusManager.subscribe(status => { + output.enqueue({ + type: 'auth_status', + isAuthenticating: status.isAuthenticating, + output: status.output, + error: status.error, + uuid: randomUUID(), + session_id: getSessionId(), + }) + }) + } + + // Set up rate limit status listener to emit SDKRateLimitEvent for all status changes. + // Emitting for all statuses (including 'allowed') ensures consumers can clear warnings + // when rate limits reset. The upstream emitStatusChange already deduplicates via isEqual. + const rateLimitListener = (limits: ClaudeAILimits) => { + const rateLimitInfo = toSDKRateLimitInfo(limits) + if (rateLimitInfo) { + output.enqueue({ + type: 'rate_limit_event', + rate_limit_info: rateLimitInfo, + uuid: randomUUID(), + session_id: getSessionId(), + }) + } + } + statusListeners.add(rateLimitListener) + + // Messages for internal tracking, directly mutated by ask(). These messages + // include Assistant, User, Attachment, and Progress messages. + // TODO: Clean up this code to avoid passing around a mutable array. + const mutableMessages: Message[] = initialMessages + + // Seed the readFileState cache from the transcript (content the model saw, + // with message timestamps) so getChangedFiles can detect external edits. + // This cache instance must persist across ask() calls, since the edit tool + // relies on this as a global state. + let readFileState = extractReadFilesFromMessages( + initialMessages, + cwd(), + READ_FILE_STATE_CACHE_SIZE, + ) + + // Client-supplied readFileState seeds (via seed_read_state control request). + // The stdin IIFE runs concurrently with ask() — a seed arriving mid-turn + // would be lost to ask()'s clone-then-replace (QueryEngine.ts finally block) + // if written directly into readFileState. Instead, seeds land here, merge + // into getReadFileCache's view (readFileState-wins-ties: seeds fill gaps), + // and are re-applied then CLEARED in setReadFileCache. One-shot: each seed + // survives exactly one clone-replace cycle, then becomes a regular + // readFileState entry subject to compact's clear like everything else. + const pendingSeeds = createFileStateCacheWithSizeLimit( + READ_FILE_STATE_CACHE_SIZE, + ) + + // Auto-resume interrupted turns on restart so CC continues from where it + // left off without requiring the SDK to re-send the prompt. + const resumeInterruptedTurnEnv = + process.env.CLAUDE_CODE_RESUME_INTERRUPTED_TURN + if ( + turnInterruptionState && + turnInterruptionState.kind !== 'none' && + resumeInterruptedTurnEnv + ) { + logForDebugging( + `[print.ts] Auto-resuming interrupted turn (kind: ${turnInterruptionState.kind})`, + ) + + // Remove the interrupted message and its sentinel, then re-enqueue so + // the model sees it exactly once. For mid-turn interruptions, the + // deserialization layer transforms them into interrupted_prompt by + // appending a synthetic "Continue from where you left off." message. + removeInterruptedMessage(mutableMessages, turnInterruptionState.message) + enqueue({ + mode: 'prompt', + value: turnInterruptionState.message.message.content, + uuid: randomUUID(), + }) + } + + const modelOptions = getModelOptions() + const modelInfos = modelOptions.map(option => { + const modelId = option.value === null ? 'default' : option.value + const resolvedModel = + modelId === 'default' + ? getDefaultMainLoopModel() + : parseUserSpecifiedModel(modelId) + const hasEffort = modelSupportsEffort(resolvedModel) + const hasAdaptiveThinking = modelSupportsAdaptiveThinking(resolvedModel) + const hasFastMode = isFastModeSupportedByModel(option.value) + const hasAutoMode = modelSupportsAutoMode(resolvedModel) + return { + value: modelId, + displayName: option.label, + description: option.description, + ...(hasEffort && { + supportsEffort: true, + supportedEffortLevels: modelSupportsMaxEffort(resolvedModel) + ? [...EFFORT_LEVELS] + : EFFORT_LEVELS.filter(l => l !== 'max'), + }), + ...(hasAdaptiveThinking && { supportsAdaptiveThinking: true }), + ...(hasFastMode && { supportsFastMode: true }), + ...(hasAutoMode && { supportsAutoMode: true }), + } + }) + let activeUserSpecifiedModel = options.userSpecifiedModel + + function injectModelSwitchBreadcrumbs( + modelArg: string, + resolvedModel: string, + ): void { + const breadcrumbs = createModelSwitchBreadcrumbs( + modelArg, + modelDisplayString(resolvedModel), + ) + mutableMessages.push(...breadcrumbs) + for (const crumb of breadcrumbs) { + if ( + typeof crumb.message.content === 'string' && + crumb.message.content.includes(`<${LOCAL_COMMAND_STDOUT_TAG}>`) + ) { + output.enqueue({ + type: 'user', + message: crumb.message, + session_id: getSessionId(), + parent_tool_use_id: null, + uuid: crumb.uuid, + timestamp: crumb.timestamp, + isReplay: true, + } satisfies SDKUserMessageReplay) + } + } + } + + // Cache SDK MCP clients to avoid reconnecting on each run + let sdkClients: MCPServerConnection[] = [] + let sdkTools: Tools = [] + + // Track which MCP clients have had elicitation handlers registered + const elicitationRegistered = new Set() + + /** + * Register elicitation request/completion handlers on connected MCP clients + * that haven't been registered yet. SDK MCP servers are excluded because they + * route through SdkControlClientTransport. Hooks run first (matching REPL + * behavior); if no hook responds, the request is forwarded to the SDK + * consumer via the control protocol. + */ + function registerElicitationHandlers(clients: MCPServerConnection[]): void { + for (const connection of clients) { + if ( + connection.type !== 'connected' || + elicitationRegistered.has(connection.name) + ) { + continue + } + // Skip SDK MCP servers — elicitation flows through SdkControlClientTransport + if (connection.config.type === 'sdk') { + continue + } + const serverName = connection.name + + // Wrapped in try/catch because setRequestHandler throws if the client wasn't + // created with elicitation capability declared (e.g., SDK-created clients). + try { + connection.client.setRequestHandler( + ElicitRequestSchema, + async (request, extra) => { + logMCPDebug( + serverName, + `Elicitation request received in print mode: ${jsonStringify(request)}`, + ) + + const mode = request.params.mode === 'url' ? 'url' : 'form' + + logEvent('tengu_mcp_elicitation_shown', { + mode: mode as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + + // Run elicitation hooks first — they can provide a response programmatically + const hookResponse = await runElicitationHooks( + serverName, + request.params, + extra.signal, + ) + if (hookResponse) { + logMCPDebug( + serverName, + `Elicitation resolved by hook: ${jsonStringify(hookResponse)}`, + ) + logEvent('tengu_mcp_elicitation_response', { + mode: mode as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + action: + hookResponse.action as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + return hookResponse + } + + // Delegate to SDK consumer via control protocol + const url = + 'url' in request.params + ? (request.params.url as string) + : undefined + const requestedSchema = + 'requestedSchema' in request.params + ? (request.params.requestedSchema as + | Record + | undefined) + : undefined + + const elicitationId = + 'elicitationId' in request.params + ? (request.params.elicitationId as string | undefined) + : undefined + + const rawResult = await structuredIO.handleElicitation( + serverName, + request.params.message, + requestedSchema, + extra.signal, + mode, + url, + elicitationId, + ) + + const result = await runElicitationResultHooks( + serverName, + rawResult, + extra.signal, + mode, + elicitationId, + ) + + logEvent('tengu_mcp_elicitation_response', { + mode: mode as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + action: + result.action as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + return result + }, + ) + + // Surface completion notifications to SDK consumers (URL mode) + connection.client.setNotificationHandler( + ElicitationCompleteNotificationSchema, + notification => { + const { elicitationId } = notification.params + logMCPDebug( + serverName, + `Elicitation completion notification: ${elicitationId}`, + ) + void executeNotificationHooks({ + message: `MCP server "${serverName}" confirmed elicitation ${elicitationId} complete`, + notificationType: 'elicitation_complete', + }) + output.enqueue({ + type: 'system', + subtype: 'elicitation_complete', + mcp_server_name: serverName, + elicitation_id: elicitationId, + uuid: randomUUID(), + session_id: getSessionId(), + }) + }, + ) + + elicitationRegistered.add(serverName) + } catch { + // setRequestHandler throws if the client wasn't created with + // elicitation capability — skip silently + } + } + } + + async function updateSdkMcp() { + // Check if SDK MCP servers need to be updated (new servers added or removed) + const currentServerNames = new Set(Object.keys(sdkMcpConfigs)) + const connectedServerNames = new Set(sdkClients.map(c => c.name)) + + // Check if there are any differences (additions or removals) + const hasNewServers = Array.from(currentServerNames).some( + name => !connectedServerNames.has(name), + ) + const hasRemovedServers = Array.from(connectedServerNames).some( + name => !currentServerNames.has(name), + ) + // Check if any SDK clients are pending and need to be upgraded + const hasPendingSdkClients = sdkClients.some(c => c.type === 'pending') + // Check if any SDK clients failed their handshake and need to be retried. + // Without this, a client that lands in 'failed' (e.g. handshake timeout on + // a WS reconnect race) stays failed forever — its name satisfies the + // connectedServerNames diff but it contributes zero tools. + const hasFailedSdkClients = sdkClients.some(c => c.type === 'failed') + + const haveServersChanged = + hasNewServers || + hasRemovedServers || + hasPendingSdkClients || + hasFailedSdkClients + + if (haveServersChanged) { + // Clean up removed servers + for (const client of sdkClients) { + if (!currentServerNames.has(client.name)) { + if (client.type === 'connected') { + await client.cleanup() + } + } + } + + // Re-initialize all SDK MCP servers with current config + const sdkSetup = await setupSdkMcpClients( + sdkMcpConfigs, + (serverName, message) => + structuredIO.sendMcpMessage(serverName, message), + ) + sdkClients = sdkSetup.clients + sdkTools = sdkSetup.tools + + // Store SDK MCP tools in appState so subagents can access them via + // assembleToolPool. Only tools are stored here — SDK clients are already + // merged separately in the query loop (allMcpClients) and mcp_status handler. + // Use both old (connectedServerNames) and new (currentServerNames) to remove + // stale SDK tools when servers are added or removed. + const allSdkNames = uniq([...connectedServerNames, ...currentServerNames]) + setAppState(prev => ({ + ...prev, + mcp: { + ...prev.mcp, + tools: [ + ...prev.mcp.tools.filter( + t => + !allSdkNames.some(name => + t.name.startsWith(getMcpPrefix(name)), + ), + ), + ...sdkTools, + ], + }, + })) + + // Set up the special internal VSCode MCP server if necessary. + setupVscodeSdkMcp(sdkClients) + } + } + + void updateSdkMcp() + + // State for dynamically added MCP servers (via mcp_set_servers control message) + // These are separate from SDK MCP servers and support all transport types + let dynamicMcpState: DynamicMcpState = { + clients: [], + tools: [], + configs: {}, + } + + // Shared tool assembly for ask() and the get_context_usage control request. + // Closes over the mutable sdkTools/dynamicMcpState bindings so both call + // sites see late-connecting servers. + const buildAllTools = (appState: AppState): Tools => { + const assembledTools = assembleToolPool( + appState.toolPermissionContext, + appState.mcp.tools, + ) + let allTools = uniqBy( + mergeAndFilterTools( + [...tools, ...sdkTools, ...dynamicMcpState.tools], + assembledTools, + appState.toolPermissionContext.mode, + ), + 'name', + ) + if (options.permissionPromptToolName) { + allTools = allTools.filter( + tool => !toolMatchesName(tool, options.permissionPromptToolName!), + ) + } + const initJsonSchema = getInitJsonSchema() + if (initJsonSchema && !options.jsonSchema) { + const syntheticOutputResult = createSyntheticOutputTool(initJsonSchema) + if ('tool' in syntheticOutputResult) { + allTools = [...allTools, syntheticOutputResult.tool] + } + } + return allTools + } + + // Bridge handle for remote-control (SDK control message). + // Mirrors the REPL's useReplBridge hook: the handle is created when + // `remote_control` is enabled and torn down when disabled. + let bridgeHandle: ReplBridgeHandle | null = null + // Cursor into mutableMessages — tracks how far we've forwarded. + // Same index-based diff as useReplBridge's lastWrittenIndexRef. + let bridgeLastForwardedIndex = 0 + + // Forward new messages from mutableMessages to the bridge. + // Called incrementally during each turn (so claude.ai sees progress + // and stays alive during permission waits) and again after the turn. + // + // writeMessages has its own UUID-based dedup (initialMessageUUIDs, + // recentPostedUUIDs) — the index cursor here is a pre-filter to avoid + // O(n) re-scanning of already-sent messages on every call. + function forwardMessagesToBridge(): void { + if (!bridgeHandle) return + // Guard against mutableMessages shrinking (compaction truncates it). + const startIndex = Math.min( + bridgeLastForwardedIndex, + mutableMessages.length, + ) + const newMessages = mutableMessages + .slice(startIndex) + .filter(m => m.type === 'user' || m.type === 'assistant') + bridgeLastForwardedIndex = mutableMessages.length + if (newMessages.length > 0) { + bridgeHandle.writeMessages(newMessages) + } + } + + // Helper to apply MCP server changes - used by both mcp_set_servers control message + // and background plugin installation. + // NOTE: Nested function required - mutates closure state (sdkMcpConfigs, sdkClients, etc.) + let mcpChangesPromise: Promise<{ + response: SDKControlMcpSetServersResponse + sdkServersChanged: boolean + }> = Promise.resolve({ + response: { + added: [] as string[], + removed: [] as string[], + errors: {} as Record, + }, + sdkServersChanged: false, + }) + + function applyMcpServerChanges( + servers: Record, + ): Promise<{ + response: SDKControlMcpSetServersResponse + sdkServersChanged: boolean + }> { + // Serialize calls to prevent race conditions between concurrent callers + // (background plugin install and mcp_set_servers control messages) + const doWork = async (): Promise<{ + response: SDKControlMcpSetServersResponse + sdkServersChanged: boolean + }> => { + const oldSdkClientNames = new Set(sdkClients.map(c => c.name)) + + const result = await handleMcpSetServers( + servers, + { configs: sdkMcpConfigs, clients: sdkClients, tools: sdkTools }, + dynamicMcpState, + setAppState, + ) + + // Update SDK state (need to mutate sdkMcpConfigs since it's shared) + for (const key of Object.keys(sdkMcpConfigs)) { + delete sdkMcpConfigs[key] + } + Object.assign(sdkMcpConfigs, result.newSdkState.configs) + sdkClients = result.newSdkState.clients + sdkTools = result.newSdkState.tools + dynamicMcpState = result.newDynamicState + + // Keep appState.mcp.tools in sync so subagents can see SDK MCP tools. + // Use both old and new SDK client names to remove stale tools. + if (result.sdkServersChanged) { + const newSdkClientNames = new Set(sdkClients.map(c => c.name)) + const allSdkNames = uniq([...oldSdkClientNames, ...newSdkClientNames]) + setAppState(prev => ({ + ...prev, + mcp: { + ...prev.mcp, + tools: [ + ...prev.mcp.tools.filter( + t => + !allSdkNames.some(name => + t.name.startsWith(getMcpPrefix(name)), + ), + ), + ...sdkTools, + ], + }, + })) + } + + return { + response: result.response, + sdkServersChanged: result.sdkServersChanged, + } + } + + mcpChangesPromise = mcpChangesPromise.then(doWork, doWork) + return mcpChangesPromise + } + + // Build McpServerStatus[] for control responses. Shared by mcp_status and + // reload_plugins handlers. Reads closure state: sdkClients, dynamicMcpState. + function buildMcpServerStatuses(): McpServerStatus[] { + const currentAppState = getAppState() + const currentMcpClients = currentAppState.mcp.clients + const allMcpTools = uniqBy( + [...currentAppState.mcp.tools, ...dynamicMcpState.tools], + 'name', + ) + const existingNames = new Set([ + ...currentMcpClients.map(c => c.name), + ...sdkClients.map(c => c.name), + ]) + return [ + ...currentMcpClients, + ...sdkClients, + ...dynamicMcpState.clients.filter(c => !existingNames.has(c.name)), + ].map(connection => { + let config + if ( + connection.config.type === 'sse' || + connection.config.type === 'http' + ) { + config = { + type: connection.config.type, + url: connection.config.url, + headers: connection.config.headers, + oauth: connection.config.oauth, + } + } else if (connection.config.type === 'claudeai-proxy') { + config = { + type: 'claudeai-proxy' as const, + url: connection.config.url, + id: connection.config.id, + } + } else if ( + connection.config.type === 'stdio' || + connection.config.type === undefined + ) { + config = { + type: 'stdio' as const, + command: connection.config.command, + args: connection.config.args, + } + } + const serverTools = + connection.type === 'connected' + ? filterToolsByServer(allMcpTools, connection.name).map(tool => ({ + name: tool.mcpInfo?.toolName ?? tool.name, + annotations: { + readOnly: tool.isReadOnly({}) || undefined, + destructive: tool.isDestructive?.({}) || undefined, + openWorld: tool.isOpenWorld?.({}) || undefined, + }, + })) + : undefined + // Capabilities passthrough with allowlist pre-filter. The IDE reads + // experimental['claude/channel'] to decide whether to show the + // Enable-channel prompt — only echo it if channel_enable would + // actually pass the allowlist. Not a security boundary (the + // handler re-runs the full gate); just avoids dead buttons. + let capabilities: { experimental?: Record } | undefined + if ( + (feature('KAIROS') || feature('KAIROS_CHANNELS')) && + connection.type === 'connected' && + connection.capabilities.experimental + ) { + const exp = { ...connection.capabilities.experimental } + if ( + exp['claude/channel'] && + (!isChannelsEnabled() || + !isChannelAllowlisted(connection.config.pluginSource)) + ) { + delete exp['claude/channel'] + } + if (Object.keys(exp).length > 0) { + capabilities = { experimental: exp } + } + } + return { + name: connection.name, + status: connection.type, + serverInfo: + connection.type === 'connected' ? connection.serverInfo : undefined, + error: connection.type === 'failed' ? connection.error : undefined, + config, + scope: connection.config.scope, + tools: serverTools, + capabilities, + } + }) + } + + // NOTE: Nested function required - needs closure access to applyMcpServerChanges and updateSdkMcp + async function installPluginsAndApplyMcpInBackground(): Promise { + try { + // Join point for user settings (fired at runHeadless entry) and managed + // settings (fired in main.tsx preAction). downloadUserSettings() caches + // its promise so this awaits the same in-flight request. + await Promise.all([ + feature('DOWNLOAD_USER_SETTINGS') && + (isEnvTruthy(process.env.CLAUDE_CODE_REMOTE) || getIsRemoteMode()) + ? withDiagnosticsTiming('headless_user_settings_download', () => + downloadUserSettings(), + ) + : Promise.resolve(), + withDiagnosticsTiming('headless_managed_settings_wait', () => + waitForRemoteManagedSettingsToLoad(), + ), + ]) + + const pluginsInstalled = await installPluginsForHeadless() + + if (pluginsInstalled) { + await applyPluginMcpDiff() + } + } catch (error) { + logError(error) + } + } + + // Background plugin installation for all headless users + // Installs marketplaces from extraKnownMarketplaces and missing enabled plugins + // CLAUDE_CODE_SYNC_PLUGIN_INSTALL=true: resolved in run() before the first + // query so plugins are guaranteed available on the first ask(). + let pluginInstallPromise: Promise | null = null + // --bare / SIMPLE: skip plugin install. Scripted calls don't add plugins + // mid-session; the next interactive run reconciles. + if (!isBareMode()) { + if (isEnvTruthy(process.env.CLAUDE_CODE_SYNC_PLUGIN_INSTALL)) { + pluginInstallPromise = installPluginsAndApplyMcpInBackground() + } else { + void installPluginsAndApplyMcpInBackground() + } + } + + // Idle timeout management + const idleTimeout = createIdleTimeoutManager(() => !running) + + // Mutable commands and agents for hot reloading + let currentCommands = commands + let currentAgents = agents + + // Clear all plugin-related caches, reload commands/agents/hooks. + // Called after CLAUDE_CODE_SYNC_PLUGIN_INSTALL completes (before first query) + // and after non-sync background install finishes. + // refreshActivePlugins calls clearAllCaches() which is required because + // loadAllPlugins() may have run during main.tsx startup BEFORE managed + // settings were fetched. Without clearing, getCommands() would rebuild + // from a stale plugin list. + async function refreshPluginState(): Promise { + // refreshActivePlugins handles the full cache sweep (clearAllCaches), + // reloads all plugin component loaders, writes AppState.plugins + + // AppState.agentDefinitions, registers hooks, and bumps mcp.pluginReconnectKey. + const { agentDefinitions: freshAgentDefs } = + await refreshActivePlugins(setAppState) + + // Headless-specific: currentCommands/currentAgents are local mutable refs + // captured by the query loop (REPL uses AppState instead). getCommands is + // fresh because refreshActivePlugins cleared its cache. + currentCommands = await getCommands(cwd()) + + // Preserve SDK-provided agents (--agents CLI flag or SDK initialize + // control_request) — both inject via parseAgentsFromJson with + // source='flagSettings'. loadMarkdownFilesForSubdir never assigns this + // source, so it cleanly discriminates "injected, not disk-loadable". + // + // The previous filter used a negative set-diff (!freshAgentTypes.has(a)) + // which also matched plugin agents that were in the poisoned initial + // currentAgents but correctly excluded from freshAgentDefs after managed + // settings applied — leaking policy-blocked agents into the init message. + // See gh-23085: isBridgeEnabled() at Commander-definition time poisoned + // the settings cache before setEligibility(true) ran. + const sdkAgents = currentAgents.filter(a => a.source === 'flagSettings') + currentAgents = [...freshAgentDefs.allAgents, ...sdkAgents] + } + + // Re-diff MCP configs after plugin state changes. Filters to + // process-transport-supported types and carries SDK-mode servers through + // so applyMcpServerChanges' diff doesn't close their transports. + // Nested: needs closure access to sdkMcpConfigs, applyMcpServerChanges, + // updateSdkMcp. + async function applyPluginMcpDiff(): Promise { + const { servers: newConfigs } = await getAllMcpConfigs() + const supportedConfigs: Record = + {} + for (const [name, config] of Object.entries(newConfigs)) { + const type = config.type + if ( + type === undefined || + type === 'stdio' || + type === 'sse' || + type === 'http' || + type === 'sdk' + ) { + supportedConfigs[name] = config + } + } + for (const [name, config] of Object.entries(sdkMcpConfigs)) { + if (config.type === 'sdk' && !(name in supportedConfigs)) { + supportedConfigs[name] = config + } + } + const { response, sdkServersChanged } = + await applyMcpServerChanges(supportedConfigs) + if (sdkServersChanged) { + void updateSdkMcp() + } + logForDebugging( + `Headless MCP refresh: added=${response.added.length}, removed=${response.removed.length}`, + ) + } + + // Subscribe to skill changes for hot reloading + const unsubscribeSkillChanges = skillChangeDetector.subscribe(() => { + clearCommandsCache() + void getCommands(cwd()).then(newCommands => { + currentCommands = newCommands + }) + }) + + // Proactive mode: schedule a tick to keep the model looping autonomously. + // setTimeout(0) yields to the event loop so pending stdin messages + // (interrupts, user messages) are processed before the tick fires. + const scheduleProactiveTick = + feature('PROACTIVE') || feature('KAIROS') + ? () => { + setTimeout(() => { + if ( + !proactiveModule?.isProactiveActive() || + proactiveModule.isProactivePaused() || + inputClosed + ) { + return + } + const tickContent = `<${TICK_TAG}>${new Date().toLocaleTimeString()}` + enqueue({ + mode: 'prompt' as const, + value: tickContent, + uuid: randomUUID(), + priority: 'later', + isMeta: true, + }) + void run() + }, 0) + } + : undefined + + // Abort the current operation when a 'now' priority message arrives. + subscribeToCommandQueue(() => { + if (abortController && getCommandsByMaxPriority('now').length > 0) { + abortController.abort('interrupt') + } + }) + + const run = async () => { + if (running) { + return + } + + running = true + runPhase = undefined + notifySessionStateChanged('running') + idleTimeout.stop() + + headlessProfilerCheckpoint('run_entry') + // TODO(custom-tool-refactor): Should move to the init message, like browser + + await updateSdkMcp() + headlessProfilerCheckpoint('after_updateSdkMcp') + + // Resolve deferred plugin installation (CLAUDE_CODE_SYNC_PLUGIN_INSTALL). + // The promise was started eagerly so installation overlaps with other init. + // Awaiting here guarantees plugins are available before the first ask(). + // If CLAUDE_CODE_SYNC_PLUGIN_INSTALL_TIMEOUT_MS is set, races against that + // deadline and proceeds without plugins on timeout (logging an error). + if (pluginInstallPromise) { + const timeoutMs = parseInt( + process.env.CLAUDE_CODE_SYNC_PLUGIN_INSTALL_TIMEOUT_MS || '', + 10, + ) + if (timeoutMs > 0) { + const timeout = sleep(timeoutMs).then(() => 'timeout' as const) + const result = await Promise.race([pluginInstallPromise, timeout]) + if (result === 'timeout') { + logError( + new Error( + `CLAUDE_CODE_SYNC_PLUGIN_INSTALL: plugin installation timed out after ${timeoutMs}ms`, + ), + ) + logEvent('tengu_sync_plugin_install_timeout', { + timeout_ms: timeoutMs, + }) + } + } else { + await pluginInstallPromise + } + pluginInstallPromise = null + + // Refresh commands, agents, and hooks now that plugins are installed + await refreshPluginState() + + // Set up hot-reload for plugin hooks now that the initial install is done. + // In sync-install mode, setup.ts skips this to avoid racing with the install. + const { setupPluginHookHotReload } = await import( + '../utils/plugins/loadPluginHooks.js' + ) + setupPluginHookHotReload() + } + + // Only main-thread commands (agentId===undefined) — subagent + // notifications are drained by the subagent's mid-turn gate in query.ts. + // Defined outside the try block so it's accessible in the post-finally + // queue re-checks at the bottom of run(). + const isMainThread = (cmd: QueuedCommand) => cmd.agentId === undefined + + try { + let command: QueuedCommand | undefined + let waitingForAgents = false + + // Extract command processing into a named function for the do-while pattern. + // Drains the queue, batching consecutive prompt-mode commands into one + // ask() call so messages that queued up during a long turn coalesce + // into a single follow-up turn instead of N separate turns. + const drainCommandQueue = async () => { + while ((command = dequeue(isMainThread))) { + if ( + command.mode !== 'prompt' && + command.mode !== 'orphaned-permission' && + command.mode !== 'task-notification' + ) { + throw new Error( + 'only prompt commands are supported in streaming mode', + ) + } + + // Non-prompt commands (task-notification, orphaned-permission) carry + // side effects or orphanedPermission state, so they process singly. + // Prompt commands greedily collect followers with matching workload. + const batch: QueuedCommand[] = [command] + if (command.mode === 'prompt') { + while (canBatchWith(command, peek(isMainThread))) { + batch.push(dequeue(isMainThread)!) + } + if (batch.length > 1) { + command = { + ...command, + value: joinPromptValues(batch.map(c => c.value)), + uuid: batch.findLast(c => c.uuid)?.uuid ?? command.uuid, + } + } + } + const batchUuids = batch.map(c => c.uuid).filter(u => u !== undefined) + + // QueryEngine will emit a replay for command.uuid (the last uuid in + // the batch) via its messagesToAck path. Emit replays here for the + // rest so consumers that track per-uuid delivery (clank's + // asyncMessages footer, CCR) see an ack for every message they sent, + // not just the one that survived the merge. + if (options.replayUserMessages && batch.length > 1) { + for (const c of batch) { + if (c.uuid && c.uuid !== command.uuid) { + output.enqueue({ + type: 'user', + message: { role: 'user', content: c.value }, + session_id: getSessionId(), + parent_tool_use_id: null, + uuid: c.uuid, + isReplay: true, + } satisfies SDKUserMessageReplay) + } + } + } + + // Combine all MCP clients. appState.mcp is populated incrementally + // per-server by main.tsx (mirrors useManageMCPConnections). Reading + // fresh per-command means late-connecting servers are visible on the + // next turn. registerElicitationHandlers is idempotent (tracking set). + const appState = getAppState() + const allMcpClients = [ + ...appState.mcp.clients, + ...sdkClients, + ...dynamicMcpState.clients, + ] + registerElicitationHandlers(allMcpClients) + // Channel handlers for servers allowlisted via --channels at + // construction time (or enableChannel() mid-session). Runs every + // turn like registerElicitationHandlers — idempotent per-client + // (setNotificationHandler replaces, not stacks) and no-ops for + // non-allowlisted servers (one feature-flag check). + for (const client of allMcpClients) { + reregisterChannelHandlerAfterReconnect(client) + } + + const allTools = buildAllTools(appState) + + for (const uuid of batchUuids) { + notifyCommandLifecycle(uuid, 'started') + } + + // Task notifications arrive when background agents complete. + // Emit an SDK system event for SDK consumers, then fall through + // to ask() so the model sees the agent result and can act on it. + // This matches TUI behavior where useQueueProcessor always feeds + // notifications to the model regardless of coordinator mode. + if (command.mode === 'task-notification') { + const notificationText = + typeof command.value === 'string' ? command.value : '' + // Parse the XML-formatted notification + const taskIdMatch = notificationText.match( + /([^<]+)<\/task-id>/, + ) + const toolUseIdMatch = notificationText.match( + /([^<]+)<\/tool-use-id>/, + ) + const outputFileMatch = notificationText.match( + /([^<]+)<\/output-file>/, + ) + const statusMatch = notificationText.match( + /([^<]+)<\/status>/, + ) + const summaryMatch = notificationText.match( + /

([^<]+)<\/summary>/, + ) + + const isValidStatus = ( + s: string | undefined, + ): s is 'completed' | 'failed' | 'stopped' | 'killed' => + s === 'completed' || + s === 'failed' || + s === 'stopped' || + s === 'killed' + const rawStatus = statusMatch?.[1] + const status = isValidStatus(rawStatus) + ? rawStatus === 'killed' + ? 'stopped' + : rawStatus + : 'completed' + + const usageMatch = notificationText.match( + /([\s\S]*?)<\/usage>/, + ) + const usageContent = usageMatch?.[1] ?? '' + const totalTokensMatch = usageContent.match( + /(\d+)<\/total_tokens>/, + ) + const toolUsesMatch = usageContent.match( + /(\d+)<\/tool_uses>/, + ) + const durationMsMatch = usageContent.match( + /(\d+)<\/duration_ms>/, + ) + + // Only emit a task_notification SDK event when a tag is + // present — that means this is a terminal notification (completed/ + // failed/stopped). Stream events from enqueueStreamEvent carry no + // (they're progress pings); emitting them here would + // default to 'completed' and falsely close the task for SDK + // consumers. Terminal bookends are now emitted directly via + // emitTaskTerminatedSdk, so skipping statusless events is safe. + if (statusMatch) { + output.enqueue({ + type: 'system', + subtype: 'task_notification', + task_id: taskIdMatch?.[1] ?? '', + tool_use_id: toolUseIdMatch?.[1], + status, + output_file: outputFileMatch?.[1] ?? '', + summary: summaryMatch?.[1] ?? '', + usage: + totalTokensMatch && toolUsesMatch + ? { + total_tokens: parseInt(totalTokensMatch[1]!, 10), + tool_uses: parseInt(toolUsesMatch[1]!, 10), + duration_ms: durationMsMatch + ? parseInt(durationMsMatch[1]!, 10) + : 0, + } + : undefined, + session_id: getSessionId(), + uuid: randomUUID(), + }) + } + // No continue -- fall through to ask() so the model processes the result + } + + const input = command.value + + if (structuredIO instanceof RemoteIO && command.mode === 'prompt') { + logEvent('tengu_bridge_message_received', { + is_repl: false, + }) + } + + // Abort any in-flight suggestion generation and track acceptance + suggestionState.abortController?.abort() + suggestionState.abortController = null + suggestionState.pendingSuggestion = null + suggestionState.pendingLastEmittedEntry = null + if (suggestionState.lastEmitted) { + if (command.mode === 'prompt') { + // SDK user messages enqueue ContentBlockParam[], not a plain string + const inputText = + typeof input === 'string' + ? input + : ( + input.find(b => b.type === 'text') as + | { type: 'text'; text: string } + | undefined + )?.text + if (typeof inputText === 'string') { + logSuggestionOutcome( + suggestionState.lastEmitted.text, + inputText, + suggestionState.lastEmitted.emittedAt, + suggestionState.lastEmitted.promptId, + suggestionState.lastEmitted.generationRequestId, + ) + } + suggestionState.lastEmitted = null + } + } + + abortController = createAbortController() + const turnStartTime = feature('FILE_PERSISTENCE') + ? Date.now() + : undefined + + headlessProfilerCheckpoint('before_ask') + startQueryProfile() + // Per-iteration ALS context so bg agents spawned inside ask() + // inherit workload across their detached awaits. In-process cron + // stamps cmd.workload; the SDK --workload flag is options.workload. + // const-capture: TS loses `while ((command = dequeue()))` narrowing + // inside the closure. + const cmd = command + await runWithWorkload(cmd.workload ?? options.workload, async () => { + for await (const message of ask({ + commands: uniqBy( + [...currentCommands, ...appState.mcp.commands], + 'name', + ), + prompt: input, + promptUuid: cmd.uuid, + isMeta: cmd.isMeta, + cwd: cwd(), + tools: allTools, + verbose: options.verbose, + mcpClients: allMcpClients, + thinkingConfig: options.thinkingConfig, + maxTurns: options.maxTurns, + maxBudgetUsd: options.maxBudgetUsd, + taskBudget: options.taskBudget, + canUseTool, + userSpecifiedModel: activeUserSpecifiedModel, + fallbackModel: options.fallbackModel, + jsonSchema: getInitJsonSchema() ?? options.jsonSchema, + mutableMessages, + getReadFileCache: () => + pendingSeeds.size === 0 + ? readFileState + : mergeFileStateCaches(readFileState, pendingSeeds), + setReadFileCache: cache => { + readFileState = cache + for (const [path, seed] of pendingSeeds.entries()) { + const existing = readFileState.get(path) + if (!existing || seed.timestamp > existing.timestamp) { + readFileState.set(path, seed) + } + } + pendingSeeds.clear() + }, + customSystemPrompt: options.systemPrompt, + appendSystemPrompt: options.appendSystemPrompt, + getAppState, + setAppState, + abortController, + replayUserMessages: options.replayUserMessages, + includePartialMessages: options.includePartialMessages, + handleElicitation: (serverName, params, elicitSignal) => + structuredIO.handleElicitation( + serverName, + params.message, + undefined, + elicitSignal, + params.mode, + params.url, + 'elicitationId' in params ? params.elicitationId : undefined, + ), + agents: currentAgents, + orphanedPermission: cmd.orphanedPermission, + setSDKStatus: status => { + output.enqueue({ + type: 'system', + subtype: 'status', + status, + session_id: getSessionId(), + uuid: randomUUID(), + }) + }, + })) { + // Forward messages to bridge incrementally (mid-turn) so + // claude.ai sees progress and the connection stays alive + // while blocked on permission requests. + forwardMessagesToBridge() + + if (message.type === 'result') { + // Flush pending SDK events so they appear before result on the stream. + for (const event of drainSdkEvents()) { + output.enqueue(event) + } + + // Hold-back: don't emit result while background agents are running + const currentState = getAppState() + if ( + getRunningTasks(currentState).some( + t => + (t.type === 'local_agent' || + t.type === 'local_workflow') && + isBackgroundTask(t), + ) + ) { + heldBackResult = message + } else { + heldBackResult = null + output.enqueue(message) + } + } else { + // Flush SDK events (task_started, task_progress) so background + // agent progress is streamed in real-time, not batched until result. + for (const event of drainSdkEvents()) { + output.enqueue(event) + } + output.enqueue(message) + } + } + }) // end runWithWorkload + + for (const uuid of batchUuids) { + notifyCommandLifecycle(uuid, 'completed') + } + + // Forward messages to bridge after each turn + forwardMessagesToBridge() + bridgeHandle?.sendResult() + + if (feature('FILE_PERSISTENCE') && turnStartTime !== undefined) { + void executeFilePersistence( + turnStartTime, + abortController.signal, + result => { + output.enqueue({ + type: 'system' as const, + subtype: 'files_persisted' as const, + files: result.files, + failed: result.failed, + processed_at: new Date().toISOString(), + uuid: randomUUID(), + session_id: getSessionId(), + }) + }, + ) + } + + // Generate and emit prompt suggestion for SDK consumers + if ( + options.promptSuggestions && + !isEnvDefinedFalsy(process.env.CLAUDE_CODE_ENABLE_PROMPT_SUGGESTION) + ) { + // TS narrows suggestionState to never in the while loop body; + // cast via unknown to reset narrowing. + const state = suggestionState as unknown as typeof suggestionState + state.abortController?.abort() + const localAbort = new AbortController() + suggestionState.abortController = localAbort + + const cacheSafeParams = getLastCacheSafeParams() + if (!cacheSafeParams) { + logSuggestionSuppressed( + 'sdk_no_params', + undefined, + undefined, + 'sdk', + ) + } else { + // Use a ref object so the IIFE's finally can compare against its own + // promise without a self-reference (which upsets TypeScript's flow analysis). + const ref: { promise: Promise | null } = { promise: null } + ref.promise = (async () => { + try { + const result = await tryGenerateSuggestion( + localAbort, + mutableMessages, + getAppState, + cacheSafeParams, + 'sdk', + ) + if (!result || localAbort.signal.aborted) return + const suggestionMsg = { + type: 'prompt_suggestion' as const, + suggestion: result.suggestion, + uuid: randomUUID(), + session_id: getSessionId(), + } + const lastEmittedEntry = { + text: result.suggestion, + emittedAt: Date.now(), + promptId: result.promptId, + generationRequestId: result.generationRequestId, + } + // Defer emission if the result is being held for background agents, + // so that prompt_suggestion always arrives after result. + // Only set lastEmitted when the suggestion is actually delivered + // to the consumer; deferred suggestions may be discarded before + // delivery if a new command arrives first. + if (heldBackResult) { + suggestionState.pendingSuggestion = suggestionMsg + suggestionState.pendingLastEmittedEntry = { + text: lastEmittedEntry.text, + promptId: lastEmittedEntry.promptId, + generationRequestId: lastEmittedEntry.generationRequestId, + } + } else { + suggestionState.lastEmitted = lastEmittedEntry + output.enqueue(suggestionMsg) + } + } catch (error) { + if ( + error instanceof Error && + (error.name === 'AbortError' || + error.name === 'APIUserAbortError') + ) { + logSuggestionSuppressed( + 'aborted', + undefined, + undefined, + 'sdk', + ) + return + } + logError(toError(error)) + } finally { + if (suggestionState.inflightPromise === ref.promise) { + suggestionState.inflightPromise = null + } + } + })() + suggestionState.inflightPromise = ref.promise + } + } + + // Log headless profiler metrics for this turn and start next turn + logHeadlessProfilerTurn() + logQueryProfileReport() + headlessProfilerStartTurn() + } + } + + // Use a do-while loop to drain commands and then wait for any + // background agents that are still running. When agents complete, + // their notifications are enqueued and the loop re-drains. + do { + // Drain SDK events (task_started, task_progress) before command queue + // so progress events precede task_notification on the stream. + for (const event of drainSdkEvents()) { + output.enqueue(event) + } + + runPhase = 'draining_commands' + await drainCommandQueue() + + // Check for running background tasks before exiting. + // Exclude in_process_teammate — teammates are long-lived by design + // (status: 'running' for their whole lifetime, cleaned up by the + // shutdown protocol, not by transitioning to 'completed'). Waiting + // on them here loops forever (gh-30008). Same exclusion already + // exists at useBackgroundTaskNavigation.ts:55 for the same reason; + // L1839 above is already narrower (type === 'local_agent') so it + // doesn't hit this. + waitingForAgents = false + { + const state = getAppState() + const hasRunningBg = getRunningTasks(state).some( + t => isBackgroundTask(t) && t.type !== 'in_process_teammate', + ) + const hasMainThreadQueued = peek(isMainThread) !== undefined + if (hasRunningBg || hasMainThreadQueued) { + waitingForAgents = true + if (!hasMainThreadQueued) { + runPhase = 'waiting_for_agents' + // No commands ready yet, wait for tasks to complete + await sleep(100) + } + // Loop back to drain any newly queued commands + } + } + } while (waitingForAgents) + + if (heldBackResult) { + output.enqueue(heldBackResult) + heldBackResult = null + if (suggestionState.pendingSuggestion) { + output.enqueue(suggestionState.pendingSuggestion) + // Now that the suggestion is actually delivered, record it for acceptance tracking + if (suggestionState.pendingLastEmittedEntry) { + suggestionState.lastEmitted = { + ...suggestionState.pendingLastEmittedEntry, + emittedAt: Date.now(), + } + suggestionState.pendingLastEmittedEntry = null + } + suggestionState.pendingSuggestion = null + } + } + } catch (error) { + // Emit error result message before shutting down + // Write directly to structuredIO to ensure immediate delivery + try { + await structuredIO.write({ + type: 'result', + subtype: 'error_during_execution', + duration_ms: 0, + duration_api_ms: 0, + is_error: true, + num_turns: 0, + stop_reason: null, + session_id: getSessionId(), + total_cost_usd: 0, + usage: EMPTY_USAGE, + modelUsage: {}, + permission_denials: [], + uuid: randomUUID(), + errors: [ + errorMessage(error), + ...getInMemoryErrors().map(_ => _.error), + ], + }) + } catch { + // If we can't emit the error result, continue with shutdown anyway + } + suggestionState.abortController?.abort() + gracefulShutdownSync(1) + return + } finally { + runPhase = 'finally_flush' + // Flush pending internal events before going idle + await structuredIO.flushInternalEvents() + runPhase = 'finally_post_flush' + if (!isShuttingDown()) { + notifySessionStateChanged('idle') + // Drain so the idle session_state_changed SDK event (plus any + // terminal task_notification bookends emitted during bg-agent + // teardown) reach the output stream before we block on the next + // command. The do-while drain above only runs while + // waitingForAgents; once we're here the next drain would be the + // top of the next run(), which won't come if input is idle. + for (const event of drainSdkEvents()) { + output.enqueue(event) + } + } + running = false + // Start idle timer when we finish processing and are waiting for input + idleTimeout.start() + } + + // Proactive tick: if proactive is active and queue is empty, inject a tick + if ( + (feature('PROACTIVE') || feature('KAIROS')) && + proactiveModule?.isProactiveActive() && + !proactiveModule.isProactivePaused() + ) { + if (peek(isMainThread) === undefined && !inputClosed) { + scheduleProactiveTick!() + return + } + } + + // Re-check the queue after releasing the mutex. A message may have + // arrived (and called run()) between the last dequeue() returning + // undefined and `running = false` above. In that case the caller + // saw `running === true` and returned immediately, leaving the + // message stranded in the queue with no one to process it. + if (peek(isMainThread) !== undefined) { + void run() + return + } + + // Check for unread teammate messages and process them + // This mirrors what useInboxPoller does in interactive REPL mode + // Poll until no more messages (teammates may still be working) + { + const currentAppState = getAppState() + const teamContext = currentAppState.teamContext + + if (teamContext && isTeamLead(teamContext)) { + const agentName = 'team-lead' + + // Poll for messages while teammates are active + // This is needed because teammates may send messages while we're waiting + // Keep polling until the team is shut down + const POLL_INTERVAL_MS = 500 + + while (true) { + // Check if teammates are still active + const refreshedState = getAppState() + const hasActiveTeammates = + hasActiveInProcessTeammates(refreshedState) || + (refreshedState.teamContext && + Object.keys(refreshedState.teamContext.teammates).length > 0) + + if (!hasActiveTeammates) { + logForDebugging( + '[print.ts] No more active teammates, stopping poll', + ) + break + } + + const unread = await readUnreadMessages( + agentName, + refreshedState.teamContext?.teamName, + ) + + if (unread.length > 0) { + logForDebugging( + `[print.ts] Team-lead found ${unread.length} unread messages`, + ) + + // Mark as read immediately to avoid duplicate processing + await markMessagesAsRead( + agentName, + refreshedState.teamContext?.teamName, + ) + + // Process shutdown_approved messages - remove teammates from team file + // This mirrors what useInboxPoller does in interactive mode (lines 546-606) + const teamName = refreshedState.teamContext?.teamName + for (const m of unread) { + const shutdownApproval = isShutdownApproved(m.text) + if (shutdownApproval && teamName) { + const teammateToRemove = shutdownApproval.from + logForDebugging( + `[print.ts] Processing shutdown_approved from ${teammateToRemove}`, + ) + + // Find the teammate ID by name + const teammateId = refreshedState.teamContext?.teammates + ? Object.entries(refreshedState.teamContext.teammates).find( + ([, t]) => t.name === teammateToRemove, + )?.[0] + : undefined + + if (teammateId) { + // Remove from team file + removeTeammateFromTeamFile(teamName, { + agentId: teammateId, + name: teammateToRemove, + }) + logForDebugging( + `[print.ts] Removed ${teammateToRemove} from team file`, + ) + + // Unassign tasks owned by this teammate + await unassignTeammateTasks( + teamName, + teammateId, + teammateToRemove, + 'shutdown', + ) + + // Remove from teamContext in AppState + setAppState(prev => { + if (!prev.teamContext?.teammates) return prev + if (!(teammateId in prev.teamContext.teammates)) return prev + const { [teammateId]: _, ...remainingTeammates } = + prev.teamContext.teammates + return { + ...prev, + teamContext: { + ...prev.teamContext, + teammates: remainingTeammates, + }, + } + }) + } + } + } + + // Format messages same as useInboxPoller + const formatted = unread + .map( + (m: { from: string; text: string; color?: string }) => + `<${TEAMMATE_MESSAGE_TAG} teammate_id="${m.from}"${m.color ? ` color="${m.color}"` : ''}>\n${m.text}\n`, + ) + .join('\n\n') + + // Enqueue and process + enqueue({ + mode: 'prompt', + value: formatted, + uuid: randomUUID(), + }) + void run() + return // run() will come back here after processing + } + + // No messages - check if we need to prompt for shutdown + // If input is closed and teammates are active, inject shutdown prompt once + if (inputClosed && !shutdownPromptInjected) { + shutdownPromptInjected = true + logForDebugging( + '[print.ts] Input closed with active teammates, injecting shutdown prompt', + ) + enqueue({ + mode: 'prompt', + value: SHUTDOWN_TEAM_PROMPT, + uuid: randomUUID(), + }) + void run() + return // run() will come back here after processing + } + + // Wait and check again + await sleep(POLL_INTERVAL_MS) + } + } + } + + if (inputClosed) { + // Check for active swarm that needs shutdown + const hasActiveSwarm = await (async () => { + // Wait for any working in-process team members to finish + const currentAppState = getAppState() + if (hasWorkingInProcessTeammates(currentAppState)) { + await waitForTeammatesToBecomeIdle(setAppState, currentAppState) + } + + // Re-fetch state after potential wait + const refreshedAppState = getAppState() + const refreshedTeamContext = refreshedAppState.teamContext + const hasTeamMembersNotCleanedUp = + refreshedTeamContext && + Object.keys(refreshedTeamContext.teammates).length > 0 + + return ( + hasTeamMembersNotCleanedUp || + hasActiveInProcessTeammates(refreshedAppState) + ) + })() + + if (hasActiveSwarm) { + // Team members are idle or pane-based - inject prompt to shut down team + enqueue({ + mode: 'prompt', + value: SHUTDOWN_TEAM_PROMPT, + uuid: randomUUID(), + }) + void run() + } else { + // Wait for any in-flight push suggestion before closing the output stream. + if (suggestionState.inflightPromise) { + await Promise.race([suggestionState.inflightPromise, sleep(5000)]) + } + suggestionState.abortController?.abort() + suggestionState.abortController = null + await finalizePendingAsyncHooks() + unsubscribeSkillChanges() + unsubscribeAuthStatus?.() + statusListeners.delete(rateLimitListener) + output.done() + } + } + } + + // Set up UDS inbox callback so the query loop is kicked off + // when a message arrives via the UDS socket in headless mode. + if (feature('UDS_INBOX')) { + /* eslint-disable @typescript-eslint/no-require-imports */ + const { setOnEnqueue } = require('../utils/udsMessaging.js') + /* eslint-enable @typescript-eslint/no-require-imports */ + setOnEnqueue(() => { + if (!inputClosed) { + void run() + } + }) + } + + // Cron scheduler: runs scheduled_tasks.json tasks in SDK/-p mode. + // Mirrors REPL's useScheduledTasks hook. Fired prompts enqueue + kick + // off run() directly — unlike REPL, there's no queue subscriber here + // that drains on enqueue while idle. The run() mutex makes this safe + // during an active turn: the call no-ops and the post-run recheck at + // the end of run() picks up the queued command. + let cronScheduler: import('../utils/cronScheduler.js').CronScheduler | null = + null + if ( + feature('AGENT_TRIGGERS') && + cronSchedulerModule && + cronGate?.isKairosCronEnabled() + ) { + cronScheduler = cronSchedulerModule.createCronScheduler({ + onFire: prompt => { + if (inputClosed) return + enqueue({ + mode: 'prompt', + value: prompt, + uuid: randomUUID(), + priority: 'later', + // System-generated — matches useScheduledTasks.ts REPL equivalent. + // Without this, messages.ts metaProp eval is {} → prompt leaks + // into visible transcript when cron fires mid-turn in -p mode. + isMeta: true, + // Threaded to cc_workload= in the billing-header attribution block + // so the API can serve cron requests at lower QoS. drainCommandQueue + // reads this per-iteration and hoists it into bootstrap state for + // the ask() call. + workload: WORKLOAD_CRON, + }) + void run() + }, + isLoading: () => running || inputClosed, + getJitterConfig: cronJitterConfigModule?.getCronJitterConfig, + isKilled: () => !cronGate?.isKairosCronEnabled(), + }) + cronScheduler.start() + } + + const sendControlResponseSuccess = function ( + message: SDKControlRequest, + response?: Record, + ) { + output.enqueue({ + type: 'control_response', + response: { + subtype: 'success', + request_id: message.request_id, + response: response, + }, + }) + } + + const sendControlResponseError = function ( + message: SDKControlRequest, + errorMessage: string, + ) { + output.enqueue({ + type: 'control_response', + response: { + subtype: 'error', + request_id: message.request_id, + error: errorMessage, + }, + }) + } + + // Handle unexpected permission responses by looking up the unresolved tool + // call in the transcript and executing it + const handledOrphanedToolUseIds = new Set() + structuredIO.setUnexpectedResponseCallback(async message => { + await handleOrphanedPermissionResponse({ + message, + setAppState, + handledToolUseIds: handledOrphanedToolUseIds, + onEnqueued: () => { + // The first message of a session might be the orphaned permission + // check rather than a user prompt, so kick off the loop. + void run() + }, + }) + }) + + // Track active OAuth flows per server so we can abort a previous flow + // when a new mcp_authenticate request arrives for the same server. + const activeOAuthFlows = new Map() + // Track manual callback URL submit functions for active OAuth flows. + // Used when localhost is not reachable (e.g., browser-based IDEs). + const oauthCallbackSubmitters = new Map< + string, + (callbackUrl: string) => void + >() + // Track servers where the manual callback was actually invoked (so the + // automatic reconnect path knows to skip — the extension will reconnect). + const oauthManualCallbackUsed = new Set() + // Track OAuth auth-only promises so mcp_oauth_callback_url can await + // token exchange completion. Reconnect is handled separately by the + // extension via handleAuthDone → mcp_reconnect. + const oauthAuthPromises = new Map>() + + // In-flight Anthropic OAuth flow (claude_authenticate). Single-slot: a + // second authenticate request cleans up the first. The service holds the + // PKCE verifier + localhost listener; the promise settles after + // installOAuthTokens — after it resolves, the in-process memoized token + // cache is already cleared and the next API call picks up the new creds. + let claudeOAuth: { + service: OAuthService + flow: Promise + } | null = null + + // This is essentially spawning a parallel async task- we have two + // running in parallel- one reading from stdin and adding to the + // queue to be processed and another reading from the queue, + // processing and returning the result of the generation. + // The process is complete when the input stream completes and + // the last generation of the queue has complete. + void (async () => { + let initialized = false + logForDiagnosticsNoPII('info', 'cli_message_loop_started') + for await (const message of structuredIO.structuredInput) { + // Non-user events are handled inline (no queue). started→completed in + // the same tick carries no information, so only fire completed. + // control_response is reported by StructuredIO.processLine (which also + // sees orphans that never yield here). + const eventId = 'uuid' in message ? message.uuid : undefined + if ( + eventId && + message.type !== 'user' && + message.type !== 'control_response' + ) { + notifyCommandLifecycle(eventId, 'completed') + } + + if (message.type === 'control_request') { + if (message.request.subtype === 'interrupt') { + // Track escapes for attribution (ant-only feature) + if (feature('COMMIT_ATTRIBUTION')) { + setAppState(prev => ({ + ...prev, + attribution: { + ...prev.attribution, + escapeCount: prev.attribution.escapeCount + 1, + }, + })) + } + if (abortController) { + abortController.abort() + } + suggestionState.abortController?.abort() + suggestionState.abortController = null + suggestionState.lastEmitted = null + suggestionState.pendingSuggestion = null + sendControlResponseSuccess(message) + } else if (message.request.subtype === 'end_session') { + logForDebugging( + `[print.ts] end_session received, reason=${message.request.reason ?? 'unspecified'}`, + ) + if (abortController) { + abortController.abort() + } + suggestionState.abortController?.abort() + suggestionState.abortController = null + suggestionState.lastEmitted = null + suggestionState.pendingSuggestion = null + sendControlResponseSuccess(message) + break // exits for-await → falls through to inputClosed=true drain below + } else if (message.request.subtype === 'initialize') { + // SDK MCP server names from the initialize message + // Populated by both browser and ProcessTransport sessions + if ( + message.request.sdkMcpServers && + message.request.sdkMcpServers.length > 0 + ) { + for (const serverName of message.request.sdkMcpServers) { + // Create placeholder config for SDK MCP servers + // The actual server connection is managed by the SDK Query class + sdkMcpConfigs[serverName] = { + type: 'sdk', + name: serverName, + } + } + } + + await handleInitializeRequest( + message.request, + message.request_id, + initialized, + output, + commands, + modelInfos, + structuredIO, + !!options.enableAuthStatus, + options, + agents, + getAppState, + ) + + // Enable prompt suggestions in AppState when SDK consumer opts in. + // shouldEnablePromptSuggestion() returns false for non-interactive + // sessions, but the SDK consumer explicitly requested suggestions. + if (message.request.promptSuggestions) { + setAppState(prev => { + if (prev.promptSuggestionEnabled) return prev + return { ...prev, promptSuggestionEnabled: true } + }) + } + + if ( + message.request.agentProgressSummaries && + getFeatureValue_CACHED_MAY_BE_STALE('tengu_slate_prism', true) + ) { + setSdkAgentProgressSummariesEnabled(true) + } + + initialized = true + + // If the auto-resume logic pre-enqueued a command, drain it now + // that initialize has set up systemPrompt, agents, hooks, etc. + if (hasCommandsInQueue()) { + void run() + } + } else if (message.request.subtype === 'set_permission_mode') { + const m = message.request // for typescript (TODO: use readonly types to avoid this) + setAppState(prev => ({ + ...prev, + toolPermissionContext: handleSetPermissionMode( + m, + message.request_id, + prev.toolPermissionContext, + output, + ), + isUltraplanMode: m.ultraplan ?? prev.isUltraplanMode, + })) + // handleSetPermissionMode sends the control_response; the + // notifySessionMetadataChanged that used to follow here is + // now fired by onChangeAppState (with externalized mode name). + } else if (message.request.subtype === 'set_model') { + const requestedModel = message.request.model ?? 'default' + const model = + requestedModel === 'default' + ? getDefaultMainLoopModel() + : requestedModel + activeUserSpecifiedModel = model + setMainLoopModelOverride(model) + notifySessionMetadataChanged({ model }) + injectModelSwitchBreadcrumbs(requestedModel, model) + + sendControlResponseSuccess(message) + } else if (message.request.subtype === 'set_max_thinking_tokens') { + if (message.request.max_thinking_tokens === null) { + options.thinkingConfig = undefined + } else if (message.request.max_thinking_tokens === 0) { + options.thinkingConfig = { type: 'disabled' } + } else { + options.thinkingConfig = { + type: 'enabled', + budgetTokens: message.request.max_thinking_tokens, + } + } + sendControlResponseSuccess(message) + } else if (message.request.subtype === 'mcp_status') { + sendControlResponseSuccess(message, { + mcpServers: buildMcpServerStatuses(), + }) + } else if (message.request.subtype === 'get_context_usage') { + try { + const appState = getAppState() + const data = await collectContextData({ + messages: mutableMessages, + getAppState, + options: { + mainLoopModel: getMainLoopModel(), + tools: buildAllTools(appState), + agentDefinitions: appState.agentDefinitions, + customSystemPrompt: options.systemPrompt, + appendSystemPrompt: options.appendSystemPrompt, + }, + }) + sendControlResponseSuccess(message, { ...data }) + } catch (error) { + sendControlResponseError(message, errorMessage(error)) + } + } else if (message.request.subtype === 'mcp_message') { + // Handle MCP notifications from SDK servers + const mcpRequest = message.request + const sdkClient = sdkClients.find( + client => client.name === mcpRequest.server_name, + ) + // Check client exists - dynamically added SDK servers may have + // placeholder clients with null client until updateSdkMcp() runs + if ( + sdkClient && + sdkClient.type === 'connected' && + sdkClient.client?.transport?.onmessage + ) { + sdkClient.client.transport.onmessage(mcpRequest.message) + } + sendControlResponseSuccess(message) + } else if (message.request.subtype === 'rewind_files') { + const appState = getAppState() + const result = await handleRewindFiles( + message.request.user_message_id as UUID, + appState, + setAppState, + message.request.dry_run ?? false, + ) + if (result.canRewind || message.request.dry_run) { + sendControlResponseSuccess(message, result) + } else { + sendControlResponseError( + message, + result.error ?? 'Unexpected error', + ) + } + } else if (message.request.subtype === 'cancel_async_message') { + const targetUuid = message.request.message_uuid + const removed = dequeueAllMatching(cmd => cmd.uuid === targetUuid) + sendControlResponseSuccess(message, { + cancelled: removed.length > 0, + }) + } else if (message.request.subtype === 'seed_read_state') { + // Client observed a Read that was later removed from context (e.g. + // by snip), so transcript-based seeding missed it. Queued into + // pendingSeeds; applied at the next clone-replace boundary. + try { + // expandPath: all other readFileState writers normalize (~, relative, + // session cwd vs process cwd). FileEditTool looks up by expandPath'd + // key — a verbatim client path would miss. + const normalizedPath = expandPath(message.request.path) + // Check disk mtime before reading content. If the file changed + // since the client's observation, readFile would return C_current + // but we'd store it with the client's M_observed — getChangedFiles + // then sees disk > cache.timestamp, re-reads, diffs C_current vs + // C_current = empty, emits no attachment, and the model is never + // told about the C_observed → C_current change. Skipping the seed + // makes Edit fail "file not read yet" → forces a fresh Read. + // Math.floor matches FileReadTool and getFileModificationTime. + const diskMtime = Math.floor((await stat(normalizedPath)).mtimeMs) + if (diskMtime <= message.request.mtime) { + const raw = await readFile(normalizedPath, 'utf-8') + // Strip BOM + normalize CRLF→LF to match readFileInRange and + // readFileSyncWithMetadata. FileEditTool's content-compare + // fallback (for Windows mtime bumps without content change) + // compares against LF-normalized disk reads. + const content = ( + raw.charCodeAt(0) === 0xfeff ? raw.slice(1) : raw + ).replaceAll('\r\n', '\n') + pendingSeeds.set(normalizedPath, { + content, + timestamp: diskMtime, + offset: undefined, + limit: undefined, + }) + } + } catch { + // ENOENT etc — skip seeding but still succeed + } + sendControlResponseSuccess(message) + } else if (message.request.subtype === 'mcp_set_servers') { + const { response, sdkServersChanged } = await applyMcpServerChanges( + message.request.servers, + ) + sendControlResponseSuccess(message, response) + + // Connect SDK servers AFTER response to avoid deadlock + if (sdkServersChanged) { + void updateSdkMcp() + } + } else if (message.request.subtype === 'reload_plugins') { + try { + if ( + feature('DOWNLOAD_USER_SETTINGS') && + (isEnvTruthy(process.env.CLAUDE_CODE_REMOTE) || getIsRemoteMode()) + ) { + // Re-pull user settings so enabledPlugins pushed from the + // user's local CLI take effect before the cache sweep. + const applied = await redownloadUserSettings() + if (applied) { + settingsChangeDetector.notifyChange('userSettings') + } + } + + const r = await refreshActivePlugins(setAppState) + + const sdkAgents = currentAgents.filter( + a => a.source === 'flagSettings', + ) + currentAgents = [...r.agentDefinitions.allAgents, ...sdkAgents] + + // Reload succeeded — gather response data best-effort so a + // read failure doesn't mask the successful state change. + // allSettled so one failure doesn't discard the others. + let plugins: SDKControlReloadPluginsResponse['plugins'] = [] + const [cmdsR, mcpR, pluginsR] = await Promise.allSettled([ + getCommands(cwd()), + applyPluginMcpDiff(), + loadAllPluginsCacheOnly(), + ]) + if (cmdsR.status === 'fulfilled') { + currentCommands = cmdsR.value + } else { + logError(cmdsR.reason) + } + if (mcpR.status === 'rejected') { + logError(mcpR.reason) + } + if (pluginsR.status === 'fulfilled') { + plugins = pluginsR.value.enabled.map(p => ({ + name: p.name, + path: p.path, + source: p.source, + })) + } else { + logError(pluginsR.reason) + } + + sendControlResponseSuccess(message, { + commands: currentCommands + .filter(cmd => cmd.userInvocable !== false) + .map(cmd => ({ + name: getCommandName(cmd), + description: formatDescriptionWithSource(cmd), + argumentHint: cmd.argumentHint || '', + })), + agents: currentAgents.map(a => ({ + name: a.agentType, + description: a.whenToUse, + model: a.model === 'inherit' ? undefined : a.model, + })), + plugins, + mcpServers: buildMcpServerStatuses(), + error_count: r.error_count, + } satisfies SDKControlReloadPluginsResponse) + } catch (error) { + sendControlResponseError(message, errorMessage(error)) + } + } else if (message.request.subtype === 'mcp_reconnect') { + const currentAppState = getAppState() + const { serverName } = message.request + elicitationRegistered.delete(serverName) + // Config-existence gate must cover the SAME sources as the + // operations below. SDK-injected servers (query({mcpServers:{...}})) + // and dynamically-added servers were missing here, so + // toggleMcpServer/reconnect returned "Server not found" even though + // the disconnect/reconnect would have worked (gh-31339 / CC-314). + const config = + getMcpConfigByName(serverName) ?? + mcpClients.find(c => c.name === serverName)?.config ?? + sdkClients.find(c => c.name === serverName)?.config ?? + dynamicMcpState.clients.find(c => c.name === serverName)?.config ?? + currentAppState.mcp.clients.find(c => c.name === serverName) + ?.config ?? + null + if (!config) { + sendControlResponseError(message, `Server not found: ${serverName}`) + } else { + const result = await reconnectMcpServerImpl(serverName, config) + // Update appState.mcp with the new client, tools, commands, and resources + const prefix = getMcpPrefix(serverName) + setAppState(prev => ({ + ...prev, + mcp: { + ...prev.mcp, + clients: prev.mcp.clients.map(c => + c.name === serverName ? result.client : c, + ), + tools: [ + ...reject(prev.mcp.tools, t => t.name?.startsWith(prefix)), + ...result.tools, + ], + commands: [ + ...reject(prev.mcp.commands, c => + commandBelongsToServer(c, serverName), + ), + ...result.commands, + ], + resources: + result.resources && result.resources.length > 0 + ? { ...prev.mcp.resources, [serverName]: result.resources } + : omit(prev.mcp.resources, serverName), + }, + })) + // Also update dynamicMcpState so run() picks up the new tools + // on the next turn (run() reads dynamicMcpState, not appState) + dynamicMcpState = { + ...dynamicMcpState, + clients: [ + ...dynamicMcpState.clients.filter(c => c.name !== serverName), + result.client, + ], + tools: [ + ...dynamicMcpState.tools.filter( + t => !t.name?.startsWith(prefix), + ), + ...result.tools, + ], + } + if (result.client.type === 'connected') { + registerElicitationHandlers([result.client]) + reregisterChannelHandlerAfterReconnect(result.client) + sendControlResponseSuccess(message) + } else { + const errorMessage = + result.client.type === 'failed' + ? (result.client.error ?? 'Connection failed') + : `Server status: ${result.client.type}` + sendControlResponseError(message, errorMessage) + } + } + } else if (message.request.subtype === 'mcp_toggle') { + const currentAppState = getAppState() + const { serverName, enabled } = message.request + elicitationRegistered.delete(serverName) + // Gate must match the client-lookup spread below (which + // includes sdkClients and dynamicMcpState.clients). Same fix as + // mcp_reconnect above (gh-31339 / CC-314). + const config = + getMcpConfigByName(serverName) ?? + mcpClients.find(c => c.name === serverName)?.config ?? + sdkClients.find(c => c.name === serverName)?.config ?? + dynamicMcpState.clients.find(c => c.name === serverName)?.config ?? + currentAppState.mcp.clients.find(c => c.name === serverName) + ?.config ?? + null + + if (!config) { + sendControlResponseError(message, `Server not found: ${serverName}`) + } else if (!enabled) { + // Disabling: persist + disconnect (matches TUI toggleMcpServer behavior) + setMcpServerEnabled(serverName, false) + const client = [ + ...mcpClients, + ...sdkClients, + ...dynamicMcpState.clients, + ...currentAppState.mcp.clients, + ].find(c => c.name === serverName) + if (client && client.type === 'connected') { + await clearServerCache(serverName, config) + } + // Update appState.mcp to reflect disabled status and remove tools/commands/resources + const prefix = getMcpPrefix(serverName) + setAppState(prev => ({ + ...prev, + mcp: { + ...prev.mcp, + clients: prev.mcp.clients.map(c => + c.name === serverName + ? { name: serverName, type: 'disabled' as const, config } + : c, + ), + tools: reject(prev.mcp.tools, t => t.name?.startsWith(prefix)), + commands: reject(prev.mcp.commands, c => + commandBelongsToServer(c, serverName), + ), + resources: omit(prev.mcp.resources, serverName), + }, + })) + sendControlResponseSuccess(message) + } else { + // Enabling: persist + reconnect + setMcpServerEnabled(serverName, true) + const result = await reconnectMcpServerImpl(serverName, config) + // Update appState.mcp with the new client, tools, commands, and resources + // This ensures the LLM sees updated tools after enabling the server + const prefix = getMcpPrefix(serverName) + setAppState(prev => ({ + ...prev, + mcp: { + ...prev.mcp, + clients: prev.mcp.clients.map(c => + c.name === serverName ? result.client : c, + ), + tools: [ + ...reject(prev.mcp.tools, t => t.name?.startsWith(prefix)), + ...result.tools, + ], + commands: [ + ...reject(prev.mcp.commands, c => + commandBelongsToServer(c, serverName), + ), + ...result.commands, + ], + resources: + result.resources && result.resources.length > 0 + ? { ...prev.mcp.resources, [serverName]: result.resources } + : omit(prev.mcp.resources, serverName), + }, + })) + if (result.client.type === 'connected') { + registerElicitationHandlers([result.client]) + reregisterChannelHandlerAfterReconnect(result.client) + sendControlResponseSuccess(message) + } else { + const errorMessage = + result.client.type === 'failed' + ? (result.client.error ?? 'Connection failed') + : `Server status: ${result.client.type}` + sendControlResponseError(message, errorMessage) + } + } + } else if (message.request.subtype === 'channel_enable') { + const currentAppState = getAppState() + handleChannelEnable( + message.request_id, + message.request.serverName, + // Pool spread matches mcp_status — all three client sources. + [ + ...currentAppState.mcp.clients, + ...sdkClients, + ...dynamicMcpState.clients, + ], + output, + ) + } else if (message.request.subtype === 'mcp_authenticate') { + const { serverName } = message.request + const currentAppState = getAppState() + const config = + getMcpConfigByName(serverName) ?? + mcpClients.find(c => c.name === serverName)?.config ?? + currentAppState.mcp.clients.find(c => c.name === serverName) + ?.config ?? + null + if (!config) { + sendControlResponseError(message, `Server not found: ${serverName}`) + } else if (config.type !== 'sse' && config.type !== 'http') { + sendControlResponseError( + message, + `Server type "${config.type}" does not support OAuth authentication`, + ) + } else { + try { + // Abort any previous in-flight OAuth flow for this server + activeOAuthFlows.get(serverName)?.abort() + const controller = new AbortController() + activeOAuthFlows.set(serverName, controller) + + // Capture the auth URL from the callback + let resolveAuthUrl: (url: string) => void + const authUrlPromise = new Promise(resolve => { + resolveAuthUrl = resolve + }) + + // Start the OAuth flow in the background + const oauthPromise = performMCPOAuthFlow( + serverName, + config, + url => resolveAuthUrl!(url), + controller.signal, + { + skipBrowserOpen: true, + onWaitingForCallback: submit => { + oauthCallbackSubmitters.set(serverName, submit) + }, + }, + ) + + // Wait for the auth URL (or the flow to complete without needing redirect) + const authUrl = await Promise.race([ + authUrlPromise, + oauthPromise.then(() => null as string | null), + ]) + + if (authUrl) { + sendControlResponseSuccess(message, { + authUrl, + requiresUserAction: true, + }) + } else { + sendControlResponseSuccess(message, { + requiresUserAction: false, + }) + } + + // Store auth-only promise for mcp_oauth_callback_url handler. + // Don't swallow errors — the callback handler needs to detect + // auth failures and report them to the caller. + oauthAuthPromises.set(serverName, oauthPromise) + + // Handle background completion — reconnect after auth. + // When manual callback is used, skip the reconnect here; + // the extension's handleAuthDone → mcp_reconnect handles it + // (which also updates dynamicMcpState for tool registration). + const fullFlowPromise = oauthPromise + .then(async () => { + // Don't reconnect if the server was disabled during the OAuth flow + if (isMcpServerDisabled(serverName)) { + return + } + // Skip reconnect if the manual callback path was used — + // handleAuthDone will do it via mcp_reconnect (which + // updates dynamicMcpState for tool registration). + if (oauthManualCallbackUsed.has(serverName)) { + return + } + // Reconnect the server after successful auth + const result = await reconnectMcpServerImpl( + serverName, + config, + ) + const prefix = getMcpPrefix(serverName) + setAppState(prev => ({ + ...prev, + mcp: { + ...prev.mcp, + clients: prev.mcp.clients.map(c => + c.name === serverName ? result.client : c, + ), + tools: [ + ...reject(prev.mcp.tools, t => + t.name?.startsWith(prefix), + ), + ...result.tools, + ], + commands: [ + ...reject(prev.mcp.commands, c => + commandBelongsToServer(c, serverName), + ), + ...result.commands, + ], + resources: + result.resources && result.resources.length > 0 + ? { + ...prev.mcp.resources, + [serverName]: result.resources, + } + : omit(prev.mcp.resources, serverName), + }, + })) + // Also update dynamicMcpState so run() picks up the new tools + // on the next turn (run() reads dynamicMcpState, not appState) + dynamicMcpState = { + ...dynamicMcpState, + clients: [ + ...dynamicMcpState.clients.filter( + c => c.name !== serverName, + ), + result.client, + ], + tools: [ + ...dynamicMcpState.tools.filter( + t => !t.name?.startsWith(prefix), + ), + ...result.tools, + ], + } + }) + .catch(error => { + logForDebugging( + `MCP OAuth failed for ${serverName}: ${error}`, + { level: 'error' }, + ) + }) + .finally(() => { + // Clean up only if this is still the active flow + if (activeOAuthFlows.get(serverName) === controller) { + activeOAuthFlows.delete(serverName) + oauthCallbackSubmitters.delete(serverName) + oauthManualCallbackUsed.delete(serverName) + oauthAuthPromises.delete(serverName) + } + }) + void fullFlowPromise + } catch (error) { + sendControlResponseError(message, errorMessage(error)) + } + } + } else if (message.request.subtype === 'mcp_oauth_callback_url') { + const { serverName, callbackUrl } = message.request + const submit = oauthCallbackSubmitters.get(serverName) + if (submit) { + // Validate the callback URL before submitting. The submit + // callback in auth.ts silently ignores URLs missing a code + // param, which would leave the auth promise unresolved and + // block the control message loop until timeout. + let hasCodeOrError = false + try { + const parsed = new URL(callbackUrl) + hasCodeOrError = + parsed.searchParams.has('code') || + parsed.searchParams.has('error') + } catch { + // Invalid URL + } + if (!hasCodeOrError) { + sendControlResponseError( + message, + 'Invalid callback URL: missing authorization code. Please paste the full redirect URL including the code parameter.', + ) + } else { + oauthManualCallbackUsed.add(serverName) + submit(callbackUrl) + // Wait for auth (token exchange) to complete before responding. + // Reconnect is handled by the extension via handleAuthDone → + // mcp_reconnect (which updates dynamicMcpState for tools). + const authPromise = oauthAuthPromises.get(serverName) + if (authPromise) { + try { + await authPromise + sendControlResponseSuccess(message) + } catch (error) { + sendControlResponseError( + message, + error instanceof Error + ? error.message + : 'OAuth authentication failed', + ) + } + } else { + sendControlResponseSuccess(message) + } + } + } else { + sendControlResponseError( + message, + `No active OAuth flow for server: ${serverName}`, + ) + } + } else if (message.request.subtype === 'claude_authenticate') { + // Anthropic OAuth over the control channel. The SDK client owns + // the user's browser (we're headless in -p mode); we hand back + // both URLs and wait. Automatic URL → localhost listener catches + // the redirect if the browser is on this host; manual URL → the + // success page shows "code#state" for claude_oauth_callback. + const { loginWithClaudeAi } = message.request + + // Clean up any prior flow. cleanup() closes the localhost listener + // and nulls the manual resolver. The prior `flow` promise is left + // pending (AuthCodeListener.close() does not reject) but its object + // graph becomes unreachable once the server handle is released and + // is GC'd — no fd or port is held. + claudeOAuth?.service.cleanup() + + logEvent('tengu_oauth_flow_start', { + loginWithClaudeAi: loginWithClaudeAi ?? true, + }) + + const service = new OAuthService() + let urlResolver!: (urls: { + manualUrl: string + automaticUrl: string + }) => void + const urlPromise = new Promise<{ + manualUrl: string + automaticUrl: string + }>(resolve => { + urlResolver = resolve + }) + + const flow = service + .startOAuthFlow( + async (manualUrl, automaticUrl) => { + // automaticUrl is always defined when skipBrowserOpen is set; + // the signature is optional only for the existing single-arg callers. + urlResolver({ manualUrl, automaticUrl: automaticUrl! }) + }, + { + loginWithClaudeAi: loginWithClaudeAi ?? true, + skipBrowserOpen: true, + }, + ) + .then(async tokens => { + // installOAuthTokens: performLogout (clear stale state) → + // store profile → saveOAuthTokensIfNeeded → clearOAuthTokenCache + // → clearAuthRelatedCaches. After this resolves, the memoized + // getClaudeAIOAuthTokens in this process is invalidated; the + // next API call re-reads keychain/file and works. No respawn. + await installOAuthTokens(tokens) + logEvent('tengu_oauth_success', { + loginWithClaudeAi: loginWithClaudeAi ?? true, + }) + }) + .finally(() => { + service.cleanup() + if (claudeOAuth?.service === service) { + claudeOAuth = null + } + }) + + claudeOAuth = { service, flow } + + // Attach the rejection handler before awaiting so a synchronous + // startOAuthFlow failure doesn't surface as an unhandled rejection. + // The claude_oauth_callback handler re-awaits flow for the manual + // path and surfaces the real error to the client. + void flow.catch(err => + logForDebugging(`claude_authenticate flow ended: ${err}`, { + level: 'info', + }), + ) + + try { + // Race against flow: if startOAuthFlow rejects before calling + // the authURLHandler (e.g. AuthCodeListener.start() fails with + // EACCES or fd exhaustion), urlPromise would pend forever and + // wedge the stdin loop. flow resolving first is unreachable in + // practice (it's suspended on the same urls we're waiting for). + const { manualUrl, automaticUrl } = await Promise.race([ + urlPromise, + flow.then(() => { + throw new Error( + 'OAuth flow completed without producing auth URLs', + ) + }), + ]) + sendControlResponseSuccess(message, { + manualUrl, + automaticUrl, + }) + } catch (error) { + sendControlResponseError(message, errorMessage(error)) + } + } else if ( + message.request.subtype === 'claude_oauth_callback' || + message.request.subtype === 'claude_oauth_wait_for_completion' + ) { + if (!claudeOAuth) { + sendControlResponseError( + message, + 'No active claude_authenticate flow', + ) + } else { + // Inject the manual code synchronously — must happen in stdin + // message order so a subsequent claude_authenticate doesn't + // replace the service before this code lands. + if (message.request.subtype === 'claude_oauth_callback') { + claudeOAuth.service.handleManualAuthCodeInput({ + authorizationCode: message.request.authorizationCode, + state: message.request.state, + }) + } + // Detach the await — the stdin reader is serial and blocking + // here deadlocks claude_oauth_wait_for_completion: flow may + // only resolve via a future claude_oauth_callback on stdin, + // which can't be read while we're parked. Capture the binding; + // claudeOAuth is nulled in flow's own .finally. + const { flow } = claudeOAuth + void flow.then( + () => { + const accountInfo = getAccountInformation() + sendControlResponseSuccess(message, { + account: { + email: accountInfo?.email, + organization: accountInfo?.organization, + subscriptionType: accountInfo?.subscription, + tokenSource: accountInfo?.tokenSource, + apiKeySource: accountInfo?.apiKeySource, + apiProvider: getAPIProvider(), + }, + }) + }, + (error: unknown) => + sendControlResponseError(message, errorMessage(error)), + ) + } + } else if (message.request.subtype === 'mcp_clear_auth') { + const { serverName } = message.request + const currentAppState = getAppState() + const config = + getMcpConfigByName(serverName) ?? + mcpClients.find(c => c.name === serverName)?.config ?? + currentAppState.mcp.clients.find(c => c.name === serverName) + ?.config ?? + null + if (!config) { + sendControlResponseError(message, `Server not found: ${serverName}`) + } else if (config.type !== 'sse' && config.type !== 'http') { + sendControlResponseError( + message, + `Cannot clear auth for server type "${config.type}"`, + ) + } else { + await revokeServerTokens(serverName, config) + const result = await reconnectMcpServerImpl(serverName, config) + const prefix = getMcpPrefix(serverName) + setAppState(prev => ({ + ...prev, + mcp: { + ...prev.mcp, + clients: prev.mcp.clients.map(c => + c.name === serverName ? result.client : c, + ), + tools: [ + ...reject(prev.mcp.tools, t => t.name?.startsWith(prefix)), + ...result.tools, + ], + commands: [ + ...reject(prev.mcp.commands, c => + commandBelongsToServer(c, serverName), + ), + ...result.commands, + ], + resources: + result.resources && result.resources.length > 0 + ? { + ...prev.mcp.resources, + [serverName]: result.resources, + } + : omit(prev.mcp.resources, serverName), + }, + })) + sendControlResponseSuccess(message, {}) + } + } else if (message.request.subtype === 'apply_flag_settings') { + // Snapshot the current model before applying — we need to detect + // model switches so we can inject breadcrumbs and notify listeners. + const prevModel = getMainLoopModel() + + // Merge the provided settings into the in-memory flag settings + const existing = getFlagSettingsInline() ?? {} + const incoming = message.request.settings + // Shallow-merge top-level keys; getSettingsForSource handles + // the deep merge with file-based flag settings via mergeWith. + // JSON serialization drops `undefined`, so callers use `null` + // to signal "clear this key". Convert nulls to deletions so + // SettingsSchema().safeParse() doesn't reject the whole object + // (z.string().optional() accepts string | undefined, not null). + const merged = { ...existing, ...incoming } + for (const key of Object.keys(merged)) { + if (merged[key as keyof typeof merged] === null) { + delete merged[key as keyof typeof merged] + } + } + setFlagSettingsInline(merged) + // Route through notifyChange so fanOut() resets the settings cache + // before listeners run. The subscriber at :392 calls + // applySettingsChange for us. Pre-#20625 this was a direct + // applySettingsChange() call that relied on its own internal reset — + // now that the reset is centralized in fanOut, a direct call here + // would read stale cached settings and silently drop the update. + // Bonus: going through notifyChange also tells the other subscribers + // (loadPluginHooks, sandbox-adapter) about the change, which the + // previous direct call skipped. + settingsChangeDetector.notifyChange('flagSettings') + + // If the incoming settings include a model change, update the + // override so getMainLoopModel() reflects it. The override has + // higher priority than the settings cascade in + // getUserSpecifiedModelSetting(), so without this update, + // getMainLoopModel() returns the stale override and the model + // change is silently ignored (matching set_model at :2811). + if ('model' in incoming) { + if (incoming.model != null) { + setMainLoopModelOverride(String(incoming.model)) + } else { + setMainLoopModelOverride(undefined) + } + } + + // If the model changed, inject breadcrumbs so the model sees the + // mid-conversation switch, and notify metadata listeners (CCR). + const newModel = getMainLoopModel() + if (newModel !== prevModel) { + activeUserSpecifiedModel = newModel + const modelArg = incoming.model ? String(incoming.model) : 'default' + notifySessionMetadataChanged({ model: newModel }) + injectModelSwitchBreadcrumbs(modelArg, newModel) + } + + sendControlResponseSuccess(message) + } else if (message.request.subtype === 'get_settings') { + const currentAppState = getAppState() + const model = getMainLoopModel() + // modelSupportsEffort gate matches claude.ts — applied.effort must + // mirror what actually goes to the API, not just what's configured. + const effort = modelSupportsEffort(model) + ? resolveAppliedEffort(model, currentAppState.effortValue) + : undefined + sendControlResponseSuccess(message, { + ...getSettingsWithSources(), + applied: { + model, + // Numeric effort (ant-only) → null; SDK schema is string-level only. + effort: typeof effort === 'string' ? effort : null, + }, + }) + } else if (message.request.subtype === 'stop_task') { + const { task_id: taskId } = message.request + try { + await stopTask(taskId, { + getAppState, + setAppState, + }) + sendControlResponseSuccess(message, {}) + } catch (error) { + sendControlResponseError(message, errorMessage(error)) + } + } else if (message.request.subtype === 'generate_session_title') { + // Fire-and-forget so the Haiku call does not block the stdin loop + // (which would delay processing of subsequent user messages / + // interrupts for the duration of the API roundtrip). + const { description, persist } = message.request + // Reuse the live controller only if it has not already been aborted + // (e.g. by interrupt()); an aborted signal would cause queryHaiku to + // immediately throw APIUserAbortError → {title: null}. + const titleSignal = ( + abortController && !abortController.signal.aborted + ? abortController + : createAbortController() + ).signal + void (async () => { + try { + const title = await generateSessionTitle(description, titleSignal) + if (title && persist) { + try { + saveAiGeneratedTitle(getSessionId() as UUID, title) + } catch (e) { + logError(e) + } + } + sendControlResponseSuccess(message, { title }) + } catch (e) { + // Unreachable in practice — generateSessionTitle wraps its + // own body and returns null, saveAiGeneratedTitle is wrapped + // above. Propagate (not swallow) so unexpected failures are + // visible to the SDK caller (hostComms.ts catches and logs). + sendControlResponseError(message, errorMessage(e)) + } + })() + } else if (message.request.subtype === 'side_question') { + // Same fire-and-forget pattern as generate_session_title above — + // the forked agent's API roundtrip must not block the stdin loop. + // + // The snapshot captured by stopHooks (for querySource === 'sdk') + // holds the exact systemPrompt/userContext/systemContext/messages + // sent on the last main-thread turn. Reusing them gives a byte- + // identical prefix → prompt cache hit. + // + // Fallback (resume before first turn completes — no snapshot yet): + // rebuild from scratch. buildSideQuestionFallbackParams mirrors + // QueryEngine.ts:ask()'s system prompt assembly (including + // --system-prompt / --append-system-prompt) so the rebuilt prefix + // matches in the common case. May still miss the cache for + // coordinator mode or memory-mechanics extras — acceptable, the + // alternative is the side question failing entirely. + const { question } = message.request + void (async () => { + try { + const saved = getLastCacheSafeParams() + const cacheSafeParams = saved + ? { + ...saved, + // If the last turn was interrupted, the snapshot holds an + // already-aborted controller; createChildAbortController in + // createSubagentContext would propagate it and the fork + // would die before sending a request. The controller is + // not part of the cache key — swapping in a fresh one is + // safe. Same guard as generate_session_title above. + toolUseContext: { + ...saved.toolUseContext, + abortController: createAbortController(), + }, + } + : await buildSideQuestionFallbackParams({ + tools: buildAllTools(getAppState()), + commands: currentCommands, + mcpClients: [ + ...getAppState().mcp.clients, + ...sdkClients, + ...dynamicMcpState.clients, + ], + messages: mutableMessages, + readFileState, + getAppState, + setAppState, + customSystemPrompt: options.systemPrompt, + appendSystemPrompt: options.appendSystemPrompt, + thinkingConfig: options.thinkingConfig, + agents: currentAgents, + }) + const result = await runSideQuestion({ + question, + cacheSafeParams, + }) + sendControlResponseSuccess(message, { response: result.response }) + } catch (e) { + sendControlResponseError(message, errorMessage(e)) + } + })() + } else if ( + (feature('PROACTIVE') || feature('KAIROS')) && + (message.request as { subtype: string }).subtype === 'set_proactive' + ) { + const req = message.request as unknown as { + subtype: string + enabled: boolean + } + if (req.enabled) { + if (!proactiveModule!.isProactiveActive()) { + proactiveModule!.activateProactive('command') + scheduleProactiveTick!() + } + } else { + proactiveModule!.deactivateProactive() + } + sendControlResponseSuccess(message) + } else if (message.request.subtype === 'remote_control') { + if (message.request.enabled) { + if (bridgeHandle) { + // Already connected + sendControlResponseSuccess(message, { + session_url: getRemoteSessionUrl( + bridgeHandle.bridgeSessionId, + bridgeHandle.sessionIngressUrl, + ), + connect_url: buildBridgeConnectUrl( + bridgeHandle.environmentId, + bridgeHandle.sessionIngressUrl, + ), + environment_id: bridgeHandle.environmentId, + }) + } else { + // initReplBridge surfaces gate-failure reasons via + // onStateChange('failed', detail) before returning null. + // Capture so the control-response error is actionable + // ("/login", "disabled by your organization's policy", etc.) + // instead of a generic "initialization failed". + let bridgeFailureDetail: string | undefined + try { + const { initReplBridge } = await import( + 'src/bridge/initReplBridge.js' + ) + const handle = await initReplBridge({ + onInboundMessage(msg) { + const fields = extractInboundMessageFields(msg) + if (!fields) return + const { content, uuid } = fields + enqueue({ + value: content, + mode: 'prompt' as const, + uuid, + skipSlashCommands: true, + }) + void run() + }, + onPermissionResponse(response) { + // Forward bridge permission responses into the + // stdin processing loop so they resolve pending + // permission requests from the SDK consumer. + structuredIO.injectControlResponse(response) + }, + onInterrupt() { + abortController?.abort() + }, + onSetModel(model) { + const resolved = + model === 'default' ? getDefaultMainLoopModel() : model + activeUserSpecifiedModel = resolved + setMainLoopModelOverride(resolved) + }, + onSetMaxThinkingTokens(maxTokens) { + if (maxTokens === null) { + options.thinkingConfig = undefined + } else if (maxTokens === 0) { + options.thinkingConfig = { type: 'disabled' } + } else { + options.thinkingConfig = { + type: 'enabled', + budgetTokens: maxTokens, + } + } + }, + onStateChange(state, detail) { + if (state === 'failed') { + bridgeFailureDetail = detail + } + logForDebugging( + `[bridge:sdk] State change: ${state}${detail ? ` — ${detail}` : ''}`, + ) + output.enqueue({ + type: 'system' as StdoutMessage['type'], + subtype: 'bridge_state' as string, + state, + detail, + uuid: randomUUID(), + session_id: getSessionId(), + } as StdoutMessage) + }, + initialMessages: + mutableMessages.length > 0 ? mutableMessages : undefined, + }) + if (!handle) { + sendControlResponseError( + message, + bridgeFailureDetail ?? + 'Remote Control initialization failed', + ) + } else { + bridgeHandle = handle + bridgeLastForwardedIndex = mutableMessages.length + // Forward permission requests to the bridge + structuredIO.setOnControlRequestSent(request => { + handle.sendControlRequest(request) + }) + // Cancel stale bridge permission prompts when the SDK + // consumer resolves a can_use_tool request first. + structuredIO.setOnControlRequestResolved(requestId => { + handle.sendControlCancelRequest(requestId) + }) + sendControlResponseSuccess(message, { + session_url: getRemoteSessionUrl( + handle.bridgeSessionId, + handle.sessionIngressUrl, + ), + connect_url: buildBridgeConnectUrl( + handle.environmentId, + handle.sessionIngressUrl, + ), + environment_id: handle.environmentId, + }) + } + } catch (err) { + sendControlResponseError(message, errorMessage(err)) + } + } + } else { + // Disable + if (bridgeHandle) { + structuredIO.setOnControlRequestSent(undefined) + structuredIO.setOnControlRequestResolved(undefined) + await bridgeHandle.teardown() + bridgeHandle = null + } + sendControlResponseSuccess(message) + } + } else { + // Unknown control request subtype — send an error response so + // the caller doesn't hang waiting for a reply that never comes. + sendControlResponseError( + message, + `Unsupported control request subtype: ${(message.request as { subtype: string }).subtype}`, + ) + } + continue + } else if (message.type === 'control_response') { + // Replay control_response messages when replay mode is enabled + if (options.replayUserMessages) { + output.enqueue(message) + } + continue + } else if (message.type === 'keep_alive') { + // Silently ignore keep-alive messages + continue + } else if (message.type === 'update_environment_variables') { + // Handled in structuredIO.ts, but TypeScript needs the type guard + continue + } else if (message.type === 'assistant' || message.type === 'system') { + // History replay from bridge: inject into mutableMessages as + // conversation context so the model sees prior turns. + const internalMsgs = toInternalMessages([message]) + mutableMessages.push(...internalMsgs) + // Echo assistant messages back so CCR displays them + if (message.type === 'assistant' && options.replayUserMessages) { + output.enqueue(message) + } + continue + } + // After handling control, keep-alive, env-var, assistant, and system + // messages above, only user messages should remain. + if (message.type !== 'user') { + continue + } + + // First prompt message implicitly initializes if not already done. + initialized = true + + // Check for duplicate user message - skip if already processed + if (message.uuid) { + const sessionId = getSessionId() as UUID + const existsInSession = await doesMessageExistInSession( + sessionId, + message.uuid, + ) + + // Check both historical duplicates (from file) and runtime duplicates (this session) + if (existsInSession || receivedMessageUuids.has(message.uuid)) { + logForDebugging(`Skipping duplicate user message: ${message.uuid}`) + // Send acknowledgment for duplicate message if replay mode is enabled + if (options.replayUserMessages) { + logForDebugging( + `Sending acknowledgment for duplicate user message: ${message.uuid}`, + ) + output.enqueue({ + type: 'user', + message: message.message, + session_id: sessionId, + parent_tool_use_id: null, + uuid: message.uuid, + timestamp: message.timestamp, + isReplay: true, + } as SDKUserMessageReplay) + } + // Historical dup = transcript already has this turn's output, so it + // ran but its lifecycle was never closed (interrupted before ack). + // Runtime dups don't need this — the original enqueue path closes them. + if (existsInSession) { + notifyCommandLifecycle(message.uuid, 'completed') + } + // Don't enqueue duplicate messages for execution + continue + } + + // Track this UUID to prevent runtime duplicates + trackReceivedMessageUuid(message.uuid) + } + + enqueue({ + mode: 'prompt' as const, + // file_attachments rides the protobuf catchall from the web composer. + // Same-ref no-op when absent (no 'file_attachments' key). + value: await resolveAndPrepend(message, message.message.content), + uuid: message.uuid, + priority: message.priority, + }) + // Increment prompt count for attribution tracking and save snapshot + // The snapshot persists promptCount so it survives compaction + if (feature('COMMIT_ATTRIBUTION')) { + setAppState(prev => ({ + ...prev, + attribution: incrementPromptCount(prev.attribution, snapshot => { + void recordAttributionSnapshot(snapshot).catch(error => { + logForDebugging(`Attribution: Failed to save snapshot: ${error}`) + }) + }), + })) + } + void run() + } + inputClosed = true + cronScheduler?.stop() + if (!running) { + // If a push-suggestion is in-flight, wait for it to emit before closing + // the output stream (5 s safety timeout to prevent hanging). + if (suggestionState.inflightPromise) { + await Promise.race([suggestionState.inflightPromise, sleep(5000)]) + } + suggestionState.abortController?.abort() + suggestionState.abortController = null + await finalizePendingAsyncHooks() + unsubscribeSkillChanges() + unsubscribeAuthStatus?.() + statusListeners.delete(rateLimitListener) + output.done() + } + })() + + return output +} + +/** + * Creates a CanUseToolFn that incorporates a custom permission prompt tool. + * This function converts the permissionPromptTool into a CanUseToolFn that can be used in ask.tsx + */ +export function createCanUseToolWithPermissionPrompt( + permissionPromptTool: PermissionPromptTool, +): CanUseToolFn { + const canUseTool: CanUseToolFn = async ( + tool, + input, + toolUseContext, + assistantMessage, + toolUseId, + forceDecision, + ) => { + const mainPermissionResult = + forceDecision ?? + (await hasPermissionsToUseTool( + tool, + input, + toolUseContext, + assistantMessage, + toolUseId, + )) + + // If the tool is allowed or denied, return the result + if ( + mainPermissionResult.behavior === 'allow' || + mainPermissionResult.behavior === 'deny' + ) { + return mainPermissionResult + } + + // Race the permission prompt tool against the abort signal. + // + // Why we need this: The permission prompt tool may block indefinitely waiting + // for user input (e.g., via stdin or a UI dialog). If the user triggers an + // interrupt (Ctrl+C), we need to detect it even while the tool is blocked. + // Without this race, the abort check would only run AFTER the tool completes, + // which may never happen if the tool is waiting for input that will never come. + // + // The second check (combinedSignal.aborted) handles a race condition where + // abort fires after Promise.race resolves but before we reach this check. + const { signal: combinedSignal, cleanup: cleanupAbortListener } = + createCombinedAbortSignal(toolUseContext.abortController.signal) + + // Check if already aborted before starting the race + if (combinedSignal.aborted) { + cleanupAbortListener() + return { + behavior: 'deny', + message: 'Permission prompt was aborted.', + decisionReason: { + type: 'permissionPromptTool' as const, + permissionPromptToolName: tool.name, + toolResult: undefined, + }, + } + } + + const abortPromise = new Promise<'aborted'>(resolve => { + combinedSignal.addEventListener('abort', () => resolve('aborted'), { + once: true, + }) + }) + + const toolCallPromise = permissionPromptTool.call( + { + tool_name: tool.name, + input, + tool_use_id: toolUseId, + }, + toolUseContext, + canUseTool, + assistantMessage, + ) + + const raceResult = await Promise.race([toolCallPromise, abortPromise]) + cleanupAbortListener() + + if (raceResult === 'aborted' || combinedSignal.aborted) { + return { + behavior: 'deny', + message: 'Permission prompt was aborted.', + decisionReason: { + type: 'permissionPromptTool' as const, + permissionPromptToolName: tool.name, + toolResult: undefined, + }, + } + } + + // TypeScript narrowing: after the abort check, raceResult must be ToolResult + const result = raceResult as Awaited + + const permissionToolResultBlockParam = + permissionPromptTool.mapToolResultToToolResultBlockParam(result.data, '1') + if ( + !permissionToolResultBlockParam.content || + !Array.isArray(permissionToolResultBlockParam.content) || + !permissionToolResultBlockParam.content[0] || + permissionToolResultBlockParam.content[0].type !== 'text' || + typeof permissionToolResultBlockParam.content[0].text !== 'string' + ) { + throw new Error( + 'Permission prompt tool returned an invalid result. Expected a single text block param with type="text" and a string text value.', + ) + } + return permissionPromptToolResultToPermissionDecision( + permissionToolOutputSchema().parse( + safeParseJSON(permissionToolResultBlockParam.content[0].text), + ), + permissionPromptTool, + input, + toolUseContext, + ) + } + return canUseTool +} + +// Exported for testing — regression: this used to crash at construction when +// getMcpTools() was empty (before per-server connects populated appState). +export function getCanUseToolFn( + permissionPromptToolName: string | undefined, + structuredIO: StructuredIO, + getMcpTools: () => Tool[], + onPermissionPrompt?: (details: RequiresActionDetails) => void, +): CanUseToolFn { + if (permissionPromptToolName === 'stdio') { + return structuredIO.createCanUseTool(onPermissionPrompt) + } + if (!permissionPromptToolName) { + return async ( + tool, + input, + toolUseContext, + assistantMessage, + toolUseId, + forceDecision, + ) => + forceDecision ?? + (await hasPermissionsToUseTool( + tool, + input, + toolUseContext, + assistantMessage, + toolUseId, + )) + } + // Lazy lookup: MCP connects are per-server incremental in print mode, so + // the tool may not be in appState yet at init time. Resolve on first call + // (first permission prompt), by which point connects have had time to finish. + let resolved: CanUseToolFn | null = null + return async ( + tool, + input, + toolUseContext, + assistantMessage, + toolUseId, + forceDecision, + ) => { + if (!resolved) { + const mcpTools = getMcpTools() + const permissionPromptTool = mcpTools.find(t => + toolMatchesName(t, permissionPromptToolName), + ) as PermissionPromptTool | undefined + if (!permissionPromptTool) { + const error = `Error: MCP tool ${permissionPromptToolName} (passed via --permission-prompt-tool) not found. Available MCP tools: ${mcpTools.map(t => t.name).join(', ') || 'none'}` + process.stderr.write(`${error}\n`) + gracefulShutdownSync(1) + throw new Error(error) + } + if (!permissionPromptTool.inputJSONSchema) { + const error = `Error: tool ${permissionPromptToolName} (passed via --permission-prompt-tool) must be an MCP tool` + process.stderr.write(`${error}\n`) + gracefulShutdownSync(1) + throw new Error(error) + } + resolved = createCanUseToolWithPermissionPrompt(permissionPromptTool) + } + return resolved( + tool, + input, + toolUseContext, + assistantMessage, + toolUseId, + forceDecision, + ) + } +} + +async function handleInitializeRequest( + request: SDKControlInitializeRequest, + requestId: string, + initialized: boolean, + output: Stream, + commands: Command[], + modelInfos: ModelInfo[], + structuredIO: StructuredIO, + enableAuthStatus: boolean, + options: { + systemPrompt: string | undefined + appendSystemPrompt: string | undefined + agent?: string | undefined + userSpecifiedModel?: string | undefined + [key: string]: unknown + }, + agents: AgentDefinition[], + getAppState: () => AppState, +): Promise { + if (initialized) { + output.enqueue({ + type: 'control_response', + response: { + subtype: 'error', + error: 'Already initialized', + request_id: requestId, + pending_permission_requests: + structuredIO.getPendingPermissionRequests(), + }, + }) + return + } + + // Apply systemPrompt/appendSystemPrompt from stdin to avoid ARG_MAX limits + if (request.systemPrompt !== undefined) { + options.systemPrompt = request.systemPrompt + } + if (request.appendSystemPrompt !== undefined) { + options.appendSystemPrompt = request.appendSystemPrompt + } + if (request.promptSuggestions !== undefined) { + options.promptSuggestions = request.promptSuggestions + } + + // Merge agents from stdin to avoid ARG_MAX limits + if (request.agents) { + const stdinAgents = parseAgentsFromJson(request.agents, 'flagSettings') + agents.push(...stdinAgents) + } + + // Re-evaluate main thread agent after SDK agents are merged + // This allows --agent to reference agents defined via SDK + if (options.agent) { + // If main.tsx already found this agent (filesystem-defined), it already + // applied systemPrompt/model/initialPrompt. Skip to avoid double-apply. + const alreadyResolved = getMainThreadAgentType() === options.agent + const mainThreadAgent = agents.find(a => a.agentType === options.agent) + if (mainThreadAgent && !alreadyResolved) { + // Update the main thread agent type in bootstrap state + setMainThreadAgentType(mainThreadAgent.agentType) + + // Apply the agent's system prompt if user hasn't specified a custom one + // SDK agents are always custom agents (not built-in), so getSystemPrompt() takes no args + if (!options.systemPrompt && !isBuiltInAgent(mainThreadAgent)) { + const agentSystemPrompt = mainThreadAgent.getSystemPrompt() + if (agentSystemPrompt) { + options.systemPrompt = agentSystemPrompt + } + } + + // Apply the agent's model if user didn't specify one and agent has a model + if ( + !options.userSpecifiedModel && + mainThreadAgent.model && + mainThreadAgent.model !== 'inherit' + ) { + const agentModel = parseUserSpecifiedModel(mainThreadAgent.model) + setMainLoopModelOverride(agentModel) + } + + // SDK-defined agents arrive via init, so main.tsx's lookup missed them. + if (mainThreadAgent.initialPrompt) { + structuredIO.prependUserMessage(mainThreadAgent.initialPrompt) + } + } else if (mainThreadAgent?.initialPrompt) { + // Filesystem-defined agent (alreadyResolved by main.tsx). main.tsx + // handles initialPrompt for the string inputPrompt case, but when + // inputPrompt is an AsyncIterable (SDK stream-json), it can't + // concatenate — fall back to prependUserMessage here. + structuredIO.prependUserMessage(mainThreadAgent.initialPrompt) + } + } + + const settings = getSettings_DEPRECATED() + const outputStyle = settings?.outputStyle || DEFAULT_OUTPUT_STYLE_NAME + const availableOutputStyles = await getAllOutputStyles(getCwd()) + + // Get account information + const accountInfo = getAccountInformation() + if (request.hooks) { + const hooks: Partial> = {} + for (const [event, matchers] of Object.entries(request.hooks)) { + hooks[event as HookEvent] = matchers.map(matcher => { + const callbacks = matcher.hookCallbackIds.map(callbackId => { + return structuredIO.createHookCallback(callbackId, matcher.timeout) + }) + return { + matcher: matcher.matcher, + hooks: callbacks, + } + }) + } + registerHookCallbacks(hooks) + } + if (request.jsonSchema) { + setInitJsonSchema(request.jsonSchema) + } + const initResponse: SDKControlInitializeResponse = { + commands: commands + .filter(cmd => cmd.userInvocable !== false) + .map(cmd => ({ + name: getCommandName(cmd), + description: formatDescriptionWithSource(cmd), + argumentHint: cmd.argumentHint || '', + })), + agents: agents.map(agent => ({ + name: agent.agentType, + description: agent.whenToUse, + // 'inherit' is an internal sentinel; normalize to undefined for the public API + model: agent.model === 'inherit' ? undefined : agent.model, + })), + output_style: outputStyle, + available_output_styles: Object.keys(availableOutputStyles), + models: modelInfos, + account: { + email: accountInfo?.email, + organization: accountInfo?.organization, + subscriptionType: accountInfo?.subscription, + tokenSource: accountInfo?.tokenSource, + apiKeySource: accountInfo?.apiKeySource, + // getAccountInformation() returns undefined under 3P providers, so the + // other fields are all absent. apiProvider disambiguates "not logged + // in" (firstParty + tokenSource:none) from "3P, login not applicable". + apiProvider: getAPIProvider(), + }, + pid: process.pid, + } + + if (isFastModeEnabled() && isFastModeAvailable()) { + const appState = getAppState() + initResponse.fast_mode_state = getFastModeState( + options.userSpecifiedModel ?? null, + appState.fastMode, + ) + } + + output.enqueue({ + type: 'control_response', + response: { + subtype: 'success', + request_id: requestId, + response: initResponse, + }, + }) + + // After the initialize message, check the auth status- + // This will get notified of changes, but we also want to send the + // initial state. + if (enableAuthStatus) { + const authStatusManager = AwsAuthStatusManager.getInstance() + const status = authStatusManager.getStatus() + if (status) { + output.enqueue({ + type: 'auth_status', + isAuthenticating: status.isAuthenticating, + output: status.output, + error: status.error, + uuid: randomUUID(), + session_id: getSessionId(), + }) + } + } +} + +async function handleRewindFiles( + userMessageId: UUID, + appState: AppState, + setAppState: (updater: (prev: AppState) => AppState) => void, + dryRun: boolean, +): Promise { + if (!fileHistoryEnabled()) { + return { canRewind: false, error: 'File rewinding is not enabled.' } + } + if (!fileHistoryCanRestore(appState.fileHistory, userMessageId)) { + return { + canRewind: false, + error: 'No file checkpoint found for this message.', + } + } + + if (dryRun) { + const diffStats = await fileHistoryGetDiffStats( + appState.fileHistory, + userMessageId, + ) + return { + canRewind: true, + filesChanged: diffStats?.filesChanged, + insertions: diffStats?.insertions, + deletions: diffStats?.deletions, + } + } + + try { + await fileHistoryRewind( + updater => + setAppState(prev => ({ + ...prev, + fileHistory: updater(prev.fileHistory), + })), + userMessageId, + ) + } catch (error) { + return { + canRewind: false, + error: `Failed to rewind: ${errorMessage(error)}`, + } + } + + return { canRewind: true } +} + +function handleSetPermissionMode( + request: { mode: InternalPermissionMode }, + requestId: string, + toolPermissionContext: ToolPermissionContext, + output: Stream, +): ToolPermissionContext { + // Check if trying to switch to bypassPermissions mode + if (request.mode === 'bypassPermissions') { + if (isBypassPermissionsModeDisabled()) { + output.enqueue({ + type: 'control_response', + response: { + subtype: 'error', + request_id: requestId, + error: + 'Cannot set permission mode to bypassPermissions because it is disabled by settings or configuration', + }, + }) + return toolPermissionContext + } + if (!toolPermissionContext.isBypassPermissionsModeAvailable) { + output.enqueue({ + type: 'control_response', + response: { + subtype: 'error', + request_id: requestId, + error: + 'Cannot set permission mode to bypassPermissions because the session was not launched with --dangerously-skip-permissions', + }, + }) + return toolPermissionContext + } + } + + // Check if trying to switch to auto mode without the classifier gate + if ( + feature('TRANSCRIPT_CLASSIFIER') && + request.mode === 'auto' && + !isAutoModeGateEnabled() + ) { + const reason = getAutoModeUnavailableReason() + output.enqueue({ + type: 'control_response', + response: { + subtype: 'error', + request_id: requestId, + error: reason + ? `Cannot set permission mode to auto: ${getAutoModeUnavailableNotification(reason)}` + : 'Cannot set permission mode to auto', + }, + }) + return toolPermissionContext + } + + // Allow the mode switch + output.enqueue({ + type: 'control_response', + response: { + subtype: 'success', + request_id: requestId, + response: { + mode: request.mode, + }, + }, + }) + + return { + ...transitionPermissionMode( + toolPermissionContext.mode, + request.mode, + toolPermissionContext, + ), + mode: request.mode, + } +} + +/** + * IDE-triggered channel enable. Derives the ChannelEntry from the connection's + * pluginSource (IDE can't spoof kind/marketplace — we only take the server + * name), appends it to session allowedChannels, and runs the full gate. On + * gate failure, rolls back the append. On success, registers a notification + * handler that enqueues channel messages at priority:'next' — drainCommandQueue + * picks them up between turns. + * + * Intentionally does NOT register the claude/channel/permission handler that + * useManageMCPConnections sets up for interactive mode. That handler resolves + * a pending dialog inside handleInteractivePermission — but print.ts never + * calls handleInteractivePermission. When SDK permission lands on 'ask', it + * goes to the consumer's canUseTool callback over stdio; there is no CLI-side + * dialog for a remote "yes tbxkq" to resolve. If an IDE wants channel-relayed + * tool approval, that's IDE-side plumbing against its own pending-map. (Also + * gated separately by tengu_harbor_permissions — not yet shipping on + * interactive either.) + */ +function handleChannelEnable( + requestId: string, + serverName: string, + connectionPool: readonly MCPServerConnection[], + output: Stream, +): void { + const respondError = (error: string) => + output.enqueue({ + type: 'control_response', + response: { subtype: 'error', request_id: requestId, error }, + }) + + if (!(feature('KAIROS') || feature('KAIROS_CHANNELS'))) { + return respondError('channels feature not available in this build') + } + + // Only a 'connected' client has .capabilities and .client to register the + // handler on. The pool spread at the call site matches mcp_status. + const connection = connectionPool.find( + c => c.name === serverName && c.type === 'connected', + ) + if (!connection || connection.type !== 'connected') { + return respondError(`server ${serverName} is not connected`) + } + + const pluginSource = connection.config.pluginSource + const parsed = pluginSource ? parsePluginIdentifier(pluginSource) : undefined + if (!parsed?.marketplace) { + // No pluginSource or @-less source — can never pass the {plugin, + // marketplace}-keyed allowlist. Short-circuit with the same reason the + // gate would produce. + return respondError( + `server ${serverName} is not plugin-sourced; channel_enable requires a marketplace plugin`, + ) + } + + const entry: ChannelEntry = { + kind: 'plugin', + name: parsed.name, + marketplace: parsed.marketplace, + } + // Idempotency: don't double-append on repeat enable. + const prior = getAllowedChannels() + const already = prior.some( + e => + e.kind === 'plugin' && + e.name === entry.name && + e.marketplace === entry.marketplace, + ) + if (!already) setAllowedChannels([...prior, entry]) + + const gate = gateChannelServer( + serverName, + connection.capabilities, + pluginSource, + ) + if (gate.action === 'skip') { + // Rollback — only remove the entry we appended. + if (!already) setAllowedChannels(prior) + return respondError(gate.reason) + } + + const pluginId = + `${entry.name}@${entry.marketplace}` as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + logMCPDebug(serverName, 'Channel notifications registered') + logEvent('tengu_mcp_channel_enable', { plugin: pluginId }) + + // Identical enqueue shape to the interactive register block in + // useManageMCPConnections. drainCommandQueue processes it between turns — + // channel messages queue at priority 'next' and are seen by the model on + // the turn after they arrive. + connection.client.setNotificationHandler( + ChannelMessageNotificationSchema(), + async notification => { + const { content, meta } = notification.params + logMCPDebug( + serverName, + `notifications/claude/channel: ${content.slice(0, 80)}`, + ) + logEvent('tengu_mcp_channel_message', { + content_length: content.length, + meta_key_count: Object.keys(meta ?? {}).length, + entry_kind: + 'plugin' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + is_dev: false, + plugin: pluginId, + }) + enqueue({ + mode: 'prompt', + value: wrapChannelMessage(serverName, content, meta), + priority: 'next', + isMeta: true, + origin: { kind: 'channel', server: serverName }, + skipSlashCommands: true, + }) + }, + ) + + output.enqueue({ + type: 'control_response', + response: { + subtype: 'success', + request_id: requestId, + response: undefined, + }, + }) +} + +/** + * Re-register the channel notification handler after mcp_reconnect / + * mcp_toggle creates a new client. handleChannelEnable bound the handler to + * the OLD client object; allowedChannels survives the reconnect but the + * handler binding does not. Without this, channel messages silently drop + * after a reconnect while the IDE still believes the channel is live. + * + * Mirrors the interactive CLI's onConnectionAttempt in + * useManageMCPConnections, which re-gates on every new connection. Paired + * with registerElicitationHandlers at the same call sites. + * + * No-op if the server was never channel-enabled: gateChannelServer calls + * findChannelEntry internally and returns skip/session for an unlisted + * server, so reconnecting a non-channel MCP server costs one feature-flag + * check. + */ +function reregisterChannelHandlerAfterReconnect( + connection: MCPServerConnection, +): void { + if (!(feature('KAIROS') || feature('KAIROS_CHANNELS'))) return + if (connection.type !== 'connected') return + + const gate = gateChannelServer( + connection.name, + connection.capabilities, + connection.config.pluginSource, + ) + if (gate.action !== 'register') return + + const entry = findChannelEntry(connection.name, getAllowedChannels()) + const pluginId = + entry?.kind === 'plugin' + ? (`${entry.name}@${entry.marketplace}` as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS) + : undefined + + logMCPDebug( + connection.name, + 'Channel notifications re-registered after reconnect', + ) + connection.client.setNotificationHandler( + ChannelMessageNotificationSchema(), + async notification => { + const { content, meta } = notification.params + logMCPDebug( + connection.name, + `notifications/claude/channel: ${content.slice(0, 80)}`, + ) + logEvent('tengu_mcp_channel_message', { + content_length: content.length, + meta_key_count: Object.keys(meta ?? {}).length, + entry_kind: + entry?.kind as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + is_dev: entry?.dev ?? false, + plugin: pluginId, + }) + enqueue({ + mode: 'prompt', + value: wrapChannelMessage(connection.name, content, meta), + priority: 'next', + isMeta: true, + origin: { kind: 'channel', server: connection.name }, + skipSlashCommands: true, + }) + }, + ) +} + +/** + * Emits an error message in the correct format based on outputFormat. + * When using stream-json, writes JSON to stdout; otherwise writes plain text to stderr. + */ +function emitLoadError( + message: string, + outputFormat: string | undefined, +): void { + if (outputFormat === 'stream-json') { + const errorResult = { + type: 'result', + subtype: 'error_during_execution', + duration_ms: 0, + duration_api_ms: 0, + is_error: true, + num_turns: 0, + stop_reason: null, + session_id: getSessionId(), + total_cost_usd: 0, + usage: EMPTY_USAGE, + modelUsage: {}, + permission_denials: [], + uuid: randomUUID(), + errors: [message], + } + process.stdout.write(jsonStringify(errorResult) + '\n') + } else { + process.stderr.write(message + '\n') + } +} + +/** + * Removes an interrupted user message and its synthetic assistant sentinel + * from the message array. Used during gateway-triggered restarts to clean up + * the message history before re-enqueuing the interrupted prompt. + * + * @internal Exported for testing + */ +export function removeInterruptedMessage( + messages: Message[], + interruptedUserMessage: NormalizedUserMessage, +): void { + const idx = messages.findIndex(m => m.uuid === interruptedUserMessage.uuid) + if (idx !== -1) { + // Remove the user message and the sentinel that immediately follows it. + // splice safely handles the case where idx is the last element. + messages.splice(idx, 2) + } +} + +type LoadInitialMessagesResult = { + messages: Message[] + turnInterruptionState?: TurnInterruptionState + agentSetting?: string +} + +async function loadInitialMessages( + setAppState: (f: (prev: AppState) => AppState) => void, + options: { + continue: boolean | undefined + teleport: string | true | null | undefined + resume: string | boolean | undefined + resumeSessionAt: string | undefined + forkSession: boolean | undefined + outputFormat: string | undefined + sessionStartHooksPromise?: ReturnType + restoredWorkerState: Promise + }, +): Promise { + const persistSession = !isSessionPersistenceDisabled() + // Handle continue in print mode + if (options.continue) { + try { + logEvent('tengu_continue_print', {}) + + const result = await loadConversationForResume( + undefined /* sessionId */, + undefined /* file path */, + ) + if (result) { + // Match coordinator mode to the resumed session's mode + if (feature('COORDINATOR_MODE') && coordinatorModeModule) { + const warning = coordinatorModeModule.matchSessionMode(result.mode) + if (warning) { + process.stderr.write(warning + '\n') + // Refresh agent definitions to reflect the mode switch + const { + getAgentDefinitionsWithOverrides, + getActiveAgentsFromList, + } = + // eslint-disable-next-line @typescript-eslint/no-require-imports + require('../tools/AgentTool/loadAgentsDir.js') as typeof import('../tools/AgentTool/loadAgentsDir.js') + getAgentDefinitionsWithOverrides.cache.clear?.() + const freshAgentDefs = await getAgentDefinitionsWithOverrides( + getCwd(), + ) + + setAppState(prev => ({ + ...prev, + agentDefinitions: { + ...freshAgentDefs, + allAgents: freshAgentDefs.allAgents, + activeAgents: getActiveAgentsFromList(freshAgentDefs.allAgents), + }, + })) + } + } + + // Reuse the resumed session's ID + if (!options.forkSession) { + if (result.sessionId) { + switchSession( + asSessionId(result.sessionId), + result.fullPath ? dirname(result.fullPath) : null, + ) + if (persistSession) { + await resetSessionFilePointer() + } + } + } + restoreSessionStateFromLog(result, setAppState) + + // Restore session metadata so it's re-appended on exit via reAppendSessionMetadata + restoreSessionMetadata( + options.forkSession + ? { ...result, worktreeSession: undefined } + : result, + ) + + // Write mode entry for the resumed session + if (feature('COORDINATOR_MODE') && coordinatorModeModule) { + saveMode( + coordinatorModeModule.isCoordinatorMode() + ? 'coordinator' + : 'normal', + ) + } + + return { + messages: result.messages, + turnInterruptionState: result.turnInterruptionState, + agentSetting: result.agentSetting, + } + } + } catch (error) { + logError(error) + gracefulShutdownSync(1) + return { messages: [] } + } + } + + // Handle teleport in print mode + if (options.teleport) { + try { + if (!isPolicyAllowed('allow_remote_sessions')) { + throw new Error( + "Remote sessions are disabled by your organization's policy.", + ) + } + + logEvent('tengu_teleport_print', {}) + + if (typeof options.teleport !== 'string') { + throw new Error('No session ID provided for teleport') + } + + const { + checkOutTeleportedSessionBranch, + processMessagesForTeleportResume, + teleportResumeCodeSession, + validateGitState, + } = await import('src/utils/teleport.js') + await validateGitState() + const teleportResult = await teleportResumeCodeSession(options.teleport) + const { branchError } = await checkOutTeleportedSessionBranch( + teleportResult.branch, + ) + return { + messages: processMessagesForTeleportResume( + teleportResult.log, + branchError, + ), + } + } catch (error) { + logError(error) + gracefulShutdownSync(1) + return { messages: [] } + } + } + + // Handle resume in print mode (accepts session ID or URL) + // URLs are [ANT-ONLY] + if (options.resume) { + try { + logEvent('tengu_resume_print', {}) + + // In print mode - we require a valid session ID, JSONL file or URL + const parsedSessionId = parseSessionIdentifier( + typeof options.resume === 'string' ? options.resume : '', + ) + if (!parsedSessionId) { + let errorMessage = + 'Error: --resume requires a valid session ID when used with --print. Usage: claude -p --resume ' + if (typeof options.resume === 'string') { + errorMessage += `. Session IDs must be in UUID format (e.g., 550e8400-e29b-41d4-a716-446655440000). Provided value "${options.resume}" is not a valid UUID` + } + emitLoadError(errorMessage, options.outputFormat) + gracefulShutdownSync(1) + return { messages: [] } + } + + // Hydrate local transcript from remote before loading + if (isEnvTruthy(process.env.CLAUDE_CODE_USE_CCR_V2)) { + // Await restore alongside hydration so SSE catchup lands on + // restored state, not a fresh default. + const [, metadata] = await Promise.all([ + hydrateFromCCRv2InternalEvents(parsedSessionId.sessionId), + options.restoredWorkerState, + ]) + if (metadata) { + setAppState(externalMetadataToAppState(metadata)) + if (typeof metadata.model === 'string') { + setMainLoopModelOverride(metadata.model) + } + } + } else if ( + parsedSessionId.isUrl && + parsedSessionId.ingressUrl && + isEnvTruthy(process.env.ENABLE_SESSION_PERSISTENCE) + ) { + // v1: fetch session logs from Session Ingress + await hydrateRemoteSession( + parsedSessionId.sessionId, + parsedSessionId.ingressUrl, + ) + } + + // Load the conversation with the specified session ID + const result = await loadConversationForResume( + parsedSessionId.sessionId, + parsedSessionId.jsonlFile || undefined, + ) + + // hydrateFromCCRv2InternalEvents writes an empty transcript file for + // fresh sessions (writeFile(sessionFile, '') with zero events), so + // loadConversationForResume returns {messages: []} not null. Treat + // empty the same as null so SessionStart still fires. + if (!result || result.messages.length === 0) { + // For URL-based or CCR v2 resume, start with empty session (it was hydrated but empty) + if ( + parsedSessionId.isUrl || + isEnvTruthy(process.env.CLAUDE_CODE_USE_CCR_V2) + ) { + // Execute SessionStart hooks for startup since we're starting a new session + return { + messages: await (options.sessionStartHooksPromise ?? + processSessionStartHooks('startup')), + } + } else { + emitLoadError( + `No conversation found with session ID: ${parsedSessionId.sessionId}`, + options.outputFormat, + ) + gracefulShutdownSync(1) + return { messages: [] } + } + } + + // Handle resumeSessionAt feature + if (options.resumeSessionAt) { + const index = result.messages.findIndex( + m => m.uuid === options.resumeSessionAt, + ) + if (index < 0) { + emitLoadError( + `No message found with message.uuid of: ${options.resumeSessionAt}`, + options.outputFormat, + ) + gracefulShutdownSync(1) + return { messages: [] } + } + + result.messages = index >= 0 ? result.messages.slice(0, index + 1) : [] + } + + // Match coordinator mode to the resumed session's mode + if (feature('COORDINATOR_MODE') && coordinatorModeModule) { + const warning = coordinatorModeModule.matchSessionMode(result.mode) + if (warning) { + process.stderr.write(warning + '\n') + // Refresh agent definitions to reflect the mode switch + const { getAgentDefinitionsWithOverrides, getActiveAgentsFromList } = + // eslint-disable-next-line @typescript-eslint/no-require-imports + require('../tools/AgentTool/loadAgentsDir.js') as typeof import('../tools/AgentTool/loadAgentsDir.js') + getAgentDefinitionsWithOverrides.cache.clear?.() + const freshAgentDefs = await getAgentDefinitionsWithOverrides( + getCwd(), + ) + + setAppState(prev => ({ + ...prev, + agentDefinitions: { + ...freshAgentDefs, + allAgents: freshAgentDefs.allAgents, + activeAgents: getActiveAgentsFromList(freshAgentDefs.allAgents), + }, + })) + } + } + + // Reuse the resumed session's ID + if (!options.forkSession && result.sessionId) { + switchSession( + asSessionId(result.sessionId), + result.fullPath ? dirname(result.fullPath) : null, + ) + if (persistSession) { + await resetSessionFilePointer() + } + } + restoreSessionStateFromLog(result, setAppState) + + // Restore session metadata so it's re-appended on exit via reAppendSessionMetadata + restoreSessionMetadata( + options.forkSession + ? { ...result, worktreeSession: undefined } + : result, + ) + + // Write mode entry for the resumed session + if (feature('COORDINATOR_MODE') && coordinatorModeModule) { + saveMode( + coordinatorModeModule.isCoordinatorMode() ? 'coordinator' : 'normal', + ) + } + + return { + messages: result.messages, + turnInterruptionState: result.turnInterruptionState, + agentSetting: result.agentSetting, + } + } catch (error) { + logError(error) + const errorMessage = + error instanceof Error + ? `Failed to resume session: ${error.message}` + : 'Failed to resume session with --print mode' + emitLoadError(errorMessage, options.outputFormat) + gracefulShutdownSync(1) + return { messages: [] } + } + } + + // Join the SessionStart hooks promise kicked in main.tsx (or run fresh if + // it wasn't kicked — e.g. --continue with no prior session falls through + // here with sessionStartHooksPromise undefined because main.tsx guards on continue) + return { + messages: await (options.sessionStartHooksPromise ?? + processSessionStartHooks('startup')), + } +} + +function getStructuredIO( + inputPrompt: string | AsyncIterable, + options: { + sdkUrl: string | undefined + replayUserMessages?: boolean + }, +): StructuredIO { + let inputStream: AsyncIterable + if (typeof inputPrompt === 'string') { + if (inputPrompt.trim() !== '') { + // Normalize to a streaming input. + inputStream = fromArray([ + jsonStringify({ + type: 'user', + session_id: '', + message: { + role: 'user', + content: inputPrompt, + }, + parent_tool_use_id: null, + } satisfies SDKUserMessage), + ]) + } else { + // Empty string - create empty stream + inputStream = fromArray([]) + } + } else { + inputStream = inputPrompt + } + + // Use RemoteIO if sdkUrl is provided, otherwise use regular StructuredIO + return options.sdkUrl + ? new RemoteIO(options.sdkUrl, inputStream, options.replayUserMessages) + : new StructuredIO(inputStream, options.replayUserMessages) +} + +/** + * Handles unexpected permission responses by looking up the unresolved tool + * call in the transcript and enqueuing it for execution. + * + * Returns true if a permission was enqueued, false otherwise. + */ +export async function handleOrphanedPermissionResponse({ + message, + setAppState, + onEnqueued, + handledToolUseIds, +}: { + message: SDKControlResponse + setAppState: (f: (prev: AppState) => AppState) => void + onEnqueued?: () => void + handledToolUseIds: Set +}): Promise { + if ( + message.response.subtype === 'success' && + message.response.response?.toolUseID && + typeof message.response.response.toolUseID === 'string' + ) { + const permissionResult = message.response.response as PermissionResult + const { toolUseID } = permissionResult + if (!toolUseID) { + return false + } + + logForDebugging( + `handleOrphanedPermissionResponse: received orphaned control_response for toolUseID=${toolUseID} request_id=${message.response.request_id}`, + ) + + // Prevent re-processing the same orphaned tool_use. Without this guard, + // duplicate control_response deliveries (e.g. from WebSocket reconnect) + // cause the same tool to be executed multiple times, producing duplicate + // tool_use IDs in the messages array and a 400 error from the API. + // Once corrupted, every retry accumulates more duplicates. + if (handledToolUseIds.has(toolUseID)) { + logForDebugging( + `handleOrphanedPermissionResponse: skipping duplicate orphaned permission for toolUseID=${toolUseID} (already handled)`, + ) + return false + } + + const assistantMessage = await findUnresolvedToolUse(toolUseID) + if (!assistantMessage) { + logForDebugging( + `handleOrphanedPermissionResponse: no unresolved tool_use found for toolUseID=${toolUseID} (already resolved in transcript)`, + ) + return false + } + + handledToolUseIds.add(toolUseID) + logForDebugging( + `handleOrphanedPermissionResponse: enqueuing orphaned permission for toolUseID=${toolUseID} messageID=${assistantMessage.message.id}`, + ) + enqueue({ + mode: 'orphaned-permission' as const, + value: [], + orphanedPermission: { + permissionResult, + assistantMessage, + }, + }) + + onEnqueued?.() + return true + } + return false +} + +export type DynamicMcpState = { + clients: MCPServerConnection[] + tools: Tools + configs: Record +} + +/** + * Converts a process transport config to a scoped config. + * The types are structurally compatible, so we just add the scope. + */ +function toScopedConfig( + config: McpServerConfigForProcessTransport, +): ScopedMcpServerConfig { + // McpServerConfigForProcessTransport is a subset of McpServerConfig + // (it excludes IDE-specific types like sse-ide and ws-ide) + // Adding scope makes it a valid ScopedMcpServerConfig + return { ...config, scope: 'dynamic' } as ScopedMcpServerConfig +} + +/** + * State for SDK MCP servers that run in the SDK process. + */ +export type SdkMcpState = { + configs: Record + clients: MCPServerConnection[] + tools: Tools +} + +/** + * Result of handleMcpSetServers - contains new state and response data. + */ +export type McpSetServersResult = { + response: SDKControlMcpSetServersResponse + newSdkState: SdkMcpState + newDynamicState: DynamicMcpState + sdkServersChanged: boolean +} + +/** + * Handles mcp_set_servers requests by processing both SDK and process-based servers. + * SDK servers run in the SDK process; process-based servers are spawned by the CLI. + * + * Applies enterprise allowedMcpServers/deniedMcpServers policy — same filter as + * --mcp-config (see filterMcpServersByPolicy call in main.tsx). Without this, + * SDK V2 Query.setMcpServers() was a second policy bypass vector. Blocked servers + * are reported in response.errors so the SDK consumer knows why they weren't added. + */ +export async function handleMcpSetServers( + servers: Record, + sdkState: SdkMcpState, + dynamicState: DynamicMcpState, + setAppState: (f: (prev: AppState) => AppState) => void, +): Promise { + // Enforce enterprise MCP policy on process-based servers (stdio/http/sse). + // Mirrors the --mcp-config filter in main.tsx — both user-controlled injection + // paths must have the same gate. type:'sdk' servers are exempt (SDK-managed, + // CLI never spawns/connects for them — see filterMcpServersByPolicy jsdoc). + // Blocked servers go into response.errors so the SDK caller sees why. + const { allowed: allowedServers, blocked } = filterMcpServersByPolicy(servers) + const policyErrors: Record = {} + for (const name of blocked) { + policyErrors[name] = + 'Blocked by enterprise policy (allowedMcpServers/deniedMcpServers)' + } + + // Separate SDK servers from process-based servers + const sdkServers: Record = {} + const processServers: Record = {} + + for (const [name, config] of Object.entries(allowedServers)) { + if (config.type === 'sdk') { + sdkServers[name] = config + } else { + processServers[name] = config + } + } + + // Handle SDK servers + const currentSdkNames = new Set(Object.keys(sdkState.configs)) + const newSdkNames = new Set(Object.keys(sdkServers)) + const sdkAdded: string[] = [] + const sdkRemoved: string[] = [] + + const newSdkConfigs = { ...sdkState.configs } + let newSdkClients = [...sdkState.clients] + let newSdkTools = [...sdkState.tools] + + // Remove SDK servers no longer in desired state + for (const name of currentSdkNames) { + if (!newSdkNames.has(name)) { + const client = newSdkClients.find(c => c.name === name) + if (client && client.type === 'connected') { + await client.cleanup() + } + newSdkClients = newSdkClients.filter(c => c.name !== name) + const prefix = `mcp__${name}__` + newSdkTools = newSdkTools.filter(t => !t.name.startsWith(prefix)) + delete newSdkConfigs[name] + sdkRemoved.push(name) + } + } + + // Add new SDK servers as pending - they'll be upgraded to connected + // when updateSdkMcp() runs on the next query + for (const [name, config] of Object.entries(sdkServers)) { + if (!currentSdkNames.has(name)) { + newSdkConfigs[name] = config + const pendingClient: MCPServerConnection = { + type: 'pending', + name, + config: { ...config, scope: 'dynamic' as const }, + } + newSdkClients = [...newSdkClients, pendingClient] + sdkAdded.push(name) + } + } + + // Handle process-based servers + const processResult = await reconcileMcpServers( + processServers, + dynamicState, + setAppState, + ) + + return { + response: { + added: [...sdkAdded, ...processResult.response.added], + removed: [...sdkRemoved, ...processResult.response.removed], + errors: { ...policyErrors, ...processResult.response.errors }, + }, + newSdkState: { + configs: newSdkConfigs, + clients: newSdkClients, + tools: newSdkTools, + }, + newDynamicState: processResult.newState, + sdkServersChanged: sdkAdded.length > 0 || sdkRemoved.length > 0, + } +} + +/** + * Reconciles the current set of dynamic MCP servers with a new desired state. + * Handles additions, removals, and config changes. + */ +export async function reconcileMcpServers( + desiredConfigs: Record, + currentState: DynamicMcpState, + setAppState: (f: (prev: AppState) => AppState) => void, +): Promise<{ + response: SDKControlMcpSetServersResponse + newState: DynamicMcpState +}> { + const currentNames = new Set(Object.keys(currentState.configs)) + const desiredNames = new Set(Object.keys(desiredConfigs)) + + const toRemove = [...currentNames].filter(n => !desiredNames.has(n)) + const toAdd = [...desiredNames].filter(n => !currentNames.has(n)) + + // Check for config changes (same name, different config) + const toCheck = [...currentNames].filter(n => desiredNames.has(n)) + const toReplace = toCheck.filter(name => { + const currentConfig = currentState.configs[name] + const desiredConfigRaw = desiredConfigs[name] + if (!currentConfig || !desiredConfigRaw) return true + const desiredConfig = toScopedConfig(desiredConfigRaw) + return !areMcpConfigsEqual(currentConfig, desiredConfig) + }) + + const removed: string[] = [] + const added: string[] = [] + const errors: Record = {} + + let newClients = [...currentState.clients] + let newTools = [...currentState.tools] + + // Remove old servers (including ones being replaced) + for (const name of [...toRemove, ...toReplace]) { + const client = newClients.find(c => c.name === name) + const config = currentState.configs[name] + if (client && config) { + if (client.type === 'connected') { + try { + await client.cleanup() + } catch (e) { + logError(e) + } + } + // Clear the memoization cache + await clearServerCache(name, config) + } + + // Remove tools from this server + const prefix = `mcp__${name}__` + newTools = newTools.filter(t => !t.name.startsWith(prefix)) + + // Remove from clients list + newClients = newClients.filter(c => c.name !== name) + + // Track removal (only for actually removed, not replaced) + if (toRemove.includes(name)) { + removed.push(name) + } + } + + // Add new servers (including replacements) + for (const name of [...toAdd, ...toReplace]) { + const config = desiredConfigs[name] + if (!config) continue + const scopedConfig = toScopedConfig(config) + + // SDK servers are managed by the SDK process, not the CLI. + // Just track them without trying to connect. + if (config.type === 'sdk') { + added.push(name) + continue + } + + try { + const client = await connectToServer(name, scopedConfig) + newClients.push(client) + + if (client.type === 'connected') { + const serverTools = await fetchToolsForClient(client) + newTools.push(...serverTools) + } else if (client.type === 'failed') { + errors[name] = client.error || 'Connection failed' + } + + added.push(name) + } catch (e) { + const err = toError(e) + errors[name] = err.message + logError(err) + } + } + + // Build new configs + const newConfigs: Record = {} + for (const name of desiredNames) { + const config = desiredConfigs[name] + if (config) { + newConfigs[name] = toScopedConfig(config) + } + } + + const newState: DynamicMcpState = { + clients: newClients, + tools: newTools, + configs: newConfigs, + } + + // Update AppState with the new tools + setAppState(prev => { + // Get all dynamic server names (current + new) + const allDynamicServerNames = new Set([ + ...Object.keys(currentState.configs), + ...Object.keys(newConfigs), + ]) + + // Remove old dynamic tools + const nonDynamicTools = prev.mcp.tools.filter(t => { + for (const serverName of allDynamicServerNames) { + if (t.name.startsWith(`mcp__${serverName}__`)) { + return false + } + } + return true + }) + + // Remove old dynamic clients + const nonDynamicClients = prev.mcp.clients.filter(c => { + return !allDynamicServerNames.has(c.name) + }) + + return { + ...prev, + mcp: { + ...prev.mcp, + tools: [...nonDynamicTools, ...newTools], + clients: [...nonDynamicClients, ...newClients], + }, + } + }) + + return { + response: { added, removed, errors }, + newState, + } +} diff --git a/src/cli/remoteIO.ts b/src/cli/remoteIO.ts new file mode 100644 index 0000000..7d82c3e --- /dev/null +++ b/src/cli/remoteIO.ts @@ -0,0 +1,255 @@ +import type { StdoutMessage } from 'src/entrypoints/sdk/controlTypes.js' +import { PassThrough } from 'stream' +import { URL } from 'url' +import { getSessionId } from '../bootstrap/state.js' +import { getPollIntervalConfig } from '../bridge/pollConfig.js' +import { registerCleanup } from '../utils/cleanupRegistry.js' +import { setCommandLifecycleListener } from '../utils/commandLifecycle.js' +import { isDebugMode, logForDebugging } from '../utils/debug.js' +import { logForDiagnosticsNoPII } from '../utils/diagLogs.js' +import { isEnvTruthy } from '../utils/envUtils.js' +import { errorMessage } from '../utils/errors.js' +import { gracefulShutdown } from '../utils/gracefulShutdown.js' +import { logError } from '../utils/log.js' +import { writeToStdout } from '../utils/process.js' +import { getSessionIngressAuthToken } from '../utils/sessionIngressAuth.js' +import { + setSessionMetadataChangedListener, + setSessionStateChangedListener, +} from '../utils/sessionState.js' +import { + setInternalEventReader, + setInternalEventWriter, +} from '../utils/sessionStorage.js' +import { ndjsonSafeStringify } from './ndjsonSafeStringify.js' +import { StructuredIO } from './structuredIO.js' +import { CCRClient, CCRInitError } from './transports/ccrClient.js' +import { SSETransport } from './transports/SSETransport.js' +import type { Transport } from './transports/Transport.js' +import { getTransportForUrl } from './transports/transportUtils.js' + +/** + * Bidirectional streaming for SDK mode with session tracking + * Supports WebSocket transport + */ +export class RemoteIO extends StructuredIO { + private url: URL + private transport: Transport + private inputStream: PassThrough + private readonly isBridge: boolean = false + private readonly isDebug: boolean = false + private ccrClient: CCRClient | null = null + private keepAliveTimer: ReturnType | null = null + + constructor( + streamUrl: string, + initialPrompt?: AsyncIterable, + replayUserMessages?: boolean, + ) { + const inputStream = new PassThrough({ encoding: 'utf8' }) + super(inputStream, replayUserMessages) + this.inputStream = inputStream + this.url = new URL(streamUrl) + + // Prepare headers with session token if available + const headers: Record = {} + const sessionToken = getSessionIngressAuthToken() + if (sessionToken) { + headers['Authorization'] = `Bearer ${sessionToken}` + } else { + logForDebugging('[remote-io] No session ingress token available', { + level: 'error', + }) + } + + // Add environment runner version if available (set by Environment Manager) + const erVersion = process.env.CLAUDE_CODE_ENVIRONMENT_RUNNER_VERSION + if (erVersion) { + headers['x-environment-runner-version'] = erVersion + } + + // Provide a callback that re-reads the session token dynamically. + // When the parent process refreshes the token (via token file or env var), + // the transport can pick it up on reconnection. + const refreshHeaders = (): Record => { + const h: Record = {} + const freshToken = getSessionIngressAuthToken() + if (freshToken) { + h['Authorization'] = `Bearer ${freshToken}` + } + const freshErVersion = process.env.CLAUDE_CODE_ENVIRONMENT_RUNNER_VERSION + if (freshErVersion) { + h['x-environment-runner-version'] = freshErVersion + } + return h + } + + // Get appropriate transport based on URL protocol + this.transport = getTransportForUrl( + this.url, + headers, + getSessionId(), + refreshHeaders, + ) + + // Set up data callback + this.isBridge = process.env.CLAUDE_CODE_ENVIRONMENT_KIND === 'bridge' + this.isDebug = isDebugMode() + this.transport.setOnData((data: string) => { + this.inputStream.write(data) + if (this.isBridge && this.isDebug) { + writeToStdout(data.endsWith('\n') ? data : data + '\n') + } + }) + + // Set up close callback to handle connection failures + this.transport.setOnClose(() => { + // End the input stream to trigger graceful shutdown + this.inputStream.end() + }) + + // Initialize CCR v2 client (heartbeats, epoch, state reporting, event writes). + // The CCRClient constructor wires the SSE received-ack handler + // synchronously, so new CCRClient() MUST run before transport.connect() — + // otherwise early SSE frames hit an unwired onEventCallback and their + // 'received' delivery acks are silently dropped. + if (isEnvTruthy(process.env.CLAUDE_CODE_USE_CCR_V2)) { + // CCR v2 is SSE+POST by definition. getTransportForUrl returns + // SSETransport under the same env var, but the two checks live in + // different files — assert the invariant so a future decoupling + // fails loudly here instead of confusingly inside CCRClient. + if (!(this.transport instanceof SSETransport)) { + throw new Error( + 'CCR v2 requires SSETransport; check getTransportForUrl', + ) + } + this.ccrClient = new CCRClient(this.transport, this.url) + const init = this.ccrClient.initialize() + this.restoredWorkerState = init.catch(() => null) + init.catch((error: unknown) => { + logForDiagnosticsNoPII('error', 'cli_worker_lifecycle_init_failed', { + reason: error instanceof CCRInitError ? error.reason : 'unknown', + }) + logError( + new Error(`CCRClient initialization failed: ${errorMessage(error)}`), + ) + void gracefulShutdown(1, 'other') + }) + registerCleanup(async () => this.ccrClient?.close()) + + // Register internal event writer for transcript persistence. + // When set, sessionStorage writes transcript messages as CCR v2 + // internal events instead of v1 Session Ingress. + setInternalEventWriter((eventType, payload, options) => + this.ccrClient!.writeInternalEvent(eventType, payload, options), + ) + + // Register internal event readers for session resume. + // When set, hydrateFromCCRv2InternalEvents() can fetch foreground + // and subagent internal events to reconstruct conversation state. + setInternalEventReader( + () => this.ccrClient!.readInternalEvents(), + () => this.ccrClient!.readSubagentInternalEvents(), + ) + + const LIFECYCLE_TO_DELIVERY = { + started: 'processing', + completed: 'processed', + } as const + setCommandLifecycleListener((uuid, state) => { + this.ccrClient?.reportDelivery(uuid, LIFECYCLE_TO_DELIVERY[state]) + }) + setSessionStateChangedListener((state, details) => { + this.ccrClient?.reportState(state, details) + }) + setSessionMetadataChangedListener(metadata => { + this.ccrClient?.reportMetadata(metadata) + }) + } + + // Start connection only after all callbacks are wired (setOnData above, + // setOnEvent inside new CCRClient() when CCR v2 is enabled). + void this.transport.connect() + + // Push a silent keep_alive frame on a fixed interval so upstream + // proxies and the session-ingress layer don't GC an otherwise-idle + // remote control session. The keep_alive type is filtered before + // reaching any client UI (Query.ts drops it; structuredIO.ts drops it; + // web/iOS/Android never see it in their message loop). Interval comes + // from GrowthBook (tengu_bridge_poll_interval_config + // session_keepalive_interval_v2_ms, default 120s); 0 = disabled. + // Bridge-only: fixes Envoy idle timeout on bridge-topology sessions + // (#21931). byoc workers ran without this before #21931 and do not + // need it — different network path. + const keepAliveIntervalMs = + getPollIntervalConfig().session_keepalive_interval_v2_ms + if (this.isBridge && keepAliveIntervalMs > 0) { + this.keepAliveTimer = setInterval(() => { + logForDebugging('[remote-io] keep_alive sent') + void this.write({ type: 'keep_alive' }).catch(err => { + logForDebugging( + `[remote-io] keep_alive write failed: ${errorMessage(err)}`, + ) + }) + }, keepAliveIntervalMs) + this.keepAliveTimer.unref?.() + } + + // Register for graceful shutdown cleanup + registerCleanup(async () => this.close()) + + // If initial prompt is provided, send it through the input stream + if (initialPrompt) { + // Convert the initial prompt to the input stream format. + // Chunks from stdin may already contain trailing newlines, so strip + // them before appending our own to avoid double-newline issues that + // cause structuredIO to parse empty lines. String() handles both + // string chunks and Buffer objects from process.stdin. + const stream = this.inputStream + void (async () => { + for await (const chunk of initialPrompt) { + stream.write(String(chunk).replace(/\n$/, '') + '\n') + } + })() + } + } + + override flushInternalEvents(): Promise { + return this.ccrClient?.flushInternalEvents() ?? Promise.resolve() + } + + override get internalEventsPending(): number { + return this.ccrClient?.internalEventsPending ?? 0 + } + + /** + * Send output to the transport. + * In bridge mode, control_request messages are always echoed to stdout so the + * bridge parent can detect permission requests. Other messages are echoed only + * in debug mode. + */ + async write(message: StdoutMessage): Promise { + if (this.ccrClient) { + await this.ccrClient.writeEvent(message) + } else { + await this.transport.write(message) + } + if (this.isBridge) { + if (message.type === 'control_request' || this.isDebug) { + writeToStdout(ndjsonSafeStringify(message) + '\n') + } + } + } + + /** + * Clean up connections gracefully + */ + close(): void { + if (this.keepAliveTimer) { + clearInterval(this.keepAliveTimer) + this.keepAliveTimer = null + } + this.transport.close() + this.inputStream.end() + } +} diff --git a/src/cli/structuredIO.ts b/src/cli/structuredIO.ts new file mode 100644 index 0000000..366b56f --- /dev/null +++ b/src/cli/structuredIO.ts @@ -0,0 +1,859 @@ +import { feature } from 'bun:bundle' +import type { + ElicitResult, + JSONRPCMessage, +} from '@modelcontextprotocol/sdk/types.js' +import { randomUUID } from 'crypto' +import type { AssistantMessage } from 'src//types/message.js' +import type { + HookInput, + HookJSONOutput, + PermissionUpdate, + SDKMessage, + SDKUserMessage, +} from 'src/entrypoints/agentSdkTypes.js' +import { SDKControlElicitationResponseSchema } from 'src/entrypoints/sdk/controlSchemas.js' +import type { + SDKControlRequest, + SDKControlResponse, + StdinMessage, + StdoutMessage, +} from 'src/entrypoints/sdk/controlTypes.js' +import type { CanUseToolFn } from 'src/hooks/useCanUseTool.js' +import type { Tool, ToolUseContext } from 'src/Tool.js' +import { type HookCallback, hookJSONOutputSchema } from 'src/types/hooks.js' +import { logForDebugging } from 'src/utils/debug.js' +import { logForDiagnosticsNoPII } from 'src/utils/diagLogs.js' +import { AbortError } from 'src/utils/errors.js' +import { + type Output as PermissionToolOutput, + permissionPromptToolResultToPermissionDecision, + outputSchema as permissionToolOutputSchema, +} from 'src/utils/permissions/PermissionPromptToolResultSchema.js' +import type { + PermissionDecision, + PermissionDecisionReason, +} from 'src/utils/permissions/PermissionResult.js' +import { hasPermissionsToUseTool } from 'src/utils/permissions/permissions.js' +import { writeToStdout } from 'src/utils/process.js' +import { jsonStringify } from 'src/utils/slowOperations.js' +import { z } from 'zod/v4' +import { notifyCommandLifecycle } from '../utils/commandLifecycle.js' +import { normalizeControlMessageKeys } from '../utils/controlMessageCompat.js' +import { executePermissionRequestHooks } from '../utils/hooks.js' +import { + applyPermissionUpdates, + persistPermissionUpdates, +} from '../utils/permissions/PermissionUpdate.js' +import { + notifySessionStateChanged, + type RequiresActionDetails, + type SessionExternalMetadata, +} from '../utils/sessionState.js' +import { jsonParse } from '../utils/slowOperations.js' +import { Stream } from '../utils/stream.js' +import { ndjsonSafeStringify } from './ndjsonSafeStringify.js' + +/** + * Synthetic tool name used when forwarding sandbox network permission + * requests via the can_use_tool control_request protocol. SDK hosts + * see this as a normal tool permission prompt. + */ +export const SANDBOX_NETWORK_ACCESS_TOOL_NAME = 'SandboxNetworkAccess' + +function serializeDecisionReason( + reason: PermissionDecisionReason | undefined, +): string | undefined { + if (!reason) { + return undefined + } + + if ( + (feature('BASH_CLASSIFIER') || feature('TRANSCRIPT_CLASSIFIER')) && + reason.type === 'classifier' + ) { + return reason.reason + } + switch (reason.type) { + case 'rule': + case 'mode': + case 'subcommandResults': + case 'permissionPromptTool': + return undefined + case 'hook': + case 'asyncAgent': + case 'sandboxOverride': + case 'workingDir': + case 'safetyCheck': + case 'other': + return reason.reason + } +} + +function buildRequiresActionDetails( + tool: Tool, + input: Record, + toolUseID: string, + requestId: string, +): RequiresActionDetails { + // Per-tool summary methods may throw on malformed input; permission + // handling must not break because of a bad description. + let description: string + try { + description = + tool.getActivityDescription?.(input) ?? + tool.getToolUseSummary?.(input) ?? + tool.userFacingName(input) + } catch { + description = tool.name + } + return { + tool_name: tool.name, + action_description: description, + tool_use_id: toolUseID, + request_id: requestId, + input, + } +} + +type PendingRequest = { + resolve: (result: T) => void + reject: (error: unknown) => void + schema?: z.Schema + request: SDKControlRequest +} + +/** + * Provides a structured way to read and write SDK messages from stdio, + * capturing the SDK protocol. + */ +// Maximum number of resolved tool_use IDs to track. Once exceeded, the oldest +// entry is evicted. This bounds memory in very long sessions while keeping +// enough history to catch duplicate control_response deliveries. +const MAX_RESOLVED_TOOL_USE_IDS = 1000 + +export class StructuredIO { + readonly structuredInput: AsyncGenerator + private readonly pendingRequests = new Map>() + + // CCR external_metadata read back on worker start; null when the + // transport doesn't restore. Assigned by RemoteIO. + restoredWorkerState: Promise = + Promise.resolve(null) + + private inputClosed = false + private unexpectedResponseCallback?: ( + response: SDKControlResponse, + ) => Promise + + // Tracks tool_use IDs that have been resolved through the normal permission + // flow (or aborted by a hook). When a duplicate control_response arrives + // after the original was already handled, this Set prevents the orphan + // handler from re-processing it — which would push duplicate assistant + // messages into mutableMessages and cause a 400 "tool_use ids must be unique" + // error from the API. + private readonly resolvedToolUseIds = new Set() + private prependedLines: string[] = [] + private onControlRequestSent?: (request: SDKControlRequest) => void + private onControlRequestResolved?: (requestId: string) => void + + // sendRequest() and print.ts both enqueue here; the drain loop is the + // only writer. Prevents control_request from overtaking queued stream_events. + readonly outbound = new Stream() + + constructor( + private readonly input: AsyncIterable, + private readonly replayUserMessages?: boolean, + ) { + this.input = input + this.structuredInput = this.read() + } + + /** + * Records a tool_use ID as resolved so that late/duplicate control_response + * messages for the same tool are ignored by the orphan handler. + */ + private trackResolvedToolUseId(request: SDKControlRequest): void { + if (request.request.subtype === 'can_use_tool') { + this.resolvedToolUseIds.add(request.request.tool_use_id) + if (this.resolvedToolUseIds.size > MAX_RESOLVED_TOOL_USE_IDS) { + // Evict the oldest entry (Sets iterate in insertion order) + const first = this.resolvedToolUseIds.values().next().value + if (first !== undefined) { + this.resolvedToolUseIds.delete(first) + } + } + } + } + + /** Flush pending internal events. No-op for non-remote IO. Overridden by RemoteIO. */ + flushInternalEvents(): Promise { + return Promise.resolve() + } + + /** Internal-event queue depth. Overridden by RemoteIO; zero otherwise. */ + get internalEventsPending(): number { + return 0 + } + + /** + * Queue a user turn to be yielded before the next message from this.input. + * Works before iteration starts and mid-stream — read() re-checks + * prependedLines between each yielded message. + */ + prependUserMessage(content: string): void { + this.prependedLines.push( + jsonStringify({ + type: 'user', + session_id: '', + message: { role: 'user', content }, + parent_tool_use_id: null, + } satisfies SDKUserMessage) + '\n', + ) + } + + private async *read() { + let content = '' + + // Called once before for-await (an empty this.input otherwise skips the + // loop body entirely), then again per block. prependedLines re-check is + // inside the while so a prepend pushed between two messages in the SAME + // block still lands first. + const splitAndProcess = async function* (this: StructuredIO) { + for (;;) { + if (this.prependedLines.length > 0) { + content = this.prependedLines.join('') + content + this.prependedLines = [] + } + const newline = content.indexOf('\n') + if (newline === -1) break + const line = content.slice(0, newline) + content = content.slice(newline + 1) + const message = await this.processLine(line) + if (message) { + logForDiagnosticsNoPII('info', 'cli_stdin_message_parsed', { + type: message.type, + }) + yield message + } + } + }.bind(this) + + yield* splitAndProcess() + + for await (const block of this.input) { + content += block + yield* splitAndProcess() + } + if (content) { + const message = await this.processLine(content) + if (message) { + yield message + } + } + this.inputClosed = true + for (const request of this.pendingRequests.values()) { + // Reject all pending requests if the input stream + request.reject( + new Error('Tool permission stream closed before response received'), + ) + } + } + + getPendingPermissionRequests() { + return Array.from(this.pendingRequests.values()) + .map(entry => entry.request) + .filter(pr => pr.request.subtype === 'can_use_tool') + } + + setUnexpectedResponseCallback( + callback: (response: SDKControlResponse) => Promise, + ): void { + this.unexpectedResponseCallback = callback + } + + /** + * Inject a control_response message to resolve a pending permission request. + * Used by the bridge to feed permission responses from claude.ai into the + * SDK permission flow. + * + * Also sends a control_cancel_request to the SDK consumer so its canUseTool + * callback is aborted via the signal — otherwise the callback hangs. + */ + injectControlResponse(response: SDKControlResponse): void { + const requestId = response.response?.request_id + if (!requestId) return + const request = this.pendingRequests.get(requestId) + if (!request) return + this.trackResolvedToolUseId(request.request) + this.pendingRequests.delete(requestId) + // Cancel the SDK consumer's canUseTool callback — the bridge won. + void this.write({ + type: 'control_cancel_request', + request_id: requestId, + }) + if (response.response.subtype === 'error') { + request.reject(new Error(response.response.error)) + } else { + const result = response.response.response + if (request.schema) { + try { + request.resolve(request.schema.parse(result)) + } catch (error) { + request.reject(error) + } + } else { + request.resolve({}) + } + } + } + + /** + * Register a callback invoked whenever a can_use_tool control_request + * is written to stdout. Used by the bridge to forward permission + * requests to claude.ai. + */ + setOnControlRequestSent( + callback: ((request: SDKControlRequest) => void) | undefined, + ): void { + this.onControlRequestSent = callback + } + + /** + * Register a callback invoked when a can_use_tool control_response arrives + * from the SDK consumer (via stdin). Used by the bridge to cancel the + * stale permission prompt on claude.ai when the SDK consumer wins the race. + */ + setOnControlRequestResolved( + callback: ((requestId: string) => void) | undefined, + ): void { + this.onControlRequestResolved = callback + } + + private async processLine( + line: string, + ): Promise { + // Skip empty lines (e.g. from double newlines in piped stdin) + if (!line) { + return undefined + } + try { + const message = normalizeControlMessageKeys(jsonParse(line)) as + | StdinMessage + | SDKMessage + if (message.type === 'keep_alive') { + // Silently ignore keep-alive messages + return undefined + } + if (message.type === 'update_environment_variables') { + // Apply environment variable updates directly to process.env. + // Used by bridge session runner for auth token refresh + // (CLAUDE_CODE_SESSION_ACCESS_TOKEN) which must be readable + // by the REPL process itself, not just child Bash commands. + const keys = Object.keys(message.variables) + for (const [key, value] of Object.entries(message.variables)) { + process.env[key] = value + } + logForDebugging( + `[structuredIO] applied update_environment_variables: ${keys.join(', ')}`, + ) + return undefined + } + if (message.type === 'control_response') { + // Close lifecycle for every control_response, including duplicates + // and orphans — orphans don't yield to print.ts's main loop, so this + // is the only path that sees them. uuid is server-injected into the + // payload. + const uuid = + 'uuid' in message && typeof message.uuid === 'string' + ? message.uuid + : undefined + if (uuid) { + notifyCommandLifecycle(uuid, 'completed') + } + const request = this.pendingRequests.get(message.response.request_id) + if (!request) { + // Check if this tool_use was already resolved through the normal + // permission flow. Duplicate control_response deliveries (e.g. from + // WebSocket reconnects) arrive after the original was handled, and + // re-processing them would push duplicate assistant messages into + // the conversation, causing API 400 errors. + const responsePayload = + message.response.subtype === 'success' + ? message.response.response + : undefined + const toolUseID = responsePayload?.toolUseID + if ( + typeof toolUseID === 'string' && + this.resolvedToolUseIds.has(toolUseID) + ) { + logForDebugging( + `Ignoring duplicate control_response for already-resolved toolUseID=${toolUseID} request_id=${message.response.request_id}`, + ) + return undefined + } + if (this.unexpectedResponseCallback) { + await this.unexpectedResponseCallback(message) + } + return undefined // Ignore responses for requests we don't know about + } + this.trackResolvedToolUseId(request.request) + this.pendingRequests.delete(message.response.request_id) + // Notify the bridge when the SDK consumer resolves a can_use_tool + // request, so it can cancel the stale permission prompt on claude.ai. + if ( + request.request.request.subtype === 'can_use_tool' && + this.onControlRequestResolved + ) { + this.onControlRequestResolved(message.response.request_id) + } + + if (message.response.subtype === 'error') { + request.reject(new Error(message.response.error)) + return undefined + } + const result = message.response.response + if (request.schema) { + try { + request.resolve(request.schema.parse(result)) + } catch (error) { + request.reject(error) + } + } else { + request.resolve({}) + } + // Propagate control responses when replay is enabled + if (this.replayUserMessages) { + return message + } + return undefined + } + if ( + message.type !== 'user' && + message.type !== 'control_request' && + message.type !== 'assistant' && + message.type !== 'system' + ) { + logForDebugging(`Ignoring unknown message type: ${message.type}`, { + level: 'warn', + }) + return undefined + } + if (message.type === 'control_request') { + if (!message.request) { + exitWithMessage(`Error: Missing request on control_request`) + } + return message + } + if (message.type === 'assistant' || message.type === 'system') { + return message + } + if (message.message.role !== 'user') { + exitWithMessage( + `Error: Expected message role 'user', got '${message.message.role}'`, + ) + } + return message + } catch (error) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.error(`Error parsing streaming input line: ${line}: ${error}`) + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(1) + } + } + + async write(message: StdoutMessage): Promise { + writeToStdout(ndjsonSafeStringify(message) + '\n') + } + + private async sendRequest( + request: SDKControlRequest['request'], + schema: z.Schema, + signal?: AbortSignal, + requestId: string = randomUUID(), + ): Promise { + const message: SDKControlRequest = { + type: 'control_request', + request_id: requestId, + request, + } + if (this.inputClosed) { + throw new Error('Stream closed') + } + if (signal?.aborted) { + throw new Error('Request aborted') + } + this.outbound.enqueue(message) + if (request.subtype === 'can_use_tool' && this.onControlRequestSent) { + this.onControlRequestSent(message) + } + const aborted = () => { + this.outbound.enqueue({ + type: 'control_cancel_request', + request_id: requestId, + }) + // Immediately reject the outstanding promise, without + // waiting for the host to acknowledge the cancellation. + const request = this.pendingRequests.get(requestId) + if (request) { + // Track the tool_use ID as resolved before rejecting, so that a + // late response from the host is ignored by the orphan handler. + this.trackResolvedToolUseId(request.request) + request.reject(new AbortError()) + } + } + if (signal) { + signal.addEventListener('abort', aborted, { + once: true, + }) + } + try { + return await new Promise((resolve, reject) => { + this.pendingRequests.set(requestId, { + request: { + type: 'control_request', + request_id: requestId, + request, + }, + resolve: result => { + resolve(result as Response) + }, + reject, + schema, + }) + }) + } finally { + if (signal) { + signal.removeEventListener('abort', aborted) + } + this.pendingRequests.delete(requestId) + } + } + + createCanUseTool( + onPermissionPrompt?: (details: RequiresActionDetails) => void, + ): CanUseToolFn { + return async ( + tool: Tool, + input: { [key: string]: unknown }, + toolUseContext: ToolUseContext, + assistantMessage: AssistantMessage, + toolUseID: string, + forceDecision?: PermissionDecision, + ): Promise => { + const mainPermissionResult = + forceDecision ?? + (await hasPermissionsToUseTool( + tool, + input, + toolUseContext, + assistantMessage, + toolUseID, + )) + // If the tool is allowed or denied, return the result + if ( + mainPermissionResult.behavior === 'allow' || + mainPermissionResult.behavior === 'deny' + ) { + return mainPermissionResult + } + + // Run PermissionRequest hooks in parallel with the SDK permission + // prompt. In the terminal CLI, hooks race against the interactive + // prompt so that e.g. a hook with --delay 20 doesn't block the UI. + // We need the same behavior here: the SDK host (VS Code, etc.) shows + // its permission dialog immediately while hooks run in the background. + // Whichever resolves first wins; the loser is cancelled/ignored. + + // AbortController used to cancel the SDK request if a hook decides first + const hookAbortController = new AbortController() + const parentSignal = toolUseContext.abortController.signal + // Forward parent abort to our local controller + const onParentAbort = () => hookAbortController.abort() + parentSignal.addEventListener('abort', onParentAbort, { once: true }) + + try { + // Start the hook evaluation (runs in background) + const hookPromise = executePermissionRequestHooksForSDK( + tool.name, + toolUseID, + input, + toolUseContext, + mainPermissionResult.suggestions, + ).then(decision => ({ source: 'hook' as const, decision })) + + // Start the SDK permission prompt immediately (don't wait for hooks) + const requestId = randomUUID() + onPermissionPrompt?.( + buildRequiresActionDetails(tool, input, toolUseID, requestId), + ) + const sdkPromise = this.sendRequest( + { + subtype: 'can_use_tool', + tool_name: tool.name, + input, + permission_suggestions: mainPermissionResult.suggestions, + blocked_path: mainPermissionResult.blockedPath, + decision_reason: serializeDecisionReason( + mainPermissionResult.decisionReason, + ), + tool_use_id: toolUseID, + agent_id: toolUseContext.agentId, + }, + permissionToolOutputSchema(), + hookAbortController.signal, + requestId, + ).then(result => ({ source: 'sdk' as const, result })) + + // Race: hook completion vs SDK prompt response. + // The hook promise always resolves (never rejects), returning + // undefined if no hook made a decision. + const winner = await Promise.race([hookPromise, sdkPromise]) + + if (winner.source === 'hook') { + if (winner.decision) { + // Hook decided — abort the pending SDK request. + // Suppress the expected AbortError rejection from sdkPromise. + sdkPromise.catch(() => {}) + hookAbortController.abort() + return winner.decision + } + // Hook passed through (no decision) — wait for the SDK prompt + const sdkResult = await sdkPromise + return permissionPromptToolResultToPermissionDecision( + sdkResult.result, + tool, + input, + toolUseContext, + ) + } + + // SDK prompt responded first — use its result (hook still running + // in background but its result will be ignored) + return permissionPromptToolResultToPermissionDecision( + winner.result, + tool, + input, + toolUseContext, + ) + } catch (error) { + return permissionPromptToolResultToPermissionDecision( + { + behavior: 'deny', + message: `Tool permission request failed: ${error}`, + toolUseID, + }, + tool, + input, + toolUseContext, + ) + } finally { + // Only transition back to 'running' if no other permission prompts + // are pending (concurrent tool execution can have multiple in-flight). + if (this.getPendingPermissionRequests().length === 0) { + notifySessionStateChanged('running') + } + parentSignal.removeEventListener('abort', onParentAbort) + } + } + } + + createHookCallback(callbackId: string, timeout?: number): HookCallback { + return { + type: 'callback', + timeout, + callback: async ( + input: HookInput, + toolUseID: string | null, + abort: AbortSignal | undefined, + ): Promise => { + try { + const result = await this.sendRequest( + { + subtype: 'hook_callback', + callback_id: callbackId, + input, + tool_use_id: toolUseID || undefined, + }, + hookJSONOutputSchema(), + abort, + ) + return result + } catch (error) { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.error(`Error in hook callback ${callbackId}:`, error) + return {} + } + }, + } + } + + /** + * Sends an elicitation request to the SDK consumer and returns the response. + */ + async handleElicitation( + serverName: string, + message: string, + requestedSchema?: Record, + signal?: AbortSignal, + mode?: 'form' | 'url', + url?: string, + elicitationId?: string, + ): Promise { + try { + const result = await this.sendRequest( + { + subtype: 'elicitation', + mcp_server_name: serverName, + message, + mode, + url, + elicitation_id: elicitationId, + requested_schema: requestedSchema, + }, + SDKControlElicitationResponseSchema(), + signal, + ) + return result + } catch { + return { action: 'cancel' as const } + } + } + + /** + * Creates a SandboxAskCallback that forwards sandbox network permission + * requests to the SDK host as can_use_tool control_requests. + * + * This piggybacks on the existing can_use_tool protocol with a synthetic + * tool name so that SDK hosts (VS Code, CCR, etc.) can prompt the user + * for network access without requiring a new protocol subtype. + */ + createSandboxAskCallback(): (hostPattern: { + host: string + port?: number + }) => Promise { + return async (hostPattern): Promise => { + try { + const result = await this.sendRequest( + { + subtype: 'can_use_tool', + tool_name: SANDBOX_NETWORK_ACCESS_TOOL_NAME, + input: { host: hostPattern.host }, + tool_use_id: randomUUID(), + description: `Allow network connection to ${hostPattern.host}?`, + }, + permissionToolOutputSchema(), + ) + return result.behavior === 'allow' + } catch { + // If the request fails (stream closed, abort, etc.), deny the connection + return false + } + } + } + + /** + * Sends an MCP message to an SDK server and waits for the response + */ + async sendMcpMessage( + serverName: string, + message: JSONRPCMessage, + ): Promise { + const response = await this.sendRequest<{ mcp_response: JSONRPCMessage }>( + { + subtype: 'mcp_message', + server_name: serverName, + message, + }, + z.object({ + mcp_response: z.any() as z.Schema, + }), + ) + return response.mcp_response + } +} + +function exitWithMessage(message: string): never { + // biome-ignore lint/suspicious/noConsole:: intentional console output + console.error(message) + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(1) +} + +/** + * Execute PermissionRequest hooks and return a decision if one is made. + * Returns undefined if no hook made a decision. + */ +async function executePermissionRequestHooksForSDK( + toolName: string, + toolUseID: string, + input: Record, + toolUseContext: ToolUseContext, + suggestions: PermissionUpdate[] | undefined, +): Promise { + const appState = toolUseContext.getAppState() + const permissionMode = appState.toolPermissionContext.mode + + // Iterate directly over the generator instead of using `all` + const hookGenerator = executePermissionRequestHooks( + toolName, + toolUseID, + input, + toolUseContext, + permissionMode, + suggestions, + toolUseContext.abortController.signal, + ) + + for await (const hookResult of hookGenerator) { + if ( + hookResult.permissionRequestResult && + (hookResult.permissionRequestResult.behavior === 'allow' || + hookResult.permissionRequestResult.behavior === 'deny') + ) { + const decision = hookResult.permissionRequestResult + if (decision.behavior === 'allow') { + const finalInput = decision.updatedInput || input + + // Apply permission updates if provided by hook ("always allow") + const permissionUpdates = decision.updatedPermissions ?? [] + if (permissionUpdates.length > 0) { + persistPermissionUpdates(permissionUpdates) + const currentAppState = toolUseContext.getAppState() + const updatedContext = applyPermissionUpdates( + currentAppState.toolPermissionContext, + permissionUpdates, + ) + // Update permission context via setAppState + toolUseContext.setAppState(prev => { + if (prev.toolPermissionContext === updatedContext) return prev + return { ...prev, toolPermissionContext: updatedContext } + }) + } + + return { + behavior: 'allow', + updatedInput: finalInput, + userModified: false, + decisionReason: { + type: 'hook', + hookName: 'PermissionRequest', + }, + } + } else { + // Hook denied the permission + return { + behavior: 'deny', + message: + decision.message || 'Permission denied by PermissionRequest hook', + decisionReason: { + type: 'hook', + hookName: 'PermissionRequest', + }, + } + } + } + } + + return undefined +} diff --git a/src/cli/transports/HybridTransport.ts b/src/cli/transports/HybridTransport.ts new file mode 100644 index 0000000..15500ec --- /dev/null +++ b/src/cli/transports/HybridTransport.ts @@ -0,0 +1,282 @@ +import axios, { type AxiosError } from 'axios' +import type { StdoutMessage } from 'src/entrypoints/sdk/controlTypes.js' +import { logForDebugging } from '../../utils/debug.js' +import { logForDiagnosticsNoPII } from '../../utils/diagLogs.js' +import { getSessionIngressAuthToken } from '../../utils/sessionIngressAuth.js' +import { SerialBatchEventUploader } from './SerialBatchEventUploader.js' +import { + WebSocketTransport, + type WebSocketTransportOptions, +} from './WebSocketTransport.js' + +const BATCH_FLUSH_INTERVAL_MS = 100 +// Per-attempt POST timeout. Bounds how long a single stuck POST can block +// the serialized queue. Without this, a hung connection stalls all writes. +const POST_TIMEOUT_MS = 15_000 +// Grace period for queued writes on close(). Covers a healthy POST (~100ms) +// plus headroom; best-effort, not a delivery guarantee under degraded network. +// Void-ed (nothing awaits it) so this is a last resort — replBridge teardown +// now closes AFTER archive so archive latency is the primary drain window. +// NOTE: gracefulShutdown's cleanup budget is 2s (not the 5s outer failsafe); +// 3s here exceeds it, but the process lives ~2s longer for hooks+analytics. +const CLOSE_GRACE_MS = 3000 + +/** + * Hybrid transport: WebSocket for reads, HTTP POST for writes. + * + * Write flow: + * + * write(stream_event) ─┐ + * │ (100ms timer) + * │ + * ▼ + * write(other) ────► uploader.enqueue() (SerialBatchEventUploader) + * ▲ │ + * writeBatch() ────────┘ │ serial, batched, retries indefinitely, + * │ backpressure at maxQueueSize + * ▼ + * postOnce() (single HTTP POST, throws on retryable) + * + * stream_event messages accumulate in streamEventBuffer for up to 100ms + * before enqueue (reduces POST count for high-volume content deltas). A + * non-stream write flushes any buffered stream_events first to preserve order. + * + * Serialization + retry + backpressure are delegated to SerialBatchEventUploader + * (same primitive CCR uses). At most one POST in-flight; events arriving during + * a POST batch into the next one. On failure, the uploader re-queues and retries + * with exponential backoff + jitter. If the queue fills past maxQueueSize, + * enqueue() blocks — giving awaiting callers backpressure. + * + * Why serialize? Bridge mode fires writes via `void transport.write()` + * (fire-and-forget). Without this, concurrent POSTs → concurrent Firestore + * writes to the same document → collisions → retry storms → pages oncall. + */ +export class HybridTransport extends WebSocketTransport { + private postUrl: string + private uploader: SerialBatchEventUploader + + // stream_event delay buffer — accumulates content deltas for up to + // BATCH_FLUSH_INTERVAL_MS before enqueueing (reduces POST count) + private streamEventBuffer: StdoutMessage[] = [] + private streamEventTimer: ReturnType | null = null + + constructor( + url: URL, + headers: Record = {}, + sessionId?: string, + refreshHeaders?: () => Record, + options?: WebSocketTransportOptions & { + maxConsecutiveFailures?: number + onBatchDropped?: (batchSize: number, failures: number) => void + }, + ) { + super(url, headers, sessionId, refreshHeaders, options) + const { maxConsecutiveFailures, onBatchDropped } = options ?? {} + this.postUrl = convertWsUrlToPostUrl(url) + this.uploader = new SerialBatchEventUploader({ + // Large cap — session-ingress accepts arbitrary batch sizes. Events + // naturally batch during in-flight POSTs; this just bounds the payload. + maxBatchSize: 500, + // Bridge callers use `void transport.write()` — backpressure doesn't + // apply (they don't await). A batch >maxQueueSize deadlocks (see + // SerialBatchEventUploader backpressure check). So set it high enough + // to be a memory bound only. Wire real backpressure in a follow-up + // once callers await. + maxQueueSize: 100_000, + baseDelayMs: 500, + maxDelayMs: 8000, + jitterMs: 1000, + // Optional cap so a persistently-failing server can't pin the drain + // loop for the lifetime of the process. Undefined = indefinite retry. + // replBridge sets this; the 1P transportUtils path does not. + maxConsecutiveFailures, + onBatchDropped: (batchSize, failures) => { + logForDiagnosticsNoPII( + 'error', + 'cli_hybrid_batch_dropped_max_failures', + { + batchSize, + failures, + }, + ) + onBatchDropped?.(batchSize, failures) + }, + send: batch => this.postOnce(batch), + }) + logForDebugging(`HybridTransport: POST URL = ${this.postUrl}`) + logForDiagnosticsNoPII('info', 'cli_hybrid_transport_initialized') + } + + /** + * Enqueue a message and wait for the queue to drain. Returning flush() + * preserves the contract that `await write()` resolves after the event is + * POSTed (relied on by tests and replBridge's initial flush). Fire-and-forget + * callers (`void transport.write()`) are unaffected — they don't await, + * so the later resolution doesn't add latency. + */ + override async write(message: StdoutMessage): Promise { + if (message.type === 'stream_event') { + // Delay: accumulate stream_events briefly before enqueueing. + // Promise resolves immediately — callers don't await stream_events. + this.streamEventBuffer.push(message) + if (!this.streamEventTimer) { + this.streamEventTimer = setTimeout( + () => this.flushStreamEvents(), + BATCH_FLUSH_INTERVAL_MS, + ) + } + return + } + // Immediate: flush any buffered stream_events (ordering), then this event. + await this.uploader.enqueue([...this.takeStreamEvents(), message]) + return this.uploader.flush() + } + + async writeBatch(messages: StdoutMessage[]): Promise { + await this.uploader.enqueue([...this.takeStreamEvents(), ...messages]) + return this.uploader.flush() + } + + /** Snapshot before/after writeBatch() to detect silent drops. */ + get droppedBatchCount(): number { + return this.uploader.droppedBatchCount + } + + /** + * Block until all pending events are POSTed. Used by bridge's initial + * history flush so onStateChange('connected') fires after persistence. + */ + flush(): Promise { + void this.uploader.enqueue(this.takeStreamEvents()) + return this.uploader.flush() + } + + /** Take ownership of buffered stream_events and clear the delay timer. */ + private takeStreamEvents(): StdoutMessage[] { + if (this.streamEventTimer) { + clearTimeout(this.streamEventTimer) + this.streamEventTimer = null + } + const buffered = this.streamEventBuffer + this.streamEventBuffer = [] + return buffered + } + + /** Delay timer fired — enqueue accumulated stream_events. */ + private flushStreamEvents(): void { + this.streamEventTimer = null + void this.uploader.enqueue(this.takeStreamEvents()) + } + + override close(): void { + if (this.streamEventTimer) { + clearTimeout(this.streamEventTimer) + this.streamEventTimer = null + } + this.streamEventBuffer = [] + // Grace period for queued writes — fallback. replBridge teardown now + // awaits archive between write and close (see CLOSE_GRACE_MS), so + // archive latency is the primary drain window and this is a last + // resort. Keep close() sync (returns immediately) but defer + // uploader.close() so any remaining queue gets a chance to finish. + const uploader = this.uploader + let graceTimer: ReturnType | undefined + void Promise.race([ + uploader.flush(), + new Promise(r => { + // eslint-disable-next-line no-restricted-syntax -- need timer ref for clearTimeout + graceTimer = setTimeout(r, CLOSE_GRACE_MS) + }), + ]).finally(() => { + clearTimeout(graceTimer) + uploader.close() + }) + super.close() + } + + /** + * Single-attempt POST. Throws on retryable failures (429, 5xx, network) + * so SerialBatchEventUploader re-queues and retries. Returns on success + * and on permanent failures (4xx non-429, no token) so the uploader moves on. + */ + private async postOnce(events: StdoutMessage[]): Promise { + const sessionToken = getSessionIngressAuthToken() + if (!sessionToken) { + logForDebugging('HybridTransport: No session token available for POST') + logForDiagnosticsNoPII('warn', 'cli_hybrid_post_no_token') + return + } + + const headers: Record = { + Authorization: `Bearer ${sessionToken}`, + 'Content-Type': 'application/json', + } + + let response + try { + response = await axios.post( + this.postUrl, + { events }, + { + headers, + validateStatus: () => true, + timeout: POST_TIMEOUT_MS, + }, + ) + } catch (error) { + const axiosError = error as AxiosError + logForDebugging(`HybridTransport: POST error: ${axiosError.message}`) + logForDiagnosticsNoPII('warn', 'cli_hybrid_post_network_error') + throw error + } + + if (response.status >= 200 && response.status < 300) { + logForDebugging(`HybridTransport: POST success count=${events.length}`) + return + } + + // 4xx (except 429) are permanent — drop, don't retry. + if ( + response.status >= 400 && + response.status < 500 && + response.status !== 429 + ) { + logForDebugging( + `HybridTransport: POST returned ${response.status} (permanent), dropping`, + ) + logForDiagnosticsNoPII('warn', 'cli_hybrid_post_client_error', { + status: response.status, + }) + return + } + + // 429 / 5xx — retryable. Throw so uploader re-queues and backs off. + logForDebugging( + `HybridTransport: POST returned ${response.status} (retryable)`, + ) + logForDiagnosticsNoPII('warn', 'cli_hybrid_post_retryable_error', { + status: response.status, + }) + throw new Error(`POST failed with ${response.status}`) + } +} + +/** + * Convert a WebSocket URL to the HTTP POST endpoint URL. + * From: wss://api.example.com/v2/session_ingress/ws/ + * To: https://api.example.com/v2/session_ingress/session//events + */ +function convertWsUrlToPostUrl(wsUrl: URL): string { + const protocol = wsUrl.protocol === 'wss:' ? 'https:' : 'http:' + + // Replace /ws/ with /session/ and append /events + let pathname = wsUrl.pathname + pathname = pathname.replace('/ws/', '/session/') + if (!pathname.endsWith('/events')) { + pathname = pathname.endsWith('/') + ? pathname + 'events' + : pathname + '/events' + } + + return `${protocol}//${wsUrl.host}${pathname}${wsUrl.search}` +} diff --git a/src/cli/transports/SSETransport.ts b/src/cli/transports/SSETransport.ts new file mode 100644 index 0000000..4f43dbe --- /dev/null +++ b/src/cli/transports/SSETransport.ts @@ -0,0 +1,711 @@ +import axios, { type AxiosError } from 'axios' +import type { StdoutMessage } from 'src/entrypoints/sdk/controlTypes.js' +import { logForDebugging } from '../../utils/debug.js' +import { logForDiagnosticsNoPII } from '../../utils/diagLogs.js' +import { errorMessage } from '../../utils/errors.js' +import { getSessionIngressAuthHeaders } from '../../utils/sessionIngressAuth.js' +import { sleep } from '../../utils/sleep.js' +import { jsonParse, jsonStringify } from '../../utils/slowOperations.js' +import { getClaudeCodeUserAgent } from '../../utils/userAgent.js' +import type { Transport } from './Transport.js' + +// --------------------------------------------------------------------------- +// Configuration +// --------------------------------------------------------------------------- + +const RECONNECT_BASE_DELAY_MS = 1000 +const RECONNECT_MAX_DELAY_MS = 30_000 +/** Time budget for reconnection attempts before giving up (10 minutes). */ +const RECONNECT_GIVE_UP_MS = 600_000 +/** Server sends keepalives every 15s; treat connection as dead after 45s of silence. */ +const LIVENESS_TIMEOUT_MS = 45_000 + +/** + * HTTP status codes that indicate a permanent server-side rejection. + * The transport transitions to 'closed' immediately without retrying. + */ +const PERMANENT_HTTP_CODES = new Set([401, 403, 404]) + +// POST retry configuration (matches HybridTransport) +const POST_MAX_RETRIES = 10 +const POST_BASE_DELAY_MS = 500 +const POST_MAX_DELAY_MS = 8000 + +/** Hoisted TextDecoder options to avoid per-chunk allocation in readStream. */ +const STREAM_DECODE_OPTS: TextDecodeOptions = { stream: true } + +/** Hoisted axios validateStatus callback to avoid per-request closure allocation. */ +function alwaysValidStatus(): boolean { + return true +} + +// --------------------------------------------------------------------------- +// SSE Frame Parser +// --------------------------------------------------------------------------- + +type SSEFrame = { + event?: string + id?: string + data?: string +} + +/** + * Incrementally parse SSE frames from a text buffer. + * Returns parsed frames and the remaining (incomplete) buffer. + * + * @internal exported for testing + */ +export function parseSSEFrames(buffer: string): { + frames: SSEFrame[] + remaining: string +} { + const frames: SSEFrame[] = [] + let pos = 0 + + // SSE frames are delimited by double newlines + let idx: number + while ((idx = buffer.indexOf('\n\n', pos)) !== -1) { + const rawFrame = buffer.slice(pos, idx) + pos = idx + 2 + + // Skip empty frames + if (!rawFrame.trim()) continue + + const frame: SSEFrame = {} + let isComment = false + + for (const line of rawFrame.split('\n')) { + if (line.startsWith(':')) { + // SSE comment (e.g., `:keepalive`) + isComment = true + continue + } + + const colonIdx = line.indexOf(':') + if (colonIdx === -1) continue + + const field = line.slice(0, colonIdx) + // Per SSE spec, strip one leading space after colon if present + const value = + line[colonIdx + 1] === ' ' + ? line.slice(colonIdx + 2) + : line.slice(colonIdx + 1) + + switch (field) { + case 'event': + frame.event = value + break + case 'id': + frame.id = value + break + case 'data': + // Per SSE spec, multiple data: lines are concatenated with \n + frame.data = frame.data ? frame.data + '\n' + value : value + break + // Ignore other fields (retry:, etc.) + } + } + + // Only emit frames that have data (or are pure comments which reset liveness) + if (frame.data || isComment) { + frames.push(frame) + } + } + + return { frames, remaining: buffer.slice(pos) } +} + +// --------------------------------------------------------------------------- +// Types +// --------------------------------------------------------------------------- + +type SSETransportState = + | 'idle' + | 'connected' + | 'reconnecting' + | 'closing' + | 'closed' + +/** + * Payload for `event: client_event` frames, matching the StreamClientEvent + * proto message in session_stream.proto. This is the only event type sent + * to worker subscribers — delivery_update, session_update, ephemeral_event, + * and catch_up_truncated are client-channel-only (see notifier.go and + * event_stream.go SubscriberClient guard). + */ +export type StreamClientEvent = { + event_id: string + sequence_num: number + event_type: string + source: string + payload: Record + created_at: string +} + +// --------------------------------------------------------------------------- +// SSETransport +// --------------------------------------------------------------------------- + +/** + * Transport that uses SSE for reading and HTTP POST for writing. + * + * Reads events via Server-Sent Events from the CCR v2 event stream endpoint. + * Writes events via HTTP POST with retry logic (same pattern as HybridTransport). + * + * Each `event: client_event` frame carries a StreamClientEvent proto JSON + * directly in `data:`. The transport extracts `payload` and passes it to + * `onData` as newline-delimited JSON for StructuredIO consumers. + * + * Supports automatic reconnection with exponential backoff and Last-Event-ID + * for resumption after disconnection. + */ +export class SSETransport implements Transport { + private state: SSETransportState = 'idle' + private onData?: (data: string) => void + private onCloseCallback?: (closeCode?: number) => void + private onEventCallback?: (event: StreamClientEvent) => void + private headers: Record + private sessionId?: string + private refreshHeaders?: () => Record + private readonly getAuthHeaders: () => Record + + // SSE connection state + private abortController: AbortController | null = null + private lastSequenceNum = 0 + private seenSequenceNums = new Set() + + // Reconnection state + private reconnectAttempts = 0 + private reconnectStartTime: number | null = null + private reconnectTimer: NodeJS.Timeout | null = null + + // Liveness detection + private livenessTimer: NodeJS.Timeout | null = null + + // POST URL (derived from SSE URL) + private postUrl: string + + // Runtime epoch for CCR v2 event format + + constructor( + private readonly url: URL, + headers: Record = {}, + sessionId?: string, + refreshHeaders?: () => Record, + initialSequenceNum?: number, + /** + * Per-instance auth header source. Omit to read the process-wide + * CLAUDE_CODE_SESSION_ACCESS_TOKEN (single-session callers). Required + * for concurrent multi-session callers — the env-var path is a process + * global and would stomp across sessions. + */ + getAuthHeaders?: () => Record, + ) { + this.headers = headers + this.sessionId = sessionId + this.refreshHeaders = refreshHeaders + this.getAuthHeaders = getAuthHeaders ?? getSessionIngressAuthHeaders + this.postUrl = convertSSEUrlToPostUrl(url) + // Seed with a caller-provided high-water mark so the first connect() + // sends from_sequence_num / Last-Event-ID. Without this, a fresh + // SSETransport always asks the server to replay from sequence 0 — + // the entire session history on every transport swap. + if (initialSequenceNum !== undefined && initialSequenceNum > 0) { + this.lastSequenceNum = initialSequenceNum + } + logForDebugging(`SSETransport: SSE URL = ${url.href}`) + logForDebugging(`SSETransport: POST URL = ${this.postUrl}`) + logForDiagnosticsNoPII('info', 'cli_sse_transport_initialized') + } + + /** + * High-water mark of sequence numbers seen on this stream. Callers that + * recreate the transport (e.g. replBridge onWorkReceived) read this before + * close() and pass it as `initialSequenceNum` to the next instance so the + * server resumes from the right point instead of replaying everything. + */ + getLastSequenceNum(): number { + return this.lastSequenceNum + } + + async connect(): Promise { + if (this.state !== 'idle' && this.state !== 'reconnecting') { + logForDebugging( + `SSETransport: Cannot connect, current state is ${this.state}`, + { level: 'error' }, + ) + logForDiagnosticsNoPII('error', 'cli_sse_connect_failed') + return + } + + this.state = 'reconnecting' + const connectStartTime = Date.now() + + // Build SSE URL with sequence number for resumption + const sseUrl = new URL(this.url.href) + if (this.lastSequenceNum > 0) { + sseUrl.searchParams.set('from_sequence_num', String(this.lastSequenceNum)) + } + + // Build headers -- use fresh auth headers (supports Cookie for session keys). + // Remove stale Authorization header from this.headers when Cookie auth is used, + // since sending both confuses the auth interceptor. + const authHeaders = this.getAuthHeaders() + const headers: Record = { + ...this.headers, + ...authHeaders, + Accept: 'text/event-stream', + 'anthropic-version': '2023-06-01', + 'User-Agent': getClaudeCodeUserAgent(), + } + if (authHeaders['Cookie']) { + delete headers['Authorization'] + } + if (this.lastSequenceNum > 0) { + headers['Last-Event-ID'] = String(this.lastSequenceNum) + } + + logForDebugging(`SSETransport: Opening ${sseUrl.href}`) + logForDiagnosticsNoPII('info', 'cli_sse_connect_opening') + + this.abortController = new AbortController() + + try { + // eslint-disable-next-line eslint-plugin-n/no-unsupported-features/node-builtins + const response = await fetch(sseUrl.href, { + headers, + signal: this.abortController.signal, + }) + + if (!response.ok) { + const isPermanent = PERMANENT_HTTP_CODES.has(response.status) + logForDebugging( + `SSETransport: HTTP ${response.status}${isPermanent ? ' (permanent)' : ''}`, + { level: 'error' }, + ) + logForDiagnosticsNoPII('error', 'cli_sse_connect_http_error', { + status: response.status, + }) + + if (isPermanent) { + this.state = 'closed' + this.onCloseCallback?.(response.status) + return + } + + this.handleConnectionError() + return + } + + if (!response.body) { + logForDebugging('SSETransport: No response body') + this.handleConnectionError() + return + } + + // Successfully connected + const connectDuration = Date.now() - connectStartTime + logForDebugging('SSETransport: Connected') + logForDiagnosticsNoPII('info', 'cli_sse_connect_connected', { + duration_ms: connectDuration, + }) + + this.state = 'connected' + this.reconnectAttempts = 0 + this.reconnectStartTime = null + this.resetLivenessTimer() + + // Read the SSE stream + await this.readStream(response.body) + } catch (error) { + if (this.abortController?.signal.aborted) { + // Intentional close + return + } + + logForDebugging( + `SSETransport: Connection error: ${errorMessage(error)}`, + { level: 'error' }, + ) + logForDiagnosticsNoPII('error', 'cli_sse_connect_error') + this.handleConnectionError() + } + } + + /** + * Read and process the SSE stream body. + */ + // eslint-disable-next-line eslint-plugin-n/no-unsupported-features/node-builtins + private async readStream(body: ReadableStream): Promise { + const reader = body.getReader() + const decoder = new TextDecoder() + let buffer = '' + + try { + while (true) { + const { done, value } = await reader.read() + if (done) break + + buffer += decoder.decode(value, STREAM_DECODE_OPTS) + const { frames, remaining } = parseSSEFrames(buffer) + buffer = remaining + + for (const frame of frames) { + // Any frame (including keepalive comments) proves the connection is alive + this.resetLivenessTimer() + + if (frame.id) { + const seqNum = parseInt(frame.id, 10) + if (!isNaN(seqNum)) { + if (this.seenSequenceNums.has(seqNum)) { + logForDebugging( + `SSETransport: DUPLICATE frame seq=${seqNum} (lastSequenceNum=${this.lastSequenceNum}, seenCount=${this.seenSequenceNums.size})`, + { level: 'warn' }, + ) + logForDiagnosticsNoPII('warn', 'cli_sse_duplicate_sequence') + } else { + this.seenSequenceNums.add(seqNum) + // Prevent unbounded growth: once we have many entries, prune + // old sequence numbers that are well below the high-water mark. + // Only sequence numbers near lastSequenceNum matter for dedup. + if (this.seenSequenceNums.size > 1000) { + const threshold = this.lastSequenceNum - 200 + for (const s of this.seenSequenceNums) { + if (s < threshold) { + this.seenSequenceNums.delete(s) + } + } + } + } + if (seqNum > this.lastSequenceNum) { + this.lastSequenceNum = seqNum + } + } + } + + if (frame.event && frame.data) { + this.handleSSEFrame(frame.event, frame.data) + } else if (frame.data) { + // data: without event: — server is emitting the old envelope format + // or a bug. Log so incidents show as a signal instead of silent drops. + logForDebugging( + 'SSETransport: Frame has data: but no event: field — dropped', + { level: 'warn' }, + ) + logForDiagnosticsNoPII('warn', 'cli_sse_frame_missing_event_field') + } + } + } + } catch (error) { + if (this.abortController?.signal.aborted) return + logForDebugging( + `SSETransport: Stream read error: ${errorMessage(error)}`, + { level: 'error' }, + ) + logForDiagnosticsNoPII('error', 'cli_sse_stream_read_error') + } finally { + reader.releaseLock() + } + + // Stream ended — reconnect unless we're closing + if (this.state !== 'closing' && this.state !== 'closed') { + logForDebugging('SSETransport: Stream ended, reconnecting') + this.handleConnectionError() + } + } + + /** + * Handle a single SSE frame. The event: field names the variant; data: + * carries the inner proto JSON directly (no envelope). + * + * Worker subscribers only receive client_event frames (see notifier.go) — + * any other event type indicates a server-side change that CC doesn't yet + * understand. Log a diagnostic so we notice in telemetry. + */ + private handleSSEFrame(eventType: string, data: string): void { + if (eventType !== 'client_event') { + logForDebugging( + `SSETransport: Unexpected SSE event type '${eventType}' on worker stream`, + { level: 'warn' }, + ) + logForDiagnosticsNoPII('warn', 'cli_sse_unexpected_event_type', { + event_type: eventType, + }) + return + } + + let ev: StreamClientEvent + try { + ev = jsonParse(data) as StreamClientEvent + } catch (error) { + logForDebugging( + `SSETransport: Failed to parse client_event data: ${errorMessage(error)}`, + { level: 'error' }, + ) + return + } + + const payload = ev.payload + if (payload && typeof payload === 'object' && 'type' in payload) { + const sessionLabel = this.sessionId ? ` session=${this.sessionId}` : '' + logForDebugging( + `SSETransport: Event seq=${ev.sequence_num} event_id=${ev.event_id} event_type=${ev.event_type} payload_type=${String(payload.type)}${sessionLabel}`, + ) + logForDiagnosticsNoPII('info', 'cli_sse_message_received') + // Pass the unwrapped payload as newline-delimited JSON, + // matching the format that StructuredIO/WebSocketTransport consumers expect + this.onData?.(jsonStringify(payload) + '\n') + } else { + logForDebugging( + `SSETransport: Ignoring client_event with no type in payload: event_id=${ev.event_id}`, + ) + } + + this.onEventCallback?.(ev) + } + + /** + * Handle connection errors with exponential backoff and time budget. + */ + private handleConnectionError(): void { + this.clearLivenessTimer() + + if (this.state === 'closing' || this.state === 'closed') return + + // Abort any in-flight SSE fetch + this.abortController?.abort() + this.abortController = null + + const now = Date.now() + if (!this.reconnectStartTime) { + this.reconnectStartTime = now + } + + const elapsed = now - this.reconnectStartTime + if (elapsed < RECONNECT_GIVE_UP_MS) { + // Clear any existing timer + if (this.reconnectTimer) { + clearTimeout(this.reconnectTimer) + this.reconnectTimer = null + } + + // Refresh headers before reconnecting + if (this.refreshHeaders) { + const freshHeaders = this.refreshHeaders() + Object.assign(this.headers, freshHeaders) + logForDebugging('SSETransport: Refreshed headers for reconnect') + } + + this.state = 'reconnecting' + this.reconnectAttempts++ + + const baseDelay = Math.min( + RECONNECT_BASE_DELAY_MS * Math.pow(2, this.reconnectAttempts - 1), + RECONNECT_MAX_DELAY_MS, + ) + // Add ±25% jitter + const delay = Math.max( + 0, + baseDelay + baseDelay * 0.25 * (2 * Math.random() - 1), + ) + + logForDebugging( + `SSETransport: Reconnecting in ${Math.round(delay)}ms (attempt ${this.reconnectAttempts}, ${Math.round(elapsed / 1000)}s elapsed)`, + ) + logForDiagnosticsNoPII('error', 'cli_sse_reconnect_attempt', { + reconnectAttempts: this.reconnectAttempts, + }) + + this.reconnectTimer = setTimeout(() => { + this.reconnectTimer = null + void this.connect() + }, delay) + } else { + logForDebugging( + `SSETransport: Reconnection time budget exhausted after ${Math.round(elapsed / 1000)}s`, + { level: 'error' }, + ) + logForDiagnosticsNoPII('error', 'cli_sse_reconnect_exhausted', { + reconnectAttempts: this.reconnectAttempts, + elapsedMs: elapsed, + }) + this.state = 'closed' + this.onCloseCallback?.() + } + } + + /** + * Bound timeout callback. Hoisted from an inline closure so that + * resetLivenessTimer (called per-frame) does not allocate a new closure + * on every SSE frame. + */ + private readonly onLivenessTimeout = (): void => { + this.livenessTimer = null + logForDebugging('SSETransport: Liveness timeout, reconnecting', { + level: 'error', + }) + logForDiagnosticsNoPII('error', 'cli_sse_liveness_timeout') + this.abortController?.abort() + this.handleConnectionError() + } + + /** + * Reset the liveness timer. If no SSE frame arrives within the timeout, + * treat the connection as dead and reconnect. + */ + private resetLivenessTimer(): void { + this.clearLivenessTimer() + this.livenessTimer = setTimeout(this.onLivenessTimeout, LIVENESS_TIMEOUT_MS) + } + + private clearLivenessTimer(): void { + if (this.livenessTimer) { + clearTimeout(this.livenessTimer) + this.livenessTimer = null + } + } + + // ----------------------------------------------------------------------- + // Write (HTTP POST) — same pattern as HybridTransport + // ----------------------------------------------------------------------- + + async write(message: StdoutMessage): Promise { + const authHeaders = this.getAuthHeaders() + if (Object.keys(authHeaders).length === 0) { + logForDebugging('SSETransport: No session token available for POST') + logForDiagnosticsNoPII('warn', 'cli_sse_post_no_token') + return + } + + const headers: Record = { + ...authHeaders, + 'Content-Type': 'application/json', + 'anthropic-version': '2023-06-01', + 'User-Agent': getClaudeCodeUserAgent(), + } + + logForDebugging( + `SSETransport: POST body keys=${Object.keys(message as Record).join(',')}`, + ) + + for (let attempt = 1; attempt <= POST_MAX_RETRIES; attempt++) { + try { + const response = await axios.post(this.postUrl, message, { + headers, + validateStatus: alwaysValidStatus, + }) + + if (response.status === 200 || response.status === 201) { + logForDebugging(`SSETransport: POST success type=${message.type}`) + return + } + + logForDebugging( + `SSETransport: POST ${response.status} body=${jsonStringify(response.data).slice(0, 200)}`, + ) + // 4xx errors (except 429) are permanent - don't retry + if ( + response.status >= 400 && + response.status < 500 && + response.status !== 429 + ) { + logForDebugging( + `SSETransport: POST returned ${response.status} (client error), not retrying`, + ) + logForDiagnosticsNoPII('warn', 'cli_sse_post_client_error', { + status: response.status, + }) + return + } + + // 429 or 5xx - retry + logForDebugging( + `SSETransport: POST returned ${response.status}, attempt ${attempt}/${POST_MAX_RETRIES}`, + ) + logForDiagnosticsNoPII('warn', 'cli_sse_post_retryable_error', { + status: response.status, + attempt, + }) + } catch (error) { + const axiosError = error as AxiosError + logForDebugging( + `SSETransport: POST error: ${axiosError.message}, attempt ${attempt}/${POST_MAX_RETRIES}`, + ) + logForDiagnosticsNoPII('warn', 'cli_sse_post_network_error', { + attempt, + }) + } + + if (attempt === POST_MAX_RETRIES) { + logForDebugging( + `SSETransport: POST failed after ${POST_MAX_RETRIES} attempts, continuing`, + ) + logForDiagnosticsNoPII('warn', 'cli_sse_post_retries_exhausted') + return + } + + const delayMs = Math.min( + POST_BASE_DELAY_MS * Math.pow(2, attempt - 1), + POST_MAX_DELAY_MS, + ) + await sleep(delayMs) + } + } + + // ----------------------------------------------------------------------- + // Transport interface + // ----------------------------------------------------------------------- + + isConnectedStatus(): boolean { + return this.state === 'connected' + } + + isClosedStatus(): boolean { + return this.state === 'closed' + } + + setOnData(callback: (data: string) => void): void { + this.onData = callback + } + + setOnClose(callback: (closeCode?: number) => void): void { + this.onCloseCallback = callback + } + + setOnEvent(callback: (event: StreamClientEvent) => void): void { + this.onEventCallback = callback + } + + close(): void { + if (this.reconnectTimer) { + clearTimeout(this.reconnectTimer) + this.reconnectTimer = null + } + this.clearLivenessTimer() + + this.state = 'closing' + this.abortController?.abort() + this.abortController = null + } +} + +// --------------------------------------------------------------------------- +// URL Conversion +// --------------------------------------------------------------------------- + +/** + * Convert an SSE URL to the HTTP POST endpoint URL. + * The SSE stream URL and POST URL share the same base; the POST endpoint + * is at `/events` (without `/stream`). + * + * From: https://api.example.com/v2/session_ingress/session//events/stream + * To: https://api.example.com/v2/session_ingress/session//events + */ +function convertSSEUrlToPostUrl(sseUrl: URL): string { + let pathname = sseUrl.pathname + // Remove /stream suffix to get the POST events endpoint + if (pathname.endsWith('/stream')) { + pathname = pathname.slice(0, -'/stream'.length) + } + return `${sseUrl.protocol}//${sseUrl.host}${pathname}` +} diff --git a/src/cli/transports/SerialBatchEventUploader.ts b/src/cli/transports/SerialBatchEventUploader.ts new file mode 100644 index 0000000..f753ca0 --- /dev/null +++ b/src/cli/transports/SerialBatchEventUploader.ts @@ -0,0 +1,275 @@ +import { jsonStringify } from '../../utils/slowOperations.js' + +/** + * Serial ordered event uploader with batching, retry, and backpressure. + * + * - enqueue() adds events to a pending buffer + * - At most 1 POST in-flight at a time + * - Drains up to maxBatchSize items per POST + * - New events accumulate while in-flight + * - On failure: exponential backoff (clamped), retries indefinitely + * until success or close() — unless maxConsecutiveFailures is set, + * in which case the failing batch is dropped and drain advances + * - flush() blocks until pending is empty and kicks drain if needed + * - Backpressure: enqueue() blocks when maxQueueSize is reached + */ + +/** + * Throw from config.send() to make the uploader wait a server-supplied + * duration before retrying (e.g. 429 with Retry-After). When retryAfterMs + * is set, it overrides exponential backoff for that attempt — clamped to + * [baseDelayMs, maxDelayMs] and jittered so a misbehaving server can + * neither hot-loop nor stall the client, and many sessions sharing a rate + * limit don't all pounce at the same instant. Without retryAfterMs, behaves + * like any other thrown error (exponential backoff). + */ +export class RetryableError extends Error { + constructor( + message: string, + readonly retryAfterMs?: number, + ) { + super(message) + } +} + +type SerialBatchEventUploaderConfig = { + /** Max items per POST (1 = no batching) */ + maxBatchSize: number + /** + * Max serialized bytes per POST. First item always goes in regardless of + * size; subsequent items only if cumulative JSON bytes stay under this. + * Undefined = no byte limit (count-only batching). + */ + maxBatchBytes?: number + /** Max pending items before enqueue() blocks */ + maxQueueSize: number + /** The actual HTTP call — caller controls payload format */ + send: (batch: T[]) => Promise + /** Base delay for exponential backoff (ms) */ + baseDelayMs: number + /** Max delay cap (ms) */ + maxDelayMs: number + /** Random jitter range added to retry delay (ms) */ + jitterMs: number + /** + * After this many consecutive send() failures, drop the failing batch + * and move on to the next pending item with a fresh failure budget. + * Undefined = retry indefinitely (default). + */ + maxConsecutiveFailures?: number + /** Called when a batch is dropped for hitting maxConsecutiveFailures. */ + onBatchDropped?: (batchSize: number, failures: number) => void +} + +export class SerialBatchEventUploader { + private pending: T[] = [] + private pendingAtClose = 0 + private draining = false + private closed = false + private backpressureResolvers: Array<() => void> = [] + private sleepResolve: (() => void) | null = null + private flushResolvers: Array<() => void> = [] + private droppedBatches = 0 + private readonly config: SerialBatchEventUploaderConfig + + constructor(config: SerialBatchEventUploaderConfig) { + this.config = config + } + + /** + * Monotonic count of batches dropped via maxConsecutiveFailures. Callers + * can snapshot before flush() and compare after to detect silent drops + * (flush() resolves normally even when batches were dropped). + */ + get droppedBatchCount(): number { + return this.droppedBatches + } + + /** + * Pending queue depth. After close(), returns the count at close time — + * close() clears the queue but shutdown diagnostics may read this after. + */ + get pendingCount(): number { + return this.closed ? this.pendingAtClose : this.pending.length + } + + /** + * Add events to the pending buffer. Returns immediately if space is + * available. Blocks (awaits) if the buffer is full — caller pauses + * until drain frees space. + */ + async enqueue(events: T | T[]): Promise { + if (this.closed) return + const items = Array.isArray(events) ? events : [events] + if (items.length === 0) return + + // Backpressure: wait until there's space + while ( + this.pending.length + items.length > this.config.maxQueueSize && + !this.closed + ) { + await new Promise(resolve => { + this.backpressureResolvers.push(resolve) + }) + } + + if (this.closed) return + this.pending.push(...items) + void this.drain() + } + + /** + * Block until all pending events have been sent. + * Used at turn boundaries and graceful shutdown. + */ + flush(): Promise { + if (this.pending.length === 0 && !this.draining) { + return Promise.resolve() + } + void this.drain() + return new Promise(resolve => { + this.flushResolvers.push(resolve) + }) + } + + /** + * Drop pending events and stop processing. + * Resolves any blocked enqueue() and flush() callers. + */ + close(): void { + if (this.closed) return + this.closed = true + this.pendingAtClose = this.pending.length + this.pending = [] + this.sleepResolve?.() + this.sleepResolve = null + for (const resolve of this.backpressureResolvers) resolve() + this.backpressureResolvers = [] + for (const resolve of this.flushResolvers) resolve() + this.flushResolvers = [] + } + + /** + * Drain loop. At most one instance runs at a time (guarded by this.draining). + * Sends batches serially. On failure, backs off and retries indefinitely. + */ + private async drain(): Promise { + if (this.draining || this.closed) return + this.draining = true + let failures = 0 + + try { + while (this.pending.length > 0 && !this.closed) { + const batch = this.takeBatch() + if (batch.length === 0) continue + + try { + await this.config.send(batch) + failures = 0 + } catch (err) { + failures++ + if ( + this.config.maxConsecutiveFailures !== undefined && + failures >= this.config.maxConsecutiveFailures + ) { + this.droppedBatches++ + this.config.onBatchDropped?.(batch.length, failures) + failures = 0 + this.releaseBackpressure() + continue + } + // Re-queue the failed batch at the front. Use concat (single + // allocation) instead of unshift(...batch) which shifts every + // pending item batch.length times. Only hit on failure path. + this.pending = batch.concat(this.pending) + const retryAfterMs = + err instanceof RetryableError ? err.retryAfterMs : undefined + await this.sleep(this.retryDelay(failures, retryAfterMs)) + continue + } + + // Release backpressure waiters if space opened up + this.releaseBackpressure() + } + } finally { + this.draining = false + // Notify flush waiters if queue is empty + if (this.pending.length === 0) { + for (const resolve of this.flushResolvers) resolve() + this.flushResolvers = [] + } + } + } + + /** + * Pull the next batch from pending. Respects both maxBatchSize and + * maxBatchBytes. The first item is always taken; subsequent items only + * if adding them keeps the cumulative JSON size under maxBatchBytes. + * + * Un-serializable items (BigInt, circular refs, throwing toJSON) are + * dropped in place — they can never be sent and leaving them at + * pending[0] would poison the queue and hang flush() forever. + */ + private takeBatch(): T[] { + const { maxBatchSize, maxBatchBytes } = this.config + if (maxBatchBytes === undefined) { + return this.pending.splice(0, maxBatchSize) + } + let bytes = 0 + let count = 0 + while (count < this.pending.length && count < maxBatchSize) { + let itemBytes: number + try { + itemBytes = Buffer.byteLength(jsonStringify(this.pending[count])) + } catch { + this.pending.splice(count, 1) + continue + } + if (count > 0 && bytes + itemBytes > maxBatchBytes) break + bytes += itemBytes + count++ + } + return this.pending.splice(0, count) + } + + private retryDelay(failures: number, retryAfterMs?: number): number { + const jitter = Math.random() * this.config.jitterMs + if (retryAfterMs !== undefined) { + // Jitter on top of the server's hint prevents thundering herd when + // many sessions share a rate limit and all receive the same + // Retry-After. Clamp first, then spread — same shape as the + // exponential path (effective ceiling is maxDelayMs + jitterMs). + const clamped = Math.max( + this.config.baseDelayMs, + Math.min(retryAfterMs, this.config.maxDelayMs), + ) + return clamped + jitter + } + const exponential = Math.min( + this.config.baseDelayMs * 2 ** (failures - 1), + this.config.maxDelayMs, + ) + return exponential + jitter + } + + private releaseBackpressure(): void { + const resolvers = this.backpressureResolvers + this.backpressureResolvers = [] + for (const resolve of resolvers) resolve() + } + + private sleep(ms: number): Promise { + return new Promise(resolve => { + this.sleepResolve = resolve + setTimeout( + (self, resolve) => { + self.sleepResolve = null + resolve() + }, + ms, + this, + resolve, + ) + }) + } +} diff --git a/src/cli/transports/WebSocketTransport.ts b/src/cli/transports/WebSocketTransport.ts new file mode 100644 index 0000000..f8e27ac --- /dev/null +++ b/src/cli/transports/WebSocketTransport.ts @@ -0,0 +1,800 @@ +import type { StdoutMessage } from 'src/entrypoints/sdk/controlTypes.js' +import type WsWebSocket from 'ws' +import { logEvent } from '../../services/analytics/index.js' +import { CircularBuffer } from '../../utils/CircularBuffer.js' +import { logForDebugging } from '../../utils/debug.js' +import { logForDiagnosticsNoPII } from '../../utils/diagLogs.js' +import { isEnvTruthy } from '../../utils/envUtils.js' +import { getWebSocketTLSOptions } from '../../utils/mtls.js' +import { + getWebSocketProxyAgent, + getWebSocketProxyUrl, +} from '../../utils/proxy.js' +import { + registerSessionActivityCallback, + unregisterSessionActivityCallback, +} from '../../utils/sessionActivity.js' +import { jsonStringify } from '../../utils/slowOperations.js' +import type { Transport } from './Transport.js' + +const KEEP_ALIVE_FRAME = '{"type":"keep_alive"}\n' + +const DEFAULT_MAX_BUFFER_SIZE = 1000 +const DEFAULT_BASE_RECONNECT_DELAY = 1000 +const DEFAULT_MAX_RECONNECT_DELAY = 30000 +/** Time budget for reconnection attempts before giving up (10 minutes). */ +const DEFAULT_RECONNECT_GIVE_UP_MS = 600_000 +const DEFAULT_PING_INTERVAL = 10000 +const DEFAULT_KEEPALIVE_INTERVAL = 300_000 // 5 minutes + +/** + * Threshold for detecting system sleep/wake. If the gap between consecutive + * reconnection attempts exceeds this, the machine likely slept. We reset + * the reconnection budget and retry — the server will reject with permanent + * close codes (4001/1002) if the session was reaped during sleep. + */ +const SLEEP_DETECTION_THRESHOLD_MS = DEFAULT_MAX_RECONNECT_DELAY * 2 // 60s + +/** + * WebSocket close codes that indicate a permanent server-side rejection. + * The transport transitions to 'closed' immediately without retrying. + */ +const PERMANENT_CLOSE_CODES = new Set([ + 1002, // protocol error — server rejected handshake (e.g. session reaped) + 4001, // session expired / not found + 4003, // unauthorized +]) + +export type WebSocketTransportOptions = { + /** When false, the transport does not attempt automatic reconnection on + * disconnect. Use this when the caller has its own recovery mechanism + * (e.g. the REPL bridge poll loop). Defaults to true. */ + autoReconnect?: boolean + /** Gates the tengu_ws_transport_* telemetry events. Set true at the + * REPL-bridge construction site so only Remote Control sessions (the + * Cloudflare-idle-timeout population) emit; print-mode workers stay + * silent. Defaults to false. */ + isBridge?: boolean +} + +type WebSocketTransportState = + | 'idle' + | 'connected' + | 'reconnecting' + | 'closing' + | 'closed' + +// Common interface between globalThis.WebSocket and ws.WebSocket +type WebSocketLike = { + close(): void + send(data: string): void + ping?(): void // Bun & ws both support this +} + +export class WebSocketTransport implements Transport { + private ws: WebSocketLike | null = null + private lastSentId: string | null = null + protected url: URL + protected state: WebSocketTransportState = 'idle' + protected onData?: (data: string) => void + private onCloseCallback?: (closeCode?: number) => void + private onConnectCallback?: () => void + private headers: Record + private sessionId?: string + private autoReconnect: boolean + private isBridge: boolean + + // Reconnection state + private reconnectAttempts = 0 + private reconnectStartTime: number | null = null + private reconnectTimer: NodeJS.Timeout | null = null + private lastReconnectAttemptTime: number | null = null + // Wall-clock of last WS data-frame activity (inbound message or outbound + // ws.send). Used to compute idle time at close — the signal for diagnosing + // proxy idle-timeout RSTs (e.g. Cloudflare 5-min). Excludes ping/pong + // control frames (proxies don't count those). + private lastActivityTime = 0 + + // Ping interval for connection health checks + private pingInterval: NodeJS.Timeout | null = null + private pongReceived = true + + // Periodic keep_alive data frames to reset proxy idle timers + private keepAliveInterval: NodeJS.Timeout | null = null + + // Message buffering for replay on reconnection + private messageBuffer: CircularBuffer + // Track which runtime's WS we're using so we can detach listeners + // with the matching API (removeEventListener vs. off). + private isBunWs = false + + // Captured at connect() time for handleOpenEvent timing. Stored as an + // instance field so the onOpen handler can be a stable class-property + // arrow function (removable in doDisconnect) instead of a closure over + // a local variable. + private connectStartTime = 0 + + private refreshHeaders?: () => Record + + constructor( + url: URL, + headers: Record = {}, + sessionId?: string, + refreshHeaders?: () => Record, + options?: WebSocketTransportOptions, + ) { + this.url = url + this.headers = headers + this.sessionId = sessionId + this.refreshHeaders = refreshHeaders + this.autoReconnect = options?.autoReconnect ?? true + this.isBridge = options?.isBridge ?? false + this.messageBuffer = new CircularBuffer(DEFAULT_MAX_BUFFER_SIZE) + } + + public async connect(): Promise { + if (this.state !== 'idle' && this.state !== 'reconnecting') { + logForDebugging( + `WebSocketTransport: Cannot connect, current state is ${this.state}`, + { level: 'error' }, + ) + logForDiagnosticsNoPII('error', 'cli_websocket_connect_failed') + return + } + this.state = 'reconnecting' + + this.connectStartTime = Date.now() + logForDebugging(`WebSocketTransport: Opening ${this.url.href}`) + logForDiagnosticsNoPII('info', 'cli_websocket_connect_opening') + + // Start with provided headers and add runtime headers + const headers = { ...this.headers } + if (this.lastSentId) { + headers['X-Last-Request-Id'] = this.lastSentId + logForDebugging( + `WebSocketTransport: Adding X-Last-Request-Id header: ${this.lastSentId}`, + ) + } + + if (typeof Bun !== 'undefined') { + // Bun's WebSocket supports headers/proxy options but the DOM typings don't + // eslint-disable-next-line eslint-plugin-n/no-unsupported-features/node-builtins + const ws = new globalThis.WebSocket(this.url.href, { + headers, + proxy: getWebSocketProxyUrl(this.url.href), + tls: getWebSocketTLSOptions() || undefined, + } as unknown as string[]) + this.ws = ws + this.isBunWs = true + + ws.addEventListener('open', this.onBunOpen) + ws.addEventListener('message', this.onBunMessage) + ws.addEventListener('error', this.onBunError) + // eslint-disable-next-line eslint-plugin-n/no-unsupported-features/node-builtins + ws.addEventListener('close', this.onBunClose) + // 'pong' is Bun-specific — not in DOM typings. + ws.addEventListener('pong', this.onPong) + } else { + const { default: WS } = await import('ws') + const ws = new WS(this.url.href, { + headers, + agent: getWebSocketProxyAgent(this.url.href), + ...getWebSocketTLSOptions(), + }) + this.ws = ws + this.isBunWs = false + + ws.on('open', this.onNodeOpen) + ws.on('message', this.onNodeMessage) + ws.on('error', this.onNodeError) + ws.on('close', this.onNodeClose) + ws.on('pong', this.onPong) + } + } + + // --- Bun (native WebSocket) event handlers --- + // Stored as class-property arrow functions so they can be removed in + // doDisconnect(). Without removal, each reconnect orphans the old WS + // object + its 5 closures until GC, which accumulates under network + // instability. Mirrors the pattern in src/utils/mcpWebSocketTransport.ts. + + private onBunOpen = () => { + this.handleOpenEvent() + // Bun's WebSocket doesn't expose upgrade response headers, + // so replay all buffered messages. The server deduplicates by UUID. + if (this.lastSentId) { + this.replayBufferedMessages('') + } + } + + private onBunMessage = (event: MessageEvent) => { + const message = + typeof event.data === 'string' ? event.data : String(event.data) + this.lastActivityTime = Date.now() + logForDiagnosticsNoPII('info', 'cli_websocket_message_received', { + length: message.length, + }) + if (this.onData) { + this.onData(message) + } + } + + private onBunError = () => { + logForDebugging('WebSocketTransport: Error', { + level: 'error', + }) + logForDiagnosticsNoPII('error', 'cli_websocket_connect_error') + // close event fires after error — let it call handleConnectionError + } + + // eslint-disable-next-line eslint-plugin-n/no-unsupported-features/node-builtins + private onBunClose = (event: CloseEvent) => { + const isClean = event.code === 1000 || event.code === 1001 + logForDebugging( + `WebSocketTransport: Closed: ${event.code}`, + isClean ? undefined : { level: 'error' }, + ) + logForDiagnosticsNoPII('error', 'cli_websocket_connect_closed') + this.handleConnectionError(event.code) + } + + // --- Node (ws package) event handlers --- + + private onNodeOpen = () => { + // Capture ws before handleOpenEvent() invokes onConnectCallback — if the + // callback synchronously closes the transport, this.ws becomes null. + // The old inline-closure code had this safety implicitly via closure capture. + const ws = this.ws + this.handleOpenEvent() + if (!ws) return + // Check for last-id in upgrade response headers (ws package only) + const nws = ws as unknown as WsWebSocket & { + upgradeReq?: { headers?: Record } + } + const upgradeResponse = nws.upgradeReq + if (upgradeResponse?.headers?.['x-last-request-id']) { + const serverLastId = upgradeResponse.headers['x-last-request-id'] + this.replayBufferedMessages(serverLastId) + } + } + + private onNodeMessage = (data: Buffer) => { + const message = data.toString() + this.lastActivityTime = Date.now() + logForDiagnosticsNoPII('info', 'cli_websocket_message_received', { + length: message.length, + }) + if (this.onData) { + this.onData(message) + } + } + + private onNodeError = (err: Error) => { + logForDebugging(`WebSocketTransport: Error: ${err.message}`, { + level: 'error', + }) + logForDiagnosticsNoPII('error', 'cli_websocket_connect_error') + // close event fires after error — let it call handleConnectionError + } + + private onNodeClose = (code: number, _reason: Buffer) => { + const isClean = code === 1000 || code === 1001 + logForDebugging( + `WebSocketTransport: Closed: ${code}`, + isClean ? undefined : { level: 'error' }, + ) + logForDiagnosticsNoPII('error', 'cli_websocket_connect_closed') + this.handleConnectionError(code) + } + + // --- Shared handlers --- + + private onPong = () => { + this.pongReceived = true + } + + private handleOpenEvent(): void { + const connectDuration = Date.now() - this.connectStartTime + logForDebugging('WebSocketTransport: Connected') + logForDiagnosticsNoPII('info', 'cli_websocket_connect_connected', { + duration_ms: connectDuration, + }) + + // Reconnect success — capture attempt count + downtime before resetting. + // reconnectStartTime is null on first connect, non-null on reopen. + if (this.isBridge && this.reconnectStartTime !== null) { + logEvent('tengu_ws_transport_reconnected', { + attempts: this.reconnectAttempts, + downtimeMs: Date.now() - this.reconnectStartTime, + }) + } + + this.reconnectAttempts = 0 + this.reconnectStartTime = null + this.lastReconnectAttemptTime = null + this.lastActivityTime = Date.now() + this.state = 'connected' + this.onConnectCallback?.() + + // Start periodic pings to detect dead connections + this.startPingInterval() + + // Start periodic keep_alive data frames to reset proxy idle timers + this.startKeepaliveInterval() + + // Register callback for session activity signals + registerSessionActivityCallback(() => { + void this.write({ type: 'keep_alive' }) + }) + } + + protected sendLine(line: string): boolean { + if (!this.ws || this.state !== 'connected') { + logForDebugging('WebSocketTransport: Not connected') + logForDiagnosticsNoPII('info', 'cli_websocket_send_not_connected') + return false + } + + try { + this.ws.send(line) + this.lastActivityTime = Date.now() + return true + } catch (error) { + logForDebugging(`WebSocketTransport: Failed to send: ${error}`, { + level: 'error', + }) + logForDiagnosticsNoPII('error', 'cli_websocket_send_error') + // Don't null this.ws here — let doDisconnect() (via handleConnectionError) + // handle cleanup so listeners are removed before the WS is released. + this.handleConnectionError() + return false + } + } + + /** + * Remove all listeners attached in connect() for the given WebSocket. + * Without this, each reconnect orphans the old WS object + its closures + * until GC — these accumulate under network instability. Mirrors the + * pattern in src/utils/mcpWebSocketTransport.ts. + */ + private removeWsListeners(ws: WebSocketLike): void { + if (this.isBunWs) { + const nws = ws as unknown as globalThis.WebSocket + nws.removeEventListener('open', this.onBunOpen) + nws.removeEventListener('message', this.onBunMessage) + nws.removeEventListener('error', this.onBunError) + // eslint-disable-next-line eslint-plugin-n/no-unsupported-features/node-builtins + nws.removeEventListener('close', this.onBunClose) + // 'pong' is Bun-specific — not in DOM typings + nws.removeEventListener('pong' as 'message', this.onPong) + } else { + const nws = ws as unknown as WsWebSocket + nws.off('open', this.onNodeOpen) + nws.off('message', this.onNodeMessage) + nws.off('error', this.onNodeError) + nws.off('close', this.onNodeClose) + nws.off('pong', this.onPong) + } + } + + protected doDisconnect(): void { + // Stop pinging and keepalive when disconnecting + this.stopPingInterval() + this.stopKeepaliveInterval() + + // Unregister session activity callback + unregisterSessionActivityCallback() + + if (this.ws) { + // Remove listeners BEFORE close() so the old WS + closures can be + // GC'd promptly instead of lingering until the next mark-and-sweep. + this.removeWsListeners(this.ws) + this.ws.close() + this.ws = null + } + } + + private handleConnectionError(closeCode?: number): void { + logForDebugging( + `WebSocketTransport: Disconnected from ${this.url.href}` + + (closeCode != null ? ` (code ${closeCode})` : ''), + ) + logForDiagnosticsNoPII('info', 'cli_websocket_disconnected') + if (this.isBridge) { + // Fire on every close — including intermediate ones during a reconnect + // storm (those never surface to the onCloseCallback consumer). For the + // Cloudflare-5min-idle hypothesis: cluster msSinceLastActivity; if the + // peak sits at ~300s with closeCode 1006, that's the proxy RST. + logEvent('tengu_ws_transport_closed', { + closeCode, + msSinceLastActivity: + this.lastActivityTime > 0 ? Date.now() - this.lastActivityTime : -1, + // 'connected' = healthy drop (the Cloudflare case); 'reconnecting' = + // connect-rejection mid-storm. State isn't mutated until the branches + // below, so this reads the pre-close value. + wasConnected: this.state === 'connected', + reconnectAttempts: this.reconnectAttempts, + }) + } + this.doDisconnect() + + if (this.state === 'closing' || this.state === 'closed') return + + // Permanent codes: don't retry — server has definitively ended the session. + // Exception: 4003 (unauthorized) can be retried when refreshHeaders is + // available and returns a new token (e.g. after the parent process mints + // a fresh session ingress token during reconnection). + let headersRefreshed = false + if (closeCode === 4003 && this.refreshHeaders) { + const freshHeaders = this.refreshHeaders() + if (freshHeaders.Authorization !== this.headers.Authorization) { + Object.assign(this.headers, freshHeaders) + headersRefreshed = true + logForDebugging( + 'WebSocketTransport: 4003 received but headers refreshed, scheduling reconnect', + ) + logForDiagnosticsNoPII('info', 'cli_websocket_4003_token_refreshed') + } + } + + if ( + closeCode != null && + PERMANENT_CLOSE_CODES.has(closeCode) && + !headersRefreshed + ) { + logForDebugging( + `WebSocketTransport: Permanent close code ${closeCode}, not reconnecting`, + { level: 'error' }, + ) + logForDiagnosticsNoPII('error', 'cli_websocket_permanent_close', { + closeCode, + }) + this.state = 'closed' + this.onCloseCallback?.(closeCode) + return + } + + // When autoReconnect is disabled, go straight to closed state. + // The caller (e.g. REPL bridge poll loop) handles recovery. + if (!this.autoReconnect) { + this.state = 'closed' + this.onCloseCallback?.(closeCode) + return + } + + // Schedule reconnection with exponential backoff and time budget + const now = Date.now() + if (!this.reconnectStartTime) { + this.reconnectStartTime = now + } + + // Detect system sleep/wake: if the gap since our last reconnection + // attempt greatly exceeds the max delay, the machine likely slept + // (e.g. laptop lid closed). Reset the budget and retry from scratch — + // the server will reject with permanent close codes (4001/1002) if + // the session was reaped while we were asleep. + if ( + this.lastReconnectAttemptTime !== null && + now - this.lastReconnectAttemptTime > SLEEP_DETECTION_THRESHOLD_MS + ) { + logForDebugging( + `WebSocketTransport: Detected system sleep (${Math.round((now - this.lastReconnectAttemptTime) / 1000)}s gap), resetting reconnection budget`, + ) + logForDiagnosticsNoPII('info', 'cli_websocket_sleep_detected', { + gapMs: now - this.lastReconnectAttemptTime, + }) + this.reconnectStartTime = now + this.reconnectAttempts = 0 + } + this.lastReconnectAttemptTime = now + + const elapsed = now - this.reconnectStartTime + if (elapsed < DEFAULT_RECONNECT_GIVE_UP_MS) { + // Clear any existing reconnection timer to avoid duplicates + if (this.reconnectTimer) { + clearTimeout(this.reconnectTimer) + this.reconnectTimer = null + } + + // Refresh headers before reconnecting (e.g. to pick up a new session token). + // Skip if already refreshed by the 4003 path above. + if (!headersRefreshed && this.refreshHeaders) { + const freshHeaders = this.refreshHeaders() + Object.assign(this.headers, freshHeaders) + logForDebugging('WebSocketTransport: Refreshed headers for reconnect') + } + + this.state = 'reconnecting' + this.reconnectAttempts++ + + const baseDelay = Math.min( + DEFAULT_BASE_RECONNECT_DELAY * Math.pow(2, this.reconnectAttempts - 1), + DEFAULT_MAX_RECONNECT_DELAY, + ) + // Add ±25% jitter to avoid thundering herd + const delay = Math.max( + 0, + baseDelay + baseDelay * 0.25 * (2 * Math.random() - 1), + ) + + logForDebugging( + `WebSocketTransport: Reconnecting in ${Math.round(delay)}ms (attempt ${this.reconnectAttempts}, ${Math.round(elapsed / 1000)}s elapsed)`, + ) + logForDiagnosticsNoPII('error', 'cli_websocket_reconnect_attempt', { + reconnectAttempts: this.reconnectAttempts, + }) + if (this.isBridge) { + logEvent('tengu_ws_transport_reconnecting', { + attempt: this.reconnectAttempts, + elapsedMs: elapsed, + delayMs: Math.round(delay), + }) + } + + this.reconnectTimer = setTimeout(() => { + this.reconnectTimer = null + void this.connect() + }, delay) + } else { + logForDebugging( + `WebSocketTransport: Reconnection time budget exhausted after ${Math.round(elapsed / 1000)}s for ${this.url.href}`, + { level: 'error' }, + ) + logForDiagnosticsNoPII('error', 'cli_websocket_reconnect_exhausted', { + reconnectAttempts: this.reconnectAttempts, + elapsedMs: elapsed, + }) + this.state = 'closed' + + // Notify close callback + if (this.onCloseCallback) { + this.onCloseCallback(closeCode) + } + } + } + + close(): void { + // Clear any pending reconnection timer + if (this.reconnectTimer) { + clearTimeout(this.reconnectTimer) + this.reconnectTimer = null + } + + // Clear ping and keepalive intervals + this.stopPingInterval() + this.stopKeepaliveInterval() + + // Unregister session activity callback + unregisterSessionActivityCallback() + + this.state = 'closing' + this.doDisconnect() + } + + private replayBufferedMessages(lastId: string): void { + const messages = this.messageBuffer.toArray() + if (messages.length === 0) return + + // Find where to start replay based on server's last received message + let startIndex = 0 + if (lastId) { + const lastConfirmedIndex = messages.findIndex( + message => 'uuid' in message && message.uuid === lastId, + ) + if (lastConfirmedIndex >= 0) { + // Server confirmed messages up to lastConfirmedIndex — evict them + startIndex = lastConfirmedIndex + 1 + // Rebuild the buffer with only unconfirmed messages + const remaining = messages.slice(startIndex) + this.messageBuffer.clear() + this.messageBuffer.addAll(remaining) + if (remaining.length === 0) { + this.lastSentId = null + } + logForDebugging( + `WebSocketTransport: Evicted ${startIndex} confirmed messages, ${remaining.length} remaining`, + ) + logForDiagnosticsNoPII( + 'info', + 'cli_websocket_evicted_confirmed_messages', + { + evicted: startIndex, + remaining: remaining.length, + }, + ) + } + } + + const messagesToReplay = messages.slice(startIndex) + if (messagesToReplay.length === 0) { + logForDebugging('WebSocketTransport: No new messages to replay') + logForDiagnosticsNoPII('info', 'cli_websocket_no_messages_to_replay') + return + } + + logForDebugging( + `WebSocketTransport: Replaying ${messagesToReplay.length} buffered messages`, + ) + logForDiagnosticsNoPII('info', 'cli_websocket_messages_to_replay', { + count: messagesToReplay.length, + }) + + for (const message of messagesToReplay) { + const line = jsonStringify(message) + '\n' + const success = this.sendLine(line) + if (!success) { + this.handleConnectionError() + break + } + } + // Do NOT clear the buffer after replay — messages remain buffered until + // the server confirms receipt on the next reconnection. This prevents + // message loss if the connection drops after replay but before the server + // processes the messages. + } + + isConnectedStatus(): boolean { + return this.state === 'connected' + } + + isClosedStatus(): boolean { + return this.state === 'closed' + } + + setOnData(callback: (data: string) => void): void { + this.onData = callback + } + + setOnConnect(callback: () => void): void { + this.onConnectCallback = callback + } + + setOnClose(callback: (closeCode?: number) => void): void { + this.onCloseCallback = callback + } + + getStateLabel(): string { + return this.state + } + + async write(message: StdoutMessage): Promise { + if ('uuid' in message && typeof message.uuid === 'string') { + this.messageBuffer.add(message) + this.lastSentId = message.uuid + } + + const line = jsonStringify(message) + '\n' + + if (this.state !== 'connected') { + // Message buffered for replay when connected (if it has a UUID) + return + } + + const sessionLabel = this.sessionId ? ` session=${this.sessionId}` : '' + const detailLabel = this.getControlMessageDetailLabel(message) + + logForDebugging( + `WebSocketTransport: Sending message type=${message.type}${sessionLabel}${detailLabel}`, + ) + + this.sendLine(line) + } + + private getControlMessageDetailLabel(message: StdoutMessage): string { + if (message.type === 'control_request') { + const { request_id, request } = message + const toolName = + request.subtype === 'can_use_tool' ? request.tool_name : '' + return ` subtype=${request.subtype} request_id=${request_id}${toolName ? ` tool=${toolName}` : ''}` + } + if (message.type === 'control_response') { + const { subtype, request_id } = message.response + return ` subtype=${subtype} request_id=${request_id}` + } + return '' + } + + private startPingInterval(): void { + // Clear any existing interval + this.stopPingInterval() + + this.pongReceived = true + let lastTickTime = Date.now() + + // Send ping periodically to detect dead connections. + // If the previous ping got no pong, treat the connection as dead. + this.pingInterval = setInterval(() => { + if (this.state === 'connected' && this.ws) { + const now = Date.now() + const gap = now - lastTickTime + lastTickTime = now + + // Process-suspension detector. If the wall-clock gap between ticks + // greatly exceeds the 10s interval, the process was suspended + // (laptop lid, SIGSTOP, VM pause). setInterval does not queue + // missed ticks — it coalesces — so on wake this callback fires + // once with a huge gap. The socket is almost certainly dead: + // NAT mappings drop in 30s–5min, and the server has been + // retransmitting into the void. Don't wait for a ping/pong + // round-trip to confirm (ws.ping() on a dead socket returns + // immediately with no error — bytes go into the kernel send + // buffer). Assume dead and reconnect now. A spurious reconnect + // after a short sleep is cheap — replayBufferedMessages() handles + // it and the server dedups by UUID. + if (gap > SLEEP_DETECTION_THRESHOLD_MS) { + logForDebugging( + `WebSocketTransport: ${Math.round(gap / 1000)}s tick gap detected — process was suspended, forcing reconnect`, + ) + logForDiagnosticsNoPII( + 'info', + 'cli_websocket_sleep_detected_on_ping', + { gapMs: gap }, + ) + this.handleConnectionError() + return + } + + if (!this.pongReceived) { + logForDebugging( + 'WebSocketTransport: No pong received, connection appears dead', + { level: 'error' }, + ) + logForDiagnosticsNoPII('error', 'cli_websocket_pong_timeout') + this.handleConnectionError() + return + } + + this.pongReceived = false + try { + this.ws.ping?.() + } catch (error) { + logForDebugging(`WebSocketTransport: Ping failed: ${error}`, { + level: 'error', + }) + logForDiagnosticsNoPII('error', 'cli_websocket_ping_failed') + } + } + }, DEFAULT_PING_INTERVAL) + } + + private stopPingInterval(): void { + if (this.pingInterval) { + clearInterval(this.pingInterval) + this.pingInterval = null + } + } + + private startKeepaliveInterval(): void { + this.stopKeepaliveInterval() + + // In CCR sessions, session activity heartbeats handle keep-alives + if (isEnvTruthy(process.env.CLAUDE_CODE_REMOTE)) { + return + } + + this.keepAliveInterval = setInterval(() => { + if (this.state === 'connected' && this.ws) { + try { + this.ws.send(KEEP_ALIVE_FRAME) + this.lastActivityTime = Date.now() + logForDebugging( + 'WebSocketTransport: Sent periodic keep_alive data frame', + ) + } catch (error) { + logForDebugging( + `WebSocketTransport: Periodic keep_alive failed: ${error}`, + { level: 'error' }, + ) + logForDiagnosticsNoPII('error', 'cli_websocket_keepalive_failed') + } + } + }, DEFAULT_KEEPALIVE_INTERVAL) + } + + private stopKeepaliveInterval(): void { + if (this.keepAliveInterval) { + clearInterval(this.keepAliveInterval) + this.keepAliveInterval = null + } + } +} diff --git a/src/cli/transports/WorkerStateUploader.ts b/src/cli/transports/WorkerStateUploader.ts new file mode 100644 index 0000000..37427b4 --- /dev/null +++ b/src/cli/transports/WorkerStateUploader.ts @@ -0,0 +1,131 @@ +import { sleep } from '../../utils/sleep.js' + +/** + * Coalescing uploader for PUT /worker (session state + metadata). + * + * - 1 in-flight PUT + 1 pending patch + * - New calls coalesce into pending (never grows beyond 1 slot) + * - On success: send pending if exists + * - On failure: exponential backoff (clamped), retries indefinitely + * until success or close(). Absorbs any pending patches before each retry. + * - No backpressure needed — naturally bounded at 2 slots + * + * Coalescing rules: + * - Top-level keys (worker_status, external_metadata) — last value wins + * - Inside external_metadata / internal_metadata — RFC 7396 merge: + * keys are added/overwritten, null values preserved (server deletes) + */ + +type WorkerStateUploaderConfig = { + send: (body: Record) => Promise + /** Base delay for exponential backoff (ms) */ + baseDelayMs: number + /** Max delay cap (ms) */ + maxDelayMs: number + /** Random jitter range added to retry delay (ms) */ + jitterMs: number +} + +export class WorkerStateUploader { + private inflight: Promise | null = null + private pending: Record | null = null + private closed = false + private readonly config: WorkerStateUploaderConfig + + constructor(config: WorkerStateUploaderConfig) { + this.config = config + } + + /** + * Enqueue a patch to PUT /worker. Coalesces with any existing pending + * patch. Fire-and-forget — callers don't need to await. + */ + enqueue(patch: Record): void { + if (this.closed) return + this.pending = this.pending ? coalescePatches(this.pending, patch) : patch + void this.drain() + } + + close(): void { + this.closed = true + this.pending = null + } + + private async drain(): Promise { + if (this.inflight || this.closed) return + if (!this.pending) return + + const payload = this.pending + this.pending = null + + this.inflight = this.sendWithRetry(payload).then(() => { + this.inflight = null + if (this.pending && !this.closed) { + void this.drain() + } + }) + } + + /** Retries indefinitely with exponential backoff until success or close(). */ + private async sendWithRetry(payload: Record): Promise { + let current = payload + let failures = 0 + while (!this.closed) { + const ok = await this.config.send(current) + if (ok) return + + failures++ + await sleep(this.retryDelay(failures)) + + // Absorb any patches that arrived during the retry + if (this.pending && !this.closed) { + current = coalescePatches(current, this.pending) + this.pending = null + } + } + } + + private retryDelay(failures: number): number { + const exponential = Math.min( + this.config.baseDelayMs * 2 ** (failures - 1), + this.config.maxDelayMs, + ) + const jitter = Math.random() * this.config.jitterMs + return exponential + jitter + } +} + +/** + * Coalesce two patches for PUT /worker. + * + * Top-level keys: overlay replaces base (last value wins). + * Metadata keys (external_metadata, internal_metadata): RFC 7396 merge + * one level deep — overlay keys are added/overwritten, null values + * preserved for server-side delete. + */ +function coalescePatches( + base: Record, + overlay: Record, +): Record { + const merged = { ...base } + + for (const [key, value] of Object.entries(overlay)) { + if ( + (key === 'external_metadata' || key === 'internal_metadata') && + merged[key] && + typeof merged[key] === 'object' && + typeof value === 'object' && + value !== null + ) { + // RFC 7396 merge — overlay keys win, nulls preserved for server + merged[key] = { + ...(merged[key] as Record), + ...(value as Record), + } + } else { + merged[key] = value + } + } + + return merged +} diff --git a/src/cli/transports/ccrClient.ts b/src/cli/transports/ccrClient.ts new file mode 100644 index 0000000..da3dc2e --- /dev/null +++ b/src/cli/transports/ccrClient.ts @@ -0,0 +1,998 @@ +import { randomUUID } from 'crypto' +import type { + SDKPartialAssistantMessage, + StdoutMessage, +} from 'src/entrypoints/sdk/controlTypes.js' +import { decodeJwtExpiry } from '../../bridge/jwtUtils.js' +import { logForDebugging } from '../../utils/debug.js' +import { logForDiagnosticsNoPII } from '../../utils/diagLogs.js' +import { errorMessage, getErrnoCode } from '../../utils/errors.js' +import { createAxiosInstance } from '../../utils/proxy.js' +import { + registerSessionActivityCallback, + unregisterSessionActivityCallback, +} from '../../utils/sessionActivity.js' +import { + getSessionIngressAuthHeaders, + getSessionIngressAuthToken, +} from '../../utils/sessionIngressAuth.js' +import type { + RequiresActionDetails, + SessionState, +} from '../../utils/sessionState.js' +import { sleep } from '../../utils/sleep.js' +import { getClaudeCodeUserAgent } from '../../utils/userAgent.js' +import { + RetryableError, + SerialBatchEventUploader, +} from './SerialBatchEventUploader.js' +import type { SSETransport, StreamClientEvent } from './SSETransport.js' +import { WorkerStateUploader } from './WorkerStateUploader.js' + +/** Default interval between heartbeat events (20s; server TTL is 60s). */ +const DEFAULT_HEARTBEAT_INTERVAL_MS = 20_000 + +/** + * stream_event messages accumulate in a delay buffer for up to this many ms + * before enqueue. Mirrors HybridTransport's batching window. text_delta + * events for the same content block accumulate into a single full-so-far + * snapshot per flush — each emitted event is self-contained so a client + * connecting mid-stream sees complete text, not a fragment. + */ +const STREAM_EVENT_FLUSH_INTERVAL_MS = 100 + +/** Hoisted axios validateStatus callback to avoid per-request closure allocation. */ +function alwaysValidStatus(): boolean { + return true +} + +export type CCRInitFailReason = + | 'no_auth_headers' + | 'missing_epoch' + | 'worker_register_failed' + +/** Thrown by initialize(); carries a typed reason for the diag classifier. */ +export class CCRInitError extends Error { + constructor(readonly reason: CCRInitFailReason) { + super(`CCRClient init failed: ${reason}`) + } +} + +/** + * Consecutive 401/403 with a VALID-LOOKING token before giving up. An + * expired JWT short-circuits this (exits immediately — deterministic, + * retry is futile). This threshold is for the uncertain case: token's + * exp is in the future but server says 401 (userauth down, KMS hiccup, + * clock skew). 10 × 20s heartbeat ≈ 200s to ride it out. + */ +const MAX_CONSECUTIVE_AUTH_FAILURES = 10 + +type EventPayload = { + uuid: string + type: string + [key: string]: unknown +} + +type ClientEvent = { + payload: EventPayload + ephemeral?: boolean +} + +/** + * Structural subset of a stream_event carrying a text_delta. Not a narrowing + * of SDKPartialAssistantMessage — RawMessageStreamEvent's delta is a union and + * narrowing through two levels defeats the discriminant. + */ +type CoalescedStreamEvent = { + type: 'stream_event' + uuid: string + session_id: string + parent_tool_use_id: string | null + event: { + type: 'content_block_delta' + index: number + delta: { type: 'text_delta'; text: string } + } +} + +/** + * Accumulator state for text_delta coalescing. Keyed by API message ID so + * lifetime is tied to the assistant message — cleared when the complete + * SDKAssistantMessage arrives (writeEvent), which is reliable even when + * abort/error paths skip content_block_stop/message_stop delivery. + */ +export type StreamAccumulatorState = { + /** API message ID (msg_...) → blocks[blockIndex] → chunk array. */ + byMessage: Map + /** + * {session_id}:{parent_tool_use_id} → active message ID. + * content_block_delta events don't carry the message ID (only + * message_start does), so we track which message is currently streaming + * for each scope. At most one message streams per scope at a time. + */ + scopeToMessage: Map +} + +export function createStreamAccumulator(): StreamAccumulatorState { + return { byMessage: new Map(), scopeToMessage: new Map() } +} + +function scopeKey(m: { + session_id: string + parent_tool_use_id: string | null +}): string { + return `${m.session_id}:${m.parent_tool_use_id ?? ''}` +} + +/** + * Accumulate text_delta stream_events into full-so-far snapshots per content + * block. Each flush emits ONE event per touched block containing the FULL + * accumulated text from the start of the block — a client connecting + * mid-stream receives a self-contained snapshot, not a fragment. + * + * Non-text-delta events pass through unchanged. message_start records the + * active message ID for the scope; content_block_delta appends chunks; + * the snapshot event reuses the first text_delta UUID seen for that block in + * this flush so server-side idempotency remains stable across retries. + * + * Cleanup happens in writeEvent when the complete assistant message arrives + * (reliable), not here on stop events (abort/error paths skip those). + */ +export function accumulateStreamEvents( + buffer: SDKPartialAssistantMessage[], + state: StreamAccumulatorState, +): EventPayload[] { + const out: EventPayload[] = [] + // chunks[] → snapshot already in `out` this flush. Keyed by the chunks + // array reference (stable per {messageId, index}) so subsequent deltas + // rewrite the same entry instead of emitting one event per delta. + const touched = new Map() + for (const msg of buffer) { + switch (msg.event.type) { + case 'message_start': { + const id = msg.event.message.id + const prevId = state.scopeToMessage.get(scopeKey(msg)) + if (prevId) state.byMessage.delete(prevId) + state.scopeToMessage.set(scopeKey(msg), id) + state.byMessage.set(id, []) + out.push(msg) + break + } + case 'content_block_delta': { + if (msg.event.delta.type !== 'text_delta') { + out.push(msg) + break + } + const messageId = state.scopeToMessage.get(scopeKey(msg)) + const blocks = messageId ? state.byMessage.get(messageId) : undefined + if (!blocks) { + // Delta without a preceding message_start (reconnect mid-stream, + // or message_start was in a prior buffer that got dropped). Pass + // through raw — can't produce a full-so-far snapshot without the + // prior chunks anyway. + out.push(msg) + break + } + const chunks = (blocks[msg.event.index] ??= []) + chunks.push(msg.event.delta.text) + const existing = touched.get(chunks) + if (existing) { + existing.event.delta.text = chunks.join('') + break + } + const snapshot: CoalescedStreamEvent = { + type: 'stream_event', + uuid: msg.uuid, + session_id: msg.session_id, + parent_tool_use_id: msg.parent_tool_use_id, + event: { + type: 'content_block_delta', + index: msg.event.index, + delta: { type: 'text_delta', text: chunks.join('') }, + }, + } + touched.set(chunks, snapshot) + out.push(snapshot) + break + } + default: + out.push(msg) + } + } + return out +} + +/** + * Clear accumulator entries for a completed assistant message. Called from + * writeEvent when the SDKAssistantMessage arrives — the reliable end-of-stream + * signal that fires even when abort/interrupt/error skip SSE stop events. + */ +export function clearStreamAccumulatorForMessage( + state: StreamAccumulatorState, + assistant: { + session_id: string + parent_tool_use_id: string | null + message: { id: string } + }, +): void { + state.byMessage.delete(assistant.message.id) + const scope = scopeKey(assistant) + if (state.scopeToMessage.get(scope) === assistant.message.id) { + state.scopeToMessage.delete(scope) + } +} + +type RequestResult = { ok: true } | { ok: false; retryAfterMs?: number } + +type WorkerEvent = { + payload: EventPayload + is_compaction?: boolean + agent_id?: string +} + +export type InternalEvent = { + event_id: string + event_type: string + payload: Record + event_metadata?: Record | null + is_compaction: boolean + created_at: string + agent_id?: string +} + +type ListInternalEventsResponse = { + data: InternalEvent[] + next_cursor?: string +} + +type WorkerStateResponse = { + worker?: { + external_metadata?: Record + } +} + +/** + * Manages the worker lifecycle protocol with CCR v2: + * - Epoch management: reads worker_epoch from CLAUDE_CODE_WORKER_EPOCH env var + * - Runtime state reporting: PUT /sessions/{id}/worker + * - Heartbeat: POST /sessions/{id}/worker/heartbeat for liveness detection + * + * All writes go through this.request(). + */ +export class CCRClient { + private workerEpoch = 0 + private readonly heartbeatIntervalMs: number + private readonly heartbeatJitterFraction: number + private heartbeatTimer: NodeJS.Timeout | null = null + private heartbeatInFlight = false + private closed = false + private consecutiveAuthFailures = 0 + private currentState: SessionState | null = null + private readonly sessionBaseUrl: string + private readonly sessionId: string + private readonly http = createAxiosInstance({ keepAlive: true }) + + // stream_event delay buffer — accumulates content deltas for up to + // STREAM_EVENT_FLUSH_INTERVAL_MS before enqueueing (reduces POST count + // and enables text_delta coalescing). Mirrors HybridTransport's pattern. + private streamEventBuffer: SDKPartialAssistantMessage[] = [] + private streamEventTimer: ReturnType | null = null + // Full-so-far text accumulator. Persists across flushes so each emitted + // text_delta event carries the complete text from the start of the block — + // mid-stream reconnects see a self-contained snapshot. Keyed by API message + // ID; cleared in writeEvent when the complete assistant message arrives. + private streamTextAccumulator = createStreamAccumulator() + + private readonly workerState: WorkerStateUploader + private readonly eventUploader: SerialBatchEventUploader + private readonly internalEventUploader: SerialBatchEventUploader + private readonly deliveryUploader: SerialBatchEventUploader<{ + eventId: string + status: 'received' | 'processing' | 'processed' + }> + + /** + * Called when the server returns 409 (a newer worker epoch superseded ours). + * Default: process.exit(1) — correct for spawn-mode children where the + * parent bridge re-spawns. In-process callers (replBridge) MUST override + * this to close gracefully instead; exit would kill the user's REPL. + */ + private readonly onEpochMismatch: () => never + + /** + * Auth header source. Defaults to the process-wide session-ingress token + * (CLAUDE_CODE_SESSION_ACCESS_TOKEN env var). Callers managing multiple + * concurrent sessions with distinct JWTs MUST inject this — the env-var + * path is a process global and would stomp across sessions. + */ + private readonly getAuthHeaders: () => Record + + constructor( + transport: SSETransport, + sessionUrl: URL, + opts?: { + onEpochMismatch?: () => never + heartbeatIntervalMs?: number + heartbeatJitterFraction?: number + /** + * Per-instance auth header source. Omit to read the process-wide + * CLAUDE_CODE_SESSION_ACCESS_TOKEN (single-session callers — REPL, + * daemon). Required for concurrent multi-session callers. + */ + getAuthHeaders?: () => Record + }, + ) { + this.onEpochMismatch = + opts?.onEpochMismatch ?? + (() => { + // eslint-disable-next-line custom-rules/no-process-exit + process.exit(1) + }) + this.heartbeatIntervalMs = + opts?.heartbeatIntervalMs ?? DEFAULT_HEARTBEAT_INTERVAL_MS + this.heartbeatJitterFraction = opts?.heartbeatJitterFraction ?? 0 + this.getAuthHeaders = opts?.getAuthHeaders ?? getSessionIngressAuthHeaders + // Session URL: https://host/v1/code/sessions/{id} + if (sessionUrl.protocol !== 'http:' && sessionUrl.protocol !== 'https:') { + throw new Error( + `CCRClient: Expected http(s) URL, got ${sessionUrl.protocol}`, + ) + } + const pathname = sessionUrl.pathname.replace(/\/$/, '') + this.sessionBaseUrl = `${sessionUrl.protocol}//${sessionUrl.host}${pathname}` + // Extract session ID from the URL path (last segment) + this.sessionId = pathname.split('/').pop() || '' + + this.workerState = new WorkerStateUploader({ + send: body => + this.request( + 'put', + '/worker', + { worker_epoch: this.workerEpoch, ...body }, + 'PUT worker', + ).then(r => r.ok), + baseDelayMs: 500, + maxDelayMs: 30_000, + jitterMs: 500, + }) + + this.eventUploader = new SerialBatchEventUploader({ + maxBatchSize: 100, + maxBatchBytes: 10 * 1024 * 1024, + // flushStreamEventBuffer() enqueues a full 100ms window of accumulated + // stream_events in one call. A burst of mixed delta types that don't + // fold into a single snapshot could exceed the old cap (50) and deadlock + // on the SerialBatchEventUploader backpressure check. Match + // HybridTransport's bound — high enough to be memory-only. + maxQueueSize: 100_000, + send: async batch => { + const result = await this.request( + 'post', + '/worker/events', + { worker_epoch: this.workerEpoch, events: batch }, + 'client events', + ) + if (!result.ok) { + throw new RetryableError( + 'client event POST failed', + result.retryAfterMs, + ) + } + }, + baseDelayMs: 500, + maxDelayMs: 30_000, + jitterMs: 500, + }) + + this.internalEventUploader = new SerialBatchEventUploader({ + maxBatchSize: 100, + maxBatchBytes: 10 * 1024 * 1024, + maxQueueSize: 200, + send: async batch => { + const result = await this.request( + 'post', + '/worker/internal-events', + { worker_epoch: this.workerEpoch, events: batch }, + 'internal events', + ) + if (!result.ok) { + throw new RetryableError( + 'internal event POST failed', + result.retryAfterMs, + ) + } + }, + baseDelayMs: 500, + maxDelayMs: 30_000, + jitterMs: 500, + }) + + this.deliveryUploader = new SerialBatchEventUploader<{ + eventId: string + status: 'received' | 'processing' | 'processed' + }>({ + maxBatchSize: 64, + maxQueueSize: 64, + send: async batch => { + const result = await this.request( + 'post', + '/worker/events/delivery', + { + worker_epoch: this.workerEpoch, + updates: batch.map(d => ({ + event_id: d.eventId, + status: d.status, + })), + }, + 'delivery batch', + ) + if (!result.ok) { + throw new RetryableError('delivery POST failed', result.retryAfterMs) + } + }, + baseDelayMs: 500, + maxDelayMs: 30_000, + jitterMs: 500, + }) + + // Ack each received client_event so CCR can track delivery status. + // Wired here (not in initialize()) so the callback is registered the + // moment new CCRClient() returns — remoteIO must be free to call + // transport.connect() immediately after without racing the first + // SSE catch-up frame against an unwired onEventCallback. + transport.setOnEvent((event: StreamClientEvent) => { + this.reportDelivery(event.event_id, 'received') + }) + } + + /** + * Initialize the session worker: + * 1. Take worker_epoch from the argument, or fall back to + * CLAUDE_CODE_WORKER_EPOCH (set by env-manager / bridge spawner) + * 2. Report state as 'idle' + * 3. Start heartbeat timer + * + * In-process callers (replBridge) pass the epoch directly — they + * registered the worker themselves and there is no parent process + * setting env vars. + */ + async initialize(epoch?: number): Promise | null> { + const startMs = Date.now() + if (Object.keys(this.getAuthHeaders()).length === 0) { + throw new CCRInitError('no_auth_headers') + } + if (epoch === undefined) { + const rawEpoch = process.env.CLAUDE_CODE_WORKER_EPOCH + epoch = rawEpoch ? parseInt(rawEpoch, 10) : NaN + } + if (isNaN(epoch)) { + throw new CCRInitError('missing_epoch') + } + this.workerEpoch = epoch + + // Concurrent with the init PUT — neither depends on the other. + const restoredPromise = this.getWorkerState() + + const result = await this.request( + 'put', + '/worker', + { + worker_status: 'idle', + worker_epoch: this.workerEpoch, + // Clear stale pending_action/task_summary left by a prior + // worker crash — the in-session clears don't survive process restart. + external_metadata: { + pending_action: null, + task_summary: null, + }, + }, + 'PUT worker (init)', + ) + if (!result.ok) { + // 409 → onEpochMismatch may throw, but request() catches it and returns + // false. Without this check we'd continue to startHeartbeat(), leaking a + // 20s timer against a dead epoch. Throw so connect()'s rejection handler + // fires instead of the success path. + throw new CCRInitError('worker_register_failed') + } + this.currentState = 'idle' + this.startHeartbeat() + + // sessionActivity's refcount-gated timer fires while an API call or tool + // is in-flight; without a write the container lease can expire mid-wait. + // v1 wires this in WebSocketTransport per-connection. + registerSessionActivityCallback(() => { + void this.writeEvent({ type: 'keep_alive' }) + }) + + logForDebugging(`CCRClient: initialized, epoch=${this.workerEpoch}`) + logForDiagnosticsNoPII('info', 'cli_worker_lifecycle_initialized', { + epoch: this.workerEpoch, + duration_ms: Date.now() - startMs, + }) + + // Await the concurrent GET and log state_restored here, after the PUT + // has succeeded — logging inside getWorkerState() raced: if the GET + // resolved before the PUT failed, diagnostics showed both init_failed + // and state_restored for the same session. + const { metadata, durationMs } = await restoredPromise + if (!this.closed) { + logForDiagnosticsNoPII('info', 'cli_worker_state_restored', { + duration_ms: durationMs, + had_state: metadata !== null, + }) + } + return metadata + } + + // Control_requests are marked processed and not re-delivered on + // restart, so read back what the prior worker wrote. + private async getWorkerState(): Promise<{ + metadata: Record | null + durationMs: number + }> { + const startMs = Date.now() + const authHeaders = this.getAuthHeaders() + if (Object.keys(authHeaders).length === 0) { + return { metadata: null, durationMs: 0 } + } + const data = await this.getWithRetry( + `${this.sessionBaseUrl}/worker`, + authHeaders, + 'worker_state', + ) + return { + metadata: data?.worker?.external_metadata ?? null, + durationMs: Date.now() - startMs, + } + } + + /** + * Send an authenticated HTTP request to CCR. Handles auth headers, + * 409 epoch mismatch, and error logging. Returns { ok: true } on 2xx. + * On 429, reads Retry-After (integer seconds) so the uploader can honor + * the server's backoff hint instead of blindly exponentiating. + */ + private async request( + method: 'post' | 'put', + path: string, + body: unknown, + label: string, + { timeout = 10_000 }: { timeout?: number } = {}, + ): Promise { + const authHeaders = this.getAuthHeaders() + if (Object.keys(authHeaders).length === 0) return { ok: false } + + try { + const response = await this.http[method]( + `${this.sessionBaseUrl}${path}`, + body, + { + headers: { + ...authHeaders, + 'Content-Type': 'application/json', + 'anthropic-version': '2023-06-01', + 'User-Agent': getClaudeCodeUserAgent(), + }, + validateStatus: alwaysValidStatus, + timeout, + }, + ) + + if (response.status >= 200 && response.status < 300) { + this.consecutiveAuthFailures = 0 + return { ok: true } + } + if (response.status === 409) { + this.handleEpochMismatch() + } + if (response.status === 401 || response.status === 403) { + // A 401 with an expired JWT is deterministic — no retry will + // ever succeed. Check the token's own exp before burning + // wall-clock on the threshold loop. + const tok = getSessionIngressAuthToken() + const exp = tok ? decodeJwtExpiry(tok) : null + if (exp !== null && exp * 1000 < Date.now()) { + logForDebugging( + `CCRClient: session_token expired (exp=${new Date(exp * 1000).toISOString()}) — no refresh was delivered, exiting`, + { level: 'error' }, + ) + logForDiagnosticsNoPII('error', 'cli_worker_token_expired_no_refresh') + this.onEpochMismatch() + } + // Token looks valid but server says 401 — possible server-side + // blip (userauth down, KMS hiccup). Count toward threshold. + this.consecutiveAuthFailures++ + if (this.consecutiveAuthFailures >= MAX_CONSECUTIVE_AUTH_FAILURES) { + logForDebugging( + `CCRClient: ${this.consecutiveAuthFailures} consecutive auth failures with a valid-looking token — server-side auth unrecoverable, exiting`, + { level: 'error' }, + ) + logForDiagnosticsNoPII('error', 'cli_worker_auth_failures_exhausted') + this.onEpochMismatch() + } + } + logForDebugging(`CCRClient: ${label} returned ${response.status}`, { + level: 'warn', + }) + logForDiagnosticsNoPII('warn', 'cli_worker_request_failed', { + method, + path, + status: response.status, + }) + if (response.status === 429) { + const raw = response.headers?.['retry-after'] + const seconds = typeof raw === 'string' ? parseInt(raw, 10) : NaN + if (!isNaN(seconds) && seconds >= 0) { + return { ok: false, retryAfterMs: seconds * 1000 } + } + } + return { ok: false } + } catch (error) { + logForDebugging(`CCRClient: ${label} failed: ${errorMessage(error)}`, { + level: 'warn', + }) + logForDiagnosticsNoPII('warn', 'cli_worker_request_error', { + method, + path, + error_code: getErrnoCode(error), + }) + return { ok: false } + } + } + + /** Report worker state to CCR via PUT /sessions/{id}/worker. */ + reportState(state: SessionState, details?: RequiresActionDetails): void { + if (state === this.currentState && !details) return + this.currentState = state + this.workerState.enqueue({ + worker_status: state, + requires_action_details: details + ? { + tool_name: details.tool_name, + action_description: details.action_description, + request_id: details.request_id, + } + : null, + }) + } + + /** Report external metadata to CCR via PUT /worker. */ + reportMetadata(metadata: Record): void { + this.workerState.enqueue({ external_metadata: metadata }) + } + + /** + * Handle epoch mismatch (409 Conflict). A newer CC instance has replaced + * this one — exit immediately. + */ + private handleEpochMismatch(): never { + logForDebugging('CCRClient: Epoch mismatch (409), shutting down', { + level: 'error', + }) + logForDiagnosticsNoPII('error', 'cli_worker_epoch_mismatch') + this.onEpochMismatch() + } + + /** Start periodic heartbeat. */ + private startHeartbeat(): void { + this.stopHeartbeat() + const schedule = (): void => { + const jitter = + this.heartbeatIntervalMs * + this.heartbeatJitterFraction * + (2 * Math.random() - 1) + this.heartbeatTimer = setTimeout(tick, this.heartbeatIntervalMs + jitter) + } + const tick = (): void => { + void this.sendHeartbeat() + // stopHeartbeat nulls the timer; check after the fire-and-forget send + // but before rescheduling so close() during sendHeartbeat is honored. + if (this.heartbeatTimer === null) return + schedule() + } + schedule() + } + + /** Stop heartbeat timer. */ + private stopHeartbeat(): void { + if (this.heartbeatTimer) { + clearTimeout(this.heartbeatTimer) + this.heartbeatTimer = null + } + } + + /** Send a heartbeat via POST /sessions/{id}/worker/heartbeat. */ + private async sendHeartbeat(): Promise { + if (this.heartbeatInFlight) return + this.heartbeatInFlight = true + try { + const result = await this.request( + 'post', + '/worker/heartbeat', + { session_id: this.sessionId, worker_epoch: this.workerEpoch }, + 'Heartbeat', + { timeout: 5_000 }, + ) + if (result.ok) { + logForDebugging('CCRClient: Heartbeat sent') + } + } finally { + this.heartbeatInFlight = false + } + } + + /** + * Write a StdoutMessage as a client event via POST /sessions/{id}/worker/events. + * These events are visible to frontend clients via the SSE stream. + * Injects a UUID if missing to ensure server-side idempotency on retry. + * + * stream_event messages are held in a 100ms delay buffer and accumulated + * (text_deltas for the same content block emit a full-so-far snapshot per + * flush). A non-stream_event write flushes the buffer first so downstream + * ordering is preserved. + */ + async writeEvent(message: StdoutMessage): Promise { + if (message.type === 'stream_event') { + this.streamEventBuffer.push(message) + if (!this.streamEventTimer) { + this.streamEventTimer = setTimeout( + () => void this.flushStreamEventBuffer(), + STREAM_EVENT_FLUSH_INTERVAL_MS, + ) + } + return + } + await this.flushStreamEventBuffer() + if (message.type === 'assistant') { + clearStreamAccumulatorForMessage(this.streamTextAccumulator, message) + } + await this.eventUploader.enqueue(this.toClientEvent(message)) + } + + /** Wrap a StdoutMessage as a ClientEvent, injecting a UUID if missing. */ + private toClientEvent(message: StdoutMessage): ClientEvent { + const msg = message as unknown as Record + return { + payload: { + ...msg, + uuid: typeof msg.uuid === 'string' ? msg.uuid : randomUUID(), + } as EventPayload, + } + } + + /** + * Drain the stream_event delay buffer: accumulate text_deltas into + * full-so-far snapshots, clear the timer, enqueue the resulting events. + * Called from the timer, from writeEvent on a non-stream message, and from + * flush(). close() drops the buffer — call flush() first if you need + * delivery. + */ + private async flushStreamEventBuffer(): Promise { + if (this.streamEventTimer) { + clearTimeout(this.streamEventTimer) + this.streamEventTimer = null + } + if (this.streamEventBuffer.length === 0) return + const buffered = this.streamEventBuffer + this.streamEventBuffer = [] + const payloads = accumulateStreamEvents( + buffered, + this.streamTextAccumulator, + ) + await this.eventUploader.enqueue( + payloads.map(payload => ({ payload, ephemeral: true })), + ) + } + + /** + * Write an internal worker event via POST /sessions/{id}/worker/internal-events. + * These events are NOT visible to frontend clients — they store worker-internal + * state (transcript messages, compaction markers) needed for session resume. + */ + async writeInternalEvent( + eventType: string, + payload: Record, + { + isCompaction = false, + agentId, + }: { + isCompaction?: boolean + agentId?: string + } = {}, + ): Promise { + const event: WorkerEvent = { + payload: { + type: eventType, + ...payload, + uuid: typeof payload.uuid === 'string' ? payload.uuid : randomUUID(), + } as EventPayload, + ...(isCompaction && { is_compaction: true }), + ...(agentId && { agent_id: agentId }), + } + await this.internalEventUploader.enqueue(event) + } + + /** + * Flush pending internal events. Call between turns and on shutdown + * to ensure transcript entries are persisted. + */ + flushInternalEvents(): Promise { + return this.internalEventUploader.flush() + } + + /** + * Flush pending client events (writeEvent queue). Call before close() + * when the caller needs delivery confirmation — close() abandons the + * queue. Resolves once the uploader drains or rejects; returns + * regardless of whether individual POSTs succeeded (check server state + * separately if that matters). + */ + async flush(): Promise { + await this.flushStreamEventBuffer() + return this.eventUploader.flush() + } + + /** + * Read foreground agent internal events from + * GET /sessions/{id}/worker/internal-events. + * Returns transcript entries from the last compaction boundary, or null on failure. + * Used for session resume. + */ + async readInternalEvents(): Promise { + return this.paginatedGet('/worker/internal-events', {}, 'internal_events') + } + + /** + * Read all subagent internal events from + * GET /sessions/{id}/worker/internal-events?subagents=true. + * Returns a merged stream across all non-foreground agents, each from its + * compaction point. Used for session resume. + */ + async readSubagentInternalEvents(): Promise { + return this.paginatedGet( + '/worker/internal-events', + { subagents: 'true' }, + 'subagent_events', + ) + } + + /** + * Paginated GET with retry. Fetches all pages from a list endpoint, + * retrying each page on failure with exponential backoff + jitter. + */ + private async paginatedGet( + path: string, + params: Record, + context: string, + ): Promise { + const authHeaders = this.getAuthHeaders() + if (Object.keys(authHeaders).length === 0) return null + + const allEvents: InternalEvent[] = [] + let cursor: string | undefined + + do { + const url = new URL(`${this.sessionBaseUrl}${path}`) + for (const [k, v] of Object.entries(params)) { + url.searchParams.set(k, v) + } + if (cursor) { + url.searchParams.set('cursor', cursor) + } + + const page = await this.getWithRetry( + url.toString(), + authHeaders, + context, + ) + if (!page) return null + + allEvents.push(...(page.data ?? [])) + cursor = page.next_cursor + } while (cursor) + + logForDebugging( + `CCRClient: Read ${allEvents.length} internal events from ${path}${params.subagents ? ' (subagents)' : ''}`, + ) + return allEvents + } + + /** + * Single GET request with retry. Returns the parsed response body + * on success, null if all retries are exhausted. + */ + private async getWithRetry( + url: string, + authHeaders: Record, + context: string, + ): Promise { + for (let attempt = 1; attempt <= 10; attempt++) { + let response + try { + response = await this.http.get(url, { + headers: { + ...authHeaders, + 'anthropic-version': '2023-06-01', + 'User-Agent': getClaudeCodeUserAgent(), + }, + validateStatus: alwaysValidStatus, + timeout: 30_000, + }) + } catch (error) { + logForDebugging( + `CCRClient: GET ${url} failed (attempt ${attempt}/10): ${errorMessage(error)}`, + { level: 'warn' }, + ) + if (attempt < 10) { + const delay = + Math.min(500 * 2 ** (attempt - 1), 30_000) + Math.random() * 500 + await sleep(delay) + } + continue + } + + if (response.status >= 200 && response.status < 300) { + return response.data + } + if (response.status === 409) { + this.handleEpochMismatch() + } + logForDebugging( + `CCRClient: GET ${url} returned ${response.status} (attempt ${attempt}/10)`, + { level: 'warn' }, + ) + + if (attempt < 10) { + const delay = + Math.min(500 * 2 ** (attempt - 1), 30_000) + Math.random() * 500 + await sleep(delay) + } + } + + logForDebugging('CCRClient: GET retries exhausted', { level: 'error' }) + logForDiagnosticsNoPII('error', 'cli_worker_get_retries_exhausted', { + context, + }) + return null + } + + /** + * Report delivery status for a client-to-worker event. + * POST /v1/code/sessions/{id}/worker/events/delivery (batch endpoint) + */ + reportDelivery( + eventId: string, + status: 'received' | 'processing' | 'processed', + ): void { + void this.deliveryUploader.enqueue({ eventId, status }) + } + + /** Get the current epoch (for external use). */ + getWorkerEpoch(): number { + return this.workerEpoch + } + + /** Internal-event queue depth — shutdown-snapshot backpressure signal. */ + get internalEventsPending(): number { + return this.internalEventUploader.pendingCount + } + + /** Clean up uploaders and timers. */ + close(): void { + this.closed = true + this.stopHeartbeat() + unregisterSessionActivityCallback() + if (this.streamEventTimer) { + clearTimeout(this.streamEventTimer) + this.streamEventTimer = null + } + this.streamEventBuffer = [] + this.streamTextAccumulator.byMessage.clear() + this.streamTextAccumulator.scopeToMessage.clear() + this.workerState.close() + this.eventUploader.close() + this.internalEventUploader.close() + this.deliveryUploader.close() + } +} diff --git a/src/cli/transports/transportUtils.ts b/src/cli/transports/transportUtils.ts new file mode 100644 index 0000000..9252473 --- /dev/null +++ b/src/cli/transports/transportUtils.ts @@ -0,0 +1,45 @@ +import { URL } from 'url' +import { isEnvTruthy } from '../../utils/envUtils.js' +import { HybridTransport } from './HybridTransport.js' +import { SSETransport } from './SSETransport.js' +import type { Transport } from './Transport.js' +import { WebSocketTransport } from './WebSocketTransport.js' + +/** + * Helper function to get the appropriate transport for a URL. + * + * Transport selection priority: + * 1. SSETransport (SSE reads + POST writes) when CLAUDE_CODE_USE_CCR_V2 is set + * 2. HybridTransport (WS reads + POST writes) when CLAUDE_CODE_POST_FOR_SESSION_INGRESS_V2 is set + * 3. WebSocketTransport (WS reads + WS writes) — default + */ +export function getTransportForUrl( + url: URL, + headers: Record = {}, + sessionId?: string, + refreshHeaders?: () => Record, +): Transport { + if (isEnvTruthy(process.env.CLAUDE_CODE_USE_CCR_V2)) { + // v2: SSE for reads, HTTP POST for writes + // --sdk-url is the session URL (.../sessions/{id}); + // derive the SSE stream URL by appending /worker/events/stream + const sseUrl = new URL(url.href) + if (sseUrl.protocol === 'wss:') { + sseUrl.protocol = 'https:' + } else if (sseUrl.protocol === 'ws:') { + sseUrl.protocol = 'http:' + } + sseUrl.pathname = + sseUrl.pathname.replace(/\/$/, '') + '/worker/events/stream' + return new SSETransport(sseUrl, headers, sessionId, refreshHeaders) + } + + if (url.protocol === 'ws:' || url.protocol === 'wss:') { + if (isEnvTruthy(process.env.CLAUDE_CODE_POST_FOR_SESSION_INGRESS_V2)) { + return new HybridTransport(url, headers, sessionId, refreshHeaders) + } + return new WebSocketTransport(url, headers, sessionId, refreshHeaders) + } else { + throw new Error(`Unsupported protocol: ${url.protocol}`) + } +} diff --git a/src/cli/update.ts b/src/cli/update.ts new file mode 100644 index 0000000..a0cd35f --- /dev/null +++ b/src/cli/update.ts @@ -0,0 +1,422 @@ +import chalk from 'chalk' +import { logEvent } from 'src/services/analytics/index.js' +import { + getLatestVersion, + type InstallStatus, + installGlobalPackage, +} from 'src/utils/autoUpdater.js' +import { regenerateCompletionCache } from 'src/utils/completionCache.js' +import { + getGlobalConfig, + type InstallMethod, + saveGlobalConfig, +} from 'src/utils/config.js' +import { logForDebugging } from 'src/utils/debug.js' +import { getDoctorDiagnostic } from 'src/utils/doctorDiagnostic.js' +import { gracefulShutdown } from 'src/utils/gracefulShutdown.js' +import { + installOrUpdateClaudePackage, + localInstallationExists, +} from 'src/utils/localInstaller.js' +import { + installLatest as installLatestNative, + removeInstalledSymlink, +} from 'src/utils/nativeInstaller/index.js' +import { getPackageManager } from 'src/utils/nativeInstaller/packageManagers.js' +import { writeToStdout } from 'src/utils/process.js' +import { gte } from 'src/utils/semver.js' +import { getInitialSettings } from 'src/utils/settings/settings.js' + +export async function update() { + logEvent('tengu_update_check', {}) + writeToStdout(`Current version: ${MACRO.VERSION}\n`) + + const channel = getInitialSettings()?.autoUpdatesChannel ?? 'latest' + writeToStdout(`Checking for updates to ${channel} version...\n`) + + logForDebugging('update: Starting update check') + + // Run diagnostic to detect potential issues + logForDebugging('update: Running diagnostic') + const diagnostic = await getDoctorDiagnostic() + logForDebugging(`update: Installation type: ${diagnostic.installationType}`) + logForDebugging( + `update: Config install method: ${diagnostic.configInstallMethod}`, + ) + + // Check for multiple installations + if (diagnostic.multipleInstallations.length > 1) { + writeToStdout('\n') + writeToStdout(chalk.yellow('Warning: Multiple installations found') + '\n') + for (const install of diagnostic.multipleInstallations) { + const current = + diagnostic.installationType === install.type + ? ' (currently running)' + : '' + writeToStdout(`- ${install.type} at ${install.path}${current}\n`) + } + } + + // Display warnings if any exist + if (diagnostic.warnings.length > 0) { + writeToStdout('\n') + for (const warning of diagnostic.warnings) { + logForDebugging(`update: Warning detected: ${warning.issue}`) + + // Don't skip PATH warnings - they're always relevant + // The user needs to know that 'which claude' points elsewhere + logForDebugging(`update: Showing warning: ${warning.issue}`) + + writeToStdout(chalk.yellow(`Warning: ${warning.issue}\n`)) + + writeToStdout(chalk.bold(`Fix: ${warning.fix}\n`)) + } + } + + // Update config if installMethod is not set (but skip for package managers) + const config = getGlobalConfig() + if ( + !config.installMethod && + diagnostic.installationType !== 'package-manager' + ) { + writeToStdout('\n') + writeToStdout('Updating configuration to track installation method...\n') + let detectedMethod: 'local' | 'native' | 'global' | 'unknown' = 'unknown' + + // Map diagnostic installation type to config install method + switch (diagnostic.installationType) { + case 'npm-local': + detectedMethod = 'local' + break + case 'native': + detectedMethod = 'native' + break + case 'npm-global': + detectedMethod = 'global' + break + default: + detectedMethod = 'unknown' + } + + saveGlobalConfig(current => ({ + ...current, + installMethod: detectedMethod, + })) + writeToStdout(`Installation method set to: ${detectedMethod}\n`) + } + + // Check if running from development build + if (diagnostic.installationType === 'development') { + writeToStdout('\n') + writeToStdout( + chalk.yellow('Warning: Cannot update development build') + '\n', + ) + await gracefulShutdown(1) + } + + // Check if running from a package manager + if (diagnostic.installationType === 'package-manager') { + const packageManager = await getPackageManager() + writeToStdout('\n') + + if (packageManager === 'homebrew') { + writeToStdout('Claude is managed by Homebrew.\n') + const latest = await getLatestVersion(channel) + if (latest && !gte(MACRO.VERSION, latest)) { + writeToStdout(`Update available: ${MACRO.VERSION} → ${latest}\n`) + writeToStdout('\n') + writeToStdout('To update, run:\n') + writeToStdout(chalk.bold(' brew upgrade claude-code') + '\n') + } else { + writeToStdout('Claude is up to date!\n') + } + } else if (packageManager === 'winget') { + writeToStdout('Claude is managed by winget.\n') + const latest = await getLatestVersion(channel) + if (latest && !gte(MACRO.VERSION, latest)) { + writeToStdout(`Update available: ${MACRO.VERSION} → ${latest}\n`) + writeToStdout('\n') + writeToStdout('To update, run:\n') + writeToStdout( + chalk.bold(' winget upgrade Anthropic.ClaudeCode') + '\n', + ) + } else { + writeToStdout('Claude is up to date!\n') + } + } else if (packageManager === 'apk') { + writeToStdout('Claude is managed by apk.\n') + const latest = await getLatestVersion(channel) + if (latest && !gte(MACRO.VERSION, latest)) { + writeToStdout(`Update available: ${MACRO.VERSION} → ${latest}\n`) + writeToStdout('\n') + writeToStdout('To update, run:\n') + writeToStdout(chalk.bold(' apk upgrade claude-code') + '\n') + } else { + writeToStdout('Claude is up to date!\n') + } + } else { + // pacman, deb, and rpm don't get specific commands because they each have + // multiple frontends (pacman: yay/paru/makepkg, deb: apt/apt-get/aptitude/nala, + // rpm: dnf/yum/zypper) + writeToStdout('Claude is managed by a package manager.\n') + writeToStdout('Please use your package manager to update.\n') + } + + await gracefulShutdown(0) + } + + // Check for config/reality mismatch (skip for package-manager installs) + if ( + config.installMethod && + diagnostic.configInstallMethod !== 'not set' && + diagnostic.installationType !== 'package-manager' + ) { + const runningType = diagnostic.installationType + const configExpects = diagnostic.configInstallMethod + + // Map installation types for comparison + const typeMapping: Record = { + 'npm-local': 'local', + 'npm-global': 'global', + native: 'native', + development: 'development', + unknown: 'unknown', + } + + const normalizedRunningType = typeMapping[runningType] || runningType + + if ( + normalizedRunningType !== configExpects && + configExpects !== 'unknown' + ) { + writeToStdout('\n') + writeToStdout(chalk.yellow('Warning: Configuration mismatch') + '\n') + writeToStdout(`Config expects: ${configExpects} installation\n`) + writeToStdout(`Currently running: ${runningType}\n`) + writeToStdout( + chalk.yellow( + `Updating the ${runningType} installation you are currently using`, + ) + '\n', + ) + + // Update config to match reality + saveGlobalConfig(current => ({ + ...current, + installMethod: normalizedRunningType as InstallMethod, + })) + writeToStdout( + `Config updated to reflect current installation method: ${normalizedRunningType}\n`, + ) + } + } + + // Handle native installation updates first + if (diagnostic.installationType === 'native') { + logForDebugging( + 'update: Detected native installation, using native updater', + ) + try { + const result = await installLatestNative(channel, true) + + // Handle lock contention gracefully + if (result.lockFailed) { + const pidInfo = result.lockHolderPid + ? ` (PID ${result.lockHolderPid})` + : '' + writeToStdout( + chalk.yellow( + `Another Claude process${pidInfo} is currently running. Please try again in a moment.`, + ) + '\n', + ) + await gracefulShutdown(0) + } + + if (!result.latestVersion) { + process.stderr.write('Failed to check for updates\n') + await gracefulShutdown(1) + } + + if (result.latestVersion === MACRO.VERSION) { + writeToStdout( + chalk.green(`Claude Code is up to date (${MACRO.VERSION})`) + '\n', + ) + } else { + writeToStdout( + chalk.green( + `Successfully updated from ${MACRO.VERSION} to version ${result.latestVersion}`, + ) + '\n', + ) + await regenerateCompletionCache() + } + await gracefulShutdown(0) + } catch (error) { + process.stderr.write('Error: Failed to install native update\n') + process.stderr.write(String(error) + '\n') + process.stderr.write('Try running "claude doctor" for diagnostics\n') + await gracefulShutdown(1) + } + } + + // Fallback to existing JS/npm-based update logic + // Remove native installer symlink since we're not using native installation + // But only if user hasn't migrated to native installation + if (config.installMethod !== 'native') { + await removeInstalledSymlink() + } + + logForDebugging('update: Checking npm registry for latest version') + logForDebugging(`update: Package URL: ${MACRO.PACKAGE_URL}`) + const npmTag = channel === 'stable' ? 'stable' : 'latest' + const npmCommand = `npm view ${MACRO.PACKAGE_URL}@${npmTag} version` + logForDebugging(`update: Running: ${npmCommand}`) + const latestVersion = await getLatestVersion(channel) + logForDebugging( + `update: Latest version from npm: ${latestVersion || 'FAILED'}`, + ) + + if (!latestVersion) { + logForDebugging('update: Failed to get latest version from npm registry') + process.stderr.write(chalk.red('Failed to check for updates') + '\n') + process.stderr.write('Unable to fetch latest version from npm registry\n') + process.stderr.write('\n') + process.stderr.write('Possible causes:\n') + process.stderr.write(' • Network connectivity issues\n') + process.stderr.write(' • npm registry is unreachable\n') + process.stderr.write(' • Corporate proxy/firewall blocking npm\n') + if (MACRO.PACKAGE_URL && !MACRO.PACKAGE_URL.startsWith('@anthropic')) { + process.stderr.write( + ' • Internal/development build not published to npm\n', + ) + } + process.stderr.write('\n') + process.stderr.write('Try:\n') + process.stderr.write(' • Check your internet connection\n') + process.stderr.write(' • Run with --debug flag for more details\n') + const packageName = + MACRO.PACKAGE_URL || + (process.env.USER_TYPE === 'ant' + ? '@anthropic-ai/claude-cli' + : '@anthropic-ai/claude-code') + process.stderr.write( + ` • Manually check: npm view ${packageName} version\n`, + ) + + process.stderr.write(' • Check if you need to login: npm whoami\n') + await gracefulShutdown(1) + } + + // Check if versions match exactly, including any build metadata (like SHA) + if (latestVersion === MACRO.VERSION) { + writeToStdout( + chalk.green(`Claude Code is up to date (${MACRO.VERSION})`) + '\n', + ) + await gracefulShutdown(0) + } + + writeToStdout( + `New version available: ${latestVersion} (current: ${MACRO.VERSION})\n`, + ) + writeToStdout('Installing update...\n') + + // Determine update method based on what's actually running + let useLocalUpdate = false + let updateMethodName = '' + + switch (diagnostic.installationType) { + case 'npm-local': + useLocalUpdate = true + updateMethodName = 'local' + break + case 'npm-global': + useLocalUpdate = false + updateMethodName = 'global' + break + case 'unknown': { + // Fallback to detection if we can't determine installation type + const isLocal = await localInstallationExists() + useLocalUpdate = isLocal + updateMethodName = isLocal ? 'local' : 'global' + writeToStdout( + chalk.yellow('Warning: Could not determine installation type') + '\n', + ) + writeToStdout( + `Attempting ${updateMethodName} update based on file detection...\n`, + ) + break + } + default: + process.stderr.write( + `Error: Cannot update ${diagnostic.installationType} installation\n`, + ) + await gracefulShutdown(1) + } + + writeToStdout(`Using ${updateMethodName} installation update method...\n`) + + logForDebugging(`update: Update method determined: ${updateMethodName}`) + logForDebugging(`update: useLocalUpdate: ${useLocalUpdate}`) + + let status: InstallStatus + + if (useLocalUpdate) { + logForDebugging( + 'update: Calling installOrUpdateClaudePackage() for local update', + ) + status = await installOrUpdateClaudePackage(channel) + } else { + logForDebugging('update: Calling installGlobalPackage() for global update') + status = await installGlobalPackage() + } + + logForDebugging(`update: Installation status: ${status}`) + + switch (status) { + case 'success': + writeToStdout( + chalk.green( + `Successfully updated from ${MACRO.VERSION} to version ${latestVersion}`, + ) + '\n', + ) + await regenerateCompletionCache() + break + case 'no_permissions': + process.stderr.write( + 'Error: Insufficient permissions to install update\n', + ) + if (useLocalUpdate) { + process.stderr.write('Try manually updating with:\n') + process.stderr.write( + ` cd ~/.claude/local && npm update ${MACRO.PACKAGE_URL}\n`, + ) + } else { + process.stderr.write('Try running with sudo or fix npm permissions\n') + process.stderr.write( + 'Or consider using native installation with: claude install\n', + ) + } + await gracefulShutdown(1) + break + case 'install_failed': + process.stderr.write('Error: Failed to install update\n') + if (useLocalUpdate) { + process.stderr.write('Try manually updating with:\n') + process.stderr.write( + ` cd ~/.claude/local && npm update ${MACRO.PACKAGE_URL}\n`, + ) + } else { + process.stderr.write( + 'Or consider using native installation with: claude install\n', + ) + } + await gracefulShutdown(1) + break + case 'in_progress': + process.stderr.write( + 'Error: Another instance is currently performing an update\n', + ) + process.stderr.write('Please wait and try again later\n') + await gracefulShutdown(1) + break + } + await gracefulShutdown(0) +} diff --git a/src/commands.ts b/src/commands.ts new file mode 100644 index 0000000..10f03b2 --- /dev/null +++ b/src/commands.ts @@ -0,0 +1,754 @@ +// biome-ignore-all assist/source/organizeImports: ANT-ONLY import markers must not be reordered +import addDir from './commands/add-dir/index.js' +import autofixPr from './commands/autofix-pr/index.js' +import backfillSessions from './commands/backfill-sessions/index.js' +import btw from './commands/btw/index.js' +import goodClaude from './commands/good-claude/index.js' +import issue from './commands/issue/index.js' +import feedback from './commands/feedback/index.js' +import clear from './commands/clear/index.js' +import color from './commands/color/index.js' +import commit from './commands/commit.js' +import copy from './commands/copy/index.js' +import desktop from './commands/desktop/index.js' +import commitPushPr from './commands/commit-push-pr.js' +import compact from './commands/compact/index.js' +import config from './commands/config/index.js' +import { context, contextNonInteractive } from './commands/context/index.js' +import cost from './commands/cost/index.js' +import diff from './commands/diff/index.js' +import ctx_viz from './commands/ctx_viz/index.js' +import doctor from './commands/doctor/index.js' +import memory from './commands/memory/index.js' +import help from './commands/help/index.js' +import ide from './commands/ide/index.js' +import init from './commands/init.js' +import initVerifiers from './commands/init-verifiers.js' +import keybindings from './commands/keybindings/index.js' +import login from './commands/login/index.js' +import logout from './commands/logout/index.js' +import installGitHubApp from './commands/install-github-app/index.js' +import installSlackApp from './commands/install-slack-app/index.js' +import breakCache from './commands/break-cache/index.js' +import mcp from './commands/mcp/index.js' +import mobile from './commands/mobile/index.js' +import onboarding from './commands/onboarding/index.js' +import pr_comments from './commands/pr_comments/index.js' +import releaseNotes from './commands/release-notes/index.js' +import rename from './commands/rename/index.js' +import resume from './commands/resume/index.js' +import review, { ultrareview } from './commands/review.js' +import session from './commands/session/index.js' +import share from './commands/share/index.js' +import skills from './commands/skills/index.js' +import status from './commands/status/index.js' +import tasks from './commands/tasks/index.js' +import teleport from './commands/teleport/index.js' +/* eslint-disable @typescript-eslint/no-require-imports */ +const agentsPlatform = + process.env.USER_TYPE === 'ant' + ? require('./commands/agents-platform/index.js').default + : null +/* eslint-enable @typescript-eslint/no-require-imports */ +import securityReview from './commands/security-review.js' +import bughunter from './commands/bughunter/index.js' +import terminalSetup from './commands/terminalSetup/index.js' +import usage from './commands/usage/index.js' +import theme from './commands/theme/index.js' +import vim from './commands/vim/index.js' +import { feature } from 'bun:bundle' +// Dead code elimination: conditional imports +/* eslint-disable @typescript-eslint/no-require-imports */ +const proactive = + feature('PROACTIVE') || feature('KAIROS') + ? require('./commands/proactive.js').default + : null +const briefCommand = + feature('KAIROS') || feature('KAIROS_BRIEF') + ? require('./commands/brief.js').default + : null +const assistantCommand = feature('KAIROS') + ? require('./commands/assistant/index.js').default + : null +const bridge = feature('BRIDGE_MODE') + ? require('./commands/bridge/index.js').default + : null +const remoteControlServerCommand = + feature('DAEMON') && feature('BRIDGE_MODE') + ? require('./commands/remoteControlServer/index.js').default + : null +const voiceCommand = feature('VOICE_MODE') + ? require('./commands/voice/index.js').default + : null +const forceSnip = feature('HISTORY_SNIP') + ? require('./commands/force-snip.js').default + : null +const workflowsCmd = feature('WORKFLOW_SCRIPTS') + ? ( + require('./commands/workflows/index.js') as typeof import('./commands/workflows/index.js') + ).default + : null +const webCmd = feature('CCR_REMOTE_SETUP') + ? ( + require('./commands/remote-setup/index.js') as typeof import('./commands/remote-setup/index.js') + ).default + : null +const clearSkillIndexCache = feature('EXPERIMENTAL_SKILL_SEARCH') + ? ( + require('./services/skillSearch/localSearch.js') as typeof import('./services/skillSearch/localSearch.js') + ).clearSkillIndexCache + : null +const subscribePr = feature('KAIROS_GITHUB_WEBHOOKS') + ? require('./commands/subscribe-pr.js').default + : null +const ultraplan = feature('ULTRAPLAN') + ? require('./commands/ultraplan.js').default + : null +const torch = feature('TORCH') ? require('./commands/torch.js').default : null +const peersCmd = feature('UDS_INBOX') + ? ( + require('./commands/peers/index.js') as typeof import('./commands/peers/index.js') + ).default + : null +const forkCmd = feature('FORK_SUBAGENT') + ? ( + require('./commands/fork/index.js') as typeof import('./commands/fork/index.js') + ).default + : null +const buddy = feature('BUDDY') + ? ( + require('./commands/buddy/index.js') as typeof import('./commands/buddy/index.js') + ).default + : null +/* eslint-enable @typescript-eslint/no-require-imports */ +import thinkback from './commands/thinkback/index.js' +import thinkbackPlay from './commands/thinkback-play/index.js' +import permissions from './commands/permissions/index.js' +import plan from './commands/plan/index.js' +import fast from './commands/fast/index.js' +import passes from './commands/passes/index.js' +import privacySettings from './commands/privacy-settings/index.js' +import hooks from './commands/hooks/index.js' +import files from './commands/files/index.js' +import branch from './commands/branch/index.js' +import agents from './commands/agents/index.js' +import plugin from './commands/plugin/index.js' +import reloadPlugins from './commands/reload-plugins/index.js' +import rewind from './commands/rewind/index.js' +import heapDump from './commands/heapdump/index.js' +import mockLimits from './commands/mock-limits/index.js' +import bridgeKick from './commands/bridge-kick.js' +import version from './commands/version.js' +import summary from './commands/summary/index.js' +import { + resetLimits, + resetLimitsNonInteractive, +} from './commands/reset-limits/index.js' +import antTrace from './commands/ant-trace/index.js' +import perfIssue from './commands/perf-issue/index.js' +import sandboxToggle from './commands/sandbox-toggle/index.js' +import chrome from './commands/chrome/index.js' +import stickers from './commands/stickers/index.js' +import advisor from './commands/advisor.js' +import { logError } from './utils/log.js' +import { toError } from './utils/errors.js' +import { logForDebugging } from './utils/debug.js' +import { + getSkillDirCommands, + clearSkillCaches, + getDynamicSkills, +} from './skills/loadSkillsDir.js' +import { getBundledSkills } from './skills/bundledSkills.js' +import { getBuiltinPluginSkillCommands } from './plugins/builtinPlugins.js' +import { + getPluginCommands, + clearPluginCommandCache, + getPluginSkills, + clearPluginSkillsCache, +} from './utils/plugins/loadPluginCommands.js' +import memoize from 'lodash-es/memoize.js' +import { isUsing3PServices, isClaudeAISubscriber } from './utils/auth.js' +import { isFirstPartyAnthropicBaseUrl } from './utils/model/providers.js' +import env from './commands/env/index.js' +import exit from './commands/exit/index.js' +import exportCommand from './commands/export/index.js' +import model from './commands/model/index.js' +import tag from './commands/tag/index.js' +import outputStyle from './commands/output-style/index.js' +import remoteEnv from './commands/remote-env/index.js' +import upgrade from './commands/upgrade/index.js' +import { + extraUsage, + extraUsageNonInteractive, +} from './commands/extra-usage/index.js' +import rateLimitOptions from './commands/rate-limit-options/index.js' +import statusline from './commands/statusline.js' +import effort from './commands/effort/index.js' +import stats from './commands/stats/index.js' +// insights.ts is 113KB (3200 lines, includes diffLines/html rendering). Lazy +// shim defers the heavy module until /insights is actually invoked. +const usageReport: Command = { + type: 'prompt', + name: 'insights', + description: 'Generate a report analyzing your Claude Code sessions', + contentLength: 0, + progressMessage: 'analyzing your sessions', + source: 'builtin', + async getPromptForCommand(args, context) { + const real = (await import('./commands/insights.js')).default + if (real.type !== 'prompt') throw new Error('unreachable') + return real.getPromptForCommand(args, context) + }, +} +import oauthRefresh from './commands/oauth-refresh/index.js' +import debugToolCall from './commands/debug-tool-call/index.js' +import { getSettingSourceName } from './utils/settings/constants.js' +import { + type Command, + getCommandName, + isCommandEnabled, +} from './types/command.js' + +// Re-export types from the centralized location +export type { + Command, + CommandBase, + CommandResultDisplay, + LocalCommandResult, + LocalJSXCommandContext, + PromptCommand, + ResumeEntrypoint, +} from './types/command.js' +export { getCommandName, isCommandEnabled } from './types/command.js' + +// Commands that get eliminated from the external build +export const INTERNAL_ONLY_COMMANDS = [ + backfillSessions, + breakCache, + bughunter, + commit, + commitPushPr, + ctx_viz, + goodClaude, + issue, + initVerifiers, + ...(forceSnip ? [forceSnip] : []), + mockLimits, + bridgeKick, + version, + ...(ultraplan ? [ultraplan] : []), + ...(subscribePr ? [subscribePr] : []), + resetLimits, + resetLimitsNonInteractive, + onboarding, + share, + summary, + teleport, + antTrace, + perfIssue, + env, + oauthRefresh, + debugToolCall, + agentsPlatform, + autofixPr, +].filter(Boolean) + +// Declared as a function so that we don't run this until getCommands is called, +// since underlying functions read from config, which can't be read at module initialization time +const COMMANDS = memoize((): Command[] => [ + addDir, + advisor, + agents, + branch, + btw, + chrome, + clear, + color, + compact, + config, + copy, + desktop, + context, + contextNonInteractive, + cost, + diff, + doctor, + effort, + exit, + fast, + files, + heapDump, + help, + ide, + init, + keybindings, + installGitHubApp, + installSlackApp, + mcp, + memory, + mobile, + model, + outputStyle, + remoteEnv, + plugin, + pr_comments, + releaseNotes, + reloadPlugins, + rename, + resume, + session, + skills, + stats, + status, + statusline, + stickers, + tag, + theme, + feedback, + review, + ultrareview, + rewind, + securityReview, + terminalSetup, + upgrade, + extraUsage, + extraUsageNonInteractive, + rateLimitOptions, + usage, + usageReport, + vim, + ...(webCmd ? [webCmd] : []), + ...(forkCmd ? [forkCmd] : []), + ...(buddy ? [buddy] : []), + ...(proactive ? [proactive] : []), + ...(briefCommand ? [briefCommand] : []), + ...(assistantCommand ? [assistantCommand] : []), + ...(bridge ? [bridge] : []), + ...(remoteControlServerCommand ? [remoteControlServerCommand] : []), + ...(voiceCommand ? [voiceCommand] : []), + thinkback, + thinkbackPlay, + permissions, + plan, + privacySettings, + hooks, + exportCommand, + sandboxToggle, + ...(!isUsing3PServices() ? [logout, login()] : []), + passes, + ...(peersCmd ? [peersCmd] : []), + tasks, + ...(workflowsCmd ? [workflowsCmd] : []), + ...(torch ? [torch] : []), + ...(process.env.USER_TYPE === 'ant' && !process.env.IS_DEMO + ? INTERNAL_ONLY_COMMANDS + : []), +]) + +export const builtInCommandNames = memoize( + (): Set => + new Set(COMMANDS().flatMap(_ => [_.name, ...(_.aliases ?? [])])), +) + +async function getSkills(cwd: string): Promise<{ + skillDirCommands: Command[] + pluginSkills: Command[] + bundledSkills: Command[] + builtinPluginSkills: Command[] +}> { + try { + const [skillDirCommands, pluginSkills] = await Promise.all([ + getSkillDirCommands(cwd).catch(err => { + logError(toError(err)) + logForDebugging( + 'Skill directory commands failed to load, continuing without them', + ) + return [] + }), + getPluginSkills().catch(err => { + logError(toError(err)) + logForDebugging('Plugin skills failed to load, continuing without them') + return [] + }), + ]) + // Bundled skills are registered synchronously at startup + const bundledSkills = getBundledSkills() + // Built-in plugin skills come from enabled built-in plugins + const builtinPluginSkills = getBuiltinPluginSkillCommands() + logForDebugging( + `getSkills returning: ${skillDirCommands.length} skill dir commands, ${pluginSkills.length} plugin skills, ${bundledSkills.length} bundled skills, ${builtinPluginSkills.length} builtin plugin skills`, + ) + return { + skillDirCommands, + pluginSkills, + bundledSkills, + builtinPluginSkills, + } + } catch (err) { + // This should never happen since we catch at the Promise level, but defensive + logError(toError(err)) + logForDebugging('Unexpected error in getSkills, returning empty') + return { + skillDirCommands: [], + pluginSkills: [], + bundledSkills: [], + builtinPluginSkills: [], + } + } +} + +/* eslint-disable @typescript-eslint/no-require-imports */ +const getWorkflowCommands = feature('WORKFLOW_SCRIPTS') + ? ( + require('./tools/WorkflowTool/createWorkflowCommand.js') as typeof import('./tools/WorkflowTool/createWorkflowCommand.js') + ).getWorkflowCommands + : null +/* eslint-enable @typescript-eslint/no-require-imports */ + +/** + * Filters commands by their declared `availability` (auth/provider requirement). + * Commands without `availability` are treated as universal. + * This runs before `isEnabled()` so that provider-gated commands are hidden + * regardless of feature-flag state. + * + * Not memoized — auth state can change mid-session (e.g. after /login), + * so this must be re-evaluated on every getCommands() call. + */ +export function meetsAvailabilityRequirement(cmd: Command): boolean { + if (!cmd.availability) return true + for (const a of cmd.availability) { + switch (a) { + case 'claude-ai': + if (isClaudeAISubscriber()) return true + break + case 'console': + // Console API key user = direct 1P API customer (not 3P, not claude.ai). + // Excludes 3P (Bedrock/Vertex/Foundry) who don't set ANTHROPIC_BASE_URL + // and gateway users who proxy through a custom base URL. + if ( + !isClaudeAISubscriber() && + !isUsing3PServices() && + isFirstPartyAnthropicBaseUrl() + ) + return true + break + default: { + const _exhaustive: never = a + void _exhaustive + break + } + } + } + return false +} + +/** + * Loads all command sources (skills, plugins, workflows). Memoized by cwd + * because loading is expensive (disk I/O, dynamic imports). + */ +const loadAllCommands = memoize(async (cwd: string): Promise => { + const [ + { skillDirCommands, pluginSkills, bundledSkills, builtinPluginSkills }, + pluginCommands, + workflowCommands, + ] = await Promise.all([ + getSkills(cwd), + getPluginCommands(), + getWorkflowCommands ? getWorkflowCommands(cwd) : Promise.resolve([]), + ]) + + return [ + ...bundledSkills, + ...builtinPluginSkills, + ...skillDirCommands, + ...workflowCommands, + ...pluginCommands, + ...pluginSkills, + ...COMMANDS(), + ] +}) + +/** + * Returns commands available to the current user. The expensive loading is + * memoized, but availability and isEnabled checks run fresh every call so + * auth changes (e.g. /login) take effect immediately. + */ +export async function getCommands(cwd: string): Promise { + const allCommands = await loadAllCommands(cwd) + + // Get dynamic skills discovered during file operations + const dynamicSkills = getDynamicSkills() + + // Build base commands without dynamic skills + const baseCommands = allCommands.filter( + _ => meetsAvailabilityRequirement(_) && isCommandEnabled(_), + ) + + if (dynamicSkills.length === 0) { + return baseCommands + } + + // Dedupe dynamic skills - only add if not already present + const baseCommandNames = new Set(baseCommands.map(c => c.name)) + const uniqueDynamicSkills = dynamicSkills.filter( + s => + !baseCommandNames.has(s.name) && + meetsAvailabilityRequirement(s) && + isCommandEnabled(s), + ) + + if (uniqueDynamicSkills.length === 0) { + return baseCommands + } + + // Insert dynamic skills after plugin skills but before built-in commands + const builtInNames = new Set(COMMANDS().map(c => c.name)) + const insertIndex = baseCommands.findIndex(c => builtInNames.has(c.name)) + + if (insertIndex === -1) { + return [...baseCommands, ...uniqueDynamicSkills] + } + + return [ + ...baseCommands.slice(0, insertIndex), + ...uniqueDynamicSkills, + ...baseCommands.slice(insertIndex), + ] +} + +/** + * Clears only the memoization caches for commands, WITHOUT clearing skill caches. + * Use this when dynamic skills are added to invalidate cached command lists. + */ +export function clearCommandMemoizationCaches(): void { + loadAllCommands.cache?.clear?.() + getSkillToolCommands.cache?.clear?.() + getSlashCommandToolSkills.cache?.clear?.() + // getSkillIndex in skillSearch/localSearch.ts is a separate memoization layer + // built ON TOP of getSkillToolCommands/getCommands. Clearing only the inner + // caches is a no-op for the outer — lodash memoize returns the cached result + // without ever reaching the cleared inners. Must clear it explicitly. + clearSkillIndexCache?.() +} + +export function clearCommandsCache(): void { + clearCommandMemoizationCaches() + clearPluginCommandCache() + clearPluginSkillsCache() + clearSkillCaches() +} + +/** + * Filter AppState.mcp.commands to MCP-provided skills (prompt-type, + * model-invocable, loaded from MCP). These live outside getCommands() so + * callers that need MCP skills in their skill index thread them through + * separately. + */ +export function getMcpSkillCommands( + mcpCommands: readonly Command[], +): readonly Command[] { + if (feature('MCP_SKILLS')) { + return mcpCommands.filter( + cmd => + cmd.type === 'prompt' && + cmd.loadedFrom === 'mcp' && + !cmd.disableModelInvocation, + ) + } + return [] +} + +// SkillTool shows ALL prompt-based commands that the model can invoke +// This includes both skills (from /skills/) and commands (from /commands/) +export const getSkillToolCommands = memoize( + async (cwd: string): Promise => { + const allCommands = await getCommands(cwd) + return allCommands.filter( + cmd => + cmd.type === 'prompt' && + !cmd.disableModelInvocation && + cmd.source !== 'builtin' && + // Always include skills from /skills/ dirs, bundled skills, and legacy /commands/ entries + // (they all get an auto-derived description from the first line if frontmatter is missing). + // Plugin/MCP commands still require an explicit description to appear in the listing. + (cmd.loadedFrom === 'bundled' || + cmd.loadedFrom === 'skills' || + cmd.loadedFrom === 'commands_DEPRECATED' || + cmd.hasUserSpecifiedDescription || + cmd.whenToUse), + ) + }, +) + +// Filters commands to include only skills. Skills are commands that provide +// specialized capabilities for the model to use. They are identified by +// loadedFrom being 'skills', 'plugin', or 'bundled', or having disableModelInvocation set. +export const getSlashCommandToolSkills = memoize( + async (cwd: string): Promise => { + try { + const allCommands = await getCommands(cwd) + return allCommands.filter( + cmd => + cmd.type === 'prompt' && + cmd.source !== 'builtin' && + (cmd.hasUserSpecifiedDescription || cmd.whenToUse) && + (cmd.loadedFrom === 'skills' || + cmd.loadedFrom === 'plugin' || + cmd.loadedFrom === 'bundled' || + cmd.disableModelInvocation), + ) + } catch (error) { + logError(toError(error)) + // Return empty array rather than throwing - skills are non-critical + // This prevents skill loading failures from breaking the entire system + logForDebugging('Returning empty skills array due to load failure') + return [] + } + }, +) + +/** + * Commands that are safe to use in remote mode (--remote). + * These only affect local TUI state and don't depend on local filesystem, + * git, shell, IDE, MCP, or other local execution context. + * + * Used in two places: + * 1. Pre-filtering commands in main.tsx before REPL renders (prevents race with CCR init) + * 2. Preserving local-only commands in REPL's handleRemoteInit after CCR filters + */ +export const REMOTE_SAFE_COMMANDS: Set = new Set([ + session, // Shows QR code / URL for remote session + exit, // Exit the TUI + clear, // Clear screen + help, // Show help + theme, // Change terminal theme + color, // Change agent color + vim, // Toggle vim mode + cost, // Show session cost (local cost tracking) + usage, // Show usage info + copy, // Copy last message + btw, // Quick note + feedback, // Send feedback + plan, // Plan mode toggle + keybindings, // Keybinding management + statusline, // Status line toggle + stickers, // Stickers + mobile, // Mobile QR code +]) + +/** + * Builtin commands of type 'local' that ARE safe to execute when received + * over the Remote Control bridge. These produce text output that streams + * back to the mobile/web client and have no terminal-only side effects. + * + * 'local-jsx' commands are blocked by type (they render Ink UI) and + * 'prompt' commands are allowed by type (they expand to text sent to the + * model) — this set only gates 'local' commands. + * + * When adding a new 'local' command that should work from mobile, add it + * here. Default is blocked. + */ +export const BRIDGE_SAFE_COMMANDS: Set = new Set( + [ + compact, // Shrink context — useful mid-session from a phone + clear, // Wipe transcript + cost, // Show session cost + summary, // Summarize conversation + releaseNotes, // Show changelog + files, // List tracked files + ].filter((c): c is Command => c !== null), +) + +/** + * Whether a slash command is safe to execute when its input arrived over the + * Remote Control bridge (mobile/web client). + * + * PR #19134 blanket-blocked all slash commands from bridge inbound because + * `/model` from iOS was popping the local Ink picker. This predicate relaxes + * that with an explicit allowlist: 'prompt' commands (skills) expand to text + * and are safe by construction; 'local' commands need an explicit opt-in via + * BRIDGE_SAFE_COMMANDS; 'local-jsx' commands render Ink UI and stay blocked. + */ +export function isBridgeSafeCommand(cmd: Command): boolean { + if (cmd.type === 'local-jsx') return false + if (cmd.type === 'prompt') return true + return BRIDGE_SAFE_COMMANDS.has(cmd) +} + +/** + * Filter commands to only include those safe for remote mode. + * Used to pre-filter commands when rendering the REPL in --remote mode, + * preventing local-only commands from being briefly available before + * the CCR init message arrives. + */ +export function filterCommandsForRemoteMode(commands: Command[]): Command[] { + return commands.filter(cmd => REMOTE_SAFE_COMMANDS.has(cmd)) +} + +export function findCommand( + commandName: string, + commands: Command[], +): Command | undefined { + return commands.find( + _ => + _.name === commandName || + getCommandName(_) === commandName || + _.aliases?.includes(commandName), + ) +} + +export function hasCommand(commandName: string, commands: Command[]): boolean { + return findCommand(commandName, commands) !== undefined +} + +export function getCommand(commandName: string, commands: Command[]): Command { + const command = findCommand(commandName, commands) + if (!command) { + throw ReferenceError( + `Command ${commandName} not found. Available commands: ${commands + .map(_ => { + const name = getCommandName(_) + return _.aliases ? `${name} (aliases: ${_.aliases.join(', ')})` : name + }) + .sort((a, b) => a.localeCompare(b)) + .join(', ')}`, + ) + } + + return command +} + +/** + * Formats a command's description with its source annotation for user-facing UI. + * Use this in typeahead, help screens, and other places where users need to see + * where a command comes from. + * + * For model-facing prompts (like SkillTool), use cmd.description directly. + */ +export function formatDescriptionWithSource(cmd: Command): string { + if (cmd.type !== 'prompt') { + return cmd.description + } + + if (cmd.kind === 'workflow') { + return `${cmd.description} (workflow)` + } + + if (cmd.source === 'plugin') { + const pluginName = cmd.pluginInfo?.pluginManifest.name + if (pluginName) { + return `(${pluginName}) ${cmd.description}` + } + return `${cmd.description} (plugin)` + } + + if (cmd.source === 'builtin' || cmd.source === 'mcp') { + return cmd.description + } + + if (cmd.source === 'bundled') { + return `${cmd.description} (bundled)` + } + + return `${cmd.description} (${getSettingSourceName(cmd.source)})` +} diff --git a/src/commands/add-dir/add-dir.tsx b/src/commands/add-dir/add-dir.tsx new file mode 100644 index 0000000..b1a2b4d --- /dev/null +++ b/src/commands/add-dir/add-dir.tsx @@ -0,0 +1,126 @@ +import { c as _c } from "react/compiler-runtime"; +import chalk from 'chalk'; +import figures from 'figures'; +import React, { useEffect } from 'react'; +import { getAdditionalDirectoriesForClaudeMd, setAdditionalDirectoriesForClaudeMd } from '../../bootstrap/state.js'; +import type { LocalJSXCommandContext } from '../../commands.js'; +import { MessageResponse } from '../../components/MessageResponse.js'; +import { AddWorkspaceDirectory } from '../../components/permissions/rules/AddWorkspaceDirectory.js'; +import { Box, Text } from '../../ink.js'; +import type { LocalJSXCommandOnDone } from '../../types/command.js'; +import { applyPermissionUpdate, persistPermissionUpdate } from '../../utils/permissions/PermissionUpdate.js'; +import type { PermissionUpdateDestination } from '../../utils/permissions/PermissionUpdateSchema.js'; +import { SandboxManager } from '../../utils/sandbox/sandbox-adapter.js'; +import { addDirHelpMessage, validateDirectoryForWorkspace } from './validation.js'; +function AddDirError(t0) { + const $ = _c(10); + const { + message, + args, + onDone + } = t0; + let t1; + let t2; + if ($[0] !== onDone) { + t1 = () => { + const timer = setTimeout(onDone, 0); + return () => clearTimeout(timer); + }; + t2 = [onDone]; + $[0] = onDone; + $[1] = t1; + $[2] = t2; + } else { + t1 = $[1]; + t2 = $[2]; + } + useEffect(t1, t2); + let t3; + if ($[3] !== args) { + t3 = {figures.pointer} /add-dir {args}; + $[3] = args; + $[4] = t3; + } else { + t3 = $[4]; + } + let t4; + if ($[5] !== message) { + t4 = {message}; + $[5] = message; + $[6] = t4; + } else { + t4 = $[6]; + } + let t5; + if ($[7] !== t3 || $[8] !== t4) { + t5 = {t3}{t4}; + $[7] = t3; + $[8] = t4; + $[9] = t5; + } else { + t5 = $[9]; + } + return t5; +} +export async function call(onDone: LocalJSXCommandOnDone, context: LocalJSXCommandContext, args?: string): Promise { + const directoryPath = (args ?? '').trim(); + const appState = context.getAppState(); + + // Helper to handle adding a directory (shared by both with-path and no-path cases) + const handleAddDirectory = async (path: string, remember = false) => { + const destination: PermissionUpdateDestination = remember ? 'localSettings' : 'session'; + const permissionUpdate = { + type: 'addDirectories' as const, + directories: [path], + destination + }; + + // Apply to session context + const latestAppState = context.getAppState(); + const updatedContext = applyPermissionUpdate(latestAppState.toolPermissionContext, permissionUpdate); + context.setAppState(prev => ({ + ...prev, + toolPermissionContext: updatedContext + })); + + // Update sandbox config so Bash commands can access the new directory. + // Bootstrap state is the source of truth for session-only dirs; persisted + // dirs are picked up via the settings subscription, but we refresh + // eagerly here to avoid a race when the user acts immediately. + const currentDirs = getAdditionalDirectoriesForClaudeMd(); + if (!currentDirs.includes(path)) { + setAdditionalDirectoriesForClaudeMd([...currentDirs, path]); + } + SandboxManager.refreshConfig(); + let message: string; + if (remember) { + try { + persistPermissionUpdate(permissionUpdate); + message = `Added ${chalk.bold(path)} as a working directory and saved to local settings`; + } catch (error) { + message = `Added ${chalk.bold(path)} as a working directory. Failed to save to local settings: ${error instanceof Error ? error.message : 'Unknown error'}`; + } + } else { + message = `Added ${chalk.bold(path)} as a working directory for this session`; + } + const messageWithHint = `${message} ${chalk.dim('· /permissions to manage')}`; + onDone(messageWithHint); + }; + + // When no path is provided, show AddWorkspaceDirectory input form directly + // and return to REPL after confirmation + if (!directoryPath) { + return { + onDone('Did not add a working directory.'); + }} />; + } + const result = await validateDirectoryForWorkspace(directoryPath, appState.toolPermissionContext); + if (result.resultType !== 'success') { + const message = addDirHelpMessage(result); + return onDone(message)} />; + } + return { + onDone(`Did not add ${chalk.bold(result.absolutePath)} as a working directory.`); + }} />; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJjaGFsayIsImZpZ3VyZXMiLCJSZWFjdCIsInVzZUVmZmVjdCIsImdldEFkZGl0aW9uYWxEaXJlY3Rvcmllc0ZvckNsYXVkZU1kIiwic2V0QWRkaXRpb25hbERpcmVjdG9yaWVzRm9yQ2xhdWRlTWQiLCJMb2NhbEpTWENvbW1hbmRDb250ZXh0IiwiTWVzc2FnZVJlc3BvbnNlIiwiQWRkV29ya3NwYWNlRGlyZWN0b3J5IiwiQm94IiwiVGV4dCIsIkxvY2FsSlNYQ29tbWFuZE9uRG9uZSIsImFwcGx5UGVybWlzc2lvblVwZGF0ZSIsInBlcnNpc3RQZXJtaXNzaW9uVXBkYXRlIiwiUGVybWlzc2lvblVwZGF0ZURlc3RpbmF0aW9uIiwiU2FuZGJveE1hbmFnZXIiLCJhZGREaXJIZWxwTWVzc2FnZSIsInZhbGlkYXRlRGlyZWN0b3J5Rm9yV29ya3NwYWNlIiwiQWRkRGlyRXJyb3IiLCJ0MCIsIiQiLCJfYyIsIm1lc3NhZ2UiLCJhcmdzIiwib25Eb25lIiwidDEiLCJ0MiIsInRpbWVyIiwic2V0VGltZW91dCIsImNsZWFyVGltZW91dCIsInQzIiwicG9pbnRlciIsInQ0IiwidDUiLCJjYWxsIiwiY29udGV4dCIsIlByb21pc2UiLCJSZWFjdE5vZGUiLCJkaXJlY3RvcnlQYXRoIiwidHJpbSIsImFwcFN0YXRlIiwiZ2V0QXBwU3RhdGUiLCJoYW5kbGVBZGREaXJlY3RvcnkiLCJwYXRoIiwicmVtZW1iZXIiLCJkZXN0aW5hdGlvbiIsInBlcm1pc3Npb25VcGRhdGUiLCJ0eXBlIiwiY29uc3QiLCJkaXJlY3RvcmllcyIsImxhdGVzdEFwcFN0YXRlIiwidXBkYXRlZENvbnRleHQiLCJ0b29sUGVybWlzc2lvbkNvbnRleHQiLCJzZXRBcHBTdGF0ZSIsInByZXYiLCJjdXJyZW50RGlycyIsImluY2x1ZGVzIiwicmVmcmVzaENvbmZpZyIsImJvbGQiLCJlcnJvciIsIkVycm9yIiwibWVzc2FnZVdpdGhIaW50IiwiZGltIiwicmVzdWx0IiwicmVzdWx0VHlwZSIsImFic29sdXRlUGF0aCJdLCJzb3VyY2VzIjpbImFkZC1kaXIudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBjaGFsayBmcm9tICdjaGFsaydcbmltcG9ydCBmaWd1cmVzIGZyb20gJ2ZpZ3VyZXMnXG5pbXBvcnQgUmVhY3QsIHsgdXNlRWZmZWN0IH0gZnJvbSAncmVhY3QnXG5pbXBvcnQge1xuICBnZXRBZGRpdGlvbmFsRGlyZWN0b3JpZXNGb3JDbGF1ZGVNZCxcbiAgc2V0QWRkaXRpb25hbERpcmVjdG9yaWVzRm9yQ2xhdWRlTWQsXG59IGZyb20gJy4uLy4uL2Jvb3RzdHJhcC9zdGF0ZS5qcydcbmltcG9ydCB0eXBlIHsgTG9jYWxKU1hDb21tYW5kQ29udGV4dCB9IGZyb20gJy4uLy4uL2NvbW1hbmRzLmpzJ1xuaW1wb3J0IHsgTWVzc2FnZVJlc3BvbnNlIH0gZnJvbSAnLi4vLi4vY29tcG9uZW50cy9NZXNzYWdlUmVzcG9uc2UuanMnXG5pbXBvcnQgeyBBZGRXb3Jrc3BhY2VEaXJlY3RvcnkgfSBmcm9tICcuLi8uLi9jb21wb25lbnRzL3Blcm1pc3Npb25zL3J1bGVzL0FkZFdvcmtzcGFjZURpcmVjdG9yeS5qcydcbmltcG9ydCB7IEJveCwgVGV4dCB9IGZyb20gJy4uLy4uL2luay5qcydcbmltcG9ydCB0eXBlIHsgTG9jYWxKU1hDb21tYW5kT25Eb25lIH0gZnJvbSAnLi4vLi4vdHlwZXMvY29tbWFuZC5qcydcbmltcG9ydCB7XG4gIGFwcGx5UGVybWlzc2lvblVwZGF0ZSxcbiAgcGVyc2lzdFBlcm1pc3Npb25VcGRhdGUsXG59IGZyb20gJy4uLy4uL3V0aWxzL3Blcm1pc3Npb25zL1Blcm1pc3Npb25VcGRhdGUuanMnXG5pbXBvcnQgdHlwZSB7IFBlcm1pc3Npb25VcGRhdGVEZXN0aW5hdGlvbiB9IGZyb20gJy4uLy4uL3V0aWxzL3Blcm1pc3Npb25zL1Blcm1pc3Npb25VcGRhdGVTY2hlbWEuanMnXG5pbXBvcnQgeyBTYW5kYm94TWFuYWdlciB9IGZyb20gJy4uLy4uL3V0aWxzL3NhbmRib3gvc2FuZGJveC1hZGFwdGVyLmpzJ1xuaW1wb3J0IHtcbiAgYWRkRGlySGVscE1lc3NhZ2UsXG4gIHZhbGlkYXRlRGlyZWN0b3J5Rm9yV29ya3NwYWNlLFxufSBmcm9tICcuL3ZhbGlkYXRpb24uanMnXG5cbmZ1bmN0aW9uIEFkZERpckVycm9yKHtcbiAgbWVzc2FnZSxcbiAgYXJncyxcbiAgb25Eb25lLFxufToge1xuICBtZXNzYWdlOiBzdHJpbmdcbiAgYXJnczogc3RyaW5nXG4gIG9uRG9uZTogKCkgPT4gdm9pZFxufSk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgLy8gV2UgbmVlZCB0byBkZWZlciBjYWxsaW5nIG9uRG9uZSB0byBhdm9pZCB0aGUgXCJyZXR1cm4gbnVsbFwiIGJ1ZyB3aGVyZVxuICAgIC8vIHRoZSBjb21wb25lbnQgdW5tb3VudHMgYmVmb3JlIFJlYWN0IGNhbiByZW5kZXIgdGhlIGVycm9yIG1lc3NhZ2UuXG4gICAgLy8gVXNpbmcgc2V0VGltZW91dCBlbnN1cmVzIHRoZSBlcnJvciBkaXNwbGF5cyBiZWZvcmUgdGhlIGNvbW1hbmQgZXhpdHMuXG4gICAgY29uc3QgdGltZXIgPSBzZXRUaW1lb3V0KG9uRG9uZSwgMClcbiAgICByZXR1cm4gKCkgPT4gY2xlYXJUaW1lb3V0KHRpbWVyKVxuICB9LCBbb25Eb25lXSlcblxuICByZXR1cm4gKFxuICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiPlxuICAgICAgPFRleHQgZGltQ29sb3I+XG4gICAgICAgIHtmaWd1cmVzLnBvaW50ZXJ9IC9hZGQtZGlyIHthcmdzfVxuICAgICAgPC9UZXh0PlxuICAgICAgPE1lc3NhZ2VSZXNwb25zZT5cbiAgICAgICAgPFRleHQ+e21lc3NhZ2V9PC9UZXh0PlxuICAgICAgPC9NZXNzYWdlUmVzcG9uc2U+XG4gICAgPC9Cb3g+XG4gIClcbn1cblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGNhbGwoXG4gIG9uRG9uZTogTG9jYWxKU1hDb21tYW5kT25Eb25lLFxuICBjb250ZXh0OiBMb2NhbEpTWENvbW1hbmRDb250ZXh0LFxuICBhcmdzPzogc3RyaW5nLFxuKTogUHJvbWlzZTxSZWFjdC5SZWFjdE5vZGU+IHtcbiAgY29uc3QgZGlyZWN0b3J5UGF0aCA9IChhcmdzID8/ICcnKS50cmltKClcbiAgY29uc3QgYXBwU3RhdGUgPSBjb250ZXh0LmdldEFwcFN0YXRlKClcblxuICAvLyBIZWxwZXIgdG8gaGFuZGxlIGFkZGluZyBhIGRpcmVjdG9yeSAoc2hhcmVkIGJ5IGJvdGggd2l0aC1wYXRoIGFuZCBuby1wYXRoIGNhc2VzKVxuICBjb25zdCBoYW5kbGVBZGREaXJlY3RvcnkgPSBhc3luYyAocGF0aDogc3RyaW5nLCByZW1lbWJlciA9IGZhbHNlKSA9PiB7XG4gICAgY29uc3QgZGVzdGluYXRpb246IFBlcm1pc3Npb25VcGRhdGVEZXN0aW5hdGlvbiA9IHJlbWVtYmVyXG4gICAgICA/ICdsb2NhbFNldHRpbmdzJ1xuICAgICAgOiAnc2Vzc2lvbidcblxuICAgIGNvbnN0IHBlcm1pc3Npb25VcGRhdGUgPSB7XG4gICAgICB0eXBlOiAnYWRkRGlyZWN0b3JpZXMnIGFzIGNvbnN0LFxuICAgICAgZGlyZWN0b3JpZXM6IFtwYXRoXSxcbiAgICAgIGRlc3RpbmF0aW9uLFxuICAgIH1cblxuICAgIC8vIEFwcGx5IHRvIHNlc3Npb24gY29udGV4dFxuICAgIGNvbnN0IGxhdGVzdEFwcFN0YXRlID0gY29udGV4dC5nZXRBcHBTdGF0ZSgpXG4gICAgY29uc3QgdXBkYXRlZENvbnRleHQgPSBhcHBseVBlcm1pc3Npb25VcGRhdGUoXG4gICAgICBsYXRlc3RBcHBTdGF0ZS50b29sUGVybWlzc2lvbkNvbnRleHQsXG4gICAgICBwZXJtaXNzaW9uVXBkYXRlLFxuICAgIClcbiAgICBjb250ZXh0LnNldEFwcFN0YXRlKHByZXYgPT4gKHtcbiAgICAgIC4uLnByZXYsXG4gICAgICB0b29sUGVybWlzc2lvbkNvbnRleHQ6IHVwZGF0ZWRDb250ZXh0LFxuICAgIH0pKVxuXG4gICAgLy8gVXBkYXRlIHNhbmRib3ggY29uZmlnIHNvIEJhc2ggY29tbWFuZHMgY2FuIGFjY2VzcyB0aGUgbmV3IGRpcmVjdG9yeS5cbiAgICAvLyBCb290c3RyYXAgc3RhdGUgaXMgdGhlIHNvdXJjZSBvZiB0cnV0aCBmb3Igc2Vzc2lvbi1vbmx5IGRpcnM7IHBlcnNpc3RlZFxuICAgIC8vIGRpcnMgYXJlIHBpY2tlZCB1cCB2aWEgdGhlIHNldHRpbmdzIHN1YnNjcmlwdGlvbiwgYnV0IHdlIHJlZnJlc2hcbiAgICAvLyBlYWdlcmx5IGhlcmUgdG8gYXZvaWQgYSByYWNlIHdoZW4gdGhlIHVzZXIgYWN0cyBpbW1lZGlhdGVseS5cbiAgICBjb25zdCBjdXJyZW50RGlycyA9IGdldEFkZGl0aW9uYWxEaXJlY3Rvcmllc0ZvckNsYXVkZU1kKClcbiAgICBpZiAoIWN1cnJlbnREaXJzLmluY2x1ZGVzKHBhdGgpKSB7XG4gICAgICBzZXRBZGRpdGlvbmFsRGlyZWN0b3JpZXNGb3JDbGF1ZGVNZChbLi4uY3VycmVudERpcnMsIHBhdGhdKVxuICAgIH1cbiAgICBTYW5kYm94TWFuYWdlci5yZWZyZXNoQ29uZmlnKClcblxuICAgIGxldCBtZXNzYWdlOiBzdHJpbmdcblxuICAgIGlmIChyZW1lbWJlcikge1xuICAgICAgdHJ5IHtcbiAgICAgICAgcGVyc2lzdFBlcm1pc3Npb25VcGRhdGUocGVybWlzc2lvblVwZGF0ZSlcbiAgICAgICAgbWVzc2FnZSA9IGBBZGRlZCAke2NoYWxrLmJvbGQocGF0aCl9IGFzIGEgd29ya2luZyBkaXJlY3RvcnkgYW5kIHNhdmVkIHRvIGxvY2FsIHNldHRpbmdzYFxuICAgICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgICAgbWVzc2FnZSA9IGBBZGRlZCAke2NoYWxrLmJvbGQocGF0aCl9IGFzIGEgd29ya2luZyBkaXJlY3RvcnkuIEZhaWxlZCB0byBzYXZlIHRvIGxvY2FsIHNldHRpbmdzOiAke2Vycm9yIGluc3RhbmNlb2YgRXJyb3IgPyBlcnJvci5tZXNzYWdlIDogJ1Vua25vd24gZXJyb3InfWBcbiAgICAgIH1cbiAgICB9IGVsc2Uge1xuICAgICAgbWVzc2FnZSA9IGBBZGRlZCAke2NoYWxrLmJvbGQocGF0aCl9IGFzIGEgd29ya2luZyBkaXJlY3RvcnkgZm9yIHRoaXMgc2Vzc2lvbmBcbiAgICB9XG5cbiAgICBjb25zdCBtZXNzYWdlV2l0aEhpbnQgPSBgJHttZXNzYWdlfSAke2NoYWxrLmRpbSgnwrcgL3Blcm1pc3Npb25zIHRvIG1hbmFnZScpfWBcbiAgICBvbkRvbmUobWVzc2FnZVdpdGhIaW50KVxuICB9XG5cbiAgLy8gV2hlbiBubyBwYXRoIGlzIHByb3ZpZGVkLCBzaG93IEFkZFdvcmtzcGFjZURpcmVjdG9yeSBpbnB1dCBmb3JtIGRpcmVjdGx5XG4gIC8vIGFuZCByZXR1cm4gdG8gUkVQTCBhZnRlciBjb25maXJtYXRpb25cbiAgaWYgKCFkaXJlY3RvcnlQYXRoKSB7XG4gICAgcmV0dXJuIChcbiAgICAgIDxBZGRXb3Jrc3BhY2VEaXJlY3RvcnlcbiAgICAgICAgcGVybWlzc2lvbkNvbnRleHQ9e2FwcFN0YXRlLnRvb2xQZXJtaXNzaW9uQ29udGV4dH1cbiAgICAgICAgb25BZGREaXJlY3Rvcnk9e2hhbmRsZUFkZERpcmVjdG9yeX1cbiAgICAgICAgb25DYW5jZWw9eygpID0+IHtcbiAgICAgICAgICBvbkRvbmUoJ0RpZCBub3QgYWRkIGEgd29ya2luZyBkaXJlY3RvcnkuJylcbiAgICAgICAgfX1cbiAgICAgIC8+XG4gICAgKVxuICB9XG5cbiAgY29uc3QgcmVzdWx0ID0gYXdhaXQgdmFsaWRhdGVEaXJlY3RvcnlGb3JXb3Jrc3BhY2UoXG4gICAgZGlyZWN0b3J5UGF0aCxcbiAgICBhcHBTdGF0ZS50b29sUGVybWlzc2lvbkNvbnRleHQsXG4gIClcblxuICBpZiAocmVzdWx0LnJlc3VsdFR5cGUgIT09ICdzdWNjZXNzJykge1xuICAgIGNvbnN0IG1lc3NhZ2UgPSBhZGREaXJIZWxwTWVzc2FnZShyZXN1bHQpXG5cbiAgICByZXR1cm4gKFxuICAgICAgPEFkZERpckVycm9yXG4gICAgICAgIG1lc3NhZ2U9e21lc3NhZ2V9XG4gICAgICAgIGFyZ3M9e2FyZ3MgPz8gJyd9XG4gICAgICAgIG9uRG9uZT17KCkgPT4gb25Eb25lKG1lc3NhZ2UpfVxuICAgICAgLz5cbiAgICApXG4gIH1cblxuICByZXR1cm4gKFxuICAgIDxBZGRXb3Jrc3BhY2VEaXJlY3RvcnlcbiAgICAgIGRpcmVjdG9yeVBhdGg9e3Jlc3VsdC5hYnNvbHV0ZVBhdGh9XG4gICAgICBwZXJtaXNzaW9uQ29udGV4dD17YXBwU3RhdGUudG9vbFBlcm1pc3Npb25Db250ZXh0fVxuICAgICAgb25BZGREaXJlY3Rvcnk9e2hhbmRsZUFkZERpcmVjdG9yeX1cbiAgICAgIG9uQ2FuY2VsPXsoKSA9PiB7XG4gICAgICAgIG9uRG9uZShcbiAgICAgICAgICBgRGlkIG5vdCBhZGQgJHtjaGFsay5ib2xkKHJlc3VsdC5hYnNvbHV0ZVBhdGgpfSBhcyBhIHdvcmtpbmcgZGlyZWN0b3J5LmAsXG4gICAgICAgIClcbiAgICAgIH19XG4gICAgLz5cbiAgKVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsT0FBT0EsS0FBSyxNQUFNLE9BQU87QUFDekIsT0FBT0MsT0FBTyxNQUFNLFNBQVM7QUFDN0IsT0FBT0MsS0FBSyxJQUFJQyxTQUFTLFFBQVEsT0FBTztBQUN4QyxTQUNFQyxtQ0FBbUMsRUFDbkNDLG1DQUFtQyxRQUM5QiwwQkFBMEI7QUFDakMsY0FBY0Msc0JBQXNCLFFBQVEsbUJBQW1CO0FBQy9ELFNBQVNDLGVBQWUsUUFBUSxxQ0FBcUM7QUFDckUsU0FBU0MscUJBQXFCLFFBQVEsNkRBQTZEO0FBQ25HLFNBQVNDLEdBQUcsRUFBRUMsSUFBSSxRQUFRLGNBQWM7QUFDeEMsY0FBY0MscUJBQXFCLFFBQVEsd0JBQXdCO0FBQ25FLFNBQ0VDLHFCQUFxQixFQUNyQkMsdUJBQXVCLFFBQ2xCLDZDQUE2QztBQUNwRCxjQUFjQywyQkFBMkIsUUFBUSxtREFBbUQ7QUFDcEcsU0FBU0MsY0FBYyxRQUFRLHdDQUF3QztBQUN2RSxTQUNFQyxp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 \ No newline at end of file diff --git a/src/commands/add-dir/index.ts b/src/commands/add-dir/index.ts new file mode 100644 index 0000000..e347549 --- /dev/null +++ b/src/commands/add-dir/index.ts @@ -0,0 +1,11 @@ +import type { Command } from '../../commands.js' + +const addDir = { + type: 'local-jsx', + name: 'add-dir', + description: 'Add a new working directory', + argumentHint: '', + load: () => import('./add-dir.js'), +} satisfies Command + +export default addDir diff --git a/src/commands/add-dir/validation.ts b/src/commands/add-dir/validation.ts new file mode 100644 index 0000000..b3627c4 --- /dev/null +++ b/src/commands/add-dir/validation.ts @@ -0,0 +1,110 @@ +import chalk from 'chalk' +import { stat } from 'fs/promises' +import { dirname, resolve } from 'path' +import type { ToolPermissionContext } from '../../Tool.js' +import { getErrnoCode } from '../../utils/errors.js' +import { expandPath } from '../../utils/path.js' +import { + allWorkingDirectories, + pathInWorkingPath, +} from '../../utils/permissions/filesystem.js' + +export type AddDirectoryResult = + | { + resultType: 'success' + absolutePath: string + } + | { + resultType: 'emptyPath' + } + | { + resultType: 'pathNotFound' | 'notADirectory' + directoryPath: string + absolutePath: string + } + | { + resultType: 'alreadyInWorkingDirectory' + directoryPath: string + workingDir: string + } + +export async function validateDirectoryForWorkspace( + directoryPath: string, + permissionContext: ToolPermissionContext, +): Promise { + if (!directoryPath) { + return { + resultType: 'emptyPath', + } + } + + // resolve() strips the trailing slash expandPath can leave on absolute + // inputs, so /foo and /foo/ map to the same storage key (CC-33). + const absolutePath = resolve(expandPath(directoryPath)) + + // Check if path exists and is a directory (single syscall) + try { + const stats = await stat(absolutePath) + if (!stats.isDirectory()) { + return { + resultType: 'notADirectory', + directoryPath, + absolutePath, + } + } + } catch (e: unknown) { + const code = getErrnoCode(e) + // Match prior existsSync() semantics: treat any of these as "not found" + // rather than re-throwing. EACCES/EPERM in particular must not crash + // startup when a settings-configured additional directory is inaccessible. + if ( + code === 'ENOENT' || + code === 'ENOTDIR' || + code === 'EACCES' || + code === 'EPERM' + ) { + return { + resultType: 'pathNotFound', + directoryPath, + absolutePath, + } + } + throw e + } + + // Get current permission context + const currentWorkingDirs = allWorkingDirectories(permissionContext) + + // Check if already within an existing working directory + for (const workingDir of currentWorkingDirs) { + if (pathInWorkingPath(absolutePath, workingDir)) { + return { + resultType: 'alreadyInWorkingDirectory', + directoryPath, + workingDir, + } + } + } + + return { + resultType: 'success', + absolutePath, + } +} + +export function addDirHelpMessage(result: AddDirectoryResult): string { + switch (result.resultType) { + case 'emptyPath': + return 'Please provide a directory path.' + case 'pathNotFound': + return `Path ${chalk.bold(result.absolutePath)} was not found.` + case 'notADirectory': { + const parentDir = dirname(result.absolutePath) + return `${chalk.bold(result.directoryPath)} is not a directory. Did you mean to add the parent directory ${chalk.bold(parentDir)}?` + } + case 'alreadyInWorkingDirectory': + return `${chalk.bold(result.directoryPath)} is already accessible within the existing working directory ${chalk.bold(result.workingDir)}.` + case 'success': + return `Added ${chalk.bold(result.absolutePath)} as a working directory.` + } +} diff --git a/src/commands/advisor.ts b/src/commands/advisor.ts new file mode 100644 index 0000000..cec3feb --- /dev/null +++ b/src/commands/advisor.ts @@ -0,0 +1,109 @@ +import type { Command } from '../commands.js' +import type { LocalCommandCall } from '../types/command.js' +import { + canUserConfigureAdvisor, + isValidAdvisorModel, + modelSupportsAdvisor, +} from '../utils/advisor.js' +import { + getDefaultMainLoopModelSetting, + normalizeModelStringForAPI, + parseUserSpecifiedModel, +} from '../utils/model/model.js' +import { validateModel } from '../utils/model/validateModel.js' +import { updateSettingsForSource } from '../utils/settings/settings.js' + +const call: LocalCommandCall = async (args, context) => { + const arg = args.trim().toLowerCase() + const baseModel = parseUserSpecifiedModel( + context.getAppState().mainLoopModel ?? getDefaultMainLoopModelSetting(), + ) + + if (!arg) { + const current = context.getAppState().advisorModel + if (!current) { + return { + type: 'text', + value: + 'Advisor: not set\nUse "/advisor " to enable (e.g. "/advisor opus").', + } + } + if (!modelSupportsAdvisor(baseModel)) { + return { + type: 'text', + value: `Advisor: ${current} (inactive)\nThe current model (${baseModel}) does not support advisors.`, + } + } + return { + type: 'text', + value: `Advisor: ${current}\nUse "/advisor unset" to disable or "/advisor " to change.`, + } + } + + if (arg === 'unset' || arg === 'off') { + const prev = context.getAppState().advisorModel + context.setAppState(s => { + if (s.advisorModel === undefined) return s + return { ...s, advisorModel: undefined } + }) + updateSettingsForSource('userSettings', { advisorModel: undefined }) + return { + type: 'text', + value: prev + ? `Advisor disabled (was ${prev}).` + : 'Advisor already unset.', + } + } + + const normalizedModel = normalizeModelStringForAPI(arg) + const resolvedModel = parseUserSpecifiedModel(arg) + const { valid, error } = await validateModel(resolvedModel) + if (!valid) { + return { + type: 'text', + value: error + ? `Invalid advisor model: ${error}` + : `Unknown model: ${arg} (${resolvedModel})`, + } + } + + if (!isValidAdvisorModel(resolvedModel)) { + return { + type: 'text', + value: `The model ${arg} (${resolvedModel}) cannot be used as an advisor`, + } + } + + context.setAppState(s => { + if (s.advisorModel === normalizedModel) return s + return { ...s, advisorModel: normalizedModel } + }) + updateSettingsForSource('userSettings', { advisorModel: normalizedModel }) + + if (!modelSupportsAdvisor(baseModel)) { + return { + type: 'text', + value: `Advisor set to ${normalizedModel}.\nNote: Your current model (${baseModel}) does not support advisors. Switch to a supported model to use the advisor.`, + } + } + + return { + type: 'text', + value: `Advisor set to ${normalizedModel}.`, + } +} + +const advisor = { + type: 'local', + name: 'advisor', + description: 'Configure the advisor model', + argumentHint: '[|off]', + isEnabled: () => canUserConfigureAdvisor(), + get isHidden() { + return !canUserConfigureAdvisor() + }, + supportsNonInteractive: true, + load: () => Promise.resolve({ call }), +} satisfies Command + +export default advisor diff --git a/src/commands/agents/agents.tsx b/src/commands/agents/agents.tsx new file mode 100644 index 0000000..3af6273 --- /dev/null +++ b/src/commands/agents/agents.tsx @@ -0,0 +1,12 @@ +import * as React from 'react'; +import { AgentsMenu } from '../../components/agents/AgentsMenu.js'; +import type { ToolUseContext } from '../../Tool.js'; +import { getTools } from '../../tools.js'; +import type { LocalJSXCommandOnDone } from '../../types/command.js'; +export async function call(onDone: LocalJSXCommandOnDone, context: ToolUseContext): Promise { + const appState = context.getAppState(); + const permissionContext = appState.toolPermissionContext; + const tools = getTools(permissionContext); + return ; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkFnZW50c01lbnUiLCJUb29sVXNlQ29udGV4dCIsImdldFRvb2xzIiwiTG9jYWxKU1hDb21tYW5kT25Eb25lIiwiY2FsbCIsIm9uRG9uZSIsImNvbnRleHQiLCJQcm9taXNlIiwiUmVhY3ROb2RlIiwiYXBwU3RhdGUiLCJnZXRBcHBTdGF0ZSIsInBlcm1pc3Npb25Db250ZXh0IiwidG9vbFBlcm1pc3Npb25Db250ZXh0IiwidG9vbHMiXSwic291cmNlcyI6WyJhZ2VudHMudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgQWdlbnRzTWVudSB9IGZyb20gJy4uLy4uL2NvbXBvbmVudHMvYWdlbnRzL0FnZW50c01lbnUuanMnXG5pbXBvcnQgdHlwZSB7IFRvb2xVc2VDb250ZXh0IH0gZnJvbSAnLi4vLi4vVG9vbC5qcydcbmltcG9ydCB7IGdldFRvb2xzIH0gZnJvbSAnLi4vLi4vdG9vbHMuanMnXG5pbXBvcnQgdHlwZSB7IExvY2FsSlNYQ29tbWFuZE9uRG9uZSB9IGZyb20gJy4uLy4uL3R5cGVzL2NvbW1hbmQuanMnXG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBjYWxsKFxuICBvbkRvbmU6IExvY2FsSlNYQ29tbWFuZE9uRG9uZSxcbiAgY29udGV4dDogVG9vbFVzZUNvbnRleHQsXG4pOiBQcm9taXNlPFJlYWN0LlJlYWN0Tm9kZT4ge1xuICBjb25zdCBhcHBTdGF0ZSA9IGNvbnRleHQuZ2V0QXBwU3RhdGUoKVxuICBjb25zdCBwZXJtaXNzaW9uQ29udGV4dCA9IGFwcFN0YXRlLnRvb2xQZXJtaXNzaW9uQ29udGV4dFxuICBjb25zdCB0b29scyA9IGdldFRvb2xzKHBlcm1pc3Npb25Db250ZXh0KVxuXG4gIHJldHVybiA8QWdlbnRzTWVudSB0b29scz17dG9vbHN9IG9uRXhpdD17b25Eb25lfSAvPlxufVxuIl0sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUtBLEtBQUssTUFBTSxPQUFPO0FBQzlCLFNBQVNDLFVBQVUsUUFBUSx1Q0FBdUM7QUFDbEUsY0FBY0MsY0FBYyxRQUFRLGVBQWU7QUFDbkQsU0FBU0MsUUFBUSxRQUFRLGdCQUFnQjtBQUN6QyxjQUFjQyxxQkFBcUIsUUFBUSx3QkFBd0I7QUFFbkUsT0FBTyxlQUFlQyxJQUFJQSxDQUN4QkMsTUFBTSxFQUFFRixxQkFBcUIsRUFDN0JHLE9BQU8sRUFBRUwsY0FBYyxDQUN4QixFQUFFTSxPQUFPLENBQUNSLEtBQUssQ0FBQ1MsU0FBUyxDQUFDLENBQUM7RUFDMUIsTUFBTUMsUUFBUSxHQUFHSCxPQUFPLENBQUNJLFdBQVcsQ0FBQyxDQUFDO0VBQ3RDLE1BQU1DLGlCQUFpQixHQUFHRixRQUFRLENBQUNHLHFCQUFxQjtFQUN4RCxNQUFNQyxLQUFLLEdBQUdYLFFBQVEsQ0FBQ1MsaUJBQWlCLENBQUM7RUFFekMsT0FBTyxDQUFDLFVBQVUsQ0FBQyxLQUFLLENBQUMsQ0FBQ0UsS0FBSyxDQUFDLENBQUMsTUFBTSxDQUFDLENBQUNSLE1BQU0sQ0FBQyxHQUFHO0FBQ3JEIiwiaWdub3JlTGlzdCI6W119 \ No newline at end of file diff --git a/src/commands/agents/index.ts b/src/commands/agents/index.ts new file mode 100644 index 0000000..ac43d2e --- /dev/null +++ b/src/commands/agents/index.ts @@ -0,0 +1,10 @@ +import type { Command } from '../../commands.js' + +const agents = { + type: 'local-jsx', + name: 'agents', + description: 'Manage agent configurations', + load: () => import('./agents.js'), +} satisfies Command + +export default agents diff --git a/src/commands/ant-trace/index.js b/src/commands/ant-trace/index.js new file mode 100644 index 0000000..e1a619d --- /dev/null +++ b/src/commands/ant-trace/index.js @@ -0,0 +1 @@ +export default { isEnabled: () => false, isHidden: true, name: 'stub' }; diff --git a/src/commands/autofix-pr/index.js b/src/commands/autofix-pr/index.js new file mode 100644 index 0000000..e1a619d --- /dev/null +++ b/src/commands/autofix-pr/index.js @@ -0,0 +1 @@ +export default { isEnabled: () => false, isHidden: true, name: 'stub' }; diff --git a/src/commands/backfill-sessions/index.js b/src/commands/backfill-sessions/index.js new file mode 100644 index 0000000..e1a619d --- /dev/null +++ b/src/commands/backfill-sessions/index.js @@ -0,0 +1 @@ +export default { isEnabled: () => false, isHidden: true, name: 'stub' }; diff --git a/src/commands/branch/branch.ts b/src/commands/branch/branch.ts new file mode 100644 index 0000000..4a7c277 --- /dev/null +++ b/src/commands/branch/branch.ts @@ -0,0 +1,296 @@ +import { randomUUID, type UUID } from 'crypto' +import { mkdir, readFile, writeFile } from 'fs/promises' +import { getOriginalCwd, getSessionId } from '../../bootstrap/state.js' +import type { LocalJSXCommandContext } from '../../commands.js' +import { logEvent } from '../../services/analytics/index.js' +import type { LocalJSXCommandOnDone } from '../../types/command.js' +import type { + ContentReplacementEntry, + Entry, + LogOption, + SerializedMessage, + TranscriptMessage, +} from '../../types/logs.js' +import { parseJSONL } from '../../utils/json.js' +import { + getProjectDir, + getTranscriptPath, + getTranscriptPathForSession, + isTranscriptMessage, + saveCustomTitle, + searchSessionsByCustomTitle, +} from '../../utils/sessionStorage.js' +import { jsonStringify } from '../../utils/slowOperations.js' +import { escapeRegExp } from '../../utils/stringUtils.js' + +type TranscriptEntry = TranscriptMessage & { + forkedFrom?: { + sessionId: string + messageUuid: UUID + } +} + +/** + * Derive a single-line title base from the first user message. + * Collapses whitespace — multiline first messages (pasted stacks, code) + * otherwise flow into the saved title and break the resume hint. + */ +export function deriveFirstPrompt( + firstUserMessage: Extract | undefined, +): string { + const content = firstUserMessage?.message?.content + if (!content) return 'Branched conversation' + const raw = + typeof content === 'string' + ? content + : content.find( + (block): block is { type: 'text'; text: string } => + block.type === 'text', + )?.text + if (!raw) return 'Branched conversation' + return ( + raw.replace(/\s+/g, ' ').trim().slice(0, 100) || 'Branched conversation' + ) +} + +/** + * Creates a fork of the current conversation by copying from the transcript file. + * Preserves all original metadata (timestamps, gitBranch, etc.) while updating + * sessionId and adding forkedFrom traceability. + */ +async function createFork(customTitle?: string): Promise<{ + sessionId: UUID + title: string | undefined + forkPath: string + serializedMessages: SerializedMessage[] + contentReplacementRecords: ContentReplacementEntry['replacements'] +}> { + const forkSessionId = randomUUID() as UUID + const originalSessionId = getSessionId() + const projectDir = getProjectDir(getOriginalCwd()) + const forkSessionPath = getTranscriptPathForSession(forkSessionId) + const currentTranscriptPath = getTranscriptPath() + + // Ensure project directory exists + await mkdir(projectDir, { recursive: true, mode: 0o700 }) + + // Read current transcript file + let transcriptContent: Buffer + try { + transcriptContent = await readFile(currentTranscriptPath) + } catch { + throw new Error('No conversation to branch') + } + + if (transcriptContent.length === 0) { + throw new Error('No conversation to branch') + } + + // Parse all transcript entries (messages + metadata entries like content-replacement) + const entries = parseJSONL(transcriptContent) + + // Filter to only main conversation messages (exclude sidechains and non-message entries) + const mainConversationEntries = entries.filter( + (entry): entry is TranscriptMessage => + isTranscriptMessage(entry) && !entry.isSidechain, + ) + + // Content-replacement entries for the original session. These record which + // tool_result blocks were replaced with previews by the per-message budget. + // Without them in the fork JSONL, `claude -r {forkId}` reconstructs state + // with an empty replacements Map → previously-replaced results are classified + // as FROZEN and sent as full content (prompt cache miss + permanent overage). + // sessionId must be rewritten since loadTranscriptFile keys lookup by the + // session's messages' sessionId. + const contentReplacementRecords = entries + .filter( + (entry): entry is ContentReplacementEntry => + entry.type === 'content-replacement' && + entry.sessionId === originalSessionId, + ) + .flatMap(entry => entry.replacements) + + if (mainConversationEntries.length === 0) { + throw new Error('No messages to branch') + } + + // Build forked entries with new sessionId and preserved metadata + let parentUuid: UUID | null = null + const lines: string[] = [] + const serializedMessages: SerializedMessage[] = [] + + for (const entry of mainConversationEntries) { + // Create forked transcript entry preserving all original metadata + const forkedEntry: TranscriptEntry = { + ...entry, + sessionId: forkSessionId, + parentUuid, + isSidechain: false, + forkedFrom: { + sessionId: originalSessionId, + messageUuid: entry.uuid, + }, + } + + // Build serialized message for LogOption + const serialized: SerializedMessage = { + ...entry, + sessionId: forkSessionId, + } + + serializedMessages.push(serialized) + lines.push(jsonStringify(forkedEntry)) + if (entry.type !== 'progress') { + parentUuid = entry.uuid + } + } + + // Append content-replacement entry (if any) with the fork's sessionId. + // Written as a SINGLE entry (same shape as insertContentReplacement) so + // loadTranscriptFile's content-replacement branch picks it up. + if (contentReplacementRecords.length > 0) { + const forkedReplacementEntry: ContentReplacementEntry = { + type: 'content-replacement', + sessionId: forkSessionId, + replacements: contentReplacementRecords, + } + lines.push(jsonStringify(forkedReplacementEntry)) + } + + // Write the fork session file + await writeFile(forkSessionPath, lines.join('\n') + '\n', { + encoding: 'utf8', + mode: 0o600, + }) + + return { + sessionId: forkSessionId, + title: customTitle, + forkPath: forkSessionPath, + serializedMessages, + contentReplacementRecords, + } +} + +/** + * Generates a unique fork name by checking for collisions with existing session names. + * If "baseName (Branch)" already exists, tries "baseName (Branch 2)", "baseName (Branch 3)", etc. + */ +async function getUniqueForkName(baseName: string): Promise { + const candidateName = `${baseName} (Branch)` + + // Check if this exact name already exists + const existingWithExactName = await searchSessionsByCustomTitle( + candidateName, + { exact: true }, + ) + + if (existingWithExactName.length === 0) { + return candidateName + } + + // Name collision - find a unique numbered suffix + // Search for all sessions that start with the base pattern + const existingForks = await searchSessionsByCustomTitle(`${baseName} (Branch`) + + // Extract existing fork numbers to find the next available + const usedNumbers = new Set([1]) // Consider " (Branch)" as number 1 + const forkNumberPattern = new RegExp( + `^${escapeRegExp(baseName)} \\(Branch(?: (\\d+))?\\)$`, + ) + + for (const session of existingForks) { + const match = session.customTitle?.match(forkNumberPattern) + if (match) { + if (match[1]) { + usedNumbers.add(parseInt(match[1], 10)) + } else { + usedNumbers.add(1) // " (Branch)" without number is treated as 1 + } + } + } + + // Find the next available number + let nextNumber = 2 + while (usedNumbers.has(nextNumber)) { + nextNumber++ + } + + return `${baseName} (Branch ${nextNumber})` +} + +export async function call( + onDone: LocalJSXCommandOnDone, + context: LocalJSXCommandContext, + args: string, +): Promise { + const customTitle = args?.trim() || undefined + + const originalSessionId = getSessionId() + + try { + const { + sessionId, + title, + forkPath, + serializedMessages, + contentReplacementRecords, + } = await createFork(customTitle) + + // Build LogOption for resume + const now = new Date() + const firstPrompt = deriveFirstPrompt( + serializedMessages.find(m => m.type === 'user'), + ) + + // Save custom title - use provided title or firstPrompt as default + // This ensures /status and /resume show the same session name + // Always add " (Branch)" suffix to make it clear this is a branched session + // Handle collisions by adding a number suffix (e.g., " (Branch 2)", " (Branch 3)") + const baseName = title ?? firstPrompt + const effectiveTitle = await getUniqueForkName(baseName) + await saveCustomTitle(sessionId, effectiveTitle, forkPath) + + logEvent('tengu_conversation_forked', { + message_count: serializedMessages.length, + has_custom_title: !!title, + }) + + const forkLog: LogOption = { + date: now.toISOString().split('T')[0]!, + messages: serializedMessages, + fullPath: forkPath, + value: now.getTime(), + created: now, + modified: now, + firstPrompt, + messageCount: serializedMessages.length, + isSidechain: false, + sessionId, + customTitle: effectiveTitle, + contentReplacements: contentReplacementRecords, + } + + // Resume into the fork + const titleInfo = title ? ` "${title}"` : '' + const resumeHint = `\nTo resume the original: claude -r ${originalSessionId}` + const successMessage = `Branched conversation${titleInfo}. You are now in the branch.${resumeHint}` + + if (context.resume) { + await context.resume(sessionId, forkLog, 'fork') + onDone(successMessage, { display: 'system' }) + } else { + // Fallback if resume not available + onDone( + `Branched conversation${titleInfo}. Resume with: /resume ${sessionId}`, + ) + } + + return null + } catch (error) { + const message = + error instanceof Error ? error.message : 'Unknown error occurred' + onDone(`Failed to branch conversation: ${message}`) + return null + } +} diff --git a/src/commands/branch/index.ts b/src/commands/branch/index.ts new file mode 100644 index 0000000..731ff39 --- /dev/null +++ b/src/commands/branch/index.ts @@ -0,0 +1,14 @@ +import { feature } from 'bun:bundle' +import type { Command } from '../../commands.js' + +const branch = { + type: 'local-jsx', + name: 'branch', + // 'fork' alias only when /fork doesn't exist as its own command + aliases: feature('FORK_SUBAGENT') ? [] : ['fork'], + description: 'Create a branch of the current conversation at this point', + argumentHint: '[name]', + load: () => import('./branch.js'), +} satisfies Command + +export default branch diff --git a/src/commands/break-cache/index.js b/src/commands/break-cache/index.js new file mode 100644 index 0000000..e1a619d --- /dev/null +++ b/src/commands/break-cache/index.js @@ -0,0 +1 @@ +export default { isEnabled: () => false, isHidden: true, name: 'stub' }; diff --git a/src/commands/bridge-kick.ts b/src/commands/bridge-kick.ts new file mode 100644 index 0000000..f8564c0 --- /dev/null +++ b/src/commands/bridge-kick.ts @@ -0,0 +1,200 @@ +import { getBridgeDebugHandle } from '../bridge/bridgeDebug.js' +import type { Command } from '../commands.js' +import type { LocalCommandCall } from '../types/command.js' + +/** + * Ant-only: inject bridge failure states to manually test recovery paths. + * + * /bridge-kick close 1002 — fire ws_closed with code 1002 + * /bridge-kick close 1006 — fire ws_closed with code 1006 + * /bridge-kick poll 404 — next poll throws 404/not_found_error + * /bridge-kick poll 404 — next poll throws 404 with error_type + * /bridge-kick poll 401 — next poll throws 401 (auth) + * /bridge-kick poll transient — next poll throws axios-style rejection + * /bridge-kick register fail — next register (inside doReconnect) transient-fails + * /bridge-kick register fail 3 — next 3 registers transient-fail + * /bridge-kick register fatal — next register 403s (terminal) + * /bridge-kick reconnect-session fail — POST /bridge/reconnect fails (→ Strategy 2) + * /bridge-kick heartbeat 401 — next heartbeat 401s (JWT expired) + * /bridge-kick reconnect — call doReconnect directly (= SIGUSR2) + * /bridge-kick status — print current bridge state + * + * Workflow: connect Remote Control, run a subcommand, `tail -f debug.log` + * and watch [bridge:repl] / [bridge:debug] lines for the recovery reaction. + * + * Composite sequences — the failure modes in the BQ data are chains, not + * single events. Queue faults then fire the trigger: + * + * # #22148 residual: ws_closed → register transient-blips → teardown? + * /bridge-kick register fail 2 + * /bridge-kick close 1002 + * → expect: doReconnect tries register, fails, returns false → teardown + * (demonstrates the retry gap that needs fixing) + * + * # Dead gate: poll 404/not_found_error → does onEnvironmentLost fire? + * /bridge-kick poll 404 + * → expect: tengu_bridge_repl_fatal_error (gate is dead — 147K/wk) + * after fix: tengu_bridge_repl_env_lost → doReconnect + */ + +const USAGE = `/bridge-kick + close fire ws_closed with the given code (e.g. 1002) + poll [type] next poll throws BridgeFatalError(status, type) + poll transient next poll throws axios-style rejection (5xx/net) + register fail [N] next N registers transient-fail (default 1) + register fatal next register 403s (terminal) + reconnect-session fail next POST /bridge/reconnect fails + heartbeat next heartbeat throws BridgeFatalError(status) + reconnect call reconnectEnvironmentWithSession directly + status print bridge state` + +const call: LocalCommandCall = async args => { + const h = getBridgeDebugHandle() + if (!h) { + return { + type: 'text', + value: + 'No bridge debug handle registered. Remote Control must be connected (USER_TYPE=ant).', + } + } + + const [sub, a, b] = args.trim().split(/\s+/) + + switch (sub) { + case 'close': { + const code = Number(a) + if (!Number.isFinite(code)) { + return { type: 'text', value: `close: need a numeric code\n${USAGE}` } + } + h.fireClose(code) + return { + type: 'text', + value: `Fired transport close(${code}). Watch debug.log for [bridge:repl] recovery.`, + } + } + + case 'poll': { + if (a === 'transient') { + h.injectFault({ + method: 'pollForWork', + kind: 'transient', + status: 503, + count: 1, + }) + h.wakePollLoop() + return { + type: 'text', + value: + 'Next poll will throw a transient (axios rejection). Poll loop woken.', + } + } + const status = Number(a) + if (!Number.isFinite(status)) { + return { + type: 'text', + value: `poll: need 'transient' or a status code\n${USAGE}`, + } + } + // Default to what the server ACTUALLY sends for 404 (BQ-verified), + // so `/bridge-kick poll 404` reproduces the real 147K/week state. + const errorType = + b ?? (status === 404 ? 'not_found_error' : 'authentication_error') + h.injectFault({ + method: 'pollForWork', + kind: 'fatal', + status, + errorType, + count: 1, + }) + h.wakePollLoop() + return { + type: 'text', + value: `Next poll will throw BridgeFatalError(${status}, ${errorType}). Poll loop woken.`, + } + } + + case 'register': { + if (a === 'fatal') { + h.injectFault({ + method: 'registerBridgeEnvironment', + kind: 'fatal', + status: 403, + errorType: 'permission_error', + count: 1, + }) + return { + type: 'text', + value: + 'Next registerBridgeEnvironment will 403. Trigger with close/reconnect.', + } + } + const n = Number(b) || 1 + h.injectFault({ + method: 'registerBridgeEnvironment', + kind: 'transient', + status: 503, + count: n, + }) + return { + type: 'text', + value: `Next ${n} registerBridgeEnvironment call(s) will transient-fail. Trigger with close/reconnect.`, + } + } + + case 'reconnect-session': { + h.injectFault({ + method: 'reconnectSession', + kind: 'fatal', + status: 404, + errorType: 'not_found_error', + count: 2, + }) + return { + type: 'text', + value: + 'Next 2 POST /bridge/reconnect calls will 404. doReconnect Strategy 1 falls through to Strategy 2.', + } + } + + case 'heartbeat': { + const status = Number(a) || 401 + h.injectFault({ + method: 'heartbeatWork', + kind: 'fatal', + status, + errorType: status === 401 ? 'authentication_error' : 'not_found_error', + count: 1, + }) + return { + type: 'text', + value: `Next heartbeat will ${status}. Watch for onHeartbeatFatal → work-state teardown.`, + } + } + + case 'reconnect': { + h.forceReconnect() + return { + type: 'text', + value: 'Called reconnectEnvironmentWithSession(). Watch debug.log.', + } + } + + case 'status': { + return { type: 'text', value: h.describe() } + } + + default: + return { type: 'text', value: USAGE } + } +} + +const bridgeKick = { + type: 'local', + name: 'bridge-kick', + description: 'Inject bridge failure states for manual recovery testing', + isEnabled: () => process.env.USER_TYPE === 'ant', + supportsNonInteractive: false, + load: () => Promise.resolve({ call }), +} satisfies Command + +export default bridgeKick diff --git a/src/commands/bridge/bridge.tsx b/src/commands/bridge/bridge.tsx new file mode 100644 index 0000000..02ca16b --- /dev/null +++ b/src/commands/bridge/bridge.tsx @@ -0,0 +1,509 @@ +import { c as _c } from "react/compiler-runtime"; +import { feature } from 'bun:bundle'; +import { toString as qrToString } from 'qrcode'; +import * as React from 'react'; +import { useEffect, useState } from 'react'; +import { getBridgeAccessToken } from '../../bridge/bridgeConfig.js'; +import { checkBridgeMinVersion, getBridgeDisabledReason, isEnvLessBridgeEnabled } from '../../bridge/bridgeEnabled.js'; +import { checkEnvLessBridgeMinVersion } from '../../bridge/envLessBridgeConfig.js'; +import { BRIDGE_LOGIN_INSTRUCTION, REMOTE_CONTROL_DISCONNECTED_MSG } from '../../bridge/types.js'; +import { Dialog } from '../../components/design-system/Dialog.js'; +import { ListItem } from '../../components/design-system/ListItem.js'; +import { shouldShowRemoteCallout } from '../../components/RemoteCallout.js'; +import { useRegisterOverlay } from '../../context/overlayContext.js'; +import { Box, Text } from '../../ink.js'; +import { useKeybindings } from '../../keybindings/useKeybinding.js'; +import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from '../../services/analytics/index.js'; +import { useAppState, useSetAppState } from '../../state/AppState.js'; +import type { ToolUseContext } from '../../Tool.js'; +import type { LocalJSXCommandContext, LocalJSXCommandOnDone } from '../../types/command.js'; +import { logForDebugging } from '../../utils/debug.js'; +type Props = { + onDone: LocalJSXCommandOnDone; + name?: string; +}; + +/** + * /remote-control command — manages the bidirectional bridge connection. + * + * When enabled, sets replBridgeEnabled in AppState, which triggers + * useReplBridge in REPL.tsx to initialize the bridge connection. + * The bridge registers an environment, creates a session with the current + * conversation, polls for work, and connects an ingress WebSocket for + * bidirectional messaging between the CLI and claude.ai. + * + * Running /remote-control when already connected shows a dialog with the session + * URL and options to disconnect or continue. + */ +function BridgeToggle(t0) { + const $ = _c(10); + const { + onDone, + name + } = t0; + const setAppState = useSetAppState(); + const replBridgeConnected = useAppState(_temp); + const replBridgeEnabled = useAppState(_temp2); + const replBridgeOutboundOnly = useAppState(_temp3); + const [showDisconnectDialog, setShowDisconnectDialog] = useState(false); + let t1; + if ($[0] !== name || $[1] !== onDone || $[2] !== replBridgeConnected || $[3] !== replBridgeEnabled || $[4] !== replBridgeOutboundOnly || $[5] !== setAppState) { + t1 = () => { + if ((replBridgeConnected || replBridgeEnabled) && !replBridgeOutboundOnly) { + setShowDisconnectDialog(true); + return; + } + let cancelled = false; + (async () => { + const error = await checkBridgePrerequisites(); + if (cancelled) { + return; + } + if (error) { + logEvent("tengu_bridge_command", { + action: "preflight_failed" as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + onDone(error, { + display: "system" + }); + return; + } + if (shouldShowRemoteCallout()) { + setAppState(prev => { + if (prev.showRemoteCallout) { + return prev; + } + return { + ...prev, + showRemoteCallout: true, + replBridgeInitialName: name + }; + }); + onDone("", { + display: "system" + }); + return; + } + logEvent("tengu_bridge_command", { + action: "connect" as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + setAppState(prev_0 => { + if (prev_0.replBridgeEnabled && !prev_0.replBridgeOutboundOnly) { + return prev_0; + } + return { + ...prev_0, + replBridgeEnabled: true, + replBridgeExplicit: true, + replBridgeOutboundOnly: false, + replBridgeInitialName: name + }; + }); + onDone("Remote Control connecting\u2026", { + display: "system" + }); + })(); + return () => { + cancelled = true; + }; + }; + $[0] = name; + $[1] = onDone; + $[2] = replBridgeConnected; + $[3] = replBridgeEnabled; + $[4] = replBridgeOutboundOnly; + $[5] = setAppState; + $[6] = t1; + } else { + t1 = $[6]; + } + let t2; + if ($[7] === Symbol.for("react.memo_cache_sentinel")) { + t2 = []; + $[7] = t2; + } else { + t2 = $[7]; + } + useEffect(t1, t2); + if (showDisconnectDialog) { + let t3; + if ($[8] !== onDone) { + t3 = ; + $[8] = onDone; + $[9] = t3; + } else { + t3 = $[9]; + } + return t3; + } + return null; +} + +/** + * Dialog shown when /remote-control is used while the bridge is already connected. + * Shows the session URL and lets the user disconnect or continue. + */ +function _temp3(s_1) { + return s_1.replBridgeOutboundOnly; +} +function _temp2(s_0) { + return s_0.replBridgeEnabled; +} +function _temp(s) { + return s.replBridgeConnected; +} +function BridgeDisconnectDialog(t0) { + const $ = _c(61); + const { + onDone + } = t0; + useRegisterOverlay("bridge-disconnect-dialog"); + const setAppState = useSetAppState(); + const sessionUrl = useAppState(_temp4); + const connectUrl = useAppState(_temp5); + const sessionActive = useAppState(_temp6); + const [focusIndex, setFocusIndex] = useState(2); + const [showQR, setShowQR] = useState(false); + const [qrText, setQrText] = useState(""); + const displayUrl = sessionActive ? sessionUrl : connectUrl; + let t1; + let t2; + if ($[0] !== displayUrl || $[1] !== showQR) { + t1 = () => { + if (!showQR || !displayUrl) { + setQrText(""); + return; + } + qrToString(displayUrl, { + type: "utf8", + errorCorrectionLevel: "L", + small: true + }).then(setQrText).catch(() => setQrText("")); + }; + t2 = [showQR, displayUrl]; + $[0] = displayUrl; + $[1] = showQR; + $[2] = t1; + $[3] = t2; + } else { + t1 = $[2]; + t2 = $[3]; + } + useEffect(t1, t2); + let t3; + if ($[4] !== onDone || $[5] !== setAppState) { + t3 = function handleDisconnect() { + setAppState(_temp7); + logEvent("tengu_bridge_command", { + action: "disconnect" as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + onDone(REMOTE_CONTROL_DISCONNECTED_MSG, { + display: "system" + }); + }; + $[4] = onDone; + $[5] = setAppState; + $[6] = t3; + } else { + t3 = $[6]; + } + const handleDisconnect = t3; + let t4; + if ($[7] === Symbol.for("react.memo_cache_sentinel")) { + t4 = function handleShowQR() { + setShowQR(_temp8); + }; + $[7] = t4; + } else { + t4 = $[7]; + } + const handleShowQR = t4; + let t5; + if ($[8] !== onDone) { + t5 = function handleContinue() { + onDone(undefined, { + display: "skip" + }); + }; + $[8] = onDone; + $[9] = t5; + } else { + t5 = $[9]; + } + const handleContinue = t5; + let t6; + let t7; + if ($[10] === Symbol.for("react.memo_cache_sentinel")) { + t6 = () => setFocusIndex(_temp9); + t7 = () => setFocusIndex(_temp0); + $[10] = t6; + $[11] = t7; + } else { + t6 = $[10]; + t7 = $[11]; + } + let t8; + if ($[12] !== focusIndex || $[13] !== handleContinue || $[14] !== handleDisconnect) { + t8 = { + "select:next": t6, + "select:previous": t7, + "select:accept": () => { + if (focusIndex === 0) { + handleDisconnect(); + } else { + if (focusIndex === 1) { + handleShowQR(); + } else { + handleContinue(); + } + } + } + }; + $[12] = focusIndex; + $[13] = handleContinue; + $[14] = handleDisconnect; + $[15] = t8; + } else { + t8 = $[15]; + } + let t9; + if ($[16] === Symbol.for("react.memo_cache_sentinel")) { + t9 = { + context: "Select" + }; + $[16] = t9; + } else { + t9 = $[16]; + } + useKeybindings(t8, t9); + let T0; + let T1; + let t10; + let t11; + let t12; + let t13; + let t14; + let t15; + let t16; + if ($[17] !== displayUrl || $[18] !== handleContinue || $[19] !== qrText || $[20] !== showQR) { + const qrLines = qrText ? qrText.split("\n").filter(_temp1) : []; + T1 = Dialog; + t14 = "Remote Control"; + t15 = handleContinue; + t16 = true; + T0 = Box; + t10 = "column"; + t11 = 1; + const t17 = displayUrl ? ` at ${displayUrl}` : ""; + if ($[30] !== t17) { + t12 = This session is available via Remote Control{t17}.; + $[30] = t17; + $[31] = t12; + } else { + t12 = $[31]; + } + t13 = showQR && qrLines.length > 0 && {qrLines.map(_temp10)}; + $[17] = displayUrl; + $[18] = handleContinue; + $[19] = qrText; + $[20] = showQR; + $[21] = T0; + $[22] = T1; + $[23] = t10; + $[24] = t11; + $[25] = t12; + $[26] = t13; + $[27] = t14; + $[28] = t15; + $[29] = t16; + } else { + T0 = $[21]; + T1 = $[22]; + t10 = $[23]; + t11 = $[24]; + t12 = $[25]; + t13 = $[26]; + t14 = $[27]; + t15 = $[28]; + t16 = $[29]; + } + const t17 = focusIndex === 0; + let t18; + if ($[32] === Symbol.for("react.memo_cache_sentinel")) { + t18 = Disconnect this session; + $[32] = t18; + } else { + t18 = $[32]; + } + let t19; + if ($[33] !== t17) { + t19 = {t18}; + $[33] = t17; + $[34] = t19; + } else { + t19 = $[34]; + } + const t20 = focusIndex === 1; + const t21 = showQR ? "Hide QR code" : "Show QR code"; + let t22; + if ($[35] !== t21) { + t22 = {t21}; + $[35] = t21; + $[36] = t22; + } else { + t22 = $[36]; + } + let t23; + if ($[37] !== t20 || $[38] !== t22) { + t23 = {t22}; + $[37] = t20; + $[38] = t22; + $[39] = t23; + } else { + t23 = $[39]; + } + const t24 = focusIndex === 2; + let t25; + if ($[40] === Symbol.for("react.memo_cache_sentinel")) { + t25 = Continue; + $[40] = t25; + } else { + t25 = $[40]; + } + let t26; + if ($[41] !== t24) { + t26 = {t25}; + $[41] = t24; + $[42] = t26; + } else { + t26 = $[42]; + } + let t27; + if ($[43] !== t19 || $[44] !== t23 || $[45] !== t26) { + t27 = {t19}{t23}{t26}; + $[43] = t19; + $[44] = t23; + $[45] = t26; + $[46] = t27; + } else { + t27 = $[46]; + } + let t28; + if ($[47] === Symbol.for("react.memo_cache_sentinel")) { + t28 = Enter to select · Esc to continue; + $[47] = t28; + } else { + t28 = $[47]; + } + let t29; + if ($[48] !== T0 || $[49] !== t10 || $[50] !== t11 || $[51] !== t12 || $[52] !== t13 || $[53] !== t27) { + t29 = {t12}{t13}{t27}{t28}; + $[48] = T0; + $[49] = t10; + $[50] = t11; + $[51] = t12; + $[52] = t13; + $[53] = t27; + $[54] = t29; + } else { + t29 = $[54]; + } + let t30; + if ($[55] !== T1 || $[56] !== t14 || $[57] !== t15 || $[58] !== t16 || $[59] !== t29) { + t30 = {t29}; + $[55] = T1; + $[56] = t14; + $[57] = t15; + $[58] = t16; + $[59] = t29; + $[60] = t30; + } else { + t30 = $[60]; + } + return t30; +} + +/** + * Check bridge prerequisites. Returns an error message if a precondition + * fails, or null if all checks pass. Awaits GrowthBook init if the disk + * cache is stale, so a user who just became entitled (e.g. upgraded to Max, + * or the flag just launched) gets an accurate result on the first try. + */ +function _temp10(line, i_1) { + return {line}; +} +function _temp1(l) { + return l.length > 0; +} +function _temp0(i_0) { + return (i_0 - 1 + 3) % 3; +} +function _temp9(i) { + return (i + 1) % 3; +} +function _temp8(prev_0) { + return !prev_0; +} +function _temp7(prev) { + if (!prev.replBridgeEnabled) { + return prev; + } + return { + ...prev, + replBridgeEnabled: false, + replBridgeExplicit: false, + replBridgeOutboundOnly: false + }; +} +function _temp6(s_1) { + return s_1.replBridgeSessionActive; +} +function _temp5(s_0) { + return s_0.replBridgeConnectUrl; +} +function _temp4(s) { + return s.replBridgeSessionUrl; +} +async function checkBridgePrerequisites(): Promise { + // Check organization policy — remote control may be disabled + const { + waitForPolicyLimitsToLoad, + isPolicyAllowed + } = await import('../../services/policyLimits/index.js'); + await waitForPolicyLimitsToLoad(); + if (!isPolicyAllowed('allow_remote_control')) { + return "Remote Control is disabled by your organization's policy."; + } + const disabledReason = await getBridgeDisabledReason(); + if (disabledReason) { + return disabledReason; + } + + // Mirror the v1/v2 branching logic in initReplBridge: env-less (v2) is used + // only when the flag is on AND the session is not perpetual. In assistant + // mode (KAIROS) useReplBridge sets perpetual=true, which forces + // initReplBridge onto the v1 path — so the prerequisite check must match. + let useV2 = isEnvLessBridgeEnabled(); + if (feature('KAIROS') && useV2) { + const { + isAssistantMode + } = await import('../../assistant/index.js'); + if (isAssistantMode()) { + useV2 = false; + } + } + const versionError = useV2 ? await checkEnvLessBridgeMinVersion() : checkBridgeMinVersion(); + if (versionError) { + return versionError; + } + if (!getBridgeAccessToken()) { + return BRIDGE_LOGIN_INSTRUCTION; + } + logForDebugging('[bridge] Prerequisites passed, enabling bridge'); + return null; +} +export async function call(onDone: LocalJSXCommandOnDone, _context: ToolUseContext & LocalJSXCommandContext, args: string): Promise { + const name = args.trim() || undefined; + return ; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJmZWF0dXJlIiwidG9TdHJpbmciLCJxclRvU3RyaW5nIiwiUmVhY3QiLCJ1c2VFZmZlY3QiLCJ1c2VTdGF0ZSIsImdldEJyaWRnZUFjY2Vzc1Rva2VuIiwiY2hlY2tCcmlkZ2VNaW5WZXJzaW9uIiwiZ2V0QnJpZGdlRGlzYWJsZWRSZWFzb24iLCJpc0Vudkxlc3NCcmlkZ2VFbmFibGVkIiwiY2hlY2tFbnZMZXNzQnJpZGdlTWluVmVyc2lvbiIsIkJSSURHRV9MT0dJTl9JTlNUUlVDVElPTiIsIlJFTU9URV9DT05UUk9MX0RJU0NPTk5FQ1RFRF9NU0ciLCJEaWFsb2ciLCJMaXN0SXRlbSIsInNob3VsZFNob3dSZW1vdGVDYWxsb3V0IiwidXNlUmVnaXN0ZXJPdmVybGF5IiwiQm94IiwiVGV4dCIsInVzZUtleWJpbmRpbmdzIiwiQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyIsImxvZ0V2ZW50IiwidXNlQXBwU3RhdGUiLCJ1c2VTZXRBcHBTdGF0ZSIsIlRvb2xVc2VDb250ZXh0IiwiTG9jYWxKU1hDb21tYW5kQ29udGV4dCIsIkxvY2FsSlNYQ29tbWFuZE9uRG9uZSIsImxvZ0ZvckRlYnVnZ2luZyIsIlByb3BzIiwib25Eb25lIiwibmFtZSIsIkJyaWRnZVRvZ2dsZSIsInQwIiwiJCIsIl9jIiwic2V0QXBwU3RhdGUiLCJyZXBsQnJpZGdlQ29ubmVjdGVkIiwiX3RlbXAiLCJyZXBsQnJpZGdlRW5hYmxlZCIsIl90ZW1wMiIsInJlcGxCcmlkZ2VPdXRib3VuZE9ubHkiLCJfdGVtcDMiLCJzaG93RGlzY29ubmVjdERpYWxvZyIsInNldFNob3dEaXNjb25uZWN0RGlhbG9nIiwidDEiLCJjYW5jZWxsZWQiLCJlcnJvciIsImNoZWNrQnJpZGdlUHJlcmVxdWlzaXRlcyIsImFjdGlvbiIsImRpc3BsYXkiLCJwcmV2Iiwic2hvd1JlbW90ZUNhbGxvdXQiLCJyZXBsQnJpZGdlSW5pdGlhbE5hbWUiLCJwcmV2XzAiLCJyZXBsQnJpZGdlRXhwbGljaXQiLCJ0MiIsIlN5bWJvbCIsImZvciIsInQzIiwic18xIiwicyIsInNfMCIsIkJyaWRnZURpc2Nvbm5lY3REaWFsb2ciLCJzZXNzaW9uVXJsIiwiX3RlbXA0IiwiY29ubmVjdFVybCIsIl90ZW1wNSIsInNlc3Npb25BY3RpdmUiLCJfdGVtcDYiLCJmb2N1c0luZGV4Iiwic2V0Rm9jdXNJbmRleCIsInNob3dRUiIsInNldFNob3dRUiIsInFyVGV4dCIsInNldFFyVGV4dCIsImRpc3BsYXlVcmwiLCJ0eXBlIiwiZXJyb3JDb3JyZWN0aW9uTGV2ZWwiLCJzbWFsbCIsInRoZW4iLCJjYXRjaCIsImhhbmRsZURpc2Nvbm5lY3QiLCJfdGVtcDciLCJ0NCIsImhhbmRsZVNob3dRUiIsIl90ZW1wOCIsInQ1IiwiaGFuZGxlQ29udGludWUiLCJ1bmRlZmluZWQiLCJ0NiIsInQ3IiwiX3RlbXA5IiwiX3RlbXAwIiwidDgiLCJzZWxlY3Q6YWNjZXB0IiwidDkiLCJjb250ZXh0IiwiVDAiLCJUMSIsInQxMCIsInQxMSIsInQxMiIsInQxMyIsInQxNCIsInQxNSIsInQxNiIsInFyTGluZXMiLCJzcGxpdCIsImZpbHRlciIsIl90ZW1wMSIsInQxNyIsImxlbmd0aCIsIm1hcCIsIl90ZW1wMTAiLCJ0MTgiLCJ0MTkiLCJ0MjAiLCJ0MjEiLCJ0MjIiLCJ0MjMiLCJ0MjQiLCJ0MjUiLCJ0MjYiLCJ0MjciLCJ0MjgiLCJ0MjkiLCJ0MzAiLCJsaW5lIiwiaV8xIiwiaSIsImwiLCJpXzAiLCJyZXBsQnJpZGdlU2Vzc2lvbkFjdGl2ZSIsInJlcGxCcmlkZ2VDb25uZWN0VXJsIiwicmVwbEJyaWRnZVNlc3Npb25VcmwiLCJQcm9taXNlIiwid2FpdEZvclBvbGljeUxpbWl0c1RvTG9hZCIsImlzUG9saWN5QWxsb3dlZCIsImRpc2FibGVkUmVhc29uIiwidXNlVjIiLCJpc0Fzc2lzdGFudE1vZGUiLCJ2ZXJzaW9uRXJyb3IiLCJjYWxsIiwiX2NvbnRleHQiLCJhcmdzIiwiUmVhY3ROb2RlIiwidHJpbSJdLCJzb3VyY2VzIjpbImJyaWRnZS50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgZmVhdHVyZSB9IGZyb20gJ2J1bjpidW5kbGUnXG5pbXBvcnQgeyB0b1N0cmluZyBhcyBxclRvU3RyaW5nIH0gZnJvbSAncXJjb2RlJ1xuaW1wb3J0ICogYXMgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyB1c2VFZmZlY3QsIHVzZVN0YXRlIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBnZXRCcmlkZ2VBY2Nlc3NUb2tlbiB9IGZyb20gJy4uLy4uL2JyaWRnZS9icmlkZ2VDb25maWcuanMnXG5pbXBvcnQge1xuICBjaGVja0JyaWRnZU1pblZlcnNpb24sXG4gIGdldEJyaWRnZURpc2FibGVkUmVhc29uLFxuICBpc0Vudkxlc3NCcmlkZ2VFbmFibGVkLFxufSBmcm9tICcuLi8uLi9icmlkZ2UvYnJpZGdlRW5hYmxlZC5qcydcbmltcG9ydCB7IGNoZWNrRW52TGVzc0JyaWRnZU1pblZlcnNpb24gfSBmcm9tICcuLi8uLi9icmlkZ2UvZW52TGVzc0JyaWRnZUNvbmZpZy5qcydcbmltcG9ydCB7XG4gIEJSSURHRV9MT0dJTl9JTlNUUlVDVElPTixcbiAgUkVNT1RFX0NPTlRST0xfRElTQ09OTkVDVEVEX01TRyxcbn0gZnJvbSAnLi4vLi4vYnJpZGdlL3R5cGVzLmpzJ1xuaW1wb3J0IHsgRGlhbG9nIH0gZnJvbSAnLi4vLi4vY29tcG9uZW50cy9kZXNpZ24tc3lzdGVtL0RpYWxvZy5qcydcbmltcG9ydCB7IExpc3RJdGVtIH0gZnJvbSAnLi4vLi4vY29tcG9uZW50cy9kZXNpZ24tc3lzdGVtL0xpc3RJdGVtLmpzJ1xuaW1wb3J0IHsgc2hvdWxkU2hvd1JlbW90ZUNhbGxvdXQgfSBmcm9tICcuLi8uLi9jb21wb25lbnRzL1JlbW90ZUNhbGxvdXQuanMnXG5pbXBvcnQgeyB1c2VSZWdpc3Rlck92ZXJsYXkgfSBmcm9tICcuLi8uLi9jb250ZXh0L292ZXJsYXlDb250ZXh0LmpzJ1xuaW1wb3J0IHsgQm94LCBUZXh0IH0gZnJvbSAnLi4vLi4vaW5rLmpzJ1xuaW1wb3J0IHsgdXNlS2V5YmluZGluZ3MgfSBmcm9tICcuLi8uLi9rZXliaW5kaW5ncy91c2VLZXliaW5kaW5nLmpzJ1xuaW1wb3J0IHtcbiAgdHlwZSBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICBsb2dFdmVudCxcbn0gZnJvbSAnLi4vLi4vc2VydmljZXMvYW5hbHl0aWNzL2luZGV4LmpzJ1xuaW1wb3J0IHsgdXNlQXBwU3RhdGUsIHVzZVNldEFwcFN0YXRlIH0gZnJvbSAnLi4vLi4vc3RhdGUvQXBwU3RhdGUuanMnXG5pbXBvcnQgdHlwZSB7IFRvb2xVc2VDb250ZXh0IH0gZnJvbSAnLi4vLi4vVG9vbC5qcydcbmltcG9ydCB0eXBlIHtcbiAgTG9jYWxKU1hDb21tYW5kQ29udGV4dCxcbiAgTG9jYWxKU1hDb21tYW5kT25Eb25lLFxufSBmcm9tICcuLi8uLi90eXBlcy9jb21tYW5kLmpzJ1xuaW1wb3J0IHsgbG9nRm9yRGVidWdnaW5nIH0gZnJvbSAnLi4vLi4vdXRpbHMvZGVidWcuanMnXG5cbnR5cGUgUHJvcHMgPSB7XG4gIG9uRG9uZTogTG9jYWxKU1hDb21tYW5kT25Eb25lXG4gIG5hbWU/OiBzdHJpbmdcbn1cblxuLyoqXG4gKiAvcmVtb3RlLWNvbnRyb2wgY29tbWFuZCDigJQgbWFuYWdlcyB0aGUgYmlkaXJlY3Rpb25hbCBicmlkZ2UgY29ubmVjdGlvbi5cbiAqXG4gKiBXaGVuIGVuYWJsZWQsIHNldHMgcmVwbEJyaWRnZUVuYWJsZWQgaW4gQXBwU3RhdGUsIHdoaWNoIHRyaWdnZXJzXG4gKiB1c2VSZXBsQnJpZGdlIGluIFJFUEwudHN4IHRvIGluaXRpYWxpemUgdGhlIGJyaWRnZSBjb25uZWN0aW9uLlxuICogVGhlIGJyaWRnZSByZWdpc3RlcnMgYW4gZW52aXJvbm1lbnQsIGNyZWF0ZXMgYSBzZXNzaW9uIHdpdGggdGhlIGN1cnJlbnRcbiAqIGNvbnZlcnNhdGlvbiwgcG9sbHMgZm9yIHdvcmssIGFuZCBjb25uZWN0cyBhbiBpbmdyZXNzIFdlYlNvY2tldCBmb3JcbiAqIGJpZGlyZWN0aW9uYWwgbWVzc2FnaW5nIGJldHdlZW4gdGhlIENMSSBhbmQgY2xhdWRlLmFpLlxuICpcbiAqIFJ1bm5pbmcgL3JlbW90ZS1jb250cm9sIHdoZW4gYWxyZWFkeSBjb25uZWN0ZWQgc2hvd3MgYSBkaWFsb2cgd2l0aCB0aGUgc2Vzc2lvblxuICogVVJMIGFuZCBvcHRpb25zIHRvIGRpc2Nvbm5lY3Qgb3IgY29udGludWUuXG4gKi9cbmZ1bmN0aW9uIEJyaWRnZVRvZ2dsZSh7IG9uRG9uZSwgbmFtZSB9OiBQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IHNldEFwcFN0YXRlID0gdXNlU2V0QXBwU3RhdGUoKVxuICBjb25zdCByZXBsQnJpZGdlQ29ubmVjdGVkID0gdXNlQXBwU3RhdGUocyA9PiBzLnJlcGxCcmlkZ2VDb25uZWN0ZWQpXG4gIGNvbnN0IHJlcGxCcmlkZ2VFbmFibGVkID0gdXNlQXBwU3RhdGUocyA9PiBzLnJlcGxCcmlkZ2VFbmFibGVkKVxuICBjb25zdCByZXBsQnJpZGdlT3V0Ym91bmRPbmx5ID0gdXNlQXBwU3RhdGUocyA9PiBzLnJlcGxCcmlkZ2VPdXRib3VuZE9ubHkpXG4gIGNvbnN0IFtzaG93RGlzY29ubmVjdERpYWxvZywgc2V0U2hvd0Rpc2Nvbm5lY3REaWFsb2ddID0gdXNlU3RhdGUoZmFsc2UpXG5cbiAgLy8gYmlvbWUtaWdub3JlIGxpbnQvY29ycmVjdG5lc3MvdXNlRXhoYXVzdGl2ZURlcGVuZGVuY2llczogYnJpZGdlIHN0YXJ0cyBvbmNlLCBzaG91bGQgbm90IHJlc3RhcnQgb24gc3RhdGUgY2hhbmdlc1xuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIC8vIElmIGFscmVhZHkgY29ubmVjdGVkIG9yIGVuYWJsZWQgaW4gZnVsbCBiaWRpcmVjdGlvbmFsIG1vZGUsIHNob3dcbiAgICAvLyBkaXNjb25uZWN0IGNvbmZpcm1hdGlvbi4gT3V0Ym91bmQtb25seSAoQ0NSIG1pcnJvcikgZG9lc24ndCBjb3VudCDigJRcbiAgICAvLyAvcmVtb3RlLWNvbnRyb2wgdXBncmFkZXMgaXQgdG8gZnVsbCBSQyBpbnN0ZWFkLlxuICAgIGlmICgocmVwbEJyaWRnZUNvbm5lY3RlZCB8fCByZXBsQnJpZGdlRW5hYmxlZCkgJiYgIXJlcGxCcmlkZ2VPdXRib3VuZE9ubHkpIHtcbiAgICAgIHNldFNob3dEaXNjb25uZWN0RGlhbG9nKHRydWUpXG4gICAgICByZXR1cm5cbiAgICB9XG5cbiAgICBsZXQgY2FuY2VsbGVkID0gZmFsc2VcbiAgICB2b2lkIChhc3luYyAoKSA9PiB7XG4gICAgICAvLyBQcmUtZmxpZ2h0IGNoZWNrcyBiZWZvcmUgZW5hYmxpbmcgKGF3YWl0cyBHcm93dGhCb29rIGluaXQgaWYgZGlza1xuICAgICAgLy8gY2FjaGUgaXMgc3RhbGUg4oCUIHNvIE1heCB1c2VycyBkb24ndCBnZXQgYSBmYWxzZSBcIm5vdCBlbmFibGVkXCIgZXJyb3IpXG4gICAgICBjb25zdCBlcnJvciA9IGF3YWl0IGNoZWNrQnJpZGdlUHJlcmVxdWlzaXRlcygpXG4gICAgICBpZiAoY2FuY2VsbGVkKSByZXR1cm5cbiAgICAgIGlmIChlcnJvcikge1xuICAgICAgICBsb2dFdmVudCgndGVuZ3VfYnJpZGdlX2NvbW1hbmQnLCB7XG4gICAgICAgICAgYWN0aW9uOlxuICAgICAgICAgICAgJ3ByZWZsaWdodF9mYWlsZWQnIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgIH0pXG4gICAgICAgIG9uRG9uZShlcnJvciwgeyBkaXNwbGF5OiAnc3lzdGVtJyB9KVxuICAgICAgICByZXR1cm5cbiAgICAgIH1cblxuICAgICAgLy8gU2hvdyBmaXJzdC10aW1lIHJlbW90ZSBkaWFsb2cgaWYgbm90IHlldCBzZWVuLlxuICAgICAgLy8gU3RvcmUgdGhlIG5hbWUgbm93IHNvIGl0J3MgaW4gQXBwU3RhdGUgd2hlbiB0aGUgY2FsbG91dCBoYW5kbGVyIGxhdGVyXG4gICAgICAvLyBlbmFibGVzIHRoZSBicmlkZ2UgKHRoZSBoYW5kbGVyIG9ubHkgc2V0cyByZXBsQnJpZGdlRW5hYmxlZCwgbm90IHRoZSBuYW1lKS5cbiAgICAgIGlmIChzaG91bGRTaG93UmVtb3RlQ2FsbG91dCgpKSB7XG4gICAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4ge1xuICAgICAgICAgIGlmIChwcmV2LnNob3dSZW1vdGVDYWxsb3V0KSByZXR1cm4gcHJldlxuICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAuLi5wcmV2LFxuICAgICAgICAgICAgc2hvd1JlbW90ZUNhbGxvdXQ6IHRydWUsXG4gICAgICAgICAgICByZXBsQnJpZGdlSW5pdGlhbE5hbWU6IG5hbWUsXG4gICAgICAgICAgfVxuICAgICAgICB9KVxuICAgICAgICBvbkRvbmUoJycsIHsgZGlzcGxheTogJ3N5c3RlbScgfSlcbiAgICAgICAgcmV0dXJuXG4gICAgICB9XG5cbiAgICAgIC8vIEVuYWJsZSB0aGUgYnJpZGdlIOKAlCB1c2VSZXBsQnJpZGdlIGluIFJFUEwudHN4IGhhbmRsZXMgdGhlIHJlc3Q6XG4gICAgICAvLyByZWdpc3RlcnMgZW52aXJvbm1lbnQsIGNyZWF0ZXMgc2Vzc2lvbiB3aXRoIGNvbnZlcnNhdGlvbiwgY29ubmVjdHMgV2ViU29ja2V0XG4gICAgICBsb2dFdmVudCgndGVuZ3VfYnJpZGdlX2NvbW1hbmQnLCB7XG4gICAgICAgIGFjdGlvbjpcbiAgICAgICAgICAnY29ubmVjdCcgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICAgIH0pXG4gICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+IHtcbiAgICAgICAgaWYgKHByZXYucmVwbEJyaWRnZUVuYWJsZWQgJiYgIXByZXYucmVwbEJyaWRnZU91dGJvdW5kT25seSkgcmV0dXJuIHByZXZcbiAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAuLi5wcmV2LFxuICAgICAgICAgIHJlcGxCcmlkZ2VFbmFibGVkOiB0cnVlLFxuICAgICAgICAgIHJlcGxCcmlkZ2VFeHBsaWNpdDogdHJ1ZSxcbiAgICAgICAgICByZXBsQnJpZGdlT3V0Ym91bmRPbmx5OiBmYWxzZSxcbiAgICAgICAgICByZXBsQnJpZGdlSW5pdGlhbE5hbWU6IG5hbWUsXG4gICAgICAgIH1cbiAgICAgIH0pXG4gICAgICBvbkRvbmUoJ1JlbW90ZSBDb250cm9sIGNvbm5lY3RpbmdcXHUyMDI2Jywge1xuICAgICAgICBkaXNwbGF5OiAnc3lzdGVtJyxcbiAgICAgIH0pXG4gICAgfSkoKVxuXG4gICAgcmV0dXJuICgpID0+IHtcbiAgICAgIGNhbmNlbGxlZCA9IHRydWVcbiAgICB9XG4gIH0sIFtdKSAvLyBlc2xpbnQtZGlzYWJsZS1saW5lIHJlYWN0LWhvb2tzL2V4aGF1c3RpdmUtZGVwcyAtLSBydW4gb25jZSBvbiBtb3VudFxuXG4gIGlmIChzaG93RGlzY29ubmVjdERpYWxvZykge1xuICAgIHJldHVybiA8QnJpZGdlRGlzY29ubmVjdERpYWxvZyBvbkRvbmU9e29uRG9uZX0gLz5cbiAgfVxuXG4gIHJldHVybiBudWxsXG59XG5cbi8qKlxuICogRGlhbG9nIHNob3duIHdoZW4gL3JlbW90ZS1jb250cm9sIGlzIHVzZWQgd2hpbGUgdGhlIGJyaWRnZSBpcyBhbHJlYWR5IGNvbm5lY3RlZC5cbiAqIFNob3dzIHRoZSBzZXNzaW9uIFVSTCBhbmQgbGV0cyB0aGUgdXNlciBkaXNjb25uZWN0IG9yIGNvbnRpbnVlLlxuICovXG5mdW5jdGlvbiBCcmlkZ2VEaXNjb25uZWN0RGlhbG9nKHsgb25Eb25lIH06IFByb3BzKTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgdXNlUmVnaXN0ZXJPdmVybGF5KCdicmlkZ2UtZGlzY29ubmVjdC1kaWFsb2cnKVxuICBjb25zdCBzZXRBcHBTdGF0ZSA9IHVzZVNldEFwcFN0YXRlKClcbiAgY29uc3Qgc2Vzc2lvblVybCA9IHVzZUFwcFN0YXRlKHMgPT4gcy5yZXBsQnJpZGdlU2Vzc2lvblVybClcbiAgY29uc3QgY29ubmVjdFVybCA9IHVzZUFwcFN0YXRlKHMgPT4gcy5yZXBsQnJpZGdlQ29ubmVjdFVybClcbiAgY29uc3Qgc2Vzc2lvbkFjdGl2ZSA9IHVzZUFwcFN0YXRlKHMgPT4gcy5yZXBsQnJpZGdlU2Vzc2lvbkFjdGl2ZSlcbiAgY29uc3QgW2ZvY3VzSW5kZXgsIHNldEZvY3VzSW5kZXhdID0gdXNlU3RhdGUoMilcbiAgY29uc3QgW3Nob3dRUiwgc2V0U2hvd1FSXSA9IHVzZVN0YXRlKGZhbHNlKVxuICBjb25zdCBbcXJUZXh0LCBzZXRRclRleHRdID0gdXNlU3RhdGUoJycpXG5cbiAgY29uc3QgZGlzcGxheVVybCA9IHNlc3Npb25BY3RpdmUgPyBzZXNzaW9uVXJsIDogY29ubmVjdFVybFxuXG4gIC8vIEdlbmVyYXRlIFFSIGNvZGUgd2hlbiBVUkwgY2hhbmdlcyBvciBRUiBpcyB0b2dnbGVkIG9uXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKCFzaG93UVIgfHwgIWRpc3BsYXlVcmwpIHtcbiAgICAgIHNldFFyVGV4dCgnJylcbiAgICAgIHJldHVyblxuICAgIH1cbiAgICBxclRvU3RyaW5nKGRpc3BsYXlVcmwsIHtcbiAgICAgIHR5cGU6ICd1dGY4JyxcbiAgICAgIGVycm9yQ29ycmVjdGlvbkxldmVsOiAnTCcsXG4gICAgICBzbWFsbDogdHJ1ZSxcbiAgICB9KVxuICAgICAgLnRoZW4oc2V0UXJUZXh0KVxuICAgICAgLmNhdGNoKCgpID0+IHNldFFyVGV4dCgnJykpXG4gIH0sIFtzaG93UVIsIGRpc3BsYXlVcmxdKVxuXG4gIGZ1bmN0aW9uIGhhbmRsZURpc2Nvbm5lY3QoKTogdm9pZCB7XG4gICAgc2V0QXBwU3RhdGUocHJldiA9PiB7XG4gICAgICBpZiAoIXByZXYucmVwbEJyaWRnZUVuYWJsZWQpIHJldHVybiBwcmV2XG4gICAgICByZXR1cm4ge1xuICAgICAgICAuLi5wcmV2LFxuICAgICAgICByZXBsQnJpZGdlRW5hYmxlZDogZmFsc2UsXG4gICAgICAgIHJlcGxCcmlkZ2VFeHBsaWNpdDogZmFsc2UsXG4gICAgICAgIHJlcGxCcmlkZ2VPdXRib3VuZE9ubHk6IGZhbHNlLFxuICAgICAgfVxuICAgIH0pXG4gICAgbG9nRXZlbnQoJ3Rlbmd1X2JyaWRnZV9jb21tYW5kJywge1xuICAgICAgYWN0aW9uOlxuICAgICAgICAnZGlzY29ubmVjdCcgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICB9KVxuICAgIG9uRG9uZShSRU1PVEVfQ09OVFJPTF9ESVNDT05ORUNURURfTVNHLCB7IGRpc3BsYXk6ICdzeXN0ZW0nIH0pXG4gIH1cblxuICBmdW5jdGlvbiBoYW5kbGVTaG93UVIoKTogdm9pZCB7XG4gICAgc2V0U2hvd1FSKHByZXYgPT4gIXByZXYpXG4gIH1cblxuICBmdW5jdGlvbiBoYW5kbGVDb250aW51ZSgpOiB2b2lkIHtcbiAgICBvbkRvbmUodW5kZWZpbmVkLCB7IGRpc3BsYXk6ICdza2lwJyB9KVxuICB9XG5cbiAgY29uc3QgSVRFTV9DT1VOVCA9IDNcblxuICB1c2VLZXliaW5kaW5ncyhcbiAgICB7XG4gICAgICAnc2VsZWN0Om5leHQnOiAoKSA9PiBzZXRGb2N1c0luZGV4KGkgPT4gKGkgKyAxKSAlIElURU1fQ09VTlQpLFxuICAgICAgJ3NlbGVjdDpwcmV2aW91cyc6ICgpID0+XG4gICAgICAgIHNldEZvY3VzSW5kZXgoaSA9PiAoaSAtIDEgKyBJVEVNX0NPVU5UKSAlIElURU1fQ09VTlQpLFxuICAgICAgJ3NlbGVjdDphY2NlcHQnOiAoKSA9PiB7XG4gICAgICAgIGlmIChmb2N1c0luZGV4ID09PSAwKSB7XG4gICAgICAgICAgaGFuZGxlRGlzY29ubmVjdCgpXG4gICAgICAgIH0gZWxzZSBpZiAoZm9jdXNJbmRleCA9PT0gMSkge1xuICAgICAgICAgIGhhbmRsZVNob3dRUigpXG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgaGFuZGxlQ29udGludWUoKVxuICAgICAgICB9XG4gICAgICB9LFxuICAgIH0sXG4gICAgeyBjb250ZXh0OiAnU2VsZWN0JyB9LFxuICApXG5cbiAgY29uc3QgcXJMaW5lcyA9IHFyVGV4dCA/IHFyVGV4dC5zcGxpdCgnXFxuJykuZmlsdGVyKGwgPT4gbC5sZW5ndGggPiAwKSA6IFtdXG5cbiAgcmV0dXJuIChcbiAgICA8RGlhbG9nIHRpdGxlPVwiUmVtb3RlIENvbnRyb2xcIiBvbkNhbmNlbD17aGFuZGxlQ29udGludWV9IGhpZGVJbnB1dEd1aWRlPlxuICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCIgZ2FwPXsxfT5cbiAgICAgICAgPFRleHQ+XG4gICAgICAgICAgVGhpcyBzZXNzaW9uIGlzIGF2YWlsYWJsZSB2aWEgUmVtb3RlIENvbnRyb2xcbiAgICAgICAgICB7ZGlzcGxheVVybCA/IGAgYXQgJHtkaXNwbGF5VXJsfWAgOiAnJ30uXG4gICAgICAgIDwvVGV4dD5cbiAgICAgICAge3Nob3dRUiAmJiBxckxpbmVzLmxlbmd0aCA+IDAgJiYgKFxuICAgICAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiPlxuICAgICAgICAgICAge3FyTGluZXMubWFwKChsaW5lLCBpKSA9PiAoXG4gICAgICAgICAgICAgIDxUZXh0IGtleT17aX0+e2xpbmV9PC9UZXh0PlxuICAgICAgICAgICAgKSl9XG4gICAgICAgICAgPC9Cb3g+XG4gICAgICAgICl9XG4gICAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiPlxuICAgICAgICAgIDxMaXN0SXRlbSBpc0ZvY3VzZWQ9e2ZvY3VzSW5kZXggPT09IDB9PlxuICAgICAgICAgICAgPFRleHQ+RGlzY29ubmVjdCB0aGlzIHNlc3Npb248L1RleHQ+XG4gICAgICAgICAgPC9MaXN0SXRlbT5cbiAgICAgICAgICA8TGlzdEl0ZW0gaXNGb2N1c2VkPXtmb2N1c0luZGV4ID09PSAxfT5cbiAgICAgICAgICAgIDxUZXh0PntzaG93UVIgPyAnSGlkZSBRUiBjb2RlJyA6ICdTaG93IFFSIGNvZGUnfTwvVGV4dD5cbiAgICAgICAgICA8L0xpc3RJdGVtPlxuICAgICAgICAgIDxMaXN0SXRlbSBpc0ZvY3VzZWQ9e2ZvY3VzSW5kZXggPT09IDJ9PlxuICAgICAgICAgICAgPFRleHQ+Q29udGludWU8L1RleHQ+XG4gICAgICAgICAgPC9MaXN0SXRlbT5cbiAgICAgICAgPC9Cb3g+XG4gICAgICAgIDxUZXh0IGRpbUNvbG9yPkVudGVyIHRvIHNlbGVjdCDCtyBFc2MgdG8gY29udGludWU8L1RleHQ+XG4gICAgICA8L0JveD5cbiAgICA8L0RpYWxvZz5cbiAgKVxufVxuXG4vKipcbiAqIENoZWNrIGJyaWRnZSBwcmVyZXF1aXNpdGVzLiBSZXR1cm5zIGFuIGVycm9yIG1lc3NhZ2UgaWYgYSBwcmVjb25kaXRpb25cbiAqIGZhaWxzLCBvciBudWxsIGlmIGFsbCBjaGVja3MgcGFzcy4gQXdhaXRzIEdyb3d0aEJvb2sgaW5pdCBpZiB0aGUgZGlza1xuICogY2FjaGUgaXMgc3RhbGUsIHNvIGEgdXNlciB3aG8ganVzdCBiZWNhbWUgZW50aXRsZWQgKGUuZy4gdXBncmFkZWQgdG8gTWF4LFxuICogb3IgdGhlIGZsYWcganVzdCBsYXVuY2hlZCkgZ2V0cyBhbiBhY2N1cmF0ZSByZXN1bHQgb24gdGhlIGZpcnN0IHRyeS5cbiAqL1xuYXN5bmMgZnVuY3Rpb24gY2hlY2tCcmlkZ2VQcmVyZXF1aXNpdGVzKCk6IFByb21pc2U8c3RyaW5nIHwgbnVsbD4ge1xuICAvLyBDaGVjayBvcmdhbml6YXRpb24gcG9saWN5IOKAlCByZW1vdGUgY29udHJvbCBtYXkgYmUgZGlzYWJsZWRcbiAgY29uc3QgeyB3YWl0Rm9yUG9saWN5TGltaXRzVG9Mb2FkLCBpc1BvbGljeUFsbG93ZWQgfSA9IGF3YWl0IGltcG9ydChcbiAgICAnLi4vLi4vc2VydmljZXMvcG9saWN5TGltaXRzL2luZGV4LmpzJ1xuICApXG4gIGF3YWl0IHdhaXRGb3JQb2xpY3lMaW1pdHNUb0xvYWQoKVxuICBpZiAoIWlzUG9saWN5QWxsb3dlZCgnYWxsb3dfcmVtb3RlX2NvbnRyb2wnKSkge1xuICAgIHJldHVybiBcIlJlbW90ZSBDb250cm9sIGlzIGRpc2FibGVkIGJ5IHlvdXIgb3JnYW5pemF0aW9uJ3MgcG9saWN5LlwiXG4gIH1cblxuICBjb25zdCBkaXNhYmxlZFJlYXNvbiA9IGF3YWl0IGdldEJyaWRnZURpc2FibGVkUmVhc29uKClcbiAgaWYgKGRpc2FibGVkUmVhc29uKSB7XG4gICAgcmV0dXJuIGRpc2FibGVkUmVhc29uXG4gIH1cblxuICAvLyBNaXJyb3IgdGhlIHYxL3YyIGJyYW5jaGluZyBsb2dpYyBpbiBpbml0UmVwbEJyaWRnZTogZW52LWxlc3MgKHYyKSBpcyB1c2VkXG4gIC8vIG9ubHkgd2hlbiB0aGUgZmxhZyBpcyBvbiBBTkQgdGhlIHNlc3Npb24gaXMgbm90IHBlcnBldHVhbC4gIEluIGFzc2lzdGFudFxuICAvLyBtb2RlIChLQUlST1MpIHVzZVJlcGxCcmlkZ2Ugc2V0cyBwZXJwZXR1YWw9dHJ1ZSwgd2hpY2ggZm9yY2VzXG4gIC8vIGluaXRSZXBsQnJpZGdlIG9udG8gdGhlIHYxIHBhdGgg4oCUIHNvIHRoZSBwcmVyZXF1aXNpdGUgY2hlY2sgbXVzdCBtYXRjaC5cbiAgbGV0IHVzZVYyID0gaXNFbnZMZXNzQnJpZGdlRW5hYmxlZCgpXG4gIGlmIChmZWF0dXJlKCdLQUlST1MnKSAmJiB1c2VWMikge1xuICAgIGNvbnN0IHsgaXNBc3Npc3RhbnRNb2RlIH0gPSBhd2FpdCBpbXBvcnQoJy4uLy4uL2Fzc2lzdGFudC9pbmRleC5qcycpXG4gICAgaWYgKGlzQXNzaXN0YW50TW9kZSgpKSB7XG4gICAgICB1c2VWMiA9IGZhbHNlXG4gICAgfVxuICB9XG4gIGNvbnN0IHZlcnNpb25FcnJvciA9IHVzZVYyXG4gICAgPyBhd2FpdCBjaGVja0Vudkxlc3NCcmlkZ2VNaW5WZXJzaW9uKClcbiAgICA6IGNoZWNrQnJpZGdlTWluVmVyc2lvbigpXG4gIGlmICh2ZXJzaW9uRXJyb3IpIHtcbiAgICByZXR1cm4gdmVyc2lvbkVycm9yXG4gIH1cblxuICBpZiAoIWdldEJyaWRnZUFjY2Vzc1Rva2VuKCkpIHtcbiAgICByZXR1cm4gQlJJREdFX0xPR0lOX0lOU1RSVUNUSU9OXG4gIH1cblxuICBsb2dGb3JEZWJ1Z2dpbmcoJ1ticmlkZ2VdIFByZXJlcXVpc2l0ZXMgcGFzc2VkLCBlbmFibGluZyBicmlkZ2UnKVxuICByZXR1cm4gbnVsbFxufVxuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gY2FsbChcbiAgb25Eb25lOiBMb2NhbEpTWENvbW1hbmRPbkRvbmUsXG4gIF9jb250ZXh0OiBUb29sVXNlQ29udGV4dCAmIExvY2FsSlNYQ29tbWFuZENvbnRleHQsXG4gIGFyZ3M6IHN0cmluZyxcbik6IFByb21pc2U8UmVhY3QuUmVhY3ROb2RlPiB7XG4gIGNvbnN0IG5hbWUgPSBhcmdzLnRyaW0oKSB8fCB1bmRlZmluZWRcbiAgcmV0dXJuIDxCcmlkZ2VUb2dnbGUgb25Eb25lPXtvbkRvbmV9IG5hbWU9e25hbWV9IC8+XG59XG4iXSwibWFwcGluZ3MiOiI7QUFBQSxTQUFTQSxPQUFPLFFBQVEsWUFBWTtBQUNwQyxTQUFTQyxRQUFRLElBQUlDLFVBQVUsUUFBUSxRQUFRO0FBQy9DLE9BQU8sS0FBS0MsS0FBSyxNQUFNLE9BQU87QUFDOUIsU0FBU0MsU0FBUyxFQUFFQyxRQUFRLFFBQVEsT0FBTztBQUMzQyxTQUFTQyxvQkFBb0IsUUFBUSw4QkFBOEI7QUFDbkUsU0FDRUMscUJBQXFCLEVBQ3JCQyx1QkFBdUIsRUFDdkJDLHNCQUFzQixRQUNqQiwrQkFBK0I7QUFDdEMsU0FBU0MsNEJBQTRCLFFBQVEscUNBQXFDO0FBQ2xGLFNBQ0VDLHdCQUF3QixFQUN4QkMsK0JBQStCLFFBQzFCLHVCQUF1QjtBQUM5QixTQUFTQyxNQUFNLFFBQVEsMENBQTBDO0FBQ2pFLFNBQVNDLFFBQVEsUUFBUSw0Q0FBNEM7QUFDckUsU0FBU0MsdUJBQXVCLFFBQVEsbUNBQW1DO0FBQzNFLFNBQVNDLGtCQUFrQixRQUFRLGlD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 \ No newline at end of file diff --git a/src/commands/bridge/index.ts b/src/commands/bridge/index.ts new file mode 100644 index 0000000..5b6fc44 --- /dev/null +++ b/src/commands/bridge/index.ts @@ -0,0 +1,26 @@ +import { feature } from 'bun:bundle' +import { isBridgeEnabled } from '../../bridge/bridgeEnabled.js' +import type { Command } from '../../commands.js' + +function isEnabled(): boolean { + if (!feature('BRIDGE_MODE')) { + return false + } + return isBridgeEnabled() +} + +const bridge = { + type: 'local-jsx', + name: 'remote-control', + aliases: ['rc'], + description: 'Connect this terminal for remote-control sessions', + argumentHint: '[name]', + isEnabled, + get isHidden() { + return !isEnabled() + }, + immediate: true, + load: () => import('./bridge.js'), +} satisfies Command + +export default bridge diff --git a/src/commands/brief.ts b/src/commands/brief.ts new file mode 100644 index 0000000..d37ffd0 --- /dev/null +++ b/src/commands/brief.ts @@ -0,0 +1,130 @@ +import { feature } from 'bun:bundle' +import { z } from 'zod/v4' +import { getKairosActive, setUserMsgOptIn } from '../bootstrap/state.js' +import { getFeatureValue_CACHED_MAY_BE_STALE } from '../services/analytics/growthbook.js' +import { + type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + logEvent, +} from '../services/analytics/index.js' +import type { ToolUseContext } from '../Tool.js' +import { isBriefEntitled } from '../tools/BriefTool/BriefTool.js' +import { BRIEF_TOOL_NAME } from '../tools/BriefTool/prompt.js' +import type { + Command, + LocalJSXCommandContext, + LocalJSXCommandOnDone, +} from '../types/command.js' +import { lazySchema } from '../utils/lazySchema.js' + +// Zod guards against fat-fingered GB pushes (same pattern as pollConfig.ts / +// cronScheduler.ts). A malformed config falls back to DEFAULT_BRIEF_CONFIG +// entirely rather than being partially trusted. +const briefConfigSchema = lazySchema(() => + z.object({ + enable_slash_command: z.boolean(), + }), +) +type BriefConfig = z.infer> + +const DEFAULT_BRIEF_CONFIG: BriefConfig = { + enable_slash_command: false, +} + +// No TTL — this gate controls slash-command *visibility*, not a kill switch. +// CACHED_MAY_BE_STALE still has one background-update flip (first call kicks +// off fetch; second call sees fresh value), but no additional flips after that. +// The tool-availability gate (tengu_kairos_brief in isBriefEnabled) keeps its +// 5-min TTL because that one IS a kill switch. +function getBriefConfig(): BriefConfig { + const raw = getFeatureValue_CACHED_MAY_BE_STALE( + 'tengu_kairos_brief_config', + DEFAULT_BRIEF_CONFIG, + ) + const parsed = briefConfigSchema().safeParse(raw) + return parsed.success ? parsed.data : DEFAULT_BRIEF_CONFIG +} + +const brief = { + type: 'local-jsx', + name: 'brief', + description: 'Toggle brief-only mode', + isEnabled: () => { + if (feature('KAIROS') || feature('KAIROS_BRIEF')) { + return getBriefConfig().enable_slash_command + } + return false + }, + immediate: true, + load: () => + Promise.resolve({ + async call( + onDone: LocalJSXCommandOnDone, + context: ToolUseContext & LocalJSXCommandContext, + ): Promise { + const current = context.getAppState().isBriefOnly + const newState = !current + + // Entitlement check only gates the on-transition — off is always + // allowed so a user whose GB gate flipped mid-session isn't stuck. + if (newState && !isBriefEntitled()) { + logEvent('tengu_brief_mode_toggled', { + enabled: false, + gated: true, + source: + 'slash_command' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + onDone('Brief tool is not enabled for your account', { + display: 'system', + }) + return null + } + + // Two-way: userMsgOptIn tracks isBriefOnly so the tool is available + // exactly when brief mode is on. This invalidates prompt cache on + // each toggle (tool list changes), but a stale tool list is worse — + // when /brief is enabled mid-session the model was previously left + // without the tool, emitting plain text the filter hides. + setUserMsgOptIn(newState) + + context.setAppState(prev => { + if (prev.isBriefOnly === newState) return prev + return { ...prev, isBriefOnly: newState } + }) + + logEvent('tengu_brief_mode_toggled', { + enabled: newState, + gated: false, + source: + 'slash_command' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + + // The tool list change alone isn't a strong enough signal mid-session + // (model may keep emitting plain text from inertia, or keep calling a + // tool that just vanished). Inject an explicit reminder into the next + // turn's context so the transition is unambiguous. + // Skip when Kairos is active: isBriefEnabled() short-circuits on + // getKairosActive() so the tool never actually leaves the list, and + // the Kairos system prompt already mandates SendUserMessage. + // Inline wrap — importing wrapInSystemReminder from + // utils/messages.ts pulls constants/xml.ts into the bridge SDK bundle + // via this module's import chain, tripping the excluded-strings check. + const metaMessages = getKairosActive() + ? undefined + : [ + `\n${ + newState + ? `Brief mode is now enabled. Use the ${BRIEF_TOOL_NAME} tool for all user-facing output — plain text outside it is hidden from the user's view.` + : `Brief mode is now disabled. The ${BRIEF_TOOL_NAME} tool is no longer available — reply with plain text.` + }\n`, + ] + + onDone( + newState ? 'Brief-only mode enabled' : 'Brief-only mode disabled', + { display: 'system', metaMessages }, + ) + return null + }, + }), +} satisfies Command + +export default brief diff --git a/src/commands/btw/btw.tsx b/src/commands/btw/btw.tsx new file mode 100644 index 0000000..f3c1c5f --- /dev/null +++ b/src/commands/btw/btw.tsx @@ -0,0 +1,243 @@ +import { c as _c } from "react/compiler-runtime"; +import * as React from 'react'; +import { useEffect, useRef, useState } from 'react'; +import { useInterval } from 'usehooks-ts'; +import type { CommandResultDisplay } from '../../commands.js'; +import { Markdown } from '../../components/Markdown.js'; +import { SpinnerGlyph } from '../../components/Spinner/SpinnerGlyph.js'; +import { DOWN_ARROW, UP_ARROW } from '../../constants/figures.js'; +import { getSystemPrompt } from '../../constants/prompts.js'; +import { useModalOrTerminalSize } from '../../context/modalContext.js'; +import { getSystemContext, getUserContext } from '../../context.js'; +import { useTerminalSize } from '../../hooks/useTerminalSize.js'; +import ScrollBox, { type ScrollBoxHandle } from '../../ink/components/ScrollBox.js'; +import type { KeyboardEvent } from '../../ink/events/keyboard-event.js'; +import { Box, Text } from '../../ink.js'; +import type { LocalJSXCommandOnDone } from '../../types/command.js'; +import type { Message } from '../../types/message.js'; +import { createAbortController } from '../../utils/abortController.js'; +import { saveGlobalConfig } from '../../utils/config.js'; +import { errorMessage } from '../../utils/errors.js'; +import { type CacheSafeParams, getLastCacheSafeParams } from '../../utils/forkedAgent.js'; +import { getMessagesAfterCompactBoundary } from '../../utils/messages.js'; +import type { ProcessUserInputContext } from '../../utils/processUserInput/processUserInput.js'; +import { runSideQuestion } from '../../utils/sideQuestion.js'; +import { asSystemPrompt } from '../../utils/systemPromptType.js'; +type BtwComponentProps = { + question: string; + context: ProcessUserInputContext; + onDone: (result?: string, options?: { + display?: CommandResultDisplay; + }) => void; +}; +const CHROME_ROWS = 5; +const OUTER_CHROME_ROWS = 6; +const SCROLL_LINES = 3; +function BtwSideQuestion(t0) { + const $ = _c(25); + const { + question, + context, + onDone + } = t0; + const [response, setResponse] = useState(null); + const [error, setError] = useState(null); + const [frame, setFrame] = useState(0); + const scrollRef = useRef(null); + const { + rows + } = useModalOrTerminalSize(useTerminalSize()); + let t1; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t1 = () => setFrame(_temp); + $[0] = t1; + } else { + t1 = $[0]; + } + useInterval(t1, response || error ? null : 80); + let t2; + if ($[1] !== onDone) { + t2 = function handleKeyDown(e) { + if (e.key === "escape" || e.key === "return" || e.key === " " || e.ctrl && (e.key === "c" || e.key === "d")) { + e.preventDefault(); + onDone(undefined, { + display: "skip" + }); + return; + } + if (e.key === "up" || e.ctrl && e.key === "p") { + e.preventDefault(); + scrollRef.current?.scrollBy(-SCROLL_LINES); + } + if (e.key === "down" || e.ctrl && e.key === "n") { + e.preventDefault(); + scrollRef.current?.scrollBy(SCROLL_LINES); + } + }; + $[1] = onDone; + $[2] = t2; + } else { + t2 = $[2]; + } + const handleKeyDown = t2; + let t3; + let t4; + if ($[3] !== context || $[4] !== question) { + t3 = () => { + const abortController = createAbortController(); + const fetchResponse = async function fetchResponse() { + ; + try { + const cacheSafeParams = await buildCacheSafeParams(context); + const result = await runSideQuestion({ + question, + cacheSafeParams + }); + if (!abortController.signal.aborted) { + if (result.response) { + setResponse(result.response); + } else { + setError("No response received"); + } + } + } catch (t5) { + const err = t5; + if (!abortController.signal.aborted) { + setError(errorMessage(err) || "Failed to get response"); + } + } + }; + fetchResponse(); + return () => { + abortController.abort(); + }; + }; + t4 = [question, context]; + $[3] = context; + $[4] = question; + $[5] = t3; + $[6] = t4; + } else { + t3 = $[5]; + t4 = $[6]; + } + useEffect(t3, t4); + const maxContentHeight = Math.max(5, rows - CHROME_ROWS - OUTER_CHROME_ROWS); + let t5; + if ($[7] === Symbol.for("react.memo_cache_sentinel")) { + t5 = /btw{" "}; + $[7] = t5; + } else { + t5 = $[7]; + } + let t6; + if ($[8] !== question) { + t6 = {t5}{question}; + $[8] = question; + $[9] = t6; + } else { + t6 = $[9]; + } + let t7; + if ($[10] !== error || $[11] !== frame || $[12] !== response) { + t7 = {error ? {error} : response ? {response} : Answering...}; + $[10] = error; + $[11] = frame; + $[12] = response; + $[13] = t7; + } else { + t7 = $[13]; + } + let t8; + if ($[14] !== maxContentHeight || $[15] !== t7) { + t8 = {t7}; + $[14] = maxContentHeight; + $[15] = t7; + $[16] = t8; + } else { + t8 = $[16]; + } + let t9; + if ($[17] !== error || $[18] !== response) { + t9 = (response || error) && {UP_ARROW}/{DOWN_ARROW} to scroll · Space, Enter, or Escape to dismiss; + $[17] = error; + $[18] = response; + $[19] = t9; + } else { + t9 = $[19]; + } + let t10; + if ($[20] !== handleKeyDown || $[21] !== t6 || $[22] !== t8 || $[23] !== t9) { + t10 = {t6}{t8}{t9}; + $[20] = handleKeyDown; + $[21] = t6; + $[22] = t8; + $[23] = t9; + $[24] = t10; + } else { + t10 = $[24]; + } + return t10; +} + +/** + * Build CacheSafeParams for the side question fork. + * + * The preferred source is getLastCacheSafeParams — the exact + * systemPrompt/userContext/systemContext bytes the main thread sent on its + * last request (captured in stopHooks). Reusing them guarantees a byte- + * identical prefix and thus a prompt cache hit. We pair these with the + * current toolUseContext (for thinkingConfig/tools) and current messages + * (for up-to-date context). + * + * Fallback (first turn before stop hooks fire, or prompt-suggestion + * disabled): rebuild from scratch. This may miss the cache if the main loop + * applied buildEffectiveSystemPrompt extras (--agent, --system-prompt, + * --append-system-prompt, coordinator mode). + */ +function _temp(f) { + return f + 1; +} +function stripInProgressAssistantMessage(messages: Message[]): Message[] { + const last = messages.at(-1); + if (last?.type === 'assistant' && last.message.stop_reason === null) { + return messages.slice(0, -1); + } + return messages; +} +async function buildCacheSafeParams(context: ProcessUserInputContext): Promise { + const forkContextMessages = getMessagesAfterCompactBoundary(stripInProgressAssistantMessage(context.messages)); + const saved = getLastCacheSafeParams(); + if (saved) { + return { + systemPrompt: saved.systemPrompt, + userContext: saved.userContext, + systemContext: saved.systemContext, + toolUseContext: context, + forkContextMessages + }; + } + const [rawSystemPrompt, userContext, systemContext] = await Promise.all([getSystemPrompt(context.options.tools, context.options.mainLoopModel, [], context.options.mcpClients), getUserContext(), getSystemContext()]); + return { + systemPrompt: asSystemPrompt(rawSystemPrompt), + userContext, + systemContext, + toolUseContext: context, + forkContextMessages + }; +} +export async function call(onDone: LocalJSXCommandOnDone, context: ProcessUserInputContext, args: string): Promise { + const question = args?.trim(); + if (!question) { + onDone('Usage: /btw ', { + display: 'system' + }); + return null; + } + saveGlobalConfig(current => ({ + ...current, + btwUseCount: current.btwUseCount + 1 + })); + return ; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsInVzZUVmZmVjdCIsInVzZVJlZiIsInVzZVN0YXRlIiwidXNlSW50ZXJ2YWwiLCJDb21tYW5kUmVzdWx0RGlzcGxheSIsIk1hcmtkb3duIiwiU3Bpbm5lckdseXBoIiwiRE9XTl9BUlJPVyIsIlVQX0FSUk9XIiwiZ2V0U3lzdGVtUHJvbXB0IiwidXNlTW9kYWxPclRlcm1pbmFsU2l6ZSIsImdldFN5c3RlbUNvbnRleHQiLCJnZXRVc2VyQ29udGV4dCIsInVzZVRlcm1pbmFsU2l6ZSIsIlNjcm9sbEJveCIsIlNjcm9sbEJveEhhbmRsZSIsIktleWJvYXJkRXZlbnQiLCJCb3giLCJUZXh0IiwiTG9jYWxKU1hDb21tYW5kT25Eb25lIiwiTWVzc2FnZSIsImNyZWF0ZUFib3J0Q29udHJvbGxlciIsInNhdmVHbG9iYWxDb25maWciLCJlcnJvck1lc3NhZ2UiLCJDYWNoZVNhZmVQYXJhbXMiLCJnZXRMYXN0Q2FjaGVTYWZlUGFyYW1zIiwiZ2V0TWVzc2FnZXNBZnRlckNvbXBhY3RCb3VuZGFyeSIsIlByb2Nlc3NVc2VySW5wdXRDb250ZXh0IiwicnVuU2lkZVF1ZXN0aW9uIiwiYXNTeXN0ZW1Qcm9tcHQiLCJCdHdDb21wb25lbnRQcm9wcyIsInF1ZXN0aW9uIiwiY29udGV4dCIsIm9uRG9uZSIsInJlc3VsdCIsIm9wdGlvbnMiLCJkaXNwbGF5IiwiQ0hST01FX1JPV1MiLCJPVVRFUl9DSFJPTUVfUk9XUyIsIlNDUk9MTF9MSU5FUyIsIkJ0d1NpZGVRdWVzdGlvbiIsInQwIiwiJCIsIl9jIiwicmVzcG9uc2UiLCJzZXRSZXNwb25zZSIsImVycm9yIiwic2V0RXJyb3IiLCJmcmFtZSIsInNldEZyYW1lIiwic2Nyb2xsUmVmIiwicm93cyIsInQxIiwiU3ltYm9sIiwiZm9yIiwiX3RlbXAiLCJ0MiIsImhhbmRsZUtleURvd24iLCJlIiwia2V5IiwiY3RybCIsInByZXZlbnREZWZhdWx0IiwidW5kZWZpbmVkIiwiY3VycmVudCIsInNjcm9sbEJ5IiwidDMiLCJ0NCIsImFib3J0Q29udHJvbGxlciIsImZldGNoUmVzcG9uc2UiLCJjYWNoZVNhZmVQYXJhbXMiLCJidWlsZENhY2hlU2FmZVBhcmFtcyIsInNpZ25hbCIsImFib3J0ZWQiLCJ0NSIsImVyciIsImFib3J0IiwibWF4Q29udGVudEhlaWdodCIsIk1hdGgiLCJtYXgiLCJ0NiIsInQ3IiwidDgiLCJ0OSIsInQxMCIsImYiLCJzdHJpcEluUHJvZ3Jlc3NBc3Npc3RhbnRNZXNzYWdlIiwibWVzc2FnZXMiLCJsYXN0IiwiYXQiLCJ0eXBlIiwibWVzc2FnZSIsInN0b3BfcmVhc29uIiwic2xpY2UiLCJQcm9taXNlIiwiZm9ya0NvbnRleHRNZXNzYWdlcyIsInNhdmVkIiwic3lzdGVtUHJvbXB0IiwidXNlckNvbnRleHQiLCJzeXN0ZW1Db250ZXh0IiwidG9vbFVzZUNvbnRleHQiLCJyYXdTeXN0ZW1Qcm9tcHQiLCJhbGwiLCJ0b29scyIsIm1haW5Mb29wTW9kZWwiLCJtY3BDbGllbnRzIiwiY2FsbCIsImFyZ3MiLCJSZWFjdE5vZGUiLCJ0cmltIiwiYnR3VXNlQ291bnQiXSwic291cmNlcyI6WyJidHcudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgdXNlRWZmZWN0LCB1c2VSZWYsIHVzZVN0YXRlIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyB1c2VJbnRlcnZhbCB9IGZyb20gJ3VzZWhvb2tzLXRzJ1xuaW1wb3J0IHR5cGUgeyBDb21tYW5kUmVzdWx0RGlzcGxheSB9IGZyb20gJy4uLy4uL2NvbW1hbmRzLmpzJ1xuaW1wb3J0IHsgTWFya2Rvd24gfSBmcm9tICcuLi8uLi9jb21wb25lbnRzL01hcmtkb3duLmpzJ1xuaW1wb3J0IHsgU3Bpbm5lckdseXBoIH0gZnJvbSAnLi4vLi4vY29tcG9uZW50cy9TcGlubmVyL1NwaW5uZXJHbHlwaC5qcydcbmltcG9ydCB7IERPV05fQVJST1csIFVQX0FSUk9XIH0gZnJvbSAnLi4vLi4vY29uc3RhbnRzL2ZpZ3VyZXMuanMnXG5pbXBvcnQgeyBnZXRTeXN0ZW1Qcm9tcHQgfSBmcm9tICcuLi8uLi9jb25zdGFudHMvcHJvbXB0cy5qcydcbmltcG9ydCB7IHVzZU1vZGFsT3JUZXJtaW5hbFNpemUgfSBmcm9tICcuLi8uLi9jb250ZXh0L21vZGFsQ29udGV4dC5qcydcbmltcG9ydCB7IGdldFN5c3RlbUNvbnRleHQsIGdldFVzZXJDb250ZXh0IH0gZnJvbSAnLi4vLi4vY29udGV4dC5qcydcbmltcG9ydCB7IHVzZVRlcm1pbmFsU2l6ZSB9IGZyb20gJy4uLy4uL2hvb2tzL3VzZVRlcm1pbmFsU2l6ZS5qcydcbmltcG9ydCBTY3JvbGxCb3gsIHtcbiAgdHlwZSBTY3JvbGxCb3hIYW5kbGUsXG59IGZyb20gJy4uLy4uL2luay9jb21wb25lbnRzL1Njcm9sbEJveC5qcydcbmltcG9ydCB0eXBlIHsgS2V5Ym9hcmRFdmVudCB9IGZyb20gJy4uLy4uL2luay9ldmVudHMva2V5Ym9hcmQtZXZlbnQuanMnXG5pbXBvcnQgeyBCb3gsIFRleHQgfSBmcm9tICcuLi8uLi9pbmsuanMnXG5pbXBvcnQgdHlwZSB7IExvY2FsSlNYQ29tbWFuZE9uRG9uZSB9IGZyb20gJy4uLy4uL3R5cGVzL2NvbW1hbmQuanMnXG5pbXBvcnQgdHlwZSB7IE1lc3NhZ2UgfSBmcm9tICcuLi8uLi90eXBlcy9tZXNzYWdlLmpzJ1xuaW1wb3J0IHsgY3JlYXRlQWJvcnRDb250cm9sbGVyIH0gZnJvbSAnLi4vLi4vdXRpbHMvYWJvcnRDb250cm9sbGVyLmpzJ1xuaW1wb3J0IHsgc2F2ZUdsb2JhbENvbmZpZyB9IGZyb20gJy4uLy4uL3V0aWxzL2NvbmZpZy5qcydcbmltcG9ydCB7IGVycm9yTWVzc2FnZSB9IGZyb20gJy4uLy4uL3V0aWxzL2Vycm9ycy5qcydcbmltcG9ydCB7XG4gIHR5cGUgQ2FjaGVTYWZlUGFyYW1zLFxuICBnZXRMYXN0Q2FjaGVTYWZlUGFyYW1zLFxufSBmcm9tICcuLi8uLi91dGlscy9mb3JrZWRBZ2VudC5qcydcbmltcG9ydCB7IGdldE1lc3NhZ2VzQWZ0ZXJDb21wYWN0Qm91bmRhcnkgfSBmcm9tICcuLi8uLi91dGlscy9tZXNzYWdlcy5qcydcbmltcG9ydCB0eXBlIHsgUHJvY2Vzc1VzZXJJbnB1dENvbnRleHQgfSBmcm9tICcuLi8uLi91dGlscy9wcm9jZXNzVXNlcklucHV0L3Byb2Nlc3NVc2VySW5wdXQuanMnXG5pbXBvcnQgeyBydW5TaWRlUXVlc3Rpb24gfSBmcm9tICcuLi8uLi91dGlscy9zaWRlUXVlc3Rpb24uanMnXG5pbXBvcnQgeyBhc1N5c3RlbVByb21wdCB9IGZyb20gJy4uLy4uL3V0aWxzL3N5c3RlbVByb21wdFR5cGUuanMnXG5cbnR5cGUgQnR3Q29tcG9uZW50UHJvcHMgPSB7XG4gIHF1ZXN0aW9uOiBzdHJpbmdcbiAgY29udGV4dDogUHJvY2Vzc1VzZXJJbnB1dENvbnRleHRcbiAgb25Eb25lOiAoXG4gICAgcmVzdWx0Pzogc3RyaW5nLFxuICAgIG9wdGlvbnM/OiB7IGRpc3BsYXk/OiBDb21tYW5kUmVzdWx0RGlzcGxheSB9LFxuICApID0+IHZvaWRcbn1cblxuY29uc3QgQ0hST01FX1JPV1MgPSA1XG5jb25zdCBPVVRFUl9DSFJPTUVfUk9XUyA9IDZcbmNvbnN0IFNDUk9MTF9MSU5FUyA9IDNcblxuZnVuY3Rpb24gQnR3U2lkZVF1ZXN0aW9uKHtcbiAgcXVlc3Rpb24sXG4gIGNvbnRleHQsXG4gIG9uRG9uZSxcbn06IEJ0d0NvbXBvbmVudFByb3BzKTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgY29uc3QgW3Jlc3BvbnNlLCBzZXRSZXNwb25zZV0gPSB1c2VTdGF0ZTxzdHJpbmcgfCBudWxsPihudWxsKVxuICBjb25zdCBbZXJyb3IsIHNldEVycm9yXSA9IHVzZVN0YXRlPHN0cmluZyB8IG51bGw+KG51bGwpXG4gIGNvbnN0IFtmcmFtZSwgc2V0RnJhbWVdID0gdXNlU3RhdGUoMClcbiAgY29uc3Qgc2Nyb2xsUmVmID0gdXNlUmVmPFNjcm9sbEJveEhhbmRsZT4obnVsbClcbiAgY29uc3QgeyByb3dzIH0gPSB1c2VNb2RhbE9yVGVybWluYWxTaXplKHVzZVRlcm1pbmFsU2l6ZSgpKVxuXG4gIC8vIEFuaW1hdGUgc3Bpbm5lciB3aGlsZSBsb2FkaW5nXG4gIHVzZUludGVydmFsKCgpID0+IHNldEZyYW1lKGYgPT4gZiArIDEpLCByZXNwb25zZSB8fCBlcnJvciA/IG51bGwgOiA4MClcblxuICBmdW5jdGlvbiBoYW5kbGVLZXlEb3duKGU6IEtleWJvYXJkRXZlbnQpOiB2b2lkIHtcbiAgICBpZiAoXG4gICAgICBlLmtleSA9PT0gJ2VzY2FwZScgfHxcbiAgICAgIGUua2V5ID09PSAncmV0dXJuJyB8fFxuICAgICAgZS5rZXkgPT09ICcgJyB8fFxuICAgICAgKGUuY3RybCAmJiAoZS5rZXkgPT09ICdjJyB8fCBlLmtleSA9PT0gJ2QnKSlcbiAgICApIHtcbiAgICAgIGUucHJldmVudERlZmF1bHQoKVxuICAgICAgb25Eb25lKHVuZGVmaW5lZCwgeyBkaXNwbGF5OiAnc2tpcCcgfSlcbiAgICAgIHJldHVyblxuICAgIH1cbiAgICBpZiAoZS5rZXkgPT09ICd1cCcgfHwgKGUuY3RybCAmJiBlLmtleSA9PT0gJ3AnKSkge1xuICAgICAgZS5wcmV2ZW50RGVmYXVsdCgpXG4gICAgICBzY3JvbGxSZWYuY3VycmVudD8uc2Nyb2xsQnkoLVNDUk9MTF9MSU5FUylcbiAgICB9XG4gICAgaWYgKGUua2V5ID09PSAnZG93bicgfHwgKGUuY3RybCAmJiBlLmtleSA9PT0gJ24nKSkge1xuICAgICAgZS5wcmV2ZW50RGVmYXVsdCgpXG4gICAgICBzY3JvbGxSZWYuY3VycmVudD8uc2Nyb2xsQnkoU0NST0xMX0xJTkVTKVxuICAgIH1cbiAgfVxuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgY29uc3QgYWJvcnRDb250cm9sbGVyID0gY3JlYXRlQWJvcnRDb250cm9sbGVyKClcblxuICAgIGFzeW5jIGZ1bmN0aW9uIGZldGNoUmVzcG9uc2UoKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgICB0cnkge1xuICAgICAgICBjb25zdCBjYWNoZVNhZmVQYXJhbXMgPSBhd2FpdCBidWlsZENhY2hlU2FmZVBhcmFtcyhjb250ZXh0KVxuICAgICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCBydW5TaWRlUXVlc3Rpb24oeyBxdWVzdGlvbiwgY2FjaGVTYWZlUGFyYW1zIH0pXG5cbiAgICAgICAgaWYgKCFhYm9ydENvbnRyb2xsZXIuc2lnbmFsLmFib3J0ZWQpIHtcbiAgICAgICAgICBpZiAocmVzdWx0LnJlc3BvbnNlKSB7XG4gICAgICAgICAgICBzZXRSZXNwb25zZShyZXN1bHQucmVzcG9uc2UpXG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHNldEVycm9yKCdObyByZXNwb25zZSByZWNlaXZlZCcpXG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9IGNhdGNoIChlcnIpIHtcbiAgICAgICAgaWYgKCFhYm9ydENvbnRyb2xsZXIuc2lnbmFsLmFib3J0ZWQpIHtcbiAgICAgICAgICBzZXRFcnJvcihlcnJvck1lc3NhZ2UoZXJyKSB8fCAnRmFpbGVkIHRvIGdldCByZXNwb25zZScpXG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG5cbiAgICB2b2lkIGZldGNoUmVzcG9uc2UoKVxuXG4gICAgcmV0dXJuICgpID0+IHtcbiAgICAgIGFib3J0Q29udHJvbGxlci5hYm9ydCgpXG4gICAgfVxuICB9LCBbcXVlc3Rpb24sIGNvbnRleHRdKVxuXG4gIGNvbnN0IG1heENvbnRlbnRIZWlnaHQgPSBNYXRoLm1heCg1LCByb3dzIC0gQ0hST01FX1JPV1MgLSBPVVRFUl9DSFJPTUVfUk9XUylcblxuICByZXR1cm4gKFxuICAgIDxCb3hcbiAgICAgIGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIlxuICAgICAgcGFkZGluZ0xlZnQ9ezJ9XG4gICAgICBtYXJnaW5Ub3A9ezF9XG4gICAgICB0YWJJbmRleD17MH1cbiAgICAgIGF1dG9Gb2N1c1xuICAgICAgb25LZXlEb3duPXtoYW5kbGVLZXlEb3dufVxuICAgID5cbiAgICAgIDxCb3g+XG4gICAgICAgIDxUZXh0IGNvbG9yPVwid2FybmluZ1wiIGJvbGQ+XG4gICAgICAgICAgL2J0d3snICd9XG4gICAgICAgIDwvVGV4dD5cbiAgICAgICAgPFRleHQgZGltQ29sb3I+e3F1ZXN0aW9ufTwvVGV4dD5cbiAgICAgIDwvQm94PlxuICAgICAgPEJveCBtYXJnaW5Ub3A9ezF9IG1hcmdpbkxlZnQ9ezJ9IG1heEhlaWdodD17bWF4Q29udGVudEhlaWdodH0+XG4gICAgICAgIDxTY3JvbGxCb3ggcmVmPXtzY3JvbGxSZWZ9IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIiBmbGV4R3Jvdz17MX0+XG4gICAgICAgICAge2Vycm9yID8gKFxuICAgICAgICAgICAgPFRleHQgY29sb3I9XCJlcnJvclwiPntlcnJvcn08L1RleHQ+XG4gICAgICAgICAgKSA6IHJlc3BvbnNlID8gKFxuICAgICAgICAgICAgPE1hcmtkb3duPntyZXNwb25zZX08L01hcmtkb3duPlxuICAgICAgICAgICkgOiAoXG4gICAgICAgICAgICA8Qm94PlxuICAgICAgICAgICAgICA8U3Bpbm5lckdseXBoIGZyYW1lPXtmcmFtZX0gbWVzc2FnZUNvbG9yPVwid2FybmluZ1wiIC8+XG4gICAgICAgICAgICAgIDxUZXh0IGNvbG9yPVwid2FybmluZ1wiPkFuc3dlcmluZy4uLjwvVGV4dD5cbiAgICAgICAgICAgIDwvQm94PlxuICAgICAgICAgICl9XG4gICAgICAgIDwvU2Nyb2xsQm94PlxuICAgICAgPC9Cb3g+XG4gICAgICB7KHJlc3BvbnNlIHx8IGVycm9yKSAmJiAoXG4gICAgICAgIDxCb3ggbWFyZ2luVG9wPXsxfT5cbiAgICAgICAgICA8VGV4dCBkaW1Db2xvcj5cbiAgICAgICAgICAgIHtVUF9BUlJPV30ve0RPV05fQVJST1d9IHRvIHNjcm9sbCDCtyBTcGFjZSwgRW50ZXIsIG9yIEVzY2FwZSB0b1xuICAgICAgICAgICAgZGlzbWlzc1xuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgPC9Cb3g+XG4gICAgICApfVxuICAgIDwvQm94PlxuICApXG59XG5cbi8qKlxuICogQnVpbGQgQ2FjaGVTYWZlUGFyYW1zIGZvciB0aGUgc2lkZSBxdWVzdGlvbiBmb3JrLlxuICpcbiAqIFRoZSBwcmVmZXJyZWQgc291cmNlIGlzIGdldExhc3RDYWNoZVNhZmVQYXJhbXMg4oCUIHRoZSBleGFjdFxuICogc3lzdGVtUHJvbXB0L3VzZXJDb250ZXh0L3N5c3RlbUNvbnRleHQgYnl0ZXMgdGhlIG1haW4gdGhyZWFkIHNlbnQgb24gaXRzXG4gKiBsYXN0IHJlcXVlc3QgKGNhcHR1cmVkIGluIHN0b3BIb29rcykuIFJldXNpbmcgdGhlbSBndWFyYW50ZWVzIGEgYnl0ZS1cbiAqIGlkZW50aWNhbCBwcmVmaXggYW5kIHRodXMgYSBwcm9tcHQgY2FjaGUgaGl0LiBXZSBwYWlyIHRoZXNlIHdpdGggdGhlXG4gKiBjdXJyZW50IHRvb2xVc2VDb250ZXh0IChmb3IgdGhpbmtpbmdDb25maWcvdG9vbHMpIGFuZCBjdXJyZW50IG1lc3NhZ2VzXG4gKiAoZm9yIHVwLXRvLWRhdGUgY29udGV4dCkuXG4gKlxuICogRmFsbGJhY2sgKGZpcnN0IHR1cm4gYmVmb3JlIHN0b3AgaG9va3MgZmlyZSwgb3IgcHJvbXB0LXN1Z2dlc3Rpb25cbiAqIGRpc2FibGVkKTogcmVidWlsZCBmcm9tIHNjcmF0Y2guIFRoaXMgbWF5IG1pc3MgdGhlIGNhY2hlIGlmIHRoZSBtYWluIGxvb3BcbiAqIGFwcGxpZWQgYnVpbGRFZmZlY3RpdmVTeXN0ZW1Qcm9tcHQgZXh0cmFzICgtLWFnZW50LCAtLXN5c3RlbS1wcm9tcHQsXG4gKiAtLWFwcGVuZC1zeXN0ZW0tcHJvbXB0LCBjb29yZGluYXRvciBtb2RlKS5cbiAqL1xuZnVuY3Rpb24gc3RyaXBJblByb2dyZXNzQXNzaXN0YW50TWVzc2FnZShtZXNzYWdlczogTWVzc2FnZVtdKTogTWVzc2FnZVtdIHtcbiAgY29uc3QgbGFzdCA9IG1lc3NhZ2VzLmF0KC0xKVxuICBpZiAobGFzdD8udHlwZSA9PT0gJ2Fzc2lzdGFudCcgJiYgbGFzdC5tZXNzYWdlLnN0b3BfcmVhc29uID09PSBudWxsKSB7XG4gICAgcmV0dXJuIG1lc3NhZ2VzLnNsaWNlKDAsIC0xKVxuICB9XG4gIHJldHVybiBtZXNzYWdlc1xufVxuXG5hc3luYyBmdW5jdGlvbiBidWlsZENhY2hlU2FmZVBhcmFtcyhcbiAgY29udGV4dDogUHJvY2Vzc1VzZXJJbnB1dENvbnRleHQsXG4pOiBQcm9taXNlPENhY2hlU2FmZVBhcmFtcz4ge1xuICBjb25zdCBmb3JrQ29udGV4dE1lc3NhZ2VzID0gZ2V0TWVzc2FnZXNBZnRlckNvbXBhY3RCb3VuZGFyeShcbiAgICBzdHJpcEluUHJvZ3Jlc3NBc3Npc3RhbnRNZXNzYWdlKGNvbnRleHQubWVzc2FnZXMpLFxuICApXG4gIGNvbnN0IHNhdmVkID0gZ2V0TGFzdENhY2hlU2FmZVBhcmFtcygpXG4gIGlmIChzYXZlZCkge1xuICAgIHJldHVybiB7XG4gICAgICBzeXN0ZW1Qcm9tcHQ6IHNhdmVkLnN5c3RlbVByb21wdCxcbiAgICAgIHVzZXJDb250ZXh0OiBzYXZlZC51c2VyQ29udGV4dCxcbiAgICAgIHN5c3RlbUNvbnRleHQ6IHNhdmVkLnN5c3RlbUNvbnRleHQsXG4gICAgICB0b29sVXNlQ29udGV4dDogY29udGV4dCxcbiAgICAgIGZvcmtDb250ZXh0TWVzc2FnZXMsXG4gICAgfVxuICB9XG4gIGNvbnN0IFtyYXdTeXN0ZW1Qcm9tcHQsIHVzZXJDb250ZXh0LCBzeXN0ZW1Db250ZXh0XSA9IGF3YWl0IFByb21pc2UuYWxsKFtcbiAgICBnZXRTeXN0ZW1Qcm9tcHQoXG4gICAgICBjb250ZXh0Lm9wdGlvbnMudG9vbHMsXG4gICAgICBjb250ZXh0Lm9wdGlvbnMubWFpbkxvb3BNb2RlbCxcbiAgICAgIFtdLFxuICAgICAgY29udGV4dC5vcHRpb25zLm1jcENsaWVudHMsXG4gICAgKSxcbiAgICBnZXRVc2VyQ29udGV4dCgpLFxuICAgIGdldFN5c3RlbUNvbnRleHQoKSxcbiAgXSlcbiAgcmV0dXJuIHtcbiAgICBzeXN0ZW1Qcm9tcHQ6IGFzU3lzdGVtUHJvbXB0KHJhd1N5c3RlbVByb21wdCksXG4gICAgdXNlckNvbnRleHQsXG4gICAgc3lzdGVtQ29udGV4dCxcbiAgICB0b29sVXNlQ29udGV4dDogY29udGV4dCxcbiAgICBmb3JrQ29udGV4dE1lc3NhZ2VzLFxuICB9XG59XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBjYWxsKFxuICBvbkRvbmU6IExvY2FsSlNYQ29tbWFuZE9uRG9uZSxcbiAgY29udGV4dDogUHJvY2Vzc1VzZXJJbnB1dENvbnRleHQsXG4gIGFyZ3M6IHN0cmluZyxcbik6IFByb21pc2U8UmVhY3QuUmVhY3ROb2RlPiB7XG4gIGNvbnN0IHF1ZXN0aW9uID0gYXJncz8udHJpbSgpXG5cbiAgaWYgKCFxdWVzdGlvbikge1xuICAgIG9uRG9uZSgnVXNhZ2U6IC9idHcgPHlvdXIgcXVlc3Rpb24+JywgeyBkaXNwbGF5OiAnc3lzdGVtJyB9KVxuICAgIHJldHVybiBudWxsXG4gIH1cblxuICBzYXZlR2xvYmFsQ29uZmlnKGN1cnJlbnQgPT4gKHtcbiAgICAuLi5jdXJyZW50LFxuICAgIGJ0d1VzZUNvdW50OiBjdXJyZW50LmJ0d1VzZUNvdW50ICsgMSxcbiAgfSkpXG5cbiAgcmV0dXJuIChcbiAgICA8QnR3U2lkZVF1ZXN0aW9uIHF1ZXN0aW9uPXtxdWVzdGlvbn0gY29udGV4dD17Y29udGV4dH0gb25Eb25lPXtvbkRvbmV9IC8+XG4gIClcbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU8sS0FBS0EsS0FBSyxNQUFNLE9BQU87QUFDOUIsU0FBU0MsU0FBUyxFQUFFQyxNQUFNLEVBQUVDLFFBQVEsUUFBUSxPQUFPO0FBQ25ELFNBQVNDLFdBQVcsUUFBUSxhQUFhO0FBQ3pDLGNBQWNDLG9CQUFvQixRQUFRLG1CQUFtQjtBQUM3RCxTQUFTQyxRQUFRLFFBQVEsOEJBQThCO0FBQ3ZELFNBQVNDLFlBQVksUUFBUSwwQ0FBMEM7QUFDdkUsU0FBU0MsVUFBVSxFQUFFQyxRQUFRLFFBQVEsNEJBQTRCO0FBQ2pFLFNBQVNDLGVBQWUs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QUFLLEVBQUU7SUFDVCxPQUFPO01BQ0xDLFlBQVksRUFBRUQsS0FBSyxDQUFDQyxZQUFZO01BQ2hDQyxXQUFXLEVBQUVGLEtBQUssQ0FBQ0UsV0FBVztNQUM5QkMsYUFBYSxFQUFFSCxLQUFLLENBQUNHLGFBQWE7TUFDbENDLGNBQWMsRUFBRW5FLE9BQU87TUFDdkI4RDtJQUNGLENBQUM7RUFDSDtFQUNBLE1BQU0sQ0FBQ00sZUFBZSxFQUFFSCxXQUFXLEVBQUVDLGFBQWEsQ0FBQyxHQUFHLE1BQU1MLE9BQU8sQ0FBQ1EsR0FBRyxDQUFDLENBQ3RFNUYsZUFBZSxDQUNidUIsT0FBTyxDQUFDRyxPQUFPLENBQUNtRSxLQUFLLEVBQ3JCdEUsT0FBTyxDQUFDRyxPQUFPLENBQUNvRSxhQUFhLEVBQzdCLEVBQUUsRUFDRnZFLE9BQU8sQ0FBQ0csT0FBTyxDQUFDcUUsVUFDbEIsQ0FBQyxFQUNENUYsY0FBYyxDQUFDLENBQUMsRUFDaEJELGdCQUFnQixDQUFDLENBQUMsQ0FDbkIsQ0FBQztFQUNGLE9BQU87SUFDTHFGLFlBQVksRUFBRW5FLGNBQWMsQ0FBQ3VFLGVBQWUsQ0FBQztJQUM3Q0gsV0FBVztJQUNYQyxhQUFhO0lBQ2JDLGNBQWMsRUFBRW5FLE9BQU87SUFDdkI4RDtFQUNGLENBQUM7QUFDSDtBQUVBLE9BQU8sZUFBZVcsSUFBSUEsQ0FDeEJ4RSxNQUFNLEVBQUVkLHFCQUFxQixFQUM3QmEsT0FBTyxFQUFFTCx1QkFBdUIsRUFDaEMrRSxJQUFJLEVBQUUsTUFBTSxDQUNiLEVBQUViLE9BQU8sQ0FBQzlGLEtBQUssQ0FBQzRHLFNBQVMsQ0FBQyxDQUFDO0VBQzFCLE1BQU01RSxRQUFRLEdBQUcyRSxJQUFJLEVBQUVFLElBQUksQ0FBQyxDQUFDO0VBRTdCLElBQUksQ0FBQzdFLFFBQVEsRUFBRTtJQUNiRSxNQUFNLENBQUMsNkJBQTZCLEVBQUU7TUFBRUcsT0FBTyxFQUFFO0lBQVMsQ0FBQyxDQUFDO0lBQzVELE9BQU8sSUFBSTtFQUNiO0VBRUFkLGdCQUFnQixDQUFDeUMsT0FBTyxLQUFLO0lBQzNCLEdBQUdBLE9BQU87SUFDVjhDLFdBQVcsRUFBRTlDLE9BQU8sQ0FBQzhDLFdBQVcsR0FBRztFQUNyQyxDQUFDLENBQUMsQ0FBQztFQUVILE9BQ0UsQ0FBQyxlQUFlLENBQUMsUUFBUSxDQUFDLENBQUM5RSxRQUFRLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQ0MsT0FBTyxDQUFDLENBQUMsTUFBTSxDQUFDLENBQUNDLE1BQU0sQ0FBQyxHQUFHO0FBRTdFIiwiaWdub3JlTGlzdCI6W119 \ No newline at end of file diff --git a/src/commands/btw/index.ts b/src/commands/btw/index.ts new file mode 100644 index 0000000..d488871 --- /dev/null +++ b/src/commands/btw/index.ts @@ -0,0 +1,13 @@ +import type { Command } from '../../commands.js' + +const btw = { + type: 'local-jsx', + name: 'btw', + description: + 'Ask a quick side question without interrupting the main conversation', + immediate: true, + argumentHint: '', + load: () => import('./btw.js'), +} satisfies Command + +export default btw diff --git a/src/commands/bughunter/index.js b/src/commands/bughunter/index.js new file mode 100644 index 0000000..e1a619d --- /dev/null +++ b/src/commands/bughunter/index.js @@ -0,0 +1 @@ +export default { isEnabled: () => false, isHidden: true, name: 'stub' }; diff --git a/src/commands/chrome/chrome.tsx b/src/commands/chrome/chrome.tsx new file mode 100644 index 0000000..c337873 --- /dev/null +++ b/src/commands/chrome/chrome.tsx @@ -0,0 +1,285 @@ +import { c as _c } from "react/compiler-runtime"; +import React, { useState } from 'react'; +import { type OptionWithDescription, Select } from '../../components/CustomSelect/select.js'; +import { Dialog } from '../../components/design-system/Dialog.js'; +import { Box, Text } from '../../ink.js'; +import { useAppState } from '../../state/AppState.js'; +import { isClaudeAISubscriber } from '../../utils/auth.js'; +import { openBrowser } from '../../utils/browser.js'; +import { CLAUDE_IN_CHROME_MCP_SERVER_NAME, openInChrome } from '../../utils/claudeInChrome/common.js'; +import { isChromeExtensionInstalled } from '../../utils/claudeInChrome/setup.js'; +import { getGlobalConfig, saveGlobalConfig } from '../../utils/config.js'; +import { env } from '../../utils/env.js'; +import { isRunningOnHomespace } from '../../utils/envUtils.js'; +const CHROME_EXTENSION_URL = 'https://claude.ai/chrome'; +const CHROME_PERMISSIONS_URL = 'https://clau.de/chrome/permissions'; +const CHROME_RECONNECT_URL = 'https://clau.de/chrome/reconnect'; +type MenuAction = 'install-extension' | 'reconnect' | 'manage-permissions' | 'toggle-default'; +type Props = { + onDone: (result?: string) => void; + isExtensionInstalled: boolean; + configEnabled: boolean | undefined; + isClaudeAISubscriber: boolean; + isWSL: boolean; +}; +function ClaudeInChromeMenu(t0) { + const $ = _c(41); + const { + onDone, + isExtensionInstalled: installed, + configEnabled, + isClaudeAISubscriber, + isWSL + } = t0; + const mcpClients = useAppState(_temp); + const [selectKey, setSelectKey] = useState(0); + const [enabledByDefault, setEnabledByDefault] = useState(configEnabled ?? false); + const [showInstallHint, setShowInstallHint] = useState(false); + const [isExtensionInstalled, setIsExtensionInstalled] = useState(installed); + let t1; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t1 = false && isRunningOnHomespace(); + $[0] = t1; + } else { + t1 = $[0]; + } + const isHomespace = t1; + let t2; + if ($[1] !== mcpClients) { + t2 = mcpClients.find(_temp2); + $[1] = mcpClients; + $[2] = t2; + } else { + t2 = $[2]; + } + const chromeClient = t2; + const isConnected = chromeClient?.type === "connected"; + let t3; + if ($[3] === Symbol.for("react.memo_cache_sentinel")) { + t3 = function openUrl(url) { + if (isHomespace) { + openBrowser(url); + } else { + openInChrome(url); + } + }; + $[3] = t3; + } else { + t3 = $[3]; + } + const openUrl = t3; + let t4; + if ($[4] !== enabledByDefault) { + t4 = function handleAction(action) { + bb22: switch (action) { + case "install-extension": + { + setSelectKey(_temp3); + setShowInstallHint(true); + openUrl(CHROME_EXTENSION_URL); + break bb22; + } + case "reconnect": + { + setSelectKey(_temp4); + isChromeExtensionInstalled().then(installed_0 => { + setIsExtensionInstalled(installed_0); + if (installed_0) { + setShowInstallHint(false); + } + }); + openUrl(CHROME_RECONNECT_URL); + break bb22; + } + case "manage-permissions": + { + setSelectKey(_temp5); + openUrl(CHROME_PERMISSIONS_URL); + break bb22; + } + case "toggle-default": + { + const newValue = !enabledByDefault; + saveGlobalConfig(current => ({ + ...current, + claudeInChromeDefaultEnabled: newValue + })); + setEnabledByDefault(newValue); + } + } + }; + $[4] = enabledByDefault; + $[5] = t4; + } else { + t4 = $[5]; + } + const handleAction = t4; + let options; + if ($[6] !== enabledByDefault || $[7] !== isExtensionInstalled) { + options = []; + const requiresExtensionSuffix = isExtensionInstalled ? "" : " (requires extension)"; + if (!isExtensionInstalled && !isHomespace) { + let t5; + if ($[9] === Symbol.for("react.memo_cache_sentinel")) { + t5 = { + label: "Install Chrome extension", + value: "install-extension" + }; + $[9] = t5; + } else { + t5 = $[9]; + } + options.push(t5); + } + let t5; + if ($[10] === Symbol.for("react.memo_cache_sentinel")) { + t5 = Manage permissions; + $[10] = t5; + } else { + t5 = $[10]; + } + let t6; + if ($[11] !== requiresExtensionSuffix) { + t6 = { + label: <>{t5}{requiresExtensionSuffix}, + value: "manage-permissions" + }; + $[11] = requiresExtensionSuffix; + $[12] = t6; + } else { + t6 = $[12]; + } + let t7; + if ($[13] === Symbol.for("react.memo_cache_sentinel")) { + t7 = Reconnect extension; + $[13] = t7; + } else { + t7 = $[13]; + } + let t8; + if ($[14] !== requiresExtensionSuffix) { + t8 = { + label: <>{t7}{requiresExtensionSuffix}, + value: "reconnect" + }; + $[14] = requiresExtensionSuffix; + $[15] = t8; + } else { + t8 = $[15]; + } + const t9 = `Enabled by default: ${enabledByDefault ? "Yes" : "No"}`; + let t10; + if ($[16] !== t9) { + t10 = { + label: t9, + value: "toggle-default" + }; + $[16] = t9; + $[17] = t10; + } else { + t10 = $[17]; + } + options.push(t6, t8, t10); + $[6] = enabledByDefault; + $[7] = isExtensionInstalled; + $[8] = options; + } else { + options = $[8]; + } + const isDisabled = isWSL || true && !isClaudeAISubscriber; + let t5; + if ($[18] !== onDone) { + t5 = () => onDone(); + $[18] = onDone; + $[19] = t5; + } else { + t5 = $[19]; + } + let t6; + if ($[20] === Symbol.for("react.memo_cache_sentinel")) { + t6 = Claude in Chrome works with the Chrome extension to let you control your browser directly from Claude Code. Navigate websites, fill forms, capture screenshots, record GIFs, and debug with console logs and network requests.; + $[20] = t6; + } else { + t6 = $[20]; + } + let t7; + if ($[21] !== isWSL) { + t7 = isWSL && Claude in Chrome is not supported in WSL at this time.; + $[21] = isWSL; + $[22] = t7; + } else { + t7 = $[22]; + } + let t8; + if ($[23] !== isClaudeAISubscriber) { + t8 = true && !isClaudeAISubscriber && Claude in Chrome requires a claude.ai subscription.; + $[23] = isClaudeAISubscriber; + $[24] = t8; + } else { + t8 = $[24]; + } + let t9; + if ($[25] !== handleAction || $[26] !== isConnected || $[27] !== isDisabled || $[28] !== isExtensionInstalled || $[29] !== options || $[30] !== selectKey || $[31] !== showInstallHint) { + t9 = !isDisabled && <>{!isHomespace && Status:{" "}{isConnected ? Enabled : Disabled}Extension:{" "}{isExtensionInstalled ? Installed : Not detected}}; + $[25] = options; + $[26] = t10; + $[27] = t9; + $[28] = t11; + } else { + t11 = $[28]; + } + let t12; + if ($[29] === Symbol.for("react.memo_cache_sentinel")) { + t12 = ; + $[29] = t12; + } else { + t12 = $[29]; + } + let t13; + if ($[30] !== handleKeyDown || $[31] !== t11) { + t13 = {t7}{t11}{t12}; + $[30] = handleKeyDown; + $[31] = t11; + $[32] = t13; + } else { + t13 = $[32]; + } + return t13; +} +function _temp2(c) { + return { + ...c, + copyFullResponse: true + }; +} +function _temp(block, index) { + const blockLines = countCharInString(block.code, "\n") + 1; + return { + label: truncateLine(block.code, 60), + value: index, + description: [block.lang, blockLines > 1 ? `${blockLines} lines` : undefined].filter(Boolean).join(", ") || undefined + }; +} +export const call: LocalJSXCommandCall = async (onDone, context, args) => { + const texts = collectRecentAssistantTexts(context.messages); + if (texts.length === 0) { + onDone('No assistant message to copy'); + return null; + } + + // /copy N reaches back N-1 messages (1 = latest, 2 = second-to-latest, ...) + let age = 0; + const arg = args?.trim(); + if (arg) { + const n = Number(arg); + if (!Number.isInteger(n) || n < 1) { + onDone(`Usage: /copy [N] where N is 1 (latest), 2, 3, \u2026 Got: ${arg}`); + return null; + } + if (n > texts.length) { + onDone(`Only ${texts.length} assistant ${texts.length === 1 ? 'message' : 'messages'} available to copy`); + return null; + } + age = n - 1; + } + const text = texts[age]!; + const codeBlocks = extractCodeBlocks(text); + const config = getGlobalConfig(); + if (codeBlocks.length === 0 || config.copyFullResponse) { + logEvent('tengu_copy', { + always: config.copyFullResponse, + block_count: codeBlocks.length, + message_age: age + }); + const result = await copyOrWriteToFile(text, RESPONSE_FILENAME); + onDone(result); + return null; + } + return ; +}; +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJta2RpciIsIndyaXRlRmlsZSIsIm1hcmtlZCIsIlRva2VucyIsInRtcGRpciIsImpvaW4iLCJSZWFjdCIsInVzZVJlZiIsIkNvbW1hbmRSZXN1bHREaXNwbGF5IiwiT3B0aW9uV2l0aERlc2NyaXB0aW9uIiwiU2VsZWN0IiwiQnlsaW5lIiwiS2V5Ym9hcmRTaG9ydGN1dEhpbnQiLCJQYW5lIiwiS2V5Ym9hcmRFdmVudCIsInN0cmluZ1dpZHRoIiwic2V0Q2xpcGJvYXJkIiwiQm94IiwiVGV4dCIsImxvZ0V2ZW50IiwiTG9jYWxKU1hDb21tYW5kQ2FsbCIsIkFzc2lzdGFudE1lc3NhZ2UiLCJNZXNzYWdlIiwiZ2V0R2xvYmFsQ29uZmlnIiwic2F2ZUdsb2JhbENvbmZpZyIsImV4dHJhY3RUZXh0Q29udGVudCIsInN0cmlwUHJvbXB0WE1MVGFncyIsImNvdW50Q2hhckluU3RyaW5nIiwiQ09QWV9ESVIiLCJSRVNQT05TRV9GSUxFTkFNRSIsIk1BWF9MT09LQkFDSyIsIkNvZGVCbG9jayIsImNvZGUiLCJsYW5nIiwiZXh0cmFjdENvZGVCbG9ja3MiLCJtYXJrZG93biIsInRva2VucyIsImxleGVyIiwiYmxvY2tzIiwidG9rZW4iLCJ0eXBlIiwiY29kZVRva2VuIiwiQ29kZSIsInB1c2giLCJ0ZXh0IiwiY29sbGVjdFJlY2VudEFzc2lzdGFudFRleHRzIiwibWVzc2FnZXMiLCJ0ZXh0cyIsImkiLCJsZW5ndGgiLCJtc2ciLCJpc0FwaUVycm9yTWVzc2FnZSIsImNvbnRlbnQiLCJtZXNzYWdlIiwiQXJyYXkiLCJpc0FycmF5IiwiZmlsZUV4dGVuc2lvbiIsInNhbml0aXplZCIsInJlcGxhY2UiLCJ3cml0ZVRvRmlsZSIsImZpbGVuYW1lIiwiUHJvbWlzZSIsImZpbGVQYXRoIiwicmVjdXJzaXZlIiwiY29weU9yV3JpdGVUb0ZpbGUiLCJyYXciLCJwcm9jZXNzIiwic3Rkb3V0Iiwid3JpdGUiLCJsaW5lQ291bnQiLCJjaGFyQ291bnQiLCJ0cnVuY2F0ZUxpbmUiLCJtYXhMZW4iLCJmaXJzdExpbmUiLCJzcGxpdCIsInJlc3VsdCIsIndpZHRoIiwidGFyZ2V0V2lkdGgiLCJjaGFyIiwiY2hhcldpZHRoIiwiUGlja2VyUHJvcHMiLCJmdWxsVGV4dCIsImNvZGVCbG9ja3MiLCJtZXNzYWdlQWdlIiwib25Eb25lIiwib3B0aW9ucyIsImRpc3BsYXkiLCJQaWNrZXJTZWxlY3Rpb24iLCJDb3B5UGlja2VyIiwidDAiLCIkIiwiX2MiLCJmb2N1c2VkUmVmIiwidDEiLCJ0MiIsImxhYmVsIiwidmFsdWUiLCJjb25zdCIsImRlc2NyaXB0aW9uIiwidDMiLCJ0NCIsIlN5bWJvbCIsImZvciIsIm1hcCIsIl90ZW1wIiwiZ2V0U2VsZWN0aW9uQ29udGVudCIsInNlbGVjdGVkIiwiYmxvY2tfMCIsImJsb2NrIiwiYmxvY2tJbmRleCIsInQ1IiwiaGFuZGxlU2VsZWN0Iiwic2VsZWN0ZWRfMCIsImNvcHlGdWxsUmVzcG9uc2UiLCJfdGVtcDIiLCJibG9ja19jb3VudCIsImFsd2F5cyIsIm1lc3NhZ2VfYWdlIiwic2VsZWN0ZWRfYmxvY2siLCJyZXN1bHRfMCIsInQ2IiwiaGFuZGxlV3JpdGUiLCJzZWxlY3RlZF8xIiwiY29udGVudF8wIiwid3JpdGVfc2hvcnRjdXQiLCJ0NyIsImUiLCJFcnJvciIsImhhbmRsZUtleURvd24iLCJlXzAiLCJrZXkiLCJwcmV2ZW50RGVmYXVsdCIsImN1cnJlbnQiLCJ0OCIsInQ5Iiwic2VsZWN0ZWRfMiIsInQxMCIsInQxMSIsInQxMiIsInQxMyIsImMiLCJpbmRleCIsImJsb2NrTGluZXMiLCJ1bmRlZmluZWQiLCJmaWx0ZXIiLCJCb29sZWFuIiwiY2FsbCIsImNvbnRleHQiLCJhcmdzIiwiYWdlIiwiYXJnIiwidHJpbSIsIm4iLCJOdW1iZXIiLCJpc0ludGVnZXIiLCJjb25maWciXSwic291cmNlcyI6WyJjb3B5LnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBta2Rpciwgd3JpdGVGaWxlIH0gZnJvbSAnZnMvcHJvbWlzZXMnXG5pbXBvcnQgeyBtYXJrZWQsIHR5cGUgVG9rZW5zIH0gZnJvbSAnbWFya2VkJ1xuaW1wb3J0IHsgdG1wZGlyIH0gZnJvbSAnb3MnXG5pbXBvcnQgeyBqb2luIH0gZnJvbSAncGF0aCdcbmltcG9ydCBSZWFjdCwgeyB1c2VSZWYgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB0eXBlIHsgQ29tbWFuZFJlc3VsdERpc3BsYXkgfSBmcm9tICcuLi8uLi9jb21tYW5kcy5qcydcbmltcG9ydCB0eXBlIHsgT3B0aW9uV2l0aERlc2NyaXB0aW9uIH0gZnJvbSAnLi4vLi4vY29tcG9uZW50cy9DdXN0b21TZWxlY3Qvc2VsZWN0LmpzJ1xuaW1wb3J0IHsgU2VsZWN0IH0gZnJvbSAnLi4vLi4vY29tcG9uZW50cy9DdXN0b21TZWxlY3Qvc2VsZWN0LmpzJ1xuaW1wb3J0IHsgQnlsaW5lIH0gZnJvbSAnLi4vLi4vY29tcG9uZW50cy9kZXNpZ24tc3lzdGVtL0J5bGluZS5qcydcbmltcG9ydCB7IEtleWJvYXJkU2hvcnRjdXRIaW50IH0gZnJvbSAnLi4vLi4vY29tcG9uZW50cy9kZXNpZ24tc3lzdGVtL0tleWJvYXJkU2hvcnRjdXRIaW50LmpzJ1xuaW1wb3J0IHsgUGFuZSB9IGZyb20gJy4uLy4uL2NvbXBvbmVudHMvZGVzaWduLXN5c3RlbS9QYW5lLmpzJ1xuaW1wb3J0IHR5cGUgeyBLZXlib2FyZEV2ZW50IH0gZnJvbSAnLi4vLi4vaW5rL2V2ZW50cy9rZXlib2FyZC1ldmVudC5qcydcbmltcG9ydCB7IHN0cmluZ1dpZHRoIH0gZnJvbSAnLi4vLi4vaW5rL3N0cmluZ1dpZHRoLmpzJ1xuaW1wb3J0IHsgc2V0Q2xpcGJvYXJkIH0gZnJvbSAnLi4vLi4vaW5rL3Rlcm1pby9vc2MuanMnXG5pbXBvcnQgeyBCb3gsIFRleHQgfSBmcm9tICcuLi8uLi9pbmsuanMnXG5pbXBvcnQgeyBsb2dFdmVudCB9IGZyb20gJy4uLy4uL3NlcnZpY2VzL2FuYWx5dGljcy9pbmRleC5qcydcbmltcG9ydCB0eXBlIHsgTG9jYWxKU1hDb21tYW5kQ2FsbCB9IGZyb20gJy4uLy4uL3R5cGVzL2NvbW1hbmQuanMnXG5pbXBvcnQgdHlwZSB7IEFzc2lzdGFudE1lc3NhZ2UsIE1lc3NhZ2UgfSBmcm9tICcuLi8uLi90eXBlcy9tZXNzYWdlLmpzJ1xuaW1wb3J0IHsgZ2V0R2xvYmFsQ29uZmlnLCBzYXZlR2xvYmFsQ29uZmlnIH0gZnJvbSAnLi4vLi4vdXRpbHMvY29uZmlnLmpzJ1xuaW1wb3J0IHsgZXh0cmFjdFRleHRDb250ZW50LCBzdHJpcFByb21wdFhNTFRhZ3MgfSBmcm9tICcuLi8uLi91dGlscy9tZXNzYWdlcy5qcydcbmltcG9ydCB7IGNvdW50Q2hhckluU3RyaW5nIH0gZnJvbSAnLi4vLi4vdXRpbHMvc3RyaW5nVXRpbHMuanMnXG5cbmNvbnN0IENPUFlfRElSID0gam9pbih0bXBkaXIoKSwgJ2NsYXVkZScpXG5jb25zdCBSRVNQT05TRV9GSUxFTkFNRSA9ICdyZXNwb25zZS5tZCdcbmNvbnN0IE1BWF9MT09LQkFDSyA9IDIwXG5cbnR5cGUgQ29kZUJsb2NrID0ge1xuICBjb2RlOiBzdHJpbmdcbiAgbGFuZzogc3RyaW5nIHwgdW5kZWZpbmVkXG59XG5cbmZ1bmN0aW9uIGV4dHJhY3RDb2RlQmxvY2tzKG1hcmtkb3duOiBzdHJpbmcpOiBDb2RlQmxvY2tbXSB7XG4gIGNvbnN0IHRva2VucyA9IG1hcmtlZC5sZXhlcihzdHJpcFByb21wdFhNTFRhZ3MobWFya2Rvd24pKVxuICBjb25zdCBibG9ja3M6IENvZGVCbG9ja1tdID0gW11cbiAgZm9yIChjb25zdCB0b2tlbiBvZiB0b2tlbnMpIHtcbiAgICBpZiAodG9rZW4udHlwZSA9PT0gJ2NvZGUnKSB7XG4gICAgICBjb25zdCBjb2RlVG9rZW4gPSB0b2tlbiBhcyBUb2tlbnMuQ29kZVxuICAgICAgYmxvY2tzLnB1c2goeyBjb2RlOiBjb2RlVG9rZW4udGV4dCwgbGFuZzogY29kZVRva2VuLmxhbmcgfSlcbiAgICB9XG4gIH1cbiAgcmV0dXJuIGJsb2Nrc1xufVxuXG4vKipcbiAqIFdhbGsgbWVzc2FnZXMgbmV3ZXN0LWZpcnN0LCByZXR1cm5pbmcgdGV4dCBmcm9tIGFzc2lzdGFudCBtZXNzYWdlcyB0aGF0XG4gKiBhY3R1YWxseSBzYWlkIHNvbWV0aGluZyAoc2tpcHMgdG9vbC11c2Utb25seSB0dXJucyBhbmQgQVBJIGVycm9ycykuXG4gKiBJbmRleCAwID0gbGF0ZXN0LCAxID0gc2Vjb25kLXRvLWxhdGVzdCwgZXRjLiBDYXBzIGF0IE1BWF9MT09LQkFDSy5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGNvbGxlY3RSZWNlbnRBc3Npc3RhbnRUZXh0cyhtZXNzYWdlczogTWVzc2FnZVtdKTogc3RyaW5nW10ge1xuICBjb25zdCB0ZXh0czogc3RyaW5nW10gPSBbXVxuICBmb3IgKFxuICAgIGxldCBpID0gbWVzc2FnZXMubGVuZ3RoIC0gMTtcbiAgICBpID49IDAgJiYgdGV4dHMubGVuZ3RoIDwgTUFYX0xPT0tCQUNLO1xuICAgIGktLVxuICApIHtcbiAgICBjb25zdCBtc2cgPSBtZXNzYWdlc1tpXVxuICAgIGlmIChtc2c/LnR5cGUgIT09ICdhc3Npc3RhbnQnIHx8IG1zZy5pc0FwaUVycm9yTWVzc2FnZSkgY29udGludWVcbiAgICBjb25zdCBjb250ZW50ID0gKG1zZyBhcyBBc3Npc3RhbnRNZXNzYWdlKS5tZXNzYWdlLmNvbnRlbnRcbiAgICBpZiAoIUFycmF5LmlzQXJyYXkoY29udGVudCkpIGNvbnRpbnVlXG4gICAgY29uc3QgdGV4dCA9IGV4dHJhY3RUZXh0Q29udGVudChjb250ZW50LCAnXFxuXFxuJylcbiAgICBpZiAodGV4dCkgdGV4dHMucHVzaCh0ZXh0KVxuICB9XG4gIHJldHVybiB0ZXh0c1xufVxuXG5leHBvcnQgZnVuY3Rpb24gZmlsZUV4dGVuc2lvbihsYW5nOiBzdHJpbmcgfCB1bmRlZmluZWQpOiBzdHJpbmcge1xuICBpZiAobGFuZykge1xuICAgIC8vIFNhbml0aXplIHRvIHByZXZlbnQgcGF0aCB0cmF2ZXJzYWwgKGUuZy4gYGBgLi4vLi4vZXRjL3Bhc3N3ZClcbiAgICAvLyBMYW5ndWFnZSBpZGVudGlmaWVycyBhcmUgYWxwaGFudW1lcmljOiBweXRob24sIHRzeCwganNvbmMsIGV0Yy5cbiAgICBjb25zdCBzYW5pdGl6ZWQgPSBsYW5nLnJlcGxhY2UoL1teYS16QS1aMC05XS9nLCAnJylcbiAgICBpZiAoc2FuaXRpemVkICYmIHNhbml0aXplZCAhPT0gJ3BsYWludGV4dCcpIHtcbiAgICAgIHJldHVybiBgLiR7c2FuaXRpemVkfWBcbiAgICB9XG4gIH1cbiAgcmV0dXJuICcudHh0J1xufVxuXG5hc3luYyBmdW5jdGlvbiB3cml0ZVRvRmlsZSh0ZXh0OiBzdHJpbmcsIGZpbGVuYW1lOiBzdHJpbmcpOiBQcm9taXNlPHN0cmluZz4ge1xuICBjb25zdCBmaWxlUGF0aCA9IGpvaW4oQ09QWV9ESVIsIGZpbGVuYW1lKVxuICBhd2FpdCBta2RpcihDT1BZX0RJUiwgeyByZWN1cnNpdmU6IHRydWUgfSlcbiAgYXdhaXQgd3JpdGVGaWxlKGZpbGVQYXRoLCB0ZXh0LCAndXRmLTgnKVxuICByZXR1cm4gZmlsZVBhdGhcbn1cblxuYXN5bmMgZnVuY3Rpb24gY29weU9yV3JpdGVUb0ZpbGUoXG4gIHRleHQ6IHN0cmluZyxcbiAgZmlsZW5hbWU6IHN0cmluZyxcbik6IFByb21pc2U8c3RyaW5nPiB7XG4gIGNvbnN0IHJhdyA9IGF3YWl0IHNldENsaXBib2FyZCh0ZXh0KVxuICBpZiAocmF3KSBwcm9jZXNzLnN0ZG91dC53cml0ZShyYXcpXG4gIGNvbnN0IGxpbmVDb3VudCA9IGNvdW50Q2hhckluU3RyaW5nKHRleHQsICdcXG4nKSArIDFcbiAgY29uc3QgY2hhckNvdW50ID0gdGV4dC5sZW5ndGhcbiAgLy8gQWxzbyB3cml0ZSB0byBhIHRlbXAgZmlsZSDigJQgY2xpcGJvYXJkIHBhdGhzIGFyZSBiZXN0LWVmZm9ydCAoT1NDIDUyIG5lZWRzXG4gIC8vIHRlcm1pbmFsIHN1cHBvcnQpLCBzbyB0aGUgZmlsZSBwcm92aWRlcyBhIHJlbGlhYmxlIGZhbGxiYWNrLlxuICB0cnkge1xuICAgIGNvbnN0IGZpbGVQYXRoID0gYXdhaXQgd3JpdGVUb0ZpbGUodGV4dCwgZmlsZW5hbWUpXG4gICAgcmV0dXJuIGBDb3BpZWQgdG8gY2xpcGJvYXJkICgke2NoYXJDb3VudH0gY2hhcmFjdGVycywgJHtsaW5lQ291bnR9IGxpbmVzKVxcbkFsc28gd3JpdHRlbiB0byAke2ZpbGVQYXRofWBcbiAgfSBjYXRjaCB7XG4gICAgcmV0dXJuIGBDb3BpZWQgdG8gY2xpcGJvYXJkICgke2NoYXJDb3VudH0gY2hhcmFjdGVycywgJHtsaW5lQ291bnR9IGxpbmVzKWBcbiAgfVxufVxuXG5mdW5jdGlvbiB0cnVuY2F0ZUxpbmUodGV4dDogc3RyaW5nLCBtYXhMZW46IG51bWJlcik6IHN0cmluZyB7XG4gIGNvbnN0IGZpcnN0TGluZSA9IHRleHQuc3BsaXQoJ1xcbicpWzBdID8/ICcnXG4gIGlmIChzdHJpbmdXaWR0aChmaXJzdExpbmUpIDw9IG1heExlbikge1xuICAgIHJldHVybiBmaXJzdExpbmVcbiAgfVxuICBsZXQgcmVzdWx0ID0gJydcbiAgbGV0IHdpZHRoID0gMFxuICBjb25zdCB0YXJnZXRXaWR0aCA9IG1heExlbiAtIDFcbiAgZm9yIChjb25zdCBjaGFyIG9mIGZpcnN0TGluZSkge1xuICAgIGNvbnN0IGNoYXJXaWR0aCA9IHN0cmluZ1dpZHRoKGNoYXIpXG4gICAgaWYgKHdpZHRoICsgY2hhcldpZHRoID4gdGFyZ2V0V2lkdGgpIGJyZWFrXG4gICAgcmVzdWx0ICs9IGNoYXJcbiAgICB3aWR0aCArPSBjaGFyV2lkdGhcbiAgfVxuICByZXR1cm4gcmVzdWx0ICsgJ1xcdTIwMjYnXG59XG5cbnR5cGUgUGlja2VyUHJvcHMgPSB7XG4gIGZ1bGxUZXh0OiBzdHJpbmdcbiAgY29kZUJsb2NrczogQ29kZUJsb2NrW11cbiAgbWVzc2FnZUFnZTogbnVtYmVyXG4gIG9uRG9uZTogKFxuICAgIHJlc3VsdD86IHN0cmluZyxcbiAgICBvcHRpb25zPzogeyBkaXNwbGF5PzogQ29tbWFuZFJlc3VsdERpc3BsYXkgfSxcbiAgKSA9PiB2b2lkXG59XG5cbnR5cGUgUGlja2VyU2VsZWN0aW9uID0gbnVtYmVyIHwgJ2Z1bGwnIHwgJ2Fsd2F5cydcblxuZnVuY3Rpb24gQ29weVBpY2tlcih7XG4gIGZ1bGxUZXh0LFxuICBjb2RlQmxvY2tzLFxuICBtZXNzYWdlQWdlLFxuICBvbkRvbmUsXG59OiBQaWNrZXJQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IGZvY3VzZWRSZWYgPSB1c2VSZWY8UGlja2VyU2VsZWN0aW9uPignZnVsbCcpXG5cbiAgY29uc3Qgb3B0aW9uczogT3B0aW9uV2l0aERlc2NyaXB0aW9uPFBpY2tlclNlbGVjdGlvbj5bXSA9IFtcbiAgICB7XG4gICAgICBsYWJlbDogJ0Z1bGwgcmVzcG9uc2UnLFxuICAgICAgdmFsdWU6ICdmdWxsJyBhcyBjb25zdCxcbiAgICAgIGRlc2NyaXB0aW9uOiBgJHtmdWxsVGV4dC5sZW5ndGh9IGNoYXJzLCAke2NvdW50Q2hhckluU3RyaW5nKGZ1bGxUZXh0LCAnXFxuJykgKyAxfSBsaW5lc2AsXG4gICAgfSxcbiAgICAuLi5jb2RlQmxvY2tzLm1hcCgoYmxvY2ssIGluZGV4KSA9PiB7XG4gICAgICBjb25zdCBibG9ja0xpbmVzID0gY291bnRDaGFySW5TdHJpbmcoYmxvY2suY29kZSwgJ1xcbicpICsgMVxuICAgICAgcmV0dXJuIHtcbiAgICAgICAgbGFiZWw6IHRydW5jYXRlTGluZShibG9jay5jb2RlLCA2MCksXG4gICAgICAgIHZhbHVlOiBpbmRleCxcbiAgICAgICAgZGVzY3JpcHRpb246XG4gICAgICAgICAgW2Jsb2NrLmxhbmcsIGJsb2NrTGluZXMgPiAxID8gYCR7YmxvY2tMaW5lc30gbGluZXNgIDogdW5kZWZpbmVkXVxuICAgICAgICAgICAgLmZpbHRlcihCb29sZWFuKVxuICAgICAgICAgICAgLmpvaW4oJywgJykgfHwgdW5kZWZpbmVkLFxuICAgICAgfVxuICAgIH0pLFxuICAgIHtcbiAgICAgIGxhYmVsOiAnQWx3YXlzIGNvcHkgZnVsbCByZXNwb25zZScsXG4gICAgICB2YWx1ZTogJ2Fsd2F5cycgYXMgY29uc3QsXG4gICAgICBkZXNjcmlwdGlvbjogJ1NraXAgdGhpcyBwaWNrZXIgaW4gdGhlIGZ1dHVyZSAocmV2ZXJ0IHZpYSAvY29uZmlnKScsXG4gICAgfSxcbiAgXVxuXG4gIGZ1bmN0aW9uIGdldFNlbGVjdGlvbkNvbnRlbnQoc2VsZWN0ZWQ6IFBpY2tlclNlbGVjdGlvbik6IHtcbiAgICB0ZXh0OiBzdHJpbmdcbiAgICBmaWxlbmFtZTogc3RyaW5nXG4gICAgYmxvY2tJbmRleD86IG51bWJlclxuICB9IHtcbiAgICBpZiAoc2VsZWN0ZWQgPT09ICdmdWxsJyB8fCBzZWxlY3RlZCA9PT0gJ2Fsd2F5cycpIHtcbiAgICAgIHJldHVybiB7IHRleHQ6IGZ1bGxUZXh0LCBmaWxlbmFtZTogUkVTUE9OU0VfRklMRU5BTUUgfVxuICAgIH1cbiAgICBjb25zdCBibG9jayA9IGNvZGVCbG9ja3Nbc2VsZWN0ZWRdIVxuICAgIHJldHVybiB7XG4gICAgICB0ZXh0OiBibG9jay5jb2RlLFxuICAgICAgZmlsZW5hbWU6IGBjb3B5JHtmaWxlRXh0ZW5zaW9uKGJsb2NrLmxhbmcpfWAsXG4gICAgICBibG9ja0luZGV4OiBzZWxlY3RlZCxcbiAgICB9XG4gIH1cblxuICBhc3luYyBmdW5jdGlvbiBoYW5kbGVTZWxlY3Qoc2VsZWN0ZWQ6IFBpY2tlclNlbGVjdGlvbik6IFByb21pc2U8dm9pZD4ge1xuICAgIGNvbnN0IGNvbnRlbnQgPSBnZXRTZWxlY3Rpb25Db250ZW50KHNlbGVjdGVkKVxuICAgIGlmIChzZWxlY3RlZCA9PT0gJ2Fsd2F5cycpIHtcbiAgICAgIGlmICghZ2V0R2xvYmFsQ29uZmlnKCkuY29weUZ1bGxSZXNwb25zZSkge1xuICAgICAgICBzYXZlR2xvYmFsQ29uZmlnKGMgPT4gKHsgLi4uYywgY29weUZ1bGxSZXNwb25zZTogdHJ1ZSB9KSlcbiAgICAgIH1cbiAgICAgIGxvZ0V2ZW50KCd0ZW5ndV9jb3B5Jywge1xuICAgICAgICBibG9ja19jb3VudDogY29kZUJsb2Nrcy5sZW5ndGgsXG4gICAgICAgIGFsd2F5czogdHJ1ZSxcbiAgICAgICAgbWVzc2FnZV9hZ2U6IG1lc3NhZ2VBZ2UsXG4gICAgICB9KVxuICAgICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgY29weU9yV3JpdGVUb0ZpbGUoY29udGVudC50ZXh0LCBjb250ZW50LmZpbGVuYW1lKVxuICAgICAgb25Eb25lKFxuICAgICAgICBgJHtyZXN1bHR9XFxuUHJlZmVyZW5jZSBzYXZlZC4gVXNlIC9jb25maWcgdG8gY2hhbmdlIGNvcHlGdWxsUmVzcG9uc2VgLFxuICAgICAgKVxuICAgICAgcmV0dXJuXG4gICAgfVxuICAgIGxvZ0V2ZW50KCd0ZW5ndV9jb3B5Jywge1xuICAgICAgc2VsZWN0ZWRfYmxvY2s6IGNvbnRlbnQuYmxvY2tJbmRleCxcbiAgICAgIGJsb2NrX2NvdW50OiBjb2RlQmxvY2tzLmxlbmd0aCxcbiAgICAgIG1lc3NhZ2VfYWdlOiBtZXNzYWdlQWdlLFxuICAgIH0pXG4gICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgY29weU9yV3JpdGVUb0ZpbGUoY29udGVudC50ZXh0LCBjb250ZW50LmZpbGVuYW1lKVxuICAgIG9uRG9uZShyZXN1bHQpXG4gIH1cblxuICBhc3luYyBmdW5jdGlvbiBoYW5kbGVXcml0ZShzZWxlY3RlZDogUGlja2VyU2VsZWN0aW9uKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgY29uc3QgY29udGVudCA9IGdldFNlbGVjdGlvbkNvbnRlbnQoc2VsZWN0ZWQpXG4gICAgbG9nRXZlbnQoJ3Rlbmd1X2NvcHknLCB7XG4gICAgICBzZWxlY3RlZF9ibG9jazogY29udGVudC5ibG9ja0luZGV4LFxuICAgICAgYmxvY2tfY291bnQ6IGNvZGVCbG9ja3MubGVuZ3RoLFxuICAgICAgbWVzc2FnZV9hZ2U6IG1lc3NhZ2VBZ2UsXG4gICAgICB3cml0ZV9zaG9ydGN1dDogdHJ1ZSxcbiAgICB9KVxuICAgIHRyeSB7XG4gICAgICBjb25zdCBmaWxlUGF0aCA9IGF3YWl0IHdyaXRlVG9GaWxlKGNvbnRlbnQudGV4dCwgY29udGVudC5maWxlbmFtZSlcbiAgICAgIG9uRG9uZShgV3JpdHRlbiB0byAke2ZpbGVQYXRofWApXG4gICAgfSBjYXRjaCAoZSkge1xuICAgICAgb25Eb25lKGBGYWlsZWQgdG8gd3JpdGUgZmlsZTogJHtlIGluc3RhbmNlb2YgRXJyb3IgPyBlLm1lc3NhZ2UgOiBlfWApXG4gICAgfVxuICB9XG5cbiAgZnVuY3Rpb24gaGFuZGxlS2V5RG93bihlOiBLZXlib2FyZEV2ZW50KTogdm9pZCB7XG4gICAgaWYgKGUua2V5ID09PSAndycpIHtcbiAgICAgIGUucHJldmVudERlZmF1bHQoKVxuICAgICAgdm9pZCBoYW5kbGVXcml0ZShmb2N1c2VkUmVmLmN1cnJlbnQpXG4gICAgfVxuICB9XG5cbiAgcmV0dXJuIChcbiAgICA8UGFuZT5cbiAgICAgIDxCb3hcbiAgICAgICAgZmxleERpcmVjdGlvbj1cImNvbHVtblwiXG4gICAgICAgIGdhcD17MX1cbiAgICAgICAgdGFiSW5kZXg9ezB9XG4gICAgICAgIGF1dG9Gb2N1c1xuICAgICAgICBvbktleURvd249e2hhbmRsZUtleURvd259XG4gICAgICA+XG4gICAgICAgIDxUZXh0IGRpbUNvbG9yPlNlbGVjdCBjb250ZW50IHRvIGNvcHk6PC9UZXh0PlxuICAgICAgICA8U2VsZWN0PFBpY2tlclNlbGVjdGlvbj5cbiAgICAgICAgICBvcHRpb25zPXtvcHRpb25zfVxuICAgICAgICAgIGhpZGVJbmRleGVzPXtmYWxzZX1cbiAgICAgICAgICBvbkZvY3VzPXt2YWx1ZSA9PiB7XG4gICAgICAgICAgICBmb2N1c2VkUmVmLmN1cnJlbnQgPSB2YWx1ZVxuICAgICAgICAgIH19XG4gICAgICAgICAgb25DaGFuZ2U9e3NlbGVjdGVkID0+IHtcbiAgICAgICAgICAgIHZvaWQgaGFuZGxlU2VsZWN0KHNlbGVjdGVkKVxuICAgICAgICAgIH19XG4gICAgICAgICAgb25DYW5jZWw9eygpID0+IHtcbiAgICAgICAgICAgIG9uRG9uZSgnQ29weSBjYW5jZWxsZWQnLCB7IGRpc3BsYXk6ICdzeXN0ZW0nIH0pXG4gICAgICAgICAgfX1cbiAgICAgICAgLz5cbiAgICAgICAgPFRleHQgZGltQ29sb3I+XG4gICAgICAgICAgPEJ5bGluZT5cbiAgICAgICAgICAgIDxLZXlib2FyZFNob3J0Y3V0SGludCBzaG9ydGN1dD1cImVudGVyXCIgYWN0aW9uPVwiY29weVwiIC8+XG4gICAgICAgICAgICA8S2V5Ym9hcmRTaG9ydGN1dEhpbnQgc2hvcnRjdXQ9XCJ3XCIgYWN0aW9uPVwid3JpdGUgdG8gZmlsZVwiIC8+XG4gICAgICAgICAgICA8S2V5Ym9hcmRTaG9ydGN1dEhpbnQgc2hvcnRjdXQ9XCJlc2NcIiBhY3Rpb249XCJjYW5jZWxcIiAvPlxuICAgICAgICAgIDwvQnlsaW5lPlxuICAgICAgICA8L1RleHQ+XG4gICAgICA8L0JveD5cbiAgICA8L1BhbmU+XG4gIClcbn1cblxuZXhwb3J0IGNvbnN0IGNhbGw6IExvY2FsSlNYQ29tbWFuZENhbGwgPSBhc3luYyAob25Eb25lLCBjb250ZXh0LCBhcmdzKSA9PiB7XG4gIGNvbnN0IHRleHRzID0gY29sbGVjdFJlY2VudEFzc2lzdGFudFRleHRzKGNvbnRleHQubWVzc2FnZXMpXG5cbiAgaWYgKHRleHRzLmxlbmd0aCA9PT0gMCkge1xuICAgIG9uRG9uZSgnTm8gYXNzaXN0YW50IG1lc3NhZ2UgdG8gY29weScpXG4gICAgcmV0dXJuIG51bGxcbiAgfVxuXG4gIC8vIC9jb3B5IE4gcmVhY2hlcyBiYWNrIE4tMSBtZXNzYWdlcyAoMSA9IGxhdGVzdCwgMiA9IHNlY29uZC10by1sYXRlc3QsIC4uLilcbiAgbGV0IGFnZSA9IDBcbiAgY29uc3QgYXJnID0gYXJncz8udHJpbSgpXG4gIGlmIChhcmcpIHtcbiAgICBjb25zdCBuID0gTnVtYmVyKGFyZylcbiAgICBpZiAoIU51bWJlci5pc0ludGVnZXIobikgfHwgbiA8IDEpIHtcbiAgICAgIG9uRG9uZShgVXNhZ2U6IC9jb3B5IFtOXSB3aGVyZSBOIGlzIDEgKGxhdGVzdCksIDIsIDMsIFxcdTIwMjYgR290OiAke2FyZ31gKVxuICAgICAgcmV0dXJuIG51bGxcbiAgICB9XG4gICAgaWYgKG4gPiB0ZXh0cy5sZW5ndGgpIHtcbiAgICAgIG9uRG9uZShcbiAgICAgICAgYE9ubHkgJHt0ZXh0cy5sZW5ndGh9IGFzc2lzdGFudCAke3RleHRzLmxlbmd0aCA9PT0gMSA/ICdtZXNzYWdlJyA6ICdtZXNzYWdlcyd9IGF2YWlsYWJsZSB0byBjb3B5YCxcbiAgICAgIClcbiAgICAgIHJldHVybiBudWxsXG4gICAgfVxuICAgIGFnZSA9IG4gLSAxXG4gIH1cblxuICBjb25zdCB0ZXh0ID0gdGV4dHNbYWdlXSFcbiAgY29uc3QgY29kZUJsb2NrcyA9IGV4dHJhY3RDb2RlQmxvY2tzKHRleHQpXG4gIGNvbnN0IGNvbmZpZyA9IGdldEdsb2JhbENvbmZpZygpXG5cbiAgaWYgKGNvZGVCbG9ja3MubGVuZ3RoID09PSAwIHx8IGNvbmZpZy5jb3B5RnVsbFJlc3BvbnNlKSB7XG4gICAgbG9nRXZlbnQoJ3Rlbmd1X2NvcHknLCB7XG4gICAgICBhbHdheXM6IGNvbmZpZy5jb3B5RnVsbFJlc3BvbnNlLFxuICAgICAgYmxvY2tfY291bnQ6IGNvZGVCbG9ja3MubGVuZ3RoLFxuICAgICAgbWVzc2FnZV9hZ2U6IGFnZSxcbiAgICB9KVxuICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IGNvcHlPcldyaXRlVG9GaWxlKHRleHQsIFJFU1BPTlNFX0ZJTEVOQU1FKVxuICAgIG9uRG9uZShyZXN1bHQpXG4gICAgcmV0dXJuIG51bGxcbiAgfVxuXG4gIHJldHVybiAoXG4gICAgPENvcHlQaWNrZXJcbiAgICAgIGZ1bGxUZXh0PXt0ZXh0fVxuICAgICAgY29kZUJsb2Nrcz17Y29kZUJsb2Nrc31cbiAgICAgIG1lc3NhZ2VBZ2U9e2FnZX1cbiAgICAgIG9uRG9uZT17b25Eb25lfVxuICAgIC8+XG4gIClcbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLFNBQVNBLEtBQUssRUFBRUMsU0FBUyxRQUFRLGFBQWE7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 \ No newline at end of file diff --git a/src/commands/copy/index.ts b/src/commands/copy/index.ts new file mode 100644 index 0000000..092c70e --- /dev/null +++ b/src/commands/copy/index.ts @@ -0,0 +1,15 @@ +/** + * Copy command - minimal metadata only. + * Implementation is lazy-loaded from copy.tsx to reduce startup time. + */ +import type { Command } from '../../commands.js' + +const copy = { + type: 'local-jsx', + name: 'copy', + description: + "Copy Claude's last response to clipboard (or /copy N for the Nth-latest)", + load: () => import('./copy.js'), +} satisfies Command + +export default copy diff --git a/src/commands/cost/cost.ts b/src/commands/cost/cost.ts new file mode 100644 index 0000000..c9fb0cb --- /dev/null +++ b/src/commands/cost/cost.ts @@ -0,0 +1,24 @@ +import { formatTotalCost } from '../../cost-tracker.js' +import { currentLimits } from '../../services/claudeAiLimits.js' +import type { LocalCommandCall } from '../../types/command.js' +import { isClaudeAISubscriber } from '../../utils/auth.js' + +export const call: LocalCommandCall = async () => { + if (isClaudeAISubscriber()) { + let value: string + + if (currentLimits.isUsingOverage) { + value = + 'You are currently using your overages to power your Claude Code usage. We will automatically switch you back to your subscription rate limits when they reset' + } else { + value = + 'You are currently using your subscription to power your Claude Code usage' + } + + if (process.env.USER_TYPE === 'ant') { + value += `\n\n[ANT-ONLY] Showing cost anyway:\n ${formatTotalCost()}` + } + return { type: 'text', value } + } + return { type: 'text', value: formatTotalCost() } +} diff --git a/src/commands/cost/index.ts b/src/commands/cost/index.ts new file mode 100644 index 0000000..d1c2d23 --- /dev/null +++ b/src/commands/cost/index.ts @@ -0,0 +1,23 @@ +/** + * Cost command - minimal metadata only. + * Implementation is lazy-loaded from cost.ts to reduce startup time. + */ +import type { Command } from '../../commands.js' +import { isClaudeAISubscriber } from '../../utils/auth.js' + +const cost = { + type: 'local', + name: 'cost', + description: 'Show the total cost and duration of the current session', + get isHidden() { + // Keep visible for Ants even if they're subscribers (they see cost breakdowns) + if (process.env.USER_TYPE === 'ant') { + return false + } + return isClaudeAISubscriber() + }, + supportsNonInteractive: true, + load: () => import('./cost.js'), +} satisfies Command + +export default cost diff --git a/src/commands/createMovedToPluginCommand.ts b/src/commands/createMovedToPluginCommand.ts new file mode 100644 index 0000000..08dee29 --- /dev/null +++ b/src/commands/createMovedToPluginCommand.ts @@ -0,0 +1,65 @@ +import type { ContentBlockParam } from '@anthropic-ai/sdk/resources/messages.js' +import type { Command } from '../commands.js' +import type { ToolUseContext } from '../Tool.js' + +type Options = { + name: string + description: string + progressMessage: string + pluginName: string + pluginCommand: string + /** + * The prompt to use while the marketplace is private. + * External users will get this prompt. Once the marketplace is public, + * this parameter and the fallback logic can be removed. + */ + getPromptWhileMarketplaceIsPrivate: ( + args: string, + context: ToolUseContext, + ) => Promise +} + +export function createMovedToPluginCommand({ + name, + description, + progressMessage, + pluginName, + pluginCommand, + getPromptWhileMarketplaceIsPrivate, +}: Options): Command { + return { + type: 'prompt', + name, + description, + progressMessage, + contentLength: 0, // Dynamic content + userFacingName() { + return name + }, + source: 'builtin', + async getPromptForCommand( + args: string, + context: ToolUseContext, + ): Promise { + if (process.env.USER_TYPE === 'ant') { + return [ + { + type: 'text', + text: `This command has been moved to a plugin. Tell the user: + +1. To install the plugin, run: + claude plugin install ${pluginName}@claude-code-marketplace + +2. After installation, use /${pluginName}:${pluginCommand} to run this command + +3. For more information, see: https://github.com/anthropics/claude-code-marketplace/blob/main/${pluginName}/README.md + +Do not attempt to run the command. Simply inform the user about the plugin installation.`, + }, + ] + } + + return getPromptWhileMarketplaceIsPrivate(args, context) + }, + } +} diff --git a/src/commands/ctx_viz/index.js b/src/commands/ctx_viz/index.js new file mode 100644 index 0000000..e1a619d --- /dev/null +++ b/src/commands/ctx_viz/index.js @@ -0,0 +1 @@ +export default { isEnabled: () => false, isHidden: true, name: 'stub' }; diff --git a/src/commands/debug-tool-call/index.js b/src/commands/debug-tool-call/index.js new file mode 100644 index 0000000..e1a619d --- /dev/null +++ b/src/commands/debug-tool-call/index.js @@ -0,0 +1 @@ +export default { isEnabled: () => false, isHidden: true, name: 'stub' }; diff --git a/src/commands/desktop/desktop.tsx b/src/commands/desktop/desktop.tsx new file mode 100644 index 0000000..a449c90 --- /dev/null +++ b/src/commands/desktop/desktop.tsx @@ -0,0 +1,9 @@ +import React from 'react'; +import type { CommandResultDisplay } from '../../commands.js'; +import { DesktopHandoff } from '../../components/DesktopHandoff.js'; +export async function call(onDone: (result?: string, options?: { + display?: CommandResultDisplay; +}) => void): Promise { + return ; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkNvbW1hbmRSZXN1bHREaXNwbGF5IiwiRGVza3RvcEhhbmRvZmYiLCJjYWxsIiwib25Eb25lIiwicmVzdWx0Iiwib3B0aW9ucyIsImRpc3BsYXkiLCJQcm9taXNlIiwiUmVhY3ROb2RlIl0sInNvdXJjZXMiOlsiZGVza3RvcC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHR5cGUgeyBDb21tYW5kUmVzdWx0RGlzcGxheSB9IGZyb20gJy4uLy4uL2NvbW1hbmRzLmpzJ1xuaW1wb3J0IHsgRGVza3RvcEhhbmRvZmYgfSBmcm9tICcuLi8uLi9jb21wb25lbnRzL0Rlc2t0b3BIYW5kb2ZmLmpzJ1xuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gY2FsbChcbiAgb25Eb25lOiAoXG4gICAgcmVzdWx0Pzogc3RyaW5nLFxuICAgIG9wdGlvbnM/OiB7IGRpc3BsYXk/OiBDb21tYW5kUmVzdWx0RGlzcGxheSB9LFxuICApID0+IHZvaWQsXG4pOiBQcm9taXNlPFJlYWN0LlJlYWN0Tm9kZT4ge1xuICByZXR1cm4gPERlc2t0b3BIYW5kb2ZmIG9uRG9uZT17b25Eb25lfSAvPlxufVxuIl0sIm1hcHBpbmdzIjoiQUFBQSxPQUFPQSxLQUFLLE1BQU0sT0FBTztBQUN6QixjQUFjQyxvQkFBb0IsUUFBUSxtQkFBbUI7QUFDN0QsU0FBU0MsY0FBYyxRQUFRLG9DQUFvQztBQUVuRSxPQUFPLGVBQWVDLElBQUlBLENBQ3hCQyxNQUFNLEVBQUUsQ0FDTkMsTUFBZSxDQUFSLEVBQUUsTUFBTSxFQUNmQyxPQUE0QyxDQUFwQyxFQUFFO0VBQUVDLE9BQU8sQ0FBQyxFQUFFTixvQkFBb0I7QUFBQyxDQUFDLEVBQzVDLEdBQUcsSUFBSSxDQUNWLEVBQUVPLE9BQU8sQ0FBQ1IsS0FBSyxDQUFDUyxTQUFTLENBQUMsQ0FBQztFQUMxQixPQUFPLENBQUMsY0FBYyxDQUFDLE1BQU0sQ0FBQyxDQUFDTCxNQUFNLENBQUMsR0FBRztBQUMzQyIsImlnbm9yZUxpc3QiOltdfQ== \ No newline at end of file diff --git a/src/commands/desktop/index.ts b/src/commands/desktop/index.ts new file mode 100644 index 0000000..d03c3ae --- /dev/null +++ b/src/commands/desktop/index.ts @@ -0,0 +1,26 @@ +import type { Command } from '../../commands.js' + +function isSupportedPlatform(): boolean { + if (process.platform === 'darwin') { + return true + } + if (process.platform === 'win32' && process.arch === 'x64') { + return true + } + return false +} + +const desktop = { + type: 'local-jsx', + name: 'desktop', + aliases: ['app'], + description: 'Continue the current session in Claude Desktop', + availability: ['claude-ai'], + isEnabled: isSupportedPlatform, + get isHidden() { + return !isSupportedPlatform() + }, + load: () => import('./desktop.js'), +} satisfies Command + +export default desktop diff --git a/src/commands/diff/diff.tsx b/src/commands/diff/diff.tsx new file mode 100644 index 0000000..f31b086 --- /dev/null +++ b/src/commands/diff/diff.tsx @@ -0,0 +1,9 @@ +import * as React from 'react'; +import type { LocalJSXCommandCall } from '../../types/command.js'; +export const call: LocalJSXCommandCall = async (onDone, context) => { + const { + DiffDialog + } = await import('../../components/diff/DiffDialog.js'); + return ; +}; +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkxvY2FsSlNYQ29tbWFuZENhbGwiLCJjYWxsIiwib25Eb25lIiwiY29udGV4dCIsIkRpZmZEaWFsb2ciLCJtZXNzYWdlcyJdLCJzb3VyY2VzIjpbImRpZmYudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHR5cGUgeyBMb2NhbEpTWENvbW1hbmRDYWxsIH0gZnJvbSAnLi4vLi4vdHlwZXMvY29tbWFuZC5qcydcblxuZXhwb3J0IGNvbnN0IGNhbGw6IExvY2FsSlNYQ29tbWFuZENhbGwgPSBhc3luYyAob25Eb25lLCBjb250ZXh0KSA9PiB7XG4gIGNvbnN0IHsgRGlmZkRpYWxvZyB9ID0gYXdhaXQgaW1wb3J0KCcuLi8uLi9jb21wb25lbnRzL2RpZmYvRGlmZkRpYWxvZy5qcycpXG4gIHJldHVybiA8RGlmZkRpYWxvZyBtZXNzYWdlcz17Y29udGV4dC5tZXNzYWdlc30gb25Eb25lPXtvbkRvbmV9IC8+XG59XG4iXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBS0EsS0FBSyxNQUFNLE9BQU87QUFDOUIsY0FBY0MsbUJBQW1CLFFBQVEsd0JBQXdCO0FBRWpFLE9BQU8sTUFBTUMsSUFBSSxFQUFFRCxtQkFBbUIsR0FBRyxNQUFBQyxDQUFPQyxNQUFNLEVBQUVDLE9BQU8sS0FBSztFQUNsRSxNQUFNO0lBQUVDO0VBQVcsQ0FBQyxHQUFHLE1BQU0sTUFBTSxDQUFDLHFDQUFxQyxDQUFDO0VBQzFFLE9BQU8sQ0FBQyxVQUFVLENBQUMsUUFBUSxDQUFDLENBQUNELE9BQU8sQ0FBQ0UsUUFBUSxDQUFDLENBQUMsTUFBTSxDQUFDLENBQUNILE1BQU0sQ0FBQyxHQUFHO0FBQ25FLENBQUMiLCJpZ25vcmVMaXN0IjpbXX0= \ No newline at end of file diff --git a/src/commands/diff/index.ts b/src/commands/diff/index.ts new file mode 100644 index 0000000..a15b819 --- /dev/null +++ b/src/commands/diff/index.ts @@ -0,0 +1,8 @@ +import type { Command } from '../../commands.js' + +export default { + type: 'local-jsx', + name: 'diff', + description: 'View uncommitted changes and per-turn diffs', + load: () => import('./diff.js'), +} satisfies Command diff --git a/src/commands/doctor/doctor.tsx b/src/commands/doctor/doctor.tsx new file mode 100644 index 0000000..447cd40 --- /dev/null +++ b/src/commands/doctor/doctor.tsx @@ -0,0 +1,7 @@ +import React from 'react'; +import { Doctor } from '../../screens/Doctor.js'; +import type { LocalJSXCommandCall } from '../../types/command.js'; +export const call: LocalJSXCommandCall = (onDone, _context, _args) => { + return Promise.resolve(); +}; +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkRvY3RvciIsIkxvY2FsSlNYQ29tbWFuZENhbGwiLCJjYWxsIiwib25Eb25lIiwiX2NvbnRleHQiLCJfYXJncyIsIlByb21pc2UiLCJyZXNvbHZlIl0sInNvdXJjZXMiOlsiZG9jdG9yLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBEb2N0b3IgfSBmcm9tICcuLi8uLi9zY3JlZW5zL0RvY3Rvci5qcydcbmltcG9ydCB0eXBlIHsgTG9jYWxKU1hDb21tYW5kQ2FsbCB9IGZyb20gJy4uLy4uL3R5cGVzL2NvbW1hbmQuanMnXG5cbmV4cG9ydCBjb25zdCBjYWxsOiBMb2NhbEpTWENvbW1hbmRDYWxsID0gKG9uRG9uZSwgX2NvbnRleHQsIF9hcmdzKSA9PiB7XG4gIHJldHVybiBQcm9taXNlLnJlc29sdmUoPERvY3RvciBvbkRvbmU9e29uRG9uZX0gLz4pXG59XG4iXSwibWFwcGluZ3MiOiJBQUFBLE9BQU9BLEtBQUssTUFBTSxPQUFPO0FBQ3pCLFNBQVNDLE1BQU0sUUFBUSx5QkFBeUI7QUFDaEQsY0FBY0MsbUJBQW1CLFFBQVEsd0JBQXdCO0FBRWpFLE9BQU8sTUFBTUMsSUFBSSxFQUFFRCxtQkFBbUIsR0FBR0MsQ0FBQ0MsTUFBTSxFQUFFQyxRQUFRLEVBQUVDLEtBQUssS0FBSztFQUNwRSxPQUFPQyxPQUFPLENBQUNDLE9BQU8sQ0FBQyxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQ0osTUFBTSxDQUFDLEdBQUcsQ0FBQztBQUNwRCxDQUFDIiwiaWdub3JlTGlzdCI6W119 \ No newline at end of file diff --git a/src/commands/doctor/index.ts b/src/commands/doctor/index.ts new file mode 100644 index 0000000..6a0b089 --- /dev/null +++ b/src/commands/doctor/index.ts @@ -0,0 +1,12 @@ +import type { Command } from '../../commands.js' +import { isEnvTruthy } from '../../utils/envUtils.js' + +const doctor: Command = { + name: 'doctor', + description: 'Diagnose and verify your Claude Code installation and settings', + isEnabled: () => !isEnvTruthy(process.env.DISABLE_DOCTOR_COMMAND), + type: 'local-jsx', + load: () => import('./doctor.js'), +} + +export default doctor diff --git a/src/commands/effort/effort.tsx b/src/commands/effort/effort.tsx new file mode 100644 index 0000000..41dd0d8 --- /dev/null +++ b/src/commands/effort/effort.tsx @@ -0,0 +1,183 @@ +import { c as _c } from "react/compiler-runtime"; +import * as React from 'react'; +import { useMainLoopModel } from '../../hooks/useMainLoopModel.js'; +import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from '../../services/analytics/index.js'; +import { useAppState, useSetAppState } from '../../state/AppState.js'; +import type { LocalJSXCommandOnDone } from '../../types/command.js'; +import { type EffortValue, getDisplayedEffortLevel, getEffortEnvOverride, getEffortValueDescription, isEffortLevel, toPersistableEffort } from '../../utils/effort.js'; +import { updateSettingsForSource } from '../../utils/settings/settings.js'; +const COMMON_HELP_ARGS = ['help', '-h', '--help']; +type EffortCommandResult = { + message: string; + effortUpdate?: { + value: EffortValue | undefined; + }; +}; +function setEffortValue(effortValue: EffortValue): EffortCommandResult { + const persistable = toPersistableEffort(effortValue); + if (persistable !== undefined) { + const result = updateSettingsForSource('userSettings', { + effortLevel: persistable + }); + if (result.error) { + return { + message: `Failed to set effort level: ${result.error.message}` + }; + } + } + logEvent('tengu_effort_command', { + effort: effortValue as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + + // Env var wins at resolveAppliedEffort time. Only flag it when it actually + // conflicts — if env matches what the user just asked for, the outcome is + // the same, so "Set effort to X" is true and the note is noise. + const envOverride = getEffortEnvOverride(); + if (envOverride !== undefined && envOverride !== effortValue) { + const envRaw = process.env.CLAUDE_CODE_EFFORT_LEVEL; + if (persistable === undefined) { + return { + message: `Not applied: CLAUDE_CODE_EFFORT_LEVEL=${envRaw} overrides effort this session, and ${effortValue} is session-only (nothing saved)`, + effortUpdate: { + value: effortValue + } + }; + } + return { + message: `CLAUDE_CODE_EFFORT_LEVEL=${envRaw} overrides this session — clear it and ${effortValue} takes over`, + effortUpdate: { + value: effortValue + } + }; + } + const description = getEffortValueDescription(effortValue); + const suffix = persistable !== undefined ? '' : ' (this session only)'; + return { + message: `Set effort level to ${effortValue}${suffix}: ${description}`, + effortUpdate: { + value: effortValue + } + }; +} +export function showCurrentEffort(appStateEffort: EffortValue | undefined, model: string): EffortCommandResult { + const envOverride = getEffortEnvOverride(); + const effectiveValue = envOverride === null ? undefined : envOverride ?? appStateEffort; + if (effectiveValue === undefined) { + const level = getDisplayedEffortLevel(model, appStateEffort); + return { + message: `Effort level: auto (currently ${level})` + }; + } + const description = getEffortValueDescription(effectiveValue); + return { + message: `Current effort level: ${effectiveValue} (${description})` + }; +} +function unsetEffortLevel(): EffortCommandResult { + const result = updateSettingsForSource('userSettings', { + effortLevel: undefined + }); + if (result.error) { + return { + message: `Failed to set effort level: ${result.error.message}` + }; + } + logEvent('tengu_effort_command', { + effort: 'auto' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + // env=auto/unset (null) matches what /effort auto asks for, so only warn + // when env is pinning a specific level that will keep overriding. + const envOverride = getEffortEnvOverride(); + if (envOverride !== undefined && envOverride !== null) { + const envRaw = process.env.CLAUDE_CODE_EFFORT_LEVEL; + return { + message: `Cleared effort from settings, but CLAUDE_CODE_EFFORT_LEVEL=${envRaw} still controls this session`, + effortUpdate: { + value: undefined + } + }; + } + return { + message: 'Effort level set to auto', + effortUpdate: { + value: undefined + } + }; +} +export function executeEffort(args: string): EffortCommandResult { + const normalized = args.toLowerCase(); + if (normalized === 'auto' || normalized === 'unset') { + return unsetEffortLevel(); + } + if (!isEffortLevel(normalized)) { + return { + message: `Invalid argument: ${args}. Valid options are: low, medium, high, max, auto` + }; + } + return setEffortValue(normalized); +} +function ShowCurrentEffort(t0) { + const { + onDone + } = t0; + const effortValue = useAppState(_temp); + const model = useMainLoopModel(); + const { + message + } = showCurrentEffort(effortValue, model); + onDone(message); + return null; +} +function _temp(s) { + return s.effortValue; +} +function ApplyEffortAndClose(t0) { + const $ = _c(6); + const { + result, + onDone + } = t0; + const setAppState = useSetAppState(); + const { + effortUpdate, + message + } = result; + let t1; + let t2; + if ($[0] !== effortUpdate || $[1] !== message || $[2] !== onDone || $[3] !== setAppState) { + t1 = () => { + if (effortUpdate) { + setAppState(prev => ({ + ...prev, + effortValue: effortUpdate.value + })); + } + onDone(message); + }; + t2 = [setAppState, effortUpdate, message, onDone]; + $[0] = effortUpdate; + $[1] = message; + $[2] = onDone; + $[3] = setAppState; + $[4] = t1; + $[5] = t2; + } else { + t1 = $[4]; + t2 = $[5]; + } + React.useEffect(t1, t2); + return null; +} +export async function call(onDone: LocalJSXCommandOnDone, _context: unknown, args?: string): Promise { + args = args?.trim() || ''; + if (COMMON_HELP_ARGS.includes(args)) { + onDone('Usage: /effort [low|medium|high|max|auto]\n\nEffort levels:\n- low: Quick, straightforward implementation\n- medium: Balanced approach with standard testing\n- high: Comprehensive implementation with extensive testing\n- max: Maximum capability with deepest reasoning (Opus 4.6 only)\n- auto: Use the default effort level for your model'); + return; + } + if (!args || args === 'current' || args === 'status') { + return ; + } + const result = executeEffort(args); + return ; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsInVzZU1haW5Mb29wTW9kZWwiLCJBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTIiwibG9nRXZlbnQiLCJ1c2VBcHBTdGF0ZSIsInVzZVNldEFwcFN0YXRlIiwiTG9jYWxKU1hDb21tYW5kT25Eb25lIiwiRWZmb3J0VmFsdWUiLCJnZXREaXNwbGF5ZWRFZmZvcnRMZXZlbCIsImdldEVmZm9ydEVudk92ZXJyaWRlIiwiZ2V0RWZmb3J0VmFsdWVEZXNjcmlwdGlvbiIsImlzRWZmb3J0TGV2ZWwiLCJ0b1BlcnNpc3RhYmxlRWZmb3J0IiwidXBkYXRlU2V0dGluZ3NGb3JTb3VyY2UiLCJDT01NT05fSEVMUF9BUkdTIiwiRWZmb3J0Q29tbWFuZFJlc3VsdCIsIm1lc3NhZ2UiLCJlZmZvcnRVcGRhdGUiLCJ2YWx1ZSIsInNldEVmZm9ydFZhbHVlIiwiZWZmb3J0VmFsdWUiLCJwZXJzaXN0YWJsZSIsInVuZGVmaW5lZCIsInJlc3VsdCIsImVmZm9ydExldmVsIiwiZXJyb3IiLCJlZmZvcnQiLCJlbnZPdmVycmlkZSIsImVudlJhdyIsInByb2Nlc3MiLCJlbnYiLCJDTEFVREVfQ09ERV9FRkZPUlRfTEVWRUwiLCJkZXNjcmlwdGlvbiIsInN1ZmZpeCIsInNob3dDdXJyZW50RWZmb3J0IiwiYXBwU3RhdGVFZmZvcnQiLCJtb2RlbCIsImVmZmVjdGl2ZVZhbHVlIiwibGV2ZWwiLCJ1bnNldEVmZm9ydExldmVsIiwiZXhlY3V0ZUVmZm9ydCIsImFyZ3MiLCJub3JtYWxpemVkIiwidG9Mb3dlckNhc2UiLCJTaG93Q3VycmVudEVmZm9ydCIsInQwIiwib25Eb25lIiwiX3RlbXAiLCJzIiwiQXBwbHlFZmZvcnRBbmRDbG9zZSIsIiQiLCJfYyIsInNldEFwcFN0YXRlIiwidDEiLCJ0MiIsInByZXYiLCJ1c2VFZmZlY3QiLCJjYWxsIiwiX2NvbnRleHQiLCJQcm9taXNlIiwiUmVhY3ROb2RlIiwidHJpbSIsImluY2x1ZGVzIl0sInNvdXJjZXMiOlsiZWZmb3J0LnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZU1haW5Mb29wTW9kZWwgfSBmcm9tICcuLi8uLi9ob29rcy91c2VNYWluTG9vcE1vZGVsLmpzJ1xuaW1wb3J0IHtcbiAgdHlwZSBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICBsb2dFdmVudCxcbn0gZnJvbSAnLi4vLi4vc2VydmljZXMvYW5hbHl0aWNzL2luZGV4LmpzJ1xuaW1wb3J0IHsgdXNlQXBwU3RhdGUsIHVzZVNldEFwcFN0YXRlIH0gZnJvbSAnLi4vLi4vc3RhdGUvQXBwU3RhdGUuanMnXG5pbXBvcnQgdHlwZSB7IExvY2FsSlNYQ29tbWFuZE9uRG9uZSB9IGZyb20gJy4uLy4uL3R5cGVzL2NvbW1hbmQuanMnXG5pbXBvcnQge1xuICB0eXBlIEVmZm9ydFZhbHVlLFxuICBnZXREaXNwbGF5ZWRFZmZvcnRMZXZlbCxcbiAgZ2V0RWZmb3J0RW52T3ZlcnJpZGUsXG4gIGdldEVmZm9ydFZhbHVlRGVzY3JpcHRpb24sXG4gIGlzRWZmb3J0TGV2ZWwsXG4gIHRvUGVyc2lzdGFibGVFZmZvcnQsXG59IGZyb20gJy4uLy4uL3V0aWxzL2VmZm9ydC5qcydcbmltcG9ydCB7IHVwZGF0ZVNldHRpbmdzRm9yU291cmNlIH0gZnJvbSAnLi4vLi4vdXRpbHMvc2V0dGluZ3Mvc2V0dGluZ3MuanMnXG5cbmNvbnN0IENPTU1PTl9IRUxQX0FSR1MgPSBbJ2hlbHAnLCAnLWgnLCAnLS1oZWxwJ11cblxudHlwZSBFZmZvcnRDb21tYW5kUmVzdWx0ID0ge1xuICBtZXNzYWdlOiBzdHJpbmdcbiAgZWZmb3J0VXBkYXRlPzogeyB2YWx1ZTogRWZmb3J0VmFsdWUgfCB1bmRlZmluZWQgfVxufVxuXG5mdW5jdGlvbiBzZXRFZmZvcnRWYWx1ZShlZmZvcnRWYWx1ZTogRWZmb3J0VmFsdWUpOiBFZmZvcnRDb21tYW5kUmVzdWx0IHtcbiAgY29uc3QgcGVyc2lzdGFibGUgPSB0b1BlcnNpc3RhYmxlRWZmb3J0KGVmZm9ydFZhbHVlKVxuICBpZiAocGVyc2lzdGFibGUgIT09IHVuZGVmaW5lZCkge1xuICAgIGNvbnN0IHJlc3VsdCA9IHVwZGF0ZVNldHRpbmdzRm9yU291cmNlKCd1c2VyU2V0dGluZ3MnLCB7XG4gICAgICBlZmZvcnRMZXZlbDogcGVyc2lzdGFibGUsXG4gICAgfSlcbiAgICBpZiAocmVzdWx0LmVycm9yKSB7XG4gICAgICByZXR1cm4ge1xuICAgICAgICBtZXNzYWdlOiBgRmFpbGVkIHRvIHNldCBlZmZvcnQgbGV2ZWw6ICR7cmVzdWx0LmVycm9yLm1lc3NhZ2V9YCxcbiAgICAgIH1cbiAgICB9XG4gIH1cbiAgbG9nRXZlbnQoJ3Rlbmd1X2VmZm9ydF9jb21tYW5kJywge1xuICAgIGVmZm9ydDpcbiAgICAgIGVmZm9ydFZhbHVlIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gIH0pXG5cbiAgLy8gRW52IHZhciB3aW5zIGF0IHJlc29sdmVBcHBsaWVkRWZmb3J0IHRpbWUuIE9ubHkgZmxhZyBpdCB3aGVuIGl0IGFjdHVhbGx5XG4gIC8vIGNvbmZsaWN0cyDigJQgaWYgZW52IG1hdGNoZXMgd2hhdCB0aGUgdXNlciBqdXN0IGFza2VkIGZvciwgdGhlIG91dGNvbWUgaXNcbiAgLy8gdGhlIHNhbWUsIHNvIFwiU2V0IGVmZm9ydCB0byBYXCIgaXMgdHJ1ZSBhbmQgdGhlIG5vdGUgaXMgbm9pc2UuXG4gIGNvbnN0IGVudk92ZXJyaWRlID0gZ2V0RWZmb3J0RW52T3ZlcnJpZGUoKVxuICBpZiAoZW52T3ZlcnJpZGUgIT09IHVuZGVmaW5lZCAmJiBlbnZPdmVycmlkZSAhPT0gZWZmb3J0VmFsdWUpIHtcbiAgICBjb25zdCBlbnZSYXcgPSBwcm9jZXNzLmVudi5DTEFVREVfQ09ERV9FRkZPUlRfTEVWRUxcbiAgICBpZiAocGVyc2lzdGFibGUgPT09IHVuZGVmaW5lZCkge1xuICAgICAgcmV0dXJuIHtcbiAgICAgICAgbWVzc2FnZTogYE5vdCBhcHBsaWVkOiBDTEFVREVfQ09ERV9FRkZPUlRfTEVWRUw9JHtlbnZSYXd9IG92ZXJyaWRlcyBlZmZvcnQgdGhpcyBzZXNzaW9uLCBhbmQgJHtlZmZvcnRWYWx1ZX0gaXMgc2Vzc2lvbi1vbmx5IChub3RoaW5nIHNhdmVkKWAsXG4gICAgICAgIGVmZm9ydFVwZGF0ZTogeyB2YWx1ZTogZWZmb3J0VmFsdWUgfSxcbiAgICAgIH1cbiAgICB9XG4gICAgcmV0dXJuIHtcbiAgICAgIG1lc3NhZ2U6IGBDTEFVREVfQ09ERV9FRkZPUlRfTEVWRUw9JHtlbnZSYXd9IG92ZXJyaWRlcyB0aGlzIHNlc3Npb24g4oCUIGNsZWFyIGl0IGFuZCAke2VmZm9ydFZhbHVlfSB0YWtlcyBvdmVyYCxcbiAgICAgIGVmZm9ydFVwZGF0ZTogeyB2YWx1ZTogZWZmb3J0VmFsdWUgfSxcbiAgICB9XG4gIH1cblxuICBjb25zdCBkZXNjcmlwdGlvbiA9IGdldEVmZm9ydFZhbHVlRGVzY3JpcHRpb24oZWZmb3J0VmFsdWUpXG4gIGNvbnN0IHN1ZmZpeCA9IHBlcnNpc3RhYmxlICE9PSB1bmRlZmluZWQgPyAnJyA6ICcgKHRoaXMgc2Vzc2lvbiBvbmx5KSdcbiAgcmV0dXJuIHtcbiAgICBtZXNzYWdlOiBgU2V0IGVmZm9ydCBsZXZlbCB0byAke2VmZm9ydFZhbHVlfSR7c3VmZml4fTogJHtkZXNjcmlwdGlvbn1gLFxuICAgIGVmZm9ydFVwZGF0ZTogeyB2YWx1ZTogZWZmb3J0VmFsdWUgfSxcbiAgfVxufVxuXG5leHBvcnQgZnVuY3Rpb24gc2hvd0N1cnJlbnRFZmZvcnQoXG4gIGFwcFN0YXRlRWZmb3J0OiBFZmZvcnRWYWx1ZSB8IHVuZGVmaW5lZCxcbiAgbW9kZWw6IHN0cmluZyxcbik6IEVmZm9ydENvbW1hbmRSZXN1bHQge1xuICBjb25zdCBlbnZPdmVycmlkZSA9IGdldEVmZm9ydEVudk92ZXJyaWRlKClcbiAgY29uc3QgZWZmZWN0aXZlVmFsdWUgPVxuICAgIGVudk92ZXJyaWRlID09PSBudWxsID8gdW5kZWZpbmVkIDogKGVudk92ZXJyaWRlID8/IGFwcFN0YXRlRWZmb3J0KVxuICBpZiAoZWZmZWN0aXZlVmFsdWUgPT09IHVuZGVmaW5lZCkge1xuICAgIGNvbnN0IGxldmVsID0gZ2V0RGlzcGxheWVkRWZmb3J0TGV2ZWwobW9kZWwsIGFwcFN0YXRlRWZmb3J0KVxuICAgIHJldHVybiB7IG1lc3NhZ2U6IGBFZmZvcnQgbGV2ZWw6IGF1dG8gKGN1cnJlbnRseSAke2xldmVsfSlgIH1cbiAgfVxuICBjb25zdCBkZXNjcmlwdGlvbiA9IGdldEVmZm9ydFZhbHVlRGVzY3JpcHRpb24oZWZmZWN0aXZlVmFsdWUpXG4gIHJldHVybiB7XG4gICAgbWVzc2FnZTogYEN1cnJlbnQgZWZmb3J0IGxldmVsOiAke2VmZmVjdGl2ZVZhbHVlfSAoJHtkZXNjcmlwdGlvbn0pYCxcbiAgfVxufVxuXG5mdW5jdGlvbiB1bnNldEVmZm9ydExldmVsKCk6IEVmZm9ydENvbW1hbmRSZXN1bHQge1xuICBjb25zdCByZXN1bHQgPSB1cGRhdGVTZXR0aW5nc0ZvclNvdXJjZSgndXNlclNldHRpbmdzJywge1xuICAgIGVmZm9ydExldmVsOiB1bmRlZmluZWQsXG4gIH0pXG4gIGlmIChyZXN1bHQuZXJyb3IpIHtcbiAgICByZXR1cm4ge1xuICAgICAgbWVzc2FnZTogYEZhaWxlZCB0byBzZXQgZWZmb3J0IGxldmVsOiAke3Jlc3VsdC5lcnJvci5tZXNzYWdlfWAsXG4gICAgfVxuICB9XG4gIGxvZ0V2ZW50KCd0ZW5ndV9lZmZvcnRfY29tbWFuZCcsIHtcbiAgICBlZmZvcnQ6XG4gICAgICAnYXV0bycgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgfSlcbiAgLy8gZW52PWF1dG8vdW5zZXQgKG51bGwpIG1hdGNoZXMgd2hhdCAvZWZmb3J0IGF1dG8gYXNrcyBmb3IsIHNvIG9ubHkgd2FyblxuICAvLyB3aGVuIGVudiBpcyBwaW5uaW5nIGEgc3BlY2lmaWMgbGV2ZWwgdGhhdCB3aWxsIGtlZXAgb3ZlcnJpZGluZy5cbiAgY29uc3QgZW52T3ZlcnJpZGUgPSBnZXRFZmZvcnRFbnZPdmVycmlkZSgpXG4gIGlmIChlbnZPdmVycmlkZSAhPT0gdW5kZWZpbmVkICYmIGVudk92ZXJyaWRlICE9PSBudWxsKSB7XG4gICAgY29uc3QgZW52UmF3ID0gcHJvY2Vzcy5lbnYuQ0xBVURFX0NPREVfRUZGT1JUX0xFVkVMXG4gICAgcmV0dXJuIHtcbiAgICAgIG1lc3NhZ2U6IGBDbGVhcmVkIGVmZm9ydCBmcm9tIHNldHRpbmdzLCBidXQgQ0xBVURFX0NPREVfRUZGT1JUX0xFVkVMPSR7ZW52UmF3fSBzdGlsbCBjb250cm9scyB0aGlzIHNlc3Npb25gLFxuICAgICAgZWZmb3J0VXBkYXRlOiB7IHZhbHVlOiB1bmRlZmluZWQgfSxcbiAgICB9XG4gIH1cbiAgcmV0dXJuIHtcbiAgICBtZXNzYWdlOiAnRWZmb3J0IGxldmVsIHNldCB0byBhdXRvJyxcbiAgICBlZmZvcnRVcGRhdGU6IHsgdmFsdWU6IHVuZGVmaW5lZCB9LFxuICB9XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBleGVjdXRlRWZmb3J0KGFyZ3M6IHN0cmluZyk6IEVmZm9ydENvbW1hbmRSZXN1bHQge1xuICBjb25zdCBub3JtYWxpemVkID0gYXJncy50b0xvd2VyQ2FzZSgpXG4gIGlmIChub3JtYWxpemVkID09PSAnYXV0bycgfHwgbm9ybWFsaXplZCA9PT0gJ3Vuc2V0Jykge1xuICAgIHJldHVybiB1bnNldEVmZm9ydExldmVsKClcbiAgfVxuXG4gIGlmICghaXNFZmZvcnRMZXZlbChub3JtYWxpemVkKSkge1xuICAgIHJldHVybiB7XG4gICAgICBtZXNzYWdlOiBgSW52YWxpZCBhcmd1bWVudDogJHthcmdzfS4gVmFsaWQgb3B0aW9ucyBhcmU6IGxvdywgbWVkaXVtLCBoaWdoLCBtYXgsIGF1dG9gLFxuICAgIH1cbiAgfVxuXG4gIHJldHVybiBzZXRFZmZvcnRWYWx1ZShub3JtYWxpemVkKVxufVxuXG5mdW5jdGlvbiBTaG93Q3VycmVudEVmZm9ydCh7XG4gIG9uRG9uZSxcbn06IHtcbiAgb25Eb25lOiAocmVzdWx0OiBzdHJpbmcpID0+IHZvaWRcbn0pOiBSZWFjdC5SZWFjdE5vZGUge1xuICBjb25zdCBlZmZvcnRWYWx1ZSA9IHVzZUFwcFN0YXRlKHMgPT4gcy5lZmZvcnRWYWx1ZSlcbiAgY29uc3QgbW9kZWwgPSB1c2VNYWluTG9vcE1vZGVsKClcbiAgY29uc3QgeyBtZXNzYWdlIH0gPSBzaG93Q3VycmVudEVmZm9ydChlZmZvcnRWYWx1ZSwgbW9kZWwpXG4gIG9uRG9uZShtZXNzYWdlKVxuICByZXR1cm4gbnVsbFxufVxuXG5mdW5jdGlvbiBBcHBseUVmZm9ydEFuZENsb3NlKHtcbiAgcmVzdWx0LFxuICBvbkRvbmUsXG59OiB7XG4gIHJlc3VsdDogRWZmb3J0Q29tbWFuZFJlc3VsdFxuICBvbkRvbmU6IChyZXN1bHQ6IHN0cmluZykgPT4gdm9pZFxufSk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IHNldEFwcFN0YXRlID0gdXNlU2V0QXBwU3RhdGUoKVxuICBjb25zdCB7IGVmZm9ydFVwZGF0ZSwgbWVzc2FnZSB9ID0gcmVzdWx0XG4gIFJlYWN0LnVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKGVmZm9ydFVwZGF0ZSkge1xuICAgICAgc2V0QXBwU3RhdGUocHJldiA9PiAoe1xuICAgICAgICAuLi5wcmV2LFxuICAgICAgICBlZmZvcnRWYWx1ZTogZWZmb3J0VXBkYXRlLnZhbHVlLFxuICAgICAgfSkpXG4gICAgfVxuICAgIG9uRG9uZShtZXNzYWdlKVxuICB9LCBbc2V0QXBwU3RhdGUsIGVmZm9ydFVwZGF0ZSwgbWVzc2FnZSwgb25Eb25lXSlcbiAgcmV0dXJuIG51bGxcbn1cblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGNhbGwoXG4gIG9uRG9uZTogTG9jYWxKU1hDb21tYW5kT25Eb25lLFxuICBfY29udGV4dDogdW5rbm93bixcbiAgYXJncz86IHN0cmluZyxcbik6IFByb21pc2U8UmVhY3QuUmVhY3ROb2RlPiB7XG4gIGFyZ3MgPSBhcmdzPy50cmltKCkgfHwgJydcblxuICBpZiAoQ09NTU9OX0hFTFBfQVJHUy5pbmNsdWRlcyhhcmdzKSkge1xuICAgIG9uRG9uZShcbiAgICAgICdVc2FnZTogL2VmZm9ydCBbbG93fG1lZGl1bXxoaWdofG1heHxhdXRvXVxcblxcbkVmZm9ydCBsZXZlbHM6XFxuLSBsb3c6IFF1aWNrLCBzdHJhaWdodGZvcndhcmQgaW1wbGVtZW50YXRpb25cXG4tIG1lZGl1bTogQmFsYW5jZWQgYXBwcm9hY2ggd2l0aCBzdGFuZGFyZCB0ZXN0aW5nXFxuLSBoaWdoOiBDb21wcmVoZW5zaXZlIGltcGxlbWVudGF0aW9uIHdpdGggZXh0ZW5zaXZlIHRlc3RpbmdcXG4tIG1heDogTWF4aW11bSBjYXBhYmlsaXR5IHdpdGggZGVlcGVzdCByZWFzb25pbmcgKE9wdXMgNC42IG9ubHkpXFxuLSBhdXRvOiBVc2UgdGhlIGRlZmF1bHQgZWZmb3J0IGxldmVsIGZvciB5b3VyIG1vZGVsJyxcbiAgICApXG4gICAgcmV0dXJuXG4gIH1cblxuICBpZiAoIWFyZ3MgfHwgYXJncyA9PT0gJ2N1cnJlbnQnIHx8IGFyZ3MgPT09ICdzdGF0dXMnKSB7XG4gICAgcmV0dXJuIDxTaG93Q3VycmVudEVmZm9ydCBvbkRvbmU9e29uRG9uZX0gLz5cbiAgfVxuXG4gIGNvbnN0IHJlc3VsdCA9IGV4ZWN1dGVFZmZvcnQoYXJncylcbiAgcmV0dXJuIDxBcHBseUVmZm9ydEFuZENsb3NlIHJlc3VsdD17cmVzdWx0fSBvbkRvbmU9e29uRG9uZX0gLz5cbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU8sS0FBS0EsS0FBSyxNQUFNLE9BQU87QUFDOUIsU0FBU0MsZ0JBQWdCLFFBQVEsaUNBQWlDO0FBQ2xFLFNBQ0UsS0FBS0MsMERBQTBELEVBQy9EQyxRQUFRLFFBQ0gsbUNBQW1DO0FBQzFDLFNBQVNDLFdBQVcs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 \ No newline at end of file diff --git a/src/commands/effort/index.ts b/src/commands/effort/index.ts new file mode 100644 index 0000000..66cd511 --- /dev/null +++ b/src/commands/effort/index.ts @@ -0,0 +1,13 @@ +import type { Command } from '../../commands.js' +import { shouldInferenceConfigCommandBeImmediate } from '../../utils/immediateCommand.js' + +export default { + type: 'local-jsx', + name: 'effort', + description: 'Set effort level for model usage', + argumentHint: '[low|medium|high|max|auto]', + get immediate() { + return shouldInferenceConfigCommandBeImmediate() + }, + load: () => import('./effort.js'), +} satisfies Command diff --git a/src/commands/env/index.js b/src/commands/env/index.js new file mode 100644 index 0000000..e1a619d --- /dev/null +++ b/src/commands/env/index.js @@ -0,0 +1 @@ +export default { isEnabled: () => false, isHidden: true, name: 'stub' }; diff --git a/src/commands/exit/exit.tsx b/src/commands/exit/exit.tsx new file mode 100644 index 0000000..0f6f49e --- /dev/null +++ b/src/commands/exit/exit.tsx @@ -0,0 +1,33 @@ +import { feature } from 'bun:bundle'; +import { spawnSync } from 'child_process'; +import sample from 'lodash-es/sample.js'; +import * as React from 'react'; +import { ExitFlow } from '../../components/ExitFlow.js'; +import type { LocalJSXCommandOnDone } from '../../types/command.js'; +import { isBgSession } from '../../utils/concurrentSessions.js'; +import { gracefulShutdown } from '../../utils/gracefulShutdown.js'; +import { getCurrentWorktreeSession } from '../../utils/worktree.js'; +const GOODBYE_MESSAGES = ['Goodbye!', 'See ya!', 'Bye!', 'Catch you later!']; +function getRandomGoodbyeMessage(): string { + return sample(GOODBYE_MESSAGES) ?? 'Goodbye!'; +} +export async function call(onDone: LocalJSXCommandOnDone): Promise { + // Inside a `claude --bg` tmux session: detach instead of kill. The REPL + // keeps running; `claude attach` can reconnect. Covers /exit, /quit, + // ctrl+c, ctrl+d — all funnel through here via REPL's handleExit. + if (feature('BG_SESSIONS') && isBgSession()) { + onDone(); + spawnSync('tmux', ['detach-client'], { + stdio: 'ignore' + }); + return null; + } + const showWorktree = getCurrentWorktreeSession() !== null; + if (showWorktree) { + return onDone()} />; + } + onDone(getRandomGoodbyeMessage()); + await gracefulShutdown(0, 'prompt_input_exit'); + return null; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJmZWF0dXJlIiwic3Bhd25TeW5jIiwic2FtcGxlIiwiUmVhY3QiLCJFeGl0RmxvdyIsIkxvY2FsSlNYQ29tbWFuZE9uRG9uZSIsImlzQmdTZXNzaW9uIiwiZ3JhY2VmdWxTaHV0ZG93biIsImdldEN1cnJlbnRXb3JrdHJlZVNlc3Npb24iLCJHT09EQllFX01FU1NBR0VTIiwiZ2V0UmFuZG9tR29vZGJ5ZU1lc3NhZ2UiLCJjYWxsIiwib25Eb25lIiwiUHJvbWlzZSIsIlJlYWN0Tm9kZSIsInN0ZGlvIiwic2hvd1dvcmt0cmVlIl0sInNvdXJjZXMiOlsiZXhpdC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgZmVhdHVyZSB9IGZyb20gJ2J1bjpidW5kbGUnXG5pbXBvcnQgeyBzcGF3blN5bmMgfSBmcm9tICdjaGlsZF9wcm9jZXNzJ1xuaW1wb3J0IHNhbXBsZSBmcm9tICdsb2Rhc2gtZXMvc2FtcGxlLmpzJ1xuaW1wb3J0ICogYXMgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBFeGl0RmxvdyB9IGZyb20gJy4uLy4uL2NvbXBvbmVudHMvRXhpdEZsb3cuanMnXG5pbXBvcnQgdHlwZSB7IExvY2FsSlNYQ29tbWFuZE9uRG9uZSB9IGZyb20gJy4uLy4uL3R5cGVzL2NvbW1hbmQuanMnXG5pbXBvcnQgeyBpc0JnU2Vzc2lvbiB9IGZyb20gJy4uLy4uL3V0aWxzL2NvbmN1cnJlbnRTZXNzaW9ucy5qcydcbmltcG9ydCB7IGdyYWNlZnVsU2h1dGRvd24gfSBmcm9tICcuLi8uLi91dGlscy9ncmFjZWZ1bFNodXRkb3duLmpzJ1xuaW1wb3J0IHsgZ2V0Q3VycmVudFdvcmt0cmVlU2Vzc2lvbiB9IGZyb20gJy4uLy4uL3V0aWxzL3dvcmt0cmVlLmpzJ1xuXG5jb25zdCBHT09EQllFX01FU1NBR0VTID0gWydHb29kYnllIScsICdTZWUgeWEhJywgJ0J5ZSEnLCAnQ2F0Y2ggeW91IGxhdGVyISddXG5cbmZ1bmN0aW9uIGdldFJhbmRvbUdvb2RieWVNZXNzYWdlKCk6IHN0cmluZyB7XG4gIHJldHVybiBzYW1wbGUoR09PREJZRV9NRVNTQUdFUykgPz8gJ0dvb2RieWUhJ1xufVxuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gY2FsbChcbiAgb25Eb25lOiBMb2NhbEpTWENvbW1hbmRPbkRvbmUsXG4pOiBQcm9taXNlPFJlYWN0LlJlYWN0Tm9kZT4ge1xuICAvLyBJbnNpZGUgYSBgY2xhdWRlIC0tYmdgIHRtdXggc2Vzc2lvbjogZGV0YWNoIGluc3RlYWQgb2Yga2lsbC4gVGhlIFJFUExcbiAgLy8ga2VlcHMgcnVubmluZzsgYGNsYXVkZSBhdHRhY2hgIGNhbiByZWNvbm5lY3QuIENvdmVycyAvZXhpdCwgL3F1aXQsXG4gIC8vIGN0cmwrYywgY3RybCtkIOKAlCBhbGwgZnVubmVsIHRocm91Z2ggaGVyZSB2aWEgUkVQTCdzIGhhbmRsZUV4aXQuXG4gIGlmIChmZWF0dXJlKCdCR19TRVNTSU9OUycpICYmIGlzQmdTZXNzaW9uKCkpIHtcbiAgICBvbkRvbmUoKVxuICAgIHNwYXduU3luYygndG11eCcsIFsnZGV0YWNoLWNsaWVudCddLCB7IHN0ZGlvOiAnaWdub3JlJyB9KVxuICAgIHJldHVybiBudWxsXG4gIH1cblxuICBjb25zdCBzaG93V29ya3RyZWUgPSBnZXRDdXJyZW50V29ya3RyZWVTZXNzaW9uKCkgIT09IG51bGxcblxuICBpZiAoc2hvd1dvcmt0cmVlKSB7XG4gICAgcmV0dXJuIChcbiAgICAgIDxFeGl0Rmxvd1xuICAgICAgICBzaG93V29ya3RyZWU9e3Nob3dXb3JrdHJlZX1cbiAgICAgICAgb25Eb25lPXtvbkRvbmV9XG4gICAgICAgIG9uQ2FuY2VsPXsoKSA9PiBvbkRvbmUoKX1cbiAgICAgIC8+XG4gICAgKVxuICB9XG5cbiAgb25Eb25lKGdldFJhbmRvbUdvb2RieWVNZXNzYWdlKCkpXG4gIGF3YWl0IGdyYWNlZnVsU2h1dGRvd24oMCwgJ3Byb21wdF9pbnB1dF9leGl0JylcbiAgcmV0dXJuIG51bGxcbn1cbiJdLCJtYXBwaW5ncyI6IkFBQUEsU0FBU0EsT0FBTyxRQUFRLFlBQVk7QUFDcEMsU0FBU0MsU0FBUyxRQUFRLGVBQWU7QUFDekMsT0FBT0MsTUFBTSxNQUFNLHFCQUFxQjtBQUN4QyxPQUFPLEtBQUtDLEtBQUssTUFBTSxPQUFPO0FBQzlCLFNBQVNDLFFBQVEsUUFBUSw4QkFBOEI7QUFDdkQsY0FBY0MscUJBQXFCLFFBQVEsd0JBQXdCO0FBQ25FLFNBQVNDLFdBQVcsUUFBUSxtQ0FBbUM7QUFDL0QsU0FBU0MsZ0JBQWdCLFFBQVEsaUNBQWlDO0FBQ2xFLFNBQVNDLHlCQUF5QixRQUFRLHlCQUF5QjtBQUVuRSxNQUFNQyxnQkFBZ0IsR0FBRyxDQUFDLFVBQVUsRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFFLGtCQUFrQixDQUFDO0FBRTVFLFNBQVNDLHVCQUF1QkEsQ0FBQSxDQUFFLEVBQUUsTUFBTSxDQUFDO0VBQ3pDLE9BQU9SLE1BQU0sQ0FBQ08sZ0JBQWdCLENBQUMsSUFBSSxVQUFVO0FBQy9DO0FBRUEsT0FBTyxlQUFlRSxJQUFJQSxDQUN4QkMsTUFBTSxFQUFFUCxxQkFBcUIsQ0FDOUIsRUFBRVEsT0FBTyxDQUFDVixLQUFLLENBQUNXLFNBQVMsQ0FBQyxDQUFDO0VBQzFCO0VBQ0E7RUFDQTtFQUNBLElBQUlkLE9BQU8sQ0FBQyxhQUFhLENBQUMsSUFBSU0sV0FBVyxDQUFDLENBQUMsRUFBRTtJQUMzQ00sTUFBTSxDQUFDLENBQUM7SUFDUlgsU0FBUyxDQUFDLE1BQU0sRUFBRSxDQUFDLGVBQWUsQ0FBQyxFQUFFO01BQUVjLEtBQUssRUFBRTtJQUFTLENBQUMsQ0FBQztJQUN6RCxPQUFPLElBQUk7RUFDYjtFQUVBLE1BQU1DLFlBQVksR0FBR1IseUJBQXlCLENBQUMsQ0FBQyxLQUFLLElBQUk7RUFFekQsSUFBSVEsWUFBWSxFQUFFO0lBQ2hCLE9BQ0UsQ0FBQyxRQUFRLENBQ1AsWUFBWSxDQUFDLENBQUNBLFlBQVksQ0FBQyxDQUMzQixNQUFNLENBQUMsQ0FBQ0osTUFBTSxDQUFDLENBQ2YsUUFBUSxDQUFDLENBQUMsTUFBTUEsTUFBTSxDQUFDLENBQUMsQ0FBQyxHQUN6QjtFQUVOO0VBRUFBLE1BQU0sQ0FBQ0YsdUJBQXVCLENBQUMsQ0FBQyxDQUFDO0VBQ2pDLE1BQU1ILGdCQUFnQixDQUFDLENBQUMsRUFBRSxtQkFBbUIsQ0FBQztFQUM5QyxPQUFPLElBQUk7QUFDYiIsImlnbm9yZUxpc3QiOltdfQ== \ No newline at end of file diff --git a/src/commands/exit/index.ts b/src/commands/exit/index.ts new file mode 100644 index 0000000..f32499e --- /dev/null +++ b/src/commands/exit/index.ts @@ -0,0 +1,12 @@ +import type { Command } from '../../commands.js' + +const exit = { + type: 'local-jsx', + name: 'exit', + aliases: ['quit'], + description: 'Exit the REPL', + immediate: true, + load: () => import('./exit.js'), +} satisfies Command + +export default exit diff --git a/src/commands/export/export.tsx b/src/commands/export/export.tsx new file mode 100644 index 0000000..c47f5cf --- /dev/null +++ b/src/commands/export/export.tsx @@ -0,0 +1,91 @@ +import { join } from 'path'; +import React from 'react'; +import { ExportDialog } from '../../components/ExportDialog.js'; +import type { ToolUseContext } from '../../Tool.js'; +import type { LocalJSXCommandOnDone } from '../../types/command.js'; +import type { Message } from '../../types/message.js'; +import { getCwd } from '../../utils/cwd.js'; +import { renderMessagesToPlainText } from '../../utils/exportRenderer.js'; +import { writeFileSync_DEPRECATED } from '../../utils/slowOperations.js'; +function formatTimestamp(date: Date): string { + const year = date.getFullYear(); + const month = String(date.getMonth() + 1).padStart(2, '0'); + const day = String(date.getDate()).padStart(2, '0'); + const hours = String(date.getHours()).padStart(2, '0'); + const minutes = String(date.getMinutes()).padStart(2, '0'); + const seconds = String(date.getSeconds()).padStart(2, '0'); + return `${year}-${month}-${day}-${hours}${minutes}${seconds}`; +} +export function extractFirstPrompt(messages: Message[]): string { + const firstUserMessage = messages.find(msg => msg.type === 'user'); + if (!firstUserMessage || firstUserMessage.type !== 'user') { + return ''; + } + const content = firstUserMessage.message?.content; + let result = ''; + if (typeof content === 'string') { + result = content.trim(); + } else if (Array.isArray(content)) { + const textContent = content.find(item => item.type === 'text'); + if (textContent && 'text' in textContent) { + result = textContent.text.trim(); + } + } + + // Take first line only and limit length + result = result.split('\n')[0] || ''; + if (result.length > 50) { + result = result.substring(0, 49) + '…'; + } + return result; +} +export function sanitizeFilename(text: string): string { + // Replace special characters with hyphens + return text.toLowerCase().replace(/[^a-z0-9\s-]/g, '') // Remove special chars + .replace(/\s+/g, '-') // Replace spaces with hyphens + .replace(/-+/g, '-') // Replace multiple hyphens with single + .replace(/^-|-$/g, ''); // Remove leading/trailing hyphens +} +async function exportWithReactRenderer(context: ToolUseContext): Promise { + const tools = context.options.tools || []; + return renderMessagesToPlainText(context.messages, tools); +} +export async function call(onDone: LocalJSXCommandOnDone, context: ToolUseContext, args: string): Promise { + // Render the conversation content + const content = await exportWithReactRenderer(context); + + // If args are provided, write directly to file and skip dialog + const filename = args.trim(); + if (filename) { + const finalFilename = filename.endsWith('.txt') ? filename : filename.replace(/\.[^.]+$/, '') + '.txt'; + const filepath = join(getCwd(), finalFilename); + try { + writeFileSync_DEPRECATED(filepath, content, { + encoding: 'utf-8', + flush: true + }); + onDone(`Conversation exported to: ${filepath}`); + return null; + } catch (error) { + onDone(`Failed to export conversation: ${error instanceof Error ? error.message : 'Unknown error'}`); + return null; + } + } + + // Generate default filename from first prompt or timestamp + const firstPrompt = extractFirstPrompt(context.messages); + const timestamp = formatTimestamp(new Date()); + let defaultFilename: string; + if (firstPrompt) { + const sanitized = sanitizeFilename(firstPrompt); + defaultFilename = sanitized ? `${timestamp}-${sanitized}.txt` : `conversation-${timestamp}.txt`; + } else { + defaultFilename = `conversation-${timestamp}.txt`; + } + + // Return the dialog component when no args provided + return { + onDone(result.message); + }} />; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJqb2luIiwiUmVhY3QiLCJFeHBvcnREaWFsb2ciLCJUb29sVXNlQ29udGV4dCIsIkxvY2FsSlNYQ29tbWFuZE9uRG9uZSIsIk1lc3NhZ2UiLCJnZXRDd2QiLCJyZW5kZXJNZXNzYWdlc1RvUGxhaW5UZXh0Iiwid3JpdGVGaWxlU3luY19ERVBSRUNBVEVEIiwiZm9ybWF0VGltZXN0YW1wIiwiZGF0ZSIsIkRhdGUiLCJ5ZWFyIiwiZ2V0RnVsbFllYXIiLCJtb250aCIsIlN0cmluZyIsImdldE1vbnRoIiwicGFkU3RhcnQiLCJkYXkiLCJnZXREYXRlIiwiaG91cnMiLCJnZXRIb3VycyIsIm1pbnV0ZXMiLCJnZXRNaW51dGVzIiwic2Vjb25kcyIsImdldFNlY29uZHMiLCJleHRyYWN0Rmlyc3RQcm9tcHQiLCJtZXNzYWdlcyIsImZpcnN0VXNlck1lc3NhZ2UiLCJmaW5kIiwibXNnIiwidHlwZSIsImNvbnRlbnQiLCJtZXNzYWdlIiwicmVzdWx0IiwidHJpbSIsIkFycmF5IiwiaXNBcnJheSIsInRleHRDb250ZW50IiwiaXRlbSIsInRleHQiLCJzcGxpdCIsImxlbmd0aCIsInN1YnN0cmluZyIsInNhbml0aXplRmlsZW5hbWUiLCJ0b0xvd2VyQ2FzZSIsInJlcGxhY2UiLCJleHBvcnRXaXRoUmVhY3RSZW5kZXJlciIsImNvbnRleHQiLCJQcm9taXNlIiwidG9vbHMiLCJvcHRpb25zIiwiY2FsbCIsIm9uRG9uZSIsImFyZ3MiLCJSZWFjdE5vZGUiLCJmaWxlbmFtZSIsImZpbmFsRmlsZW5hbWUiLCJlbmRzV2l0aCIsImZpbGVwYXRoIiwiZW5jb2RpbmciLCJmbHVzaCIsImVycm9yIiwiRXJyb3IiLCJmaXJzdFByb21wdCIsInRpbWVzdGFtcCIsImRlZmF1bHRGaWxlbmFtZSIsInNhbml0aXplZCJdLCJzb3VyY2VzIjpbImV4cG9ydC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgam9pbiB9IGZyb20gJ3BhdGgnXG5pbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBFeHBvcnREaWFsb2cgfSBmcm9tICcuLi8uLi9jb21wb25lbnRzL0V4cG9ydERpYWxvZy5qcydcbmltcG9ydCB0eXBlIHsgVG9vbFVzZUNvbnRleHQgfSBmcm9tICcuLi8uLi9Ub29sLmpzJ1xuaW1wb3J0IHR5cGUgeyBMb2NhbEpTWENvbW1hbmRPbkRvbmUgfSBmcm9tICcuLi8uLi90eXBlcy9jb21tYW5kLmpzJ1xuaW1wb3J0IHR5cGUgeyBNZXNzYWdlIH0gZnJvbSAnLi4vLi4vdHlwZXMvbWVzc2FnZS5qcydcbmltcG9ydCB7IGdldEN3ZCB9IGZyb20gJy4uLy4uL3V0aWxzL2N3ZC5qcydcbmltcG9ydCB7IHJlbmRlck1lc3NhZ2VzVG9QbGFpblRleHQgfSBmcm9tICcuLi8uLi91dGlscy9leHBvcnRSZW5kZXJlci5qcydcbmltcG9ydCB7IHdyaXRlRmlsZVN5bmNfREVQUkVDQVRFRCB9IGZyb20gJy4uLy4uL3V0aWxzL3Nsb3dPcGVyYXRpb25zLmpzJ1xuXG5mdW5jdGlvbiBmb3JtYXRUaW1lc3RhbXAoZGF0ZTogRGF0ZSk6IHN0cmluZyB7XG4gIGNvbnN0IHllYXIgPSBkYXRlLmdldEZ1bGxZZWFyKClcbiAgY29uc3QgbW9udGggPSBTdHJpbmcoZGF0ZS5nZXRNb250aCgpICsgMSkucGFkU3RhcnQoMiwgJzAnKVxuICBjb25zdCBkYXkgPSBTdHJpbmcoZGF0ZS5nZXREYXRlKCkpLnBhZFN0YXJ0KDIsICcwJylcbiAgY29uc3QgaG91cnMgPSBTdHJpbmcoZGF0ZS5nZXRIb3VycygpKS5wYWRTdGFydCgyLCAnMCcpXG4gIGNvbnN0IG1pbnV0ZXMgPSBTdHJpbmcoZGF0ZS5nZXRNaW51dGVzKCkpLnBhZFN0YXJ0KDIsICcwJylcbiAgY29uc3Qgc2Vjb25kcyA9IFN0cmluZyhkYXRlLmdldFNlY29uZHMoKSkucGFkU3RhcnQoMiwgJzAnKVxuICByZXR1cm4gYCR7eWVhcn0tJHttb250aH0tJHtkYXl9LSR7aG91cnN9JHttaW51dGVzfSR7c2Vjb25kc31gXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBleHRyYWN0Rmlyc3RQcm9tcHQobWVzc2FnZXM6IE1lc3NhZ2VbXSk6IHN0cmluZyB7XG4gIGNvbnN0IGZpcnN0VXNlck1lc3NhZ2UgPSBtZXNzYWdlcy5maW5kKG1zZyA9PiBtc2cudHlwZSA9PT0gJ3VzZXInKVxuXG4gIGlmICghZmlyc3RVc2VyTWVzc2FnZSB8fCBmaXJzdFVzZXJNZXNzYWdlLnR5cGUgIT09ICd1c2VyJykge1xuICAgIHJldHVybiAnJ1xuICB9XG5cbiAgY29uc3QgY29udGVudCA9IGZpcnN0VXNlck1lc3NhZ2UubWVzc2FnZT8uY29udGVudFxuICBsZXQgcmVzdWx0ID0gJydcblxuICBpZiAodHlwZW9mIGNvbnRlbnQgPT09ICdzdHJpbmcnKSB7XG4gICAgcmVzdWx0ID0gY29udGVudC50cmltKClcbiAgfSBlbHNlIGlmIChBcnJheS5pc0FycmF5KGNvbnRlbnQpKSB7XG4gICAgY29uc3QgdGV4dENvbnRlbnQgPSBjb250ZW50LmZpbmQoaXRlbSA9PiBpdGVtLnR5cGUgPT09ICd0ZXh0JylcbiAgICBpZiAodGV4dENvbnRlbnQgJiYgJ3RleHQnIGluIHRleHRDb250ZW50KSB7XG4gICAgICByZXN1bHQgPSB0ZXh0Q29udGVudC50ZXh0LnRyaW0oKVxuICAgIH1cbiAgfVxuXG4gIC8vIFRha2UgZmlyc3QgbGluZSBvbmx5IGFuZCBsaW1pdCBsZW5ndGhcbiAgcmVzdWx0ID0gcmVzdWx0LnNwbGl0KCdcXG4nKVswXSB8fCAnJ1xuICBpZiAocmVzdWx0Lmxlbmd0aCA+IDUwKSB7XG4gICAgcmVzdWx0ID0gcmVzdWx0LnN1YnN0cmluZygwLCA0OSkgKyAn4oCmJ1xuICB9XG5cbiAgcmV0dXJuIHJlc3VsdFxufVxuXG5leHBvcnQgZnVuY3Rpb24gc2FuaXRpemVGaWxlbmFtZSh0ZXh0OiBzdHJpbmcpOiBzdHJpbmcge1xuICAvLyBSZXBsYWNlIHNwZWNpYWwgY2hhcmFjdGVycyB3aXRoIGh5cGhlbnNcbiAgcmV0dXJuIHRleHRcbiAgICAudG9Mb3dlckNhc2UoKVxuICAgIC5yZXBsYWNlKC9bXmEtejAtOVxccy1dL2csICcnKSAvLyBSZW1vdmUgc3BlY2lhbCBjaGFyc1xuICAgIC5yZXBsYWNlKC9cXHMrL2csICctJykgLy8gUmVwbGFjZSBzcGFjZXMgd2l0aCBoeXBoZW5zXG4gICAgLnJlcGxhY2UoLy0rL2csICctJykgLy8gUmVwbGFjZSBtdWx0aXBsZSBoeXBoZW5zIHdpdGggc2luZ2xlXG4gICAgLnJlcGxhY2UoL14tfC0kL2csICcnKSAvLyBSZW1vdmUgbGVhZGluZy90cmFpbGluZyBoeXBoZW5zXG59XG5cbmFzeW5jIGZ1bmN0aW9uIGV4cG9ydFdpdGhSZWFjdFJlbmRlcmVyKFxuICBjb250ZXh0OiBUb29sVXNlQ29udGV4dCxcbik6IFByb21pc2U8c3RyaW5nPiB7XG4gIGNvbnN0IHRvb2xzID0gY29udGV4dC5vcHRpb25zLnRvb2xzIHx8IFtdXG4gIHJldHVybiByZW5kZXJNZXNzYWdlc1RvUGxhaW5UZXh0KGNvbnRleHQubWVzc2FnZXMsIHRvb2xzKVxufVxuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gY2FsbChcbiAgb25Eb25lOiBMb2NhbEpTWENvbW1hbmRPbkRvbmUsXG4gIGNvbnRleHQ6IFRvb2xVc2VDb250ZXh0LFxuICBhcmdzOiBzdHJpbmcsXG4pOiBQcm9taXNlPFJlYWN0LlJlYWN0Tm9kZT4ge1xuICAvLyBSZW5kZXIgdGhlIGNvbnZlcnNhdGlvbiBjb250ZW50XG4gIGNvbnN0IGNvbnRlbnQgPSBhd2FpdCBleHBvcnRXaXRoUmVhY3RSZW5kZXJlcihjb250ZXh0KVxuXG4gIC8vIElmIGFyZ3MgYXJlIHByb3ZpZGVkLCB3cml0ZSBkaXJlY3RseSB0byBmaWxlIGFuZCBza2lwIGRpYWxvZ1xuICBjb25zdCBmaWxlbmFtZSA9IGFyZ3MudHJpbSgpXG4gIGlmIChmaWxlbmFtZSkge1xuICAgIGNvbnN0IGZpbmFsRmlsZW5hbWUgPSBmaWxlbmFtZS5lbmRzV2l0aCgnLnR4dCcpXG4gICAgICA/IGZpbGVuYW1lXG4gICAgICA6IGZpbGVuYW1lLnJlcGxhY2UoL1xcLlteLl0rJC8sICcnKSArICcudHh0J1xuICAgIGNvbnN0IGZpbGVwYXRoID0gam9pbihnZXRDd2QoKSwgZmluYWxGaWxlbmFtZSlcblxuICAgIHRyeSB7XG4gICAgICB3cml0ZUZpbGVTeW5jX0RFUFJFQ0FURUQoZmlsZXBhdGgsIGNvbnRlbnQsIHtcbiAgICAgICAgZW5jb2Rpbmc6ICd1dGYtOCcsXG4gICAgICAgIGZsdXNoOiB0cnVlLFxuICAgICAgfSlcbiAgICAgIG9uRG9uZShgQ29udmVyc2F0aW9uIGV4cG9ydGVkIHRvOiAke2ZpbGVwYXRofWApXG4gICAgICByZXR1cm4gbnVsbFxuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICBvbkRvbmUoXG4gICAgICAgIGBGYWlsZWQgdG8gZXhwb3J0IGNvbnZlcnNhdGlvbjogJHtlcnJvciBpbnN0YW5jZW9mIEVycm9yID8gZXJyb3IubWVzc2FnZSA6ICdVbmtub3duIGVycm9yJ31gLFxuICAgICAgKVxuICAgICAgcmV0dXJuIG51bGxcbiAgICB9XG4gIH1cblxuICAvLyBHZW5lcmF0ZSBkZWZhdWx0IGZpbGVuYW1lIGZyb20gZmlyc3QgcHJvbXB0IG9yIHRpbWVzdGFtcFxuICBjb25zdCBmaXJzdFByb21wdCA9IGV4dHJhY3RGaXJzdFByb21wdChjb250ZXh0Lm1lc3NhZ2VzKVxuICBjb25zdCB0aW1lc3RhbXAgPSBmb3JtYXRUaW1lc3RhbXAobmV3IERhdGUoKSlcblxuICBsZXQgZGVmYXVsdEZpbGVuYW1lOiBzdHJpbmdcbiAgaWYgKGZpcnN0UHJvbXB0KSB7XG4gICAgY29uc3Qgc2FuaXRpemVkID0gc2FuaXRpemVGaWxlbmFtZShmaXJzdFByb21wdClcbiAgICBkZWZhdWx0RmlsZW5hbWUgPSBzYW5pdGl6ZWRcbiAgICAgID8gYCR7dGltZXN0YW1wfS0ke3Nhbml0aXplZH0udHh0YFxuICAgICAgOiBgY29udmVyc2F0aW9uLSR7dGltZXN0YW1wfS50eHRgXG4gIH0gZWxzZSB7XG4gICAgZGVmYXVsdEZpbGVuYW1lID0gYGNvbnZlcnNhdGlvbi0ke3RpbWVzdGFtcH0udHh0YFxuICB9XG5cbiAgLy8gUmV0dXJuIHRoZSBkaWFsb2cgY29tcG9uZW50IHdoZW4gbm8gYXJncyBwcm92aWRlZFxuICByZXR1cm4gKFxuICAgIDxFeHBvcnREaWFsb2dcbiAgICAgIGNvbnRlbnQ9e2NvbnRlbnR9XG4gICAgICBkZWZhdWx0RmlsZW5hbWU9e2RlZmF1bHRGaWxlbmFtZX1cbiAgICAgIG9uRG9uZT17cmVzdWx0ID0+IHtcbiAgICAgICAgb25Eb25lKHJlc3VsdC5tZXNzYWdlKVxuICAgICAgfX1cbiAgICAvPlxuICApXG59XG4iXSwibWFwcGluZ3MiOiJBQUFBLFNBQVNBLElBQUksUUFBUSxNQUFNO0FBQzNCLE9BQU9DLEtBQUssTUFBTSxPQUFPO0FBQ3pCLFNBQVNDLFlBQVksUUFBUSxrQ0FBa0M7QUFDL0QsY0FBY0MsY0FBYyxRQUFRLGVBQWU7QUFDbkQsY0FBY0MscUJBQXFCLFFBQVEsd0JBQXdCO0FBQ25FLGNBQWNDLE9BQU8sUUFBUSx3QkFBd0I7QUFDckQsU0FBU0MsTUFBTSxRQUFRLG9CQUFvQjtBQUMzQyxTQUFTQyx5QkFBeUIsUUFBUSwrQkFBK0I7QUFDekUsU0FBU0Msd0JBQXdCLFFBQVEsK0JBQStCO0FBRXhFLFNBQVNDLGVBQWVBLENBQUNDLElBQUksRUFBRUMsSUFBSSxDQUFDLEVBQUUsTUFBTSxDQUFDO0VBQzNDLE1BQU1DLElBQUksR0FBR0YsSUFBSSxDQUFDRyxXQUFXLENBQUMsQ0FBQztFQUMvQixNQUFNQyxLQUFLLEdBQUdDLE1BQU0sQ0FBQ0wsSUFBSSxDQUFDTSxRQUFRLENBQUMsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDQyxRQUFRLENBQUMsQ0FBQyxFQUFFLEdBQUcsQ0FBQztFQUMxRCxNQUFNQyxHQUFHLEdBQUdILE1BQU0sQ0FBQ0wsSUFBSSxDQUFDUyxPQUFPLENBQUMsQ0FBQyxDQUFDLENBQUNGLFFBQVEsQ0FBQyxDQUFDLEVBQUUsR0FBRyxDQUFDO0VBQ25ELE1BQU1HLEtBQUssR0FBR0wsTUFBTSxD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 \ No newline at end of file diff --git a/src/commands/export/index.ts b/src/commands/export/index.ts new file mode 100644 index 0000000..a3d8bb2 --- /dev/null +++ b/src/commands/export/index.ts @@ -0,0 +1,11 @@ +import type { Command } from '../../commands.js' + +const exportCommand = { + type: 'local-jsx', + name: 'export', + description: 'Export the current conversation to a file or clipboard', + argumentHint: '[filename]', + load: () => import('./export.js'), +} satisfies Command + +export default exportCommand diff --git a/src/commands/extra-usage/extra-usage-core.ts b/src/commands/extra-usage/extra-usage-core.ts new file mode 100644 index 0000000..4a8c03b --- /dev/null +++ b/src/commands/extra-usage/extra-usage-core.ts @@ -0,0 +1,118 @@ +import { + checkAdminRequestEligibility, + createAdminRequest, + getMyAdminRequests, +} from '../../services/api/adminRequests.js' +import { invalidateOverageCreditGrantCache } from '../../services/api/overageCreditGrant.js' +import { type ExtraUsage, fetchUtilization } from '../../services/api/usage.js' +import { getSubscriptionType } from '../../utils/auth.js' +import { hasClaudeAiBillingAccess } from '../../utils/billing.js' +import { openBrowser } from '../../utils/browser.js' +import { getGlobalConfig, saveGlobalConfig } from '../../utils/config.js' +import { logError } from '../../utils/log.js' + +type ExtraUsageResult = + | { type: 'message'; value: string } + | { type: 'browser-opened'; url: string; opened: boolean } + +export async function runExtraUsage(): Promise { + if (!getGlobalConfig().hasVisitedExtraUsage) { + saveGlobalConfig(prev => ({ ...prev, hasVisitedExtraUsage: true })) + } + // Invalidate only the current org's entry so a follow-up read refetches + // the granted state. Separate from the visited flag since users may run + // /extra-usage more than once while iterating on the claim flow. + invalidateOverageCreditGrantCache() + + const subscriptionType = getSubscriptionType() + const isTeamOrEnterprise = + subscriptionType === 'team' || subscriptionType === 'enterprise' + const hasBillingAccess = hasClaudeAiBillingAccess() + + if (!hasBillingAccess && isTeamOrEnterprise) { + // Mirror apps/claude-ai useHasUnlimitedOverage(): if overage is enabled + // with no monthly cap, there is nothing to request. On fetch error, fall + // through and let the user ask (matching web's "err toward show" behavior). + let extraUsage: ExtraUsage | null | undefined + try { + const utilization = await fetchUtilization() + extraUsage = utilization?.extra_usage + } catch (error) { + logError(error as Error) + } + + if (extraUsage?.is_enabled && extraUsage.monthly_limit === null) { + return { + type: 'message', + value: + 'Your organization already has unlimited extra usage. No request needed.', + } + } + + try { + const eligibility = await checkAdminRequestEligibility('limit_increase') + if (eligibility?.is_allowed === false) { + return { + type: 'message', + value: 'Please contact your admin to manage extra usage settings.', + } + } + } catch (error) { + logError(error as Error) + // If eligibility check fails, continue — the create endpoint will enforce if necessary + } + + try { + const pendingOrDismissedRequests = await getMyAdminRequests( + 'limit_increase', + ['pending', 'dismissed'], + ) + if (pendingOrDismissedRequests && pendingOrDismissedRequests.length > 0) { + return { + type: 'message', + value: + 'You have already submitted a request for extra usage to your admin.', + } + } + } catch (error) { + logError(error as Error) + // Fall through to creating a new request below + } + + try { + await createAdminRequest({ + request_type: 'limit_increase', + details: null, + }) + return { + type: 'message', + value: extraUsage?.is_enabled + ? 'Request sent to your admin to increase extra usage.' + : 'Request sent to your admin to enable extra usage.', + } + } catch (error) { + logError(error as Error) + // Fall through to generic message below + } + + return { + type: 'message', + value: 'Please contact your admin to manage extra usage settings.', + } + } + + const url = isTeamOrEnterprise + ? 'https://claude.ai/admin-settings/usage' + : 'https://claude.ai/settings/usage' + + try { + const opened = await openBrowser(url) + return { type: 'browser-opened', url, opened } + } catch (error) { + logError(error as Error) + return { + type: 'message', + value: `Failed to open browser. Please visit ${url} to manage extra usage.`, + } + } +} diff --git a/src/commands/extra-usage/extra-usage-noninteractive.ts b/src/commands/extra-usage/extra-usage-noninteractive.ts new file mode 100644 index 0000000..b4eabe8 --- /dev/null +++ b/src/commands/extra-usage/extra-usage-noninteractive.ts @@ -0,0 +1,16 @@ +import { runExtraUsage } from './extra-usage-core.js' + +export async function call(): Promise<{ type: 'text'; value: string }> { + const result = await runExtraUsage() + + if (result.type === 'message') { + return { type: 'text', value: result.value } + } + + return { + type: 'text', + value: result.opened + ? `Browser opened to manage extra usage. If it didn't open, visit: ${result.url}` + : `Please visit ${result.url} to manage extra usage.`, + } +} diff --git a/src/commands/extra-usage/extra-usage.tsx b/src/commands/extra-usage/extra-usage.tsx new file mode 100644 index 0000000..ca27f39 --- /dev/null +++ b/src/commands/extra-usage/extra-usage.tsx @@ -0,0 +1,17 @@ +import React from 'react'; +import type { LocalJSXCommandContext } from '../../commands.js'; +import type { LocalJSXCommandOnDone } from '../../types/command.js'; +import { Login } from '../login/login.js'; +import { runExtraUsage } from './extra-usage-core.js'; +export async function call(onDone: LocalJSXCommandOnDone, context: LocalJSXCommandContext): Promise { + const result = await runExtraUsage(); + if (result.type === 'message') { + onDone(result.value); + return null; + } + return { + context.onChangeAPIKey(); + onDone(success ? 'Login successful' : 'Login interrupted'); + }} />; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkxvY2FsSlNYQ29tbWFuZENvbnRleHQiLCJMb2NhbEpTWENvbW1hbmRPbkRvbmUiLCJMb2dpbiIsInJ1bkV4dHJhVXNhZ2UiLCJjYWxsIiwib25Eb25lIiwiY29udGV4dCIsIlByb21pc2UiLCJSZWFjdE5vZGUiLCJyZXN1bHQiLCJ0eXBlIiwidmFsdWUiLCJzdWNjZXNzIiwib25DaGFuZ2VBUElLZXkiXSwic291cmNlcyI6WyJleHRyYS11c2FnZS50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHR5cGUgeyBMb2NhbEpTWENvbW1hbmRDb250ZXh0IH0gZnJvbSAnLi4vLi4vY29tbWFuZHMuanMnXG5pbXBvcnQgdHlwZSB7IExvY2FsSlNYQ29tbWFuZE9uRG9uZSB9IGZyb20gJy4uLy4uL3R5cGVzL2NvbW1hbmQuanMnXG5pbXBvcnQgeyBMb2dpbiB9IGZyb20gJy4uL2xvZ2luL2xvZ2luLmpzJ1xuaW1wb3J0IHsgcnVuRXh0cmFVc2FnZSB9IGZyb20gJy4vZXh0cmEtdXNhZ2UtY29yZS5qcydcblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGNhbGwoXG4gIG9uRG9uZTogTG9jYWxKU1hDb21tYW5kT25Eb25lLFxuICBjb250ZXh0OiBMb2NhbEpTWENvbW1hbmRDb250ZXh0LFxuKTogUHJvbWlzZTxSZWFjdC5SZWFjdE5vZGUgfCBudWxsPiB7XG4gIGNvbnN0IHJlc3VsdCA9IGF3YWl0IHJ1bkV4dHJhVXNhZ2UoKVxuXG4gIGlmIChyZXN1bHQudHlwZSA9PT0gJ21lc3NhZ2UnKSB7XG4gICAgb25Eb25lKHJlc3VsdC52YWx1ZSlcbiAgICByZXR1cm4gbnVsbFxuICB9XG5cbiAgcmV0dXJuIChcbiAgICA8TG9naW5cbiAgICAgIHN0YXJ0aW5nTWVzc2FnZT17XG4gICAgICAgICdTdGFydGluZyBuZXcgbG9naW4gZm9sbG93aW5nIC9leHRyYS11c2FnZS4gRXhpdCB3aXRoIEN0cmwtQyB0byB1c2UgZXhpc3RpbmcgYWNjb3VudC4nXG4gICAgICB9XG4gICAgICBvbkRvbmU9e3N1Y2Nlc3MgPT4ge1xuICAgICAgICBjb250ZXh0Lm9uQ2hhbmdlQVBJS2V5KClcbiAgICAgICAgb25Eb25lKHN1Y2Nlc3MgPyAnTG9naW4gc3VjY2Vzc2Z1bCcgOiAnTG9naW4gaW50ZXJydXB0ZWQnKVxuICAgICAgfX1cbiAgICAvPlxuICApXG59XG4iXSwibWFwcGluZ3MiOiJBQUFBLE9BQU9BLEtBQUssTUFBTSxPQUFPO0FBQ3pCLGNBQWNDLHNCQUFzQixRQUFRLG1CQUFtQjtBQUMvRCxjQUFjQyxxQkFBcUIsUUFBUSx3QkFBd0I7QUFDbkUsU0FBU0MsS0FBSyxRQUFRLG1CQUFtQjtBQUN6QyxTQUFTQyxhQUFhLFFBQVEsdUJBQXVCO0FBRXJELE9BQU8sZUFBZUMsSUFBSUEsQ0FDeEJDLE1BQU0sRUFBRUoscUJBQXFCLEVBQzdCSyxPQUFPLEVBQUVOLHNCQUFzQixDQUNoQyxFQUFFTyxPQUFPLENBQUNSLEtBQUssQ0FBQ1MsU0FBUyxHQUFHLElBQUksQ0FBQyxDQUFDO0VBQ2pDLE1BQU1DLE1BQU0sR0FBRyxNQUFNTixhQUFhLENBQUMsQ0FBQztFQUVwQyxJQUFJTSxNQUFNLENBQUNDLElBQUksS0FBSyxTQUFTLEVBQUU7SUFDN0JMLE1BQU0sQ0FBQ0ksTUFBTSxDQUFDRSxLQUFLLENBQUM7SUFDcEIsT0FBTyxJQUFJO0VBQ2I7RUFFQSxPQUNFLENBQUMsS0FBSyxDQUNKLGVBQWUsQ0FBQyxDQUNkLHNGQUNGLENBQUMsQ0FDRCxNQUFNLENBQUMsQ0FBQ0MsT0FBTyxJQUFJO0lBQ2pCTixPQUFPLENBQUNPLGNBQWMsQ0FBQyxDQUFDO0lBQ3hCUixNQUFNLENBQUNPLE9BQU8sR0FBRyxrQkFBa0IsR0FBRyxtQkFBbUIsQ0FBQztFQUM1RCxDQUFDLENBQUMsR0FDRjtBQUVOIiwiaWdub3JlTGlzdCI6W119 \ No newline at end of file diff --git a/src/commands/extra-usage/index.ts b/src/commands/extra-usage/index.ts new file mode 100644 index 0000000..cea0ba4 --- /dev/null +++ b/src/commands/extra-usage/index.ts @@ -0,0 +1,31 @@ +import { getIsNonInteractiveSession } from '../../bootstrap/state.js' +import type { Command } from '../../commands.js' +import { isOverageProvisioningAllowed } from '../../utils/auth.js' +import { isEnvTruthy } from '../../utils/envUtils.js' + +function isExtraUsageAllowed(): boolean { + if (isEnvTruthy(process.env.DISABLE_EXTRA_USAGE_COMMAND)) { + return false + } + return isOverageProvisioningAllowed() +} + +export const extraUsage = { + type: 'local-jsx', + name: 'extra-usage', + description: 'Configure extra usage to keep working when limits are hit', + isEnabled: () => isExtraUsageAllowed() && !getIsNonInteractiveSession(), + load: () => import('./extra-usage.js'), +} satisfies Command + +export const extraUsageNonInteractive = { + type: 'local', + name: 'extra-usage', + supportsNonInteractive: true, + description: 'Configure extra usage to keep working when limits are hit', + isEnabled: () => isExtraUsageAllowed() && getIsNonInteractiveSession(), + get isHidden() { + return !getIsNonInteractiveSession() + }, + load: () => import('./extra-usage-noninteractive.js'), +} satisfies Command diff --git a/src/commands/fast/fast.tsx b/src/commands/fast/fast.tsx new file mode 100644 index 0000000..398c3c3 --- /dev/null +++ b/src/commands/fast/fast.tsx @@ -0,0 +1,269 @@ +import { c as _c } from "react/compiler-runtime"; +import * as React from 'react'; +import { useState } from 'react'; +import type { CommandResultDisplay, LocalJSXCommandContext } from '../../commands.js'; +import { Dialog } from '../../components/design-system/Dialog.js'; +import { FastIcon, getFastIconString } from '../../components/FastIcon.js'; +import { Box, Link, Text } from '../../ink.js'; +import { useKeybindings } from '../../keybindings/useKeybinding.js'; +import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from '../../services/analytics/index.js'; +import { type AppState, useAppState, useSetAppState } from '../../state/AppState.js'; +import type { LocalJSXCommandOnDone } from '../../types/command.js'; +import { clearFastModeCooldown, FAST_MODE_MODEL_DISPLAY, getFastModeModel, getFastModeRuntimeState, getFastModeUnavailableReason, isFastModeEnabled, isFastModeSupportedByModel, prefetchFastModeStatus } from '../../utils/fastMode.js'; +import { formatDuration } from '../../utils/format.js'; +import { formatModelPricing, getOpus46CostTier } from '../../utils/modelCost.js'; +import { updateSettingsForSource } from '../../utils/settings/settings.js'; +function applyFastMode(enable: boolean, setAppState: (f: (prev: AppState) => AppState) => void): void { + clearFastModeCooldown(); + updateSettingsForSource('userSettings', { + fastMode: enable ? true : undefined + }); + if (enable) { + setAppState(prev => { + // Only switch model if current model doesn't support fast mode + const needsModelSwitch = !isFastModeSupportedByModel(prev.mainLoopModel); + return { + ...prev, + ...(needsModelSwitch ? { + mainLoopModel: getFastModeModel(), + mainLoopModelForSession: null + } : {}), + fastMode: true + }; + }); + } else { + setAppState(prev => ({ + ...prev, + fastMode: false + })); + } +} +export function FastModePicker(t0) { + const $ = _c(30); + const { + onDone, + unavailableReason + } = t0; + const model = useAppState(_temp); + const initialFastMode = useAppState(_temp2); + const setAppState = useSetAppState(); + const [enableFastMode, setEnableFastMode] = useState(initialFastMode ?? false); + let t1; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t1 = getFastModeRuntimeState(); + $[0] = t1; + } else { + t1 = $[0]; + } + const runtimeState = t1; + const isCooldown = runtimeState.status === "cooldown"; + const isUnavailable = unavailableReason !== null; + let t2; + if ($[1] === Symbol.for("react.memo_cache_sentinel")) { + t2 = formatModelPricing(getOpus46CostTier(true)); + $[1] = t2; + } else { + t2 = $[1]; + } + const pricing = t2; + let t3; + if ($[2] !== enableFastMode || $[3] !== isUnavailable || $[4] !== model || $[5] !== onDone || $[6] !== setAppState) { + t3 = function handleConfirm() { + if (isUnavailable) { + return; + } + applyFastMode(enableFastMode, setAppState); + logEvent("tengu_fast_mode_toggled", { + enabled: enableFastMode, + source: "picker" as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + if (enableFastMode) { + const fastIcon = getFastIconString(enableFastMode); + const modelUpdated = !isFastModeSupportedByModel(model) ? ` · model set to ${FAST_MODE_MODEL_DISPLAY}` : ""; + onDone(`${fastIcon} Fast mode ON${modelUpdated} · ${pricing}`); + } else { + setAppState(_temp3); + onDone("Fast mode OFF"); + } + }; + $[2] = enableFastMode; + $[3] = isUnavailable; + $[4] = model; + $[5] = onDone; + $[6] = setAppState; + $[7] = t3; + } else { + t3 = $[7]; + } + const handleConfirm = t3; + let t4; + if ($[8] !== initialFastMode || $[9] !== isUnavailable || $[10] !== onDone || $[11] !== setAppState) { + t4 = function handleCancel() { + if (isUnavailable) { + if (initialFastMode) { + applyFastMode(false, setAppState); + } + onDone("Fast mode OFF", { + display: "system" + }); + return; + } + const message = initialFastMode ? `${getFastIconString()} Kept Fast mode ON` : "Kept Fast mode OFF"; + onDone(message, { + display: "system" + }); + }; + $[8] = initialFastMode; + $[9] = isUnavailable; + $[10] = onDone; + $[11] = setAppState; + $[12] = t4; + } else { + t4 = $[12]; + } + const handleCancel = t4; + let t5; + if ($[13] !== isUnavailable) { + t5 = function handleToggle() { + if (isUnavailable) { + return; + } + setEnableFastMode(_temp4); + }; + $[13] = isUnavailable; + $[14] = t5; + } else { + t5 = $[14]; + } + const handleToggle = t5; + let t6; + if ($[15] !== handleConfirm || $[16] !== handleToggle) { + t6 = { + "confirm:yes": handleConfirm, + "confirm:nextField": handleToggle, + "confirm:next": handleToggle, + "confirm:previous": handleToggle, + "confirm:cycleMode": handleToggle, + "confirm:toggle": handleToggle + }; + $[15] = handleConfirm; + $[16] = handleToggle; + $[17] = t6; + } else { + t6 = $[17]; + } + let t7; + if ($[18] === Symbol.for("react.memo_cache_sentinel")) { + t7 = { + context: "Confirmation" + }; + $[18] = t7; + } else { + t7 = $[18]; + } + useKeybindings(t6, t7); + let t8; + if ($[19] === Symbol.for("react.memo_cache_sentinel")) { + t8 = Fast mode (research preview); + $[19] = t8; + } else { + t8 = $[19]; + } + const title = t8; + let t9; + if ($[20] !== isUnavailable) { + t9 = exitState => exitState.pending ? Press {exitState.keyName} again to exit : isUnavailable ? Esc to cancel : Tab to toggle · Enter to confirm · Esc to cancel; + $[20] = isUnavailable; + $[21] = t9; + } else { + t9 = $[21]; + } + let t10; + if ($[22] !== enableFastMode || $[23] !== unavailableReason) { + t10 = unavailableReason ? {unavailableReason} : <>Fast mode{enableFastMode ? "ON " : "OFF"}{pricing}{isCooldown && runtimeState.status === "cooldown" && {runtimeState.reason === "overloaded" ? "Fast mode overloaded and is temporarily unavailable" : "You've hit your fast limit"}{" \xB7 resets in "}{formatDuration(runtimeState.resetAt - Date.now(), { + hideTrailingZeros: true + })}}; + $[22] = enableFastMode; + $[23] = unavailableReason; + $[24] = t10; + } else { + t10 = $[24]; + } + let t11; + if ($[25] === Symbol.for("react.memo_cache_sentinel")) { + t11 = Learn more:{" "}https://code.claude.com/docs/en/fast-mode; + $[25] = t11; + } else { + t11 = $[25]; + } + let t12; + if ($[26] !== handleCancel || $[27] !== t10 || $[28] !== t9) { + t12 = {t10}{t11}; + $[26] = handleCancel; + $[27] = t10; + $[28] = t9; + $[29] = t12; + } else { + t12 = $[29]; + } + return t12; +} +function _temp4(prev_0) { + return !prev_0; +} +function _temp3(prev) { + return { + ...prev, + fastMode: false + }; +} +function _temp2(s_0) { + return s_0.fastMode; +} +function _temp(s) { + return s.mainLoopModel; +} +async function handleFastModeShortcut(enable: boolean, getAppState: () => AppState, setAppState: (f: (prev: AppState) => AppState) => void): Promise { + const unavailableReason = getFastModeUnavailableReason(); + if (unavailableReason) { + return `Fast mode unavailable: ${unavailableReason}`; + } + const { + mainLoopModel + } = getAppState(); + applyFastMode(enable, setAppState); + logEvent('tengu_fast_mode_toggled', { + enabled: enable, + source: 'shortcut' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + if (enable) { + const fastIcon = getFastIconString(true); + const modelUpdated = !isFastModeSupportedByModel(mainLoopModel) ? ` · model set to ${FAST_MODE_MODEL_DISPLAY}` : ''; + const pricing = formatModelPricing(getOpus46CostTier(true)); + return `${fastIcon} Fast mode ON${modelUpdated} · ${pricing}`; + } else { + return `Fast mode OFF`; + } +} +export async function call(onDone: LocalJSXCommandOnDone, context: LocalJSXCommandContext, args?: string): Promise { + if (!isFastModeEnabled()) { + return null; + } + + // Fetch org fast mode status before showing the picker. We must know + // whether the org has disabled fast mode before allowing any toggle. + // If a startup prefetch is already in flight, this awaits it. + await prefetchFastModeStatus(); + const arg = args?.trim().toLowerCase(); + if (arg === 'on' || arg === 'off') { + const result = await handleFastModeShortcut(arg === 'on', context.getAppState, context.setAppState); + onDone(result); + return null; + } + const unavailableReason = getFastModeUnavailableReason(); + logEvent('tengu_fast_mode_picker_shown', { + unavailable_reason: (unavailableReason ?? '') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + return ; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsInVzZVN0YXRlIiwiQ29tbWFuZFJlc3VsdERpc3BsYXkiLCJMb2NhbEpTWENvbW1hbmRDb250ZXh0IiwiRGlhbG9nIiwiRmFzdEljb24iLCJnZXRGYXN0SWNvblN0cmluZyIsIkJveCIsIkxpbmsiLCJUZXh0IiwidXNlS2V5YmluZGluZ3MiLCJBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTIiwibG9nRXZlbnQiLCJBcHBTdGF0ZSIsInVzZUFwcFN0YXRlIiwidXNlU2V0QXBwU3RhdGUiLCJMb2NhbEpTWENvbW1hbmRPbkRvbmUiLCJjbGVhckZhc3RNb2RlQ29vbGRvd24iLCJGQVNUX01PREVfTU9ERUxfRElTUExBWSIsImdldEZhc3RNb2RlTW9kZWwiLCJnZXRGYXN0TW9kZVJ1bnRpbWVTdGF0ZSIsImdldEZhc3RNb2RlVW5hdmFpbGFibGVSZWFzb24iLCJpc0Zhc3RNb2RlRW5hYmxlZCIsImlzRmFzdE1vZGVTdXBwb3J0ZWRCeU1vZGVsIiwicHJlZmV0Y2hGYXN0TW9kZVN0YXR1cyIsImZvcm1hdER1cmF0aW9uIiwiZm9ybWF0TW9kZWxQcmljaW5nIiwiZ2V0T3B1czQ2Q29zdFRpZXIiLCJ1cGRhdGVTZXR0aW5nc0ZvclNvdXJjZSIsImFwcGx5RmFzdE1vZGUiLCJlbmFibGUiLCJzZXRBcHBTdGF0ZSIsImYiLCJwcmV2IiwiZmFzdE1vZGUiLCJ1bmRlZmluZWQiLCJuZWVkc01vZGVsU3dpdGNoIiwibWFpbkxvb3BNb2RlbCIsIm1haW5Mb29wTW9kZWxGb3JTZXNzaW9uIiwiRmFzdE1vZGVQaWNrZXIiLCJ0MCIsIiQiLCJfYyIsIm9uRG9uZSIsInVuYXZhaWxhYmxlUmVhc29uIiwibW9kZWwiLCJfdGVtcCIsImluaXRpYWxGYXN0TW9kZSIsIl90ZW1wMiIsImVuYWJsZUZhc3RNb2RlIiwic2V0RW5hYmxlRmFzdE1vZGUiLCJ0MSIsIlN5bWJvbCIsImZvciIsInJ1bnRpbWVTdGF0ZSIsImlzQ29vbGRvd24iLCJzdGF0dXMiLCJpc1VuYXZhaWxhYmxlIiwidDIiLCJwcmljaW5nIiwidDMiLCJoYW5kbGVDb25maXJtIiwiZW5hYmxlZCIsInNvdXJjZSIsImZhc3RJY29uIiwibW9kZWxVcGRhdGVkIiwiX3RlbXAzIiwidDQiLCJoYW5kbGVDYW5jZWwiLCJkaXNwbGF5IiwibWVzc2FnZSIsInQ1IiwiaGFuZGxlVG9nZ2xlIiwiX3RlbXA0IiwidDYiLCJ0NyIsImNvbnRleHQiLCJ0OCIsInRpdGxlIiwidDkiLCJleGl0U3RhdGUiLCJwZW5kaW5nIiwia2V5TmFtZSIsInQxMCIsInJlYXNvbiIsInJlc2V0QXQiLCJEYXRlIiwibm93IiwiaGlkZVRyYWlsaW5nWmVyb3MiLCJ0MTEiLCJ0MTIiLCJwcmV2XzAiLCJzXzAiLCJzIiwiaGFuZGxlRmFzdE1vZGVTaG9ydGN1dCIsImdldEFwcFN0YXRlIiwiUHJvbWlzZSIsImNhbGwiLCJhcmdzIiwiUmVhY3ROb2RlIiwiYXJnIiwidHJpbSIsInRvTG93ZXJDYXNlIiwicmVzdWx0IiwidW5hdmFpbGFibGVfcmVhc29uIl0sInNvdXJjZXMiOlsiZmFzdC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHR5cGUge1xuICBDb21tYW5kUmVzdWx0RGlzcGxheSxcbiAgTG9jYWxKU1hDb21tYW5kQ29udGV4dCxcbn0gZnJvbSAnLi4vLi4vY29tbWFuZHMuanMnXG5pbXBvcnQgeyBEaWFsb2cgfSBmcm9tICcuLi8uLi9jb21wb25lbnRzL2Rlc2lnbi1zeXN0ZW0vRGlhbG9nLmpzJ1xuaW1wb3J0IHsgRmFzdEljb24sIGdldEZhc3RJY29uU3RyaW5nIH0gZnJvbSAnLi4vLi4vY29tcG9uZW50cy9GYXN0SWNvbi5qcydcbmltcG9ydCB7IEJveCwgTGluaywgVGV4dCB9IGZyb20gJy4uLy4uL2luay5qcydcbmltcG9ydCB7IHVzZUtleWJpbmRpbmdzIH0gZnJvbSAnLi4vLi4va2V5YmluZGluZ3MvdXNlS2V5YmluZGluZy5qcydcbmltcG9ydCB7XG4gIHR5cGUgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgbG9nRXZlbnQsXG59IGZyb20gJy4uLy4uL3NlcnZpY2VzL2FuYWx5dGljcy9pbmRleC5qcydcbmltcG9ydCB7XG4gIHR5cGUgQXBwU3RhdGUsXG4gIHVzZUFwcFN0YXRlLFxuICB1c2VTZXRBcHBTdGF0ZSxcbn0gZnJvbSAnLi4vLi4vc3RhdGUvQXBwU3RhdGUuanMnXG5pbXBvcnQgdHlwZSB7IExvY2FsSlNYQ29tbWFuZE9uRG9uZSB9IGZyb20gJy4uLy4uL3R5cGVzL2NvbW1hbmQuanMnXG5pbXBvcnQge1xuICBjbGVhckZhc3RNb2RlQ29vbGRvd24sXG4gIEZBU1RfTU9ERV9NT0RFTF9ESVNQTEFZLFxuICBnZXRGYXN0TW9kZU1vZGVsLFxuICBnZXRGYXN0TW9kZVJ1bnRpbWVTdGF0ZSxcbiAgZ2V0RmFzdE1vZGVVbmF2YWlsYWJsZVJlYXNvbixcbiAgaXNGYXN0TW9kZUVuYWJsZWQsXG4gIGlzRmFzdE1vZGVTdXBwb3J0ZWRCeU1vZGVsLFxuICBwcmVmZXRjaEZhc3RNb2RlU3RhdHVzLFxufSBmcm9tICcuLi8uLi91dGlscy9mYXN0TW9kZS5qcydcbmltcG9ydCB7IGZvcm1hdER1cmF0aW9uIH0gZnJvbSAnLi4vLi4vdXRpbHMvZm9ybWF0LmpzJ1xuaW1wb3J0IHsgZm9ybWF0TW9kZWxQcmljaW5nLCBnZXRPcHVzNDZDb3N0VGllciB9IGZyb20gJy4uLy4uL3V0aWxzL21vZGVsQ29zdC5qcydcbmltcG9ydCB7IHVwZGF0ZVNldHRpbmdzRm9yU291cmNlIH0gZnJvbSAnLi4vLi4vdXRpbHMvc2V0dGluZ3Mvc2V0dGluZ3MuanMnXG5cbmZ1bmN0aW9uIGFwcGx5RmFzdE1vZGUoXG4gIGVuYWJsZTogYm9vbGVhbixcbiAgc2V0QXBwU3RhdGU6IChmOiAocHJldjogQXBwU3RhdGUpID0+IEFwcFN0YXRlKSA9PiB2b2lkLFxuKTogdm9pZCB7XG4gIGNsZWFyRmFzdE1vZGVDb29sZG93bigpXG4gIHVwZGF0ZVNldHRpbmdzRm9yU291cmNlKCd1c2VyU2V0dGluZ3MnLCB7XG4gICAgZmFzdE1vZGU6IGVuYWJsZSA/IHRydWUgOiB1bmRlZmluZWQsXG4gIH0pXG4gIGlmIChlbmFibGUpIHtcbiAgICBzZXRBcHBTdGF0ZShwcmV2ID0+IHtcbiAgICAgIC8vIE9ubHkgc3dpdGNoIG1vZGVsIGlmIGN1cnJlbnQgbW9kZWwgZG9lc24ndCBzdXBwb3J0IGZhc3QgbW9kZVxuICAgICAgY29uc3QgbmVlZHNNb2RlbFN3aXRjaCA9ICFpc0Zhc3RNb2RlU3VwcG9ydGVkQnlNb2RlbChwcmV2Lm1haW5Mb29wTW9kZWwpXG4gICAgICByZXR1cm4ge1xuICAgICAgICAuLi5wcmV2LFxuICAgICAgICAuLi4obmVlZHNNb2RlbFN3aXRjaFxuICAgICAgICAgID8geyBtYWluTG9vcE1vZGVsOiBnZXRGYXN0TW9kZU1vZGVsKCksIG1haW5Mb29wTW9kZWxGb3JTZXNzaW9uOiBudWxsIH1cbiAgICAgICAgICA6IHt9KSxcbiAgICAgICAgZmFzdE1vZGU6IHRydWUsXG4gICAgICB9XG4gICAgfSlcbiAgfSBlbHNlIHtcbiAgICBzZXRBcHBTdGF0ZShwcmV2ID0+ICh7IC4uLnByZXYsIGZhc3RNb2RlOiBmYWxzZSB9KSlcbiAgfVxufVxuXG5leHBvcnQgZnVuY3Rpb24gRmFzdE1vZGVQaWNrZXIoe1xuICBvbkRvbmUsXG4gIHVuYXZhaWxhYmxlUmVhc29uLFxufToge1xuICBvbkRvbmU6IChcbiAgICByZXN1bHQ/OiBzdHJpbmcsXG4gICAgb3B0aW9ucz86IHsgZGlzcGxheT86IENvbW1hbmRSZXN1bHREaXNwbGF5IH0sXG4gICkgPT4gdm9pZFxuICB1bmF2YWlsYWJsZVJlYXNvbjogc3RyaW5nIHwgbnVsbFxufSk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IG1vZGVsID0gdXNlQXBwU3RhdGUocyA9PiBzLm1haW5Mb29wTW9kZWwpXG4gIGNvbnN0IGluaXRpYWxGYXN0TW9kZSA9IHVzZUFwcFN0YXRlKHMgPT4gcy5mYXN0TW9kZSlcbiAgY29uc3Qgc2V0QXBwU3RhdGUgPSB1c2VTZXRBcHBTdGF0ZSgpXG4gIGNvbnN0IFtlbmFibGVGYXN0TW9kZSwgc2V0RW5hYmxlRmFzdE1vZGVdID0gdXNlU3RhdGUoaW5pdGlhbEZhc3RNb2RlID8/IGZhbHNlKVxuICBjb25zdCBydW50aW1lU3RhdGUgPSBnZXRGYXN0TW9kZVJ1bnRpbWVTdGF0ZSgpXG4gIGNvbnN0IGlzQ29vbGRvd24gPSBydW50aW1lU3RhdGUuc3RhdHVzID09PSAnY29vbGRvd24nXG4gIGNvbnN0IGlzVW5hdmFpbGFibGUgPSB1bmF2YWlsYWJsZVJlYXNvbiAhPT0gbnVsbFxuICBjb25zdCBwcmljaW5nID0gZm9ybWF0TW9kZWxQcmljaW5nKGdldE9wdXM0NkNvc3RUaWVyKHRydWUpKVxuXG4gIGZ1bmN0aW9uIGhhbmRsZUNvbmZpcm0oKTogdm9pZCB7XG4gICAgaWYgKGlzVW5hdmFpbGFibGUpIHJldHVyblxuICAgIGFwcGx5RmFzdE1vZGUoZW5hYmxlRmFzdE1vZGUsIHNldEFwcFN0YXRlKVxuICAgIGxvZ0V2ZW50KCd0ZW5ndV9mYXN0X21vZGVfdG9nZ2xlZCcsIHtcbiAgICAgIGVuYWJsZWQ6IGVuYWJsZUZhc3RNb2RlLFxuICAgICAgc291cmNlOlxuICAgICAgICAncGlja2VyJyBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgIH0pXG4gICAgaWYgKGVuYWJsZUZhc3RNb2RlKSB7XG4gICAgICBjb25zdCBmYXN0SWNvbiA9IGdldEZhc3RJY29uU3RyaW5nKGVuYWJsZUZhc3RNb2RlKVxuICAgICAgY29uc3QgbW9kZWxVcGRhdGVkID0gIWlzRmFzdE1vZGVTdXBwb3J0ZWRCeU1vZGVsKG1vZGVsKVxuICAgICAgICA/IGAgwrcgbW9kZWwgc2V0IHRvICR7RkFTVF9NT0RFX01PREVMX0RJU1BMQVl9YFxuICAgICAgICA6ICcnXG4gICAgICBvbkRvbmUoYCR7ZmFzdEljb259IEZhc3QgbW9kZSBPTiR7bW9kZWxVcGRhdGVkfSDCtyAke3ByaWNpbmd9YClcbiAgICB9IGVsc2Uge1xuICAgICAgc2V0QXBwU3RhdGUocHJldiA9PiAoeyAuLi5wcmV2LCBmYXN0TW9kZTogZmFsc2UgfSkpXG4gICAgICBvbkRvbmUoYEZhc3QgbW9kZSBPRkZgKVxuICAgIH1cbiAgfVxuXG4gIGZ1bmN0aW9uIGhhbmRsZUNhbmNlbCgpOiB2b2lkIHtcbiAgICBpZiAoaXNVbmF2YWlsYWJsZSkge1xuICAgICAgLy8gRW5zdXJlIGZhc3QgbW9kZSBpcyBvZmYgaWYgdGhlIG9yZyBoYXMgZGlzYWJsZWQgaXRcbiAgICAgIGlmIChpbml0aWFsRmFzdE1vZGUpIHtcbiAgICAgICAgYXBwbHlGYXN0TW9kZShmYWxzZSwgc2V0QXBwU3RhdGUpXG4gICAgICB9XG4gICAgICBvbkRvbmUoJ0Zhc3QgbW9kZSBPRkYnLCB7IGRpc3BsYXk6ICdzeXN0ZW0nIH0pXG4gICAgICByZXR1cm5cbiAgICB9XG4gICAgY29uc3QgbWVzc2FnZSA9IGluaXRpYWxGYXN0TW9kZVxuICAgICAgPyBgJHtnZXRGYXN0SWNvblN0cmluZygpfSBLZXB0IEZhc3QgbW9kZSBPTmBcbiAgICAgIDogYEtlcHQgRmFzdCBtb2RlIE9GRmBcbiAgICBvbkRvbmUobWVzc2FnZSwgeyBkaXNwbGF5OiAnc3lzdGVtJyB9KVxuICB9XG5cbiAgZnVuY3Rpb24gaGFuZGxlVG9nZ2xlKCk6IHZvaWQge1xuICAgIGlmIChpc1VuYXZhaWxhYmxlKSByZXR1cm5cbiAgICBzZXRFbmFibGVGYXN0TW9kZShwcmV2ID0+ICFwcmV2KVxuICB9XG5cbiAgdXNlS2V5YmluZGluZ3MoXG4gICAge1xuICAgICAgJ2NvbmZpcm06eWVzJzogaGFuZGxlQ29uZmlybSxcbiAgICAgICdjb25maXJtOm5leHRGaWVsZCc6IGhhbmRsZVRvZ2dsZSxcbiAgICAgICdjb25maXJtOm5leHQnOiBoYW5kbGVUb2dnbGUsXG4gICAgICAnY29uZmlybTpwcmV2aW91cyc6IGhhbmRsZVRvZ2dsZSxcbiAgICAgICdjb25maXJtOmN5Y2xlTW9kZSc6IGhhbmRsZVRvZ2dsZSxcbiAgICAgICdjb25maXJtOnRvZ2dsZSc6IGhhbmRsZVRvZ2dsZSxcbiAgICB9LFxuICAgIHsgY29udGV4dDogJ0NvbmZpcm1hdGlvbicgfSxcbiAgKVxuXG4gIGNvbnN0IHRpdGxlID0gKFxuICAgIDxUZXh0PlxuICAgICAgPEZhc3RJY29uIGNvb2xkb3duPXtpc0Nvb2xkb3dufSAvPiBGYXN0IG1vZGUgKHJlc2VhcmNoIHByZXZpZXcpXG4gICAgPC9UZXh0PlxuICApXG5cbiAgcmV0dXJuIChcbiAgICA8RGlhbG9nXG4gICAgICB0aXRsZT17dGl0bGV9XG4gICAgICBzdWJ0aXRsZT17YEhpZ2gtc3BlZWQgbW9kZSBmb3IgJHtGQVNUX01PREVfTU9ERUxfRElTUExBWX0uIEJpbGxlZCBhcyBleHRyYSB1c2FnZSBhdCBhIHByZW1pdW0gcmF0ZS4gU2VwYXJhdGUgcmF0ZSBsaW1pdHMgYXBwbHkuYH1cbiAgICAgIG9uQ2FuY2VsPXtoYW5kbGVDYW5jZWx9XG4gICAgICBjb2xvcj1cImZhc3RNb2RlXCJcbiAgICAgIGlucHV0R3VpZGU9e2V4aXRTdGF0ZSA9PlxuICAgICAgICBleGl0U3RhdGUucGVuZGluZyA/IChcbiAgICAgICAgICA8VGV4dD5QcmVzcyB7ZXhpdFN0YXRlLmtleU5hbWV9IGFnYWluIHRvIGV4aXQ8L1RleHQ+XG4gICAgICAgICkgOiBpc1VuYXZhaWxhYmxlID8gKFxuICAgICAgICAgIDxUZXh0PkVzYyB0byBjYW5jZWw8L1RleHQ+XG4gICAgICAgICkgOiAoXG4gICAgICAgICAgPFRleHQ+VGFiIHRvIHRvZ2dsZSDCtyBFbnRlciB0byBjb25maXJtIMK3IEVzYyB0byBjYW5jZWw8L1RleHQ+XG4gICAgICAgIClcbiAgICAgIH1cbiAgICA+XG4gICAgICB7dW5hdmFpbGFibGVSZWFzb24gPyAoXG4gICAgICAgIDxCb3ggbWFyZ2luTGVmdD17Mn0+XG4gICAgICAgICAgPFRleHQgY29sb3I9XCJlcnJvclwiPnt1bmF2YWlsYWJsZVJlYXNvbn08L1RleHQ+XG4gICAgICAgIDwvQm94PlxuICAgICAgKSA6IChcbiAgICAgICAgPD5cbiAgICAgICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIiBnYXA9ezB9IG1hcmdpbkxlZnQ9ezJ9PlxuICAgICAgICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwicm93XCIgZ2FwPXsyfT5cbiAgICAgICAgICAgICAgPFRleHQgYm9sZD5GYXN0IG1vZGU8L1RleHQ+XG4gICAgICAgICAgICAgIDxUZXh0XG4gICAgICAgICAgICAgICAgY29sb3I9e2VuYWJsZUZhc3RNb2RlID8gJ2Zhc3RNb2RlJyA6IHVuZGVmaW5lZH1cbiAgICAgICAgICAgICAgICBib2xkPXtlbmFibGVGYXN0TW9kZX1cbiAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgIHtlbmFibGVGYXN0TW9kZSA/ICdPTiAnIDogJ09GRid9XG4gICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgPFRleHQgZGltQ29sb3I+e3ByaWNpbmd9PC9UZXh0PlxuICAgICAgICAgICAgPC9Cb3g+XG4gICAgICAgICAgPC9Cb3g+XG5cbiAgICAgICAgICB7aXNDb29sZG93biAmJiBydW50aW1lU3RhdGUuc3RhdHVzID09PSAnY29vbGRvd24nICYmIChcbiAgICAgICAgICAgIDxCb3ggbWFyZ2luTGVmdD17Mn0+XG4gICAgICAgICAgICAgIDxUZXh0IGNvbG9yPVwid2FybmluZ1wiPlxuICAgICAgICAgICAgICAgIHtydW50aW1lU3RhdGUucmVhc29uID09PSAnb3ZlcmxvYWRlZCdcbiAgICAgICAgICAgICAgICAgID8gJ0Zhc3QgbW9kZSBvdmVybG9hZGVkIGFuZCBpcyB0ZW1wb3JhcmlseSB1bmF2YWlsYWJsZSdcbiAgICAgICAgICAgICAgICAgIDogXCJZb3UndmUgaGl0IHlvdXIgZmFzdCBsaW1pdFwifVxuICAgICAgICAgICAgICAgIHsnIMK3IHJlc2V0cyBpbiAnfVxuICAgICAgICAgICAgICAgIHtmb3JtYXREdXJhdGlvbihydW50aW1lU3RhdGUucmVzZXRBdCAtIERhdGUubm93KCksIHtcbiAgICAgICAgICAgICAgICAgIGhpZGVUcmFpbGluZ1plcm9zOiB0cnVlLFxuICAgICAgICAgICAgICAgIH0pfVxuICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICA8L0JveD5cbiAgICAgICAgICApfVxuICAgICAgICA8Lz5cbiAgICAgICl9XG4gICAgICA8VGV4dCBkaW1Db2xvcj5cbiAgICAgICAgTGVhcm4gbW9yZTp7JyAnfVxuICAgICAgICA8TGluayB1cmw9XCJodHRwczovL2NvZGUuY2xhdWRlLmNvbS9kb2NzL2VuL2Zhc3QtbW9kZVwiPlxuICAgICAgICAgIGh0dHBzOi8vY29kZS5jbGF1ZGUuY29tL2RvY3MvZW4vZmFzdC1tb2RlXG4gICAgICAgIDwvTGluaz5cbiAgICAgIDwvVGV4dD5cbiAgICA8L0RpYWxvZz5cbiAgKVxufVxuXG5hc3luYyBmdW5jdGlvbiBoYW5kbGVGYXN0TW9kZVNob3J0Y3V0KFxuICBlbmFibGU6IGJvb2xlYW4sXG4gIGdldEFwcFN0YXRlOiAoKSA9PiBBcHBTdGF0ZSxcbiAgc2V0QXBwU3RhdGU6IChmOiAocHJldjogQXBwU3RhdGUpID0+IEFwcFN0YXRlKSA9PiB2b2lkLFxuKTogUHJvbWlzZTxzdHJpbmc+IHtcbiAgY29uc3QgdW5hdmFpbGFibGVSZWFzb24gPSBnZXRGYXN0TW9kZVVuYXZhaWxhYmxlUmVhc29uKClcbiAgaWYgKHVuYXZhaWxhYmxlUmVhc29uKSB7XG4gICAgcmV0dXJuIGBGYXN0IG1vZGUgdW5hdmFpbGFibGU6ICR7dW5hdmFpbGFibGVSZWFzb259YFxuICB9XG5cbiAgY29uc3QgeyBtYWluTG9vcE1vZGVsIH0gPSBnZXRBcHBTdGF0ZSgpXG4gIGFwcGx5RmFzdE1vZGUoZW5hYmxlLCBzZXRBcHBTdGF0ZSlcbiAgbG9nRXZlbnQoJ3Rlbmd1X2Zhc3RfbW9kZV90b2dnbGVkJywge1xuICAgIGVuYWJsZWQ6IGVuYWJsZSxcbiAgICBzb3VyY2U6XG4gICAgICAnc2hvcnRjdXQnIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gIH0pXG5cbiAgaWYgKGVuYWJsZSkge1xuICAgIGNvbnN0IGZhc3RJY29uID0gZ2V0RmFzdEljb25TdHJpbmcodHJ1ZSlcbiAgICBjb25zdCBtb2RlbFVwZGF0ZWQgPSAhaXNGYXN0TW9kZVN1cHBvcnRlZEJ5TW9kZWwobWFpbkxvb3BNb2RlbClcbiAgICAgID8gYCDCtyBtb2RlbCBzZXQgdG8gJHtGQVNUX01PREVfTU9ERUxfRElTUExBWX1gXG4gICAgICA6ICcnXG4gICAgY29uc3QgcHJpY2luZyA9IGZvcm1hdE1vZGVsUHJpY2luZyhnZXRPcHVzNDZDb3N0VGllcih0cnVlKSlcbiAgICByZXR1cm4gYCR7ZmFzdEljb259IEZhc3QgbW9kZSBPTiR7bW9kZWxVcGRhdGVkfSDCtyAke3ByaWNpbmd9YFxuICB9IGVsc2Uge1xuICAgIHJldHVybiBgRmFzdCBtb2RlIE9GRmBcbiAgfVxufVxuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gY2FsbChcbiAgb25Eb25lOiBMb2NhbEpTWENvbW1hbmRPbkRvbmUsXG4gIGNvbnRleHQ6IExvY2FsSlNYQ29tbWFuZENvbnRleHQsXG4gIGFyZ3M/OiBzdHJpbmcsXG4pOiBQcm9taXNlPFJlYWN0LlJlYWN0Tm9kZSB8IG51bGw+IHtcbiAgaWYgKCFpc0Zhc3RNb2RlRW5hYmxlZCgpKSB7XG4gICAgcmV0dXJuIG51bGxcbiAgfVxuXG4gIC8vIEZldGNoIG9yZyBmYXN0IG1vZGUgc3RhdHVzIGJlZm9yZSBzaG93aW5nIHRoZSBwaWNrZXIuIFdlIG11c3Qga25vd1xuICAvLyB3aGV0aGVyIHRoZSBvcmcgaGFzIGRpc2FibGVkIGZhc3QgbW9kZSBiZWZvcmUgYWxsb3dpbmcgYW55IHRvZ2dsZS5cbiAgLy8gSWYgYSBzdGFydHVwIHByZWZldGNoIGlzIGFscmVhZHkgaW4gZmxpZ2h0LCB0aGlzIGF3YWl0cyBpdC5cbiAgYXdhaXQgcHJlZmV0Y2hGYXN0TW9kZVN0YXR1cygpXG5cbiAgY29uc3QgYXJnID0gYXJncz8udHJpbSgpLnRvTG93ZXJDYXNlKClcbiAgaWYgKGFyZyA9PT0gJ29uJyB8fCBhcmcgPT09ICdvZmYnKSB7XG4gICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgaGFuZGxlRmFzdE1vZGVTaG9ydGN1dChcbiAgICAgIGFyZyA9PT0gJ29uJyxcbiAgICAgIGNvbnRleHQuZ2V0QXBwU3RhdGUsXG4gICAgICBjb250ZXh0LnNldEFwcFN0YXRlLFxuICAgIClcbiAgICBvbkRvbmUocmVzdWx0KVxuICAgIHJldHVybiBudWxsXG4gIH1cblxuICBjb25zdCB1bmF2YWlsYWJsZVJlYXNvbiA9IGdldEZhc3RNb2RlVW5hdmFpbGFibGVSZWFzb24oKVxuICBsb2dFdmVudCgndGVuZ3VfZmFzdF9tb2RlX3BpY2tlcl9zaG93bicsIHtcbiAgICB1bmF2YWlsYWJsZV9yZWFzb246ICh1bmF2YWlsYWJsZVJlYXNvbiA/P1xuICAgICAgJycpIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gIH0pXG4gIHJldHVybiAoXG4gICAgPEZhc3RNb2RlUGlja2VyIG9uRG9uZT17b25Eb25lfSB1bmF2YWlsYWJsZVJlYXNvbj17dW5hdmFpbGFibGVSZWFzb259IC8+XG4gIClcbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU8sS0FBS0EsS0FBSyxNQUFNLE9BQU87QUFDOUIsU0FBU0MsUUFBUSxRQUFRLE9BQU87QUFDaEMsY0FDRUMsb0JBQW9CLEVBQ3BCQyxzQkFBc0IsUUFDakIsbUJBQW1CO0FBQzFCLFNBQVNDLE1BQU0sUUFBUSwwQ0FBMEM7QUFDakUsU0FBU0MsUUFBUSxFQUFFQyxpQkFBaUIsUUFBUSw4QkFBOEI7QUFDMUUsU0FBU0Ms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QUltQixNQUFNLEVBQUU7SUFDVixNQUFNa0MsUUFBUSxHQUFHMUQsaUJBQWlCLENBQUMsSUFBSSxDQUFDO0lBQ3hDLE1BQU0yRCxZQUFZLEdBQUcsQ0FBQzFDLDBCQUEwQixDQUFDYyxhQUFhLENBQUMsR0FDM0QsbUJBQW1CbkIsdUJBQXVCLEVBQUUsR0FDNUMsRUFBRTtJQUNOLE1BQU15QyxPQUFPLEdBQUdqQyxrQkFBa0IsQ0FBQ0MsaUJBQWlCLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDM0QsT0FBTyxHQUFHcUMsUUFBUSxnQkFBZ0JDLFlBQVksTUFBTU4sT0FBTyxFQUFFO0VBQy9ELENBQUMsTUFBTTtJQUNMLE9BQU8sZUFBZTtFQUN4QjtBQUNGO0FBRUEsT0FBTyxlQUFlc0MsSUFBSUEsQ0FDeEJ0RCxNQUFNLEVBQUUzQixxQkFBcUIsRUFDN0I0RCxPQUFPLEVBQUV6RSxzQkFBc0IsRUFDL0IrRixJQUFhLENBQVIsRUFBRSxNQUFNLENBQ2QsRUFBRUYsT0FBTyxDQUFDaEcsS0FBSyxDQUFDbUcsU0FBUyxHQUFHLElBQUksQ0FBQyxDQUFDO0VBQ2pDLElBQUksQ0FBQzdFLGlCQUFpQixDQUFDLENBQUMsRUFBRTtJQUN4QixPQUFPLElBQUk7RUFDYjs7RUFFQTtFQUNBO0VBQ0E7RUFDQSxNQUFNRSxzQkFBc0IsQ0FBQyxDQUFDO0VBRTlCLE1BQU00RSxHQUFHLEdBQUdGLElBQUksRUFBRUcsSUFBSSxDQUFDLENBQUMsQ0FBQ0MsV0FBVyxDQUFDLENBQUM7RUFDdEMsSUFBSUYsR0FBRyxLQUFLLElBQUksSUFBSUEsR0FBRyxLQUFLLEtBQUssRUFBRTtJQUNqQyxNQUFNRyxNQUFNLEdBQUcsTUFBTVQsc0JBQXNCLENBQ3pDTSxHQUFHLEtBQUssSUFBSSxFQUNaeEIsT0FBTyxDQUFDbUIsV0FBVyxFQUNuQm5CLE9BQU8sQ0FBQzdDLFdBQ1YsQ0FBQztJQUNEWSxNQUFNLENBQUM0RCxNQUFNLENBQUM7SUFDZCxPQUFPLElBQUk7RUFDYjtFQUVBLE1BQU0zRCxpQkFBaUIsR0FBR3ZCLDRCQUE0QixDQUFDLENBQUM7RUFDeERULFFBQVEsQ0FBQyw4QkFBOEIsRUFBRTtJQUN2QzRGLGtCQUFrQixFQUFFLENBQUM1RCxpQkFBaUIsSUFDcEMsRUFBRSxLQUFLakM7RUFDWCxDQUFDLENBQUM7RUFDRixPQUNFLENBQUMsY0FBYyxDQUFDLE1BQU0sQ0FBQyxDQUFDZ0MsTUFBTSxDQUFDLENBQUMsaUJBQWlCLENBQUMsQ0FBQ0MsaUJBQWlCLENBQUMsR0FBRztBQUU1RSIsImlnbm9yZUxpc3QiOltdfQ== \ No newline at end of file diff --git a/src/commands/fast/index.ts b/src/commands/fast/index.ts new file mode 100644 index 0000000..88ed550 --- /dev/null +++ b/src/commands/fast/index.ts @@ -0,0 +1,26 @@ +import type { Command } from '../../commands.js' +import { + FAST_MODE_MODEL_DISPLAY, + isFastModeEnabled, +} from '../../utils/fastMode.js' +import { shouldInferenceConfigCommandBeImmediate } from '../../utils/immediateCommand.js' + +const fast = { + type: 'local-jsx', + name: 'fast', + get description() { + return `Toggle fast mode (${FAST_MODE_MODEL_DISPLAY} only)` + }, + availability: ['claude-ai', 'console'], + isEnabled: () => isFastModeEnabled(), + get isHidden() { + return !isFastModeEnabled() + }, + argumentHint: '[on|off]', + get immediate() { + return shouldInferenceConfigCommandBeImmediate() + }, + load: () => import('./fast.js'), +} satisfies Command + +export default fast diff --git a/src/commands/feedback/feedback.tsx b/src/commands/feedback/feedback.tsx new file mode 100644 index 0000000..43828b0 --- /dev/null +++ b/src/commands/feedback/feedback.tsx @@ -0,0 +1,25 @@ +import * as React from 'react'; +import type { CommandResultDisplay, LocalJSXCommandContext } from '../../commands.js'; +import { Feedback } from '../../components/Feedback.js'; +import type { LocalJSXCommandOnDone } from '../../types/command.js'; +import type { Message } from '../../types/message.js'; + +// Shared function to render the Feedback component +export function renderFeedbackComponent(onDone: (result?: string, options?: { + display?: CommandResultDisplay; +}) => void, abortSignal: AbortSignal, messages: Message[], initialDescription: string = '', backgroundTasks: { + [taskId: string]: { + type: string; + identity?: { + agentId: string; + }; + messages?: Message[]; + }; +} = {}): React.ReactNode { + return ; +} +export async function call(onDone: LocalJSXCommandOnDone, context: LocalJSXCommandContext, args?: string): Promise { + const initialDescription = args || ''; + return renderFeedbackComponent(onDone, context.abortController.signal, context.messages, initialDescription); +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkNvbW1hbmRSZXN1bHREaXNwbGF5IiwiTG9jYWxKU1hDb21tYW5kQ29udGV4dCIsIkZlZWRiYWNrIiwiTG9jYWxKU1hDb21tYW5kT25Eb25lIiwiTWVzc2FnZSIsInJlbmRlckZlZWRiYWNrQ29tcG9uZW50Iiwib25Eb25lIiwicmVzdWx0Iiwib3B0aW9ucyIsImRpc3BsYXkiLCJhYm9ydFNpZ25hbCIsIkFib3J0U2lnbmFsIiwibWVzc2FnZXMiLCJpbml0aWFsRGVzY3JpcHRpb24iLCJiYWNrZ3JvdW5kVGFza3MiLCJ0YXNrSWQiLCJ0eXBlIiwiaWRlbnRpdHkiLCJhZ2VudElkIiwiUmVhY3ROb2RlIiwiY2FsbCIsImNvbnRleHQiLCJhcmdzIiwiUHJvbWlzZSIsImFib3J0Q29udHJvbGxlciIsInNpZ25hbCJdLCJzb3VyY2VzIjpbImZlZWRiYWNrLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB0eXBlIHtcbiAgQ29tbWFuZFJlc3VsdERpc3BsYXksXG4gIExvY2FsSlNYQ29tbWFuZENvbnRleHQsXG59IGZyb20gJy4uLy4uL2NvbW1hbmRzLmpzJ1xuaW1wb3J0IHsgRmVlZGJhY2sgfSBmcm9tICcuLi8uLi9jb21wb25lbnRzL0ZlZWRiYWNrLmpzJ1xuaW1wb3J0IHR5cGUgeyBMb2NhbEpTWENvbW1hbmRPbkRvbmUgfSBmcm9tICcuLi8uLi90eXBlcy9jb21tYW5kLmpzJ1xuaW1wb3J0IHR5cGUgeyBNZXNzYWdlIH0gZnJvbSAnLi4vLi4vdHlwZXMvbWVzc2FnZS5qcydcblxuLy8gU2hhcmVkIGZ1bmN0aW9uIHRvIHJlbmRlciB0aGUgRmVlZGJhY2sgY29tcG9uZW50XG5leHBvcnQgZnVuY3Rpb24gcmVuZGVyRmVlZGJhY2tDb21wb25lbnQoXG4gIG9uRG9uZTogKFxuICAgIHJlc3VsdD86IHN0cmluZyxcbiAgICBvcHRpb25zPzogeyBkaXNwbGF5PzogQ29tbWFuZFJlc3VsdERpc3BsYXkgfSxcbiAgKSA9PiB2b2lkLFxuICBhYm9ydFNpZ25hbDogQWJvcnRTaWduYWwsXG4gIG1lc3NhZ2VzOiBNZXNzYWdlW10sXG4gIGluaXRpYWxEZXNjcmlwdGlvbjogc3RyaW5nID0gJycsXG4gIGJhY2tncm91bmRUYXNrczoge1xuICAgIFt0YXNrSWQ6IHN0cmluZ106IHtcbiAgICAgIHR5cGU6IHN0cmluZ1xuICAgICAgaWRlbnRpdHk/OiB7IGFnZW50SWQ6IHN0cmluZyB9XG4gICAgICBtZXNzYWdlcz86IE1lc3NhZ2VbXVxuICAgIH1cbiAgfSA9IHt9LFxuKTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgcmV0dXJuIChcbiAgICA8RmVlZGJhY2tcbiAgICAgIGFib3J0U2lnbmFsPXthYm9ydFNpZ25hbH1cbiAgICAgIG1lc3NhZ2VzPXttZXNzYWdlc31cbiAgICAgIGluaXRpYWxEZXNjcmlwdGlvbj17aW5pdGlhbERlc2NyaXB0aW9ufVxuICAgICAgb25Eb25lPXtvbkRvbmV9XG4gICAgICBiYWNrZ3JvdW5kVGFza3M9e2JhY2tncm91bmRUYXNrc31cbiAgICAvPlxuICApXG59XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBjYWxsKFxuICBvbkRvbmU6IExvY2FsSlNYQ29tbWFuZE9uRG9uZSxcbiAgY29udGV4dDogTG9jYWxKU1hDb21tYW5kQ29udGV4dCxcbiAgYXJncz86IHN0cmluZyxcbik6IFByb21pc2U8UmVhY3QuUmVhY3ROb2RlPiB7XG4gIGNvbnN0IGluaXRpYWxEZXNjcmlwdGlvbiA9IGFyZ3MgfHwgJydcbiAgcmV0dXJuIHJlbmRlckZlZWRiYWNrQ29tcG9uZW50KFxuICAgIG9uRG9uZSxcbiAgICBjb250ZXh0LmFib3J0Q29udHJvbGxlci5zaWduYWwsXG4gICAgY29udGV4dC5tZXNzYWdlcyxcbiAgICBpbml0aWFsRGVzY3JpcHRpb24sXG4gIClcbn1cbiJdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLQSxLQUFLLE1BQU0sT0FBTztBQUM5QixjQUNFQyxvQkFBb0IsRUFDcEJDLHNCQUFzQixRQUNqQixtQkFBbUI7QUFDMUIsU0FBU0MsUUFBUSxRQUFRLDhCQUE4QjtBQUN2RCxjQUFjQyxxQkFBcUIsUUFBUSx3QkFBd0I7QUFDbkUsY0FBY0MsT0FBTyxRQUFRLHdCQUF3Qjs7QUFFckQ7QUFDQSxPQUFPLFNBQVNDLHVCQUF1QkEsQ0FDckNDLE1BQU0sRUFBRSxDQUNOQyxNQUFlLENBQVIsRUFBRSxNQUFNLEVBQ2ZDLE9BQTRDLENBQXBDLEVBQUU7RUFBRUMsT0FBTyxDQUFDLEVBQUVULG9CQUFvQjtBQUFDLENBQUMsRUFDNUMsR0FBRyxJQUFJLEVBQ1RVLFdBQVcsRUFBRUMsV0FBVyxFQUN4QkMsUUFBUSxFQUFFUixPQUFPLEVBQUUsRUFDbkJTLGtCQUFrQixFQUFFLE1BQU0sR0FBRyxFQUFFLEVBQy9CQyxlQUFlLEVBQUU7RUFDZixDQUFDQyxNQUFNLEVBQUUsTUFBTSxDQUFDLEVBQUU7SUFDaEJDLElBQUksRUFBRSxNQUFNO0lBQ1pDLFFBQVEsQ0FBQyxFQUFFO01BQUVDLE9BQU8sRUFBRSxNQUFNO0lBQUMsQ0FBQztJQUM5Qk4sUUFBUSxDQUFDLEVBQUVSLE9BQU8sRUFBRTtFQUN0QixDQUFDO0FBQ0gsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUNQLEVBQUVMLEtBQUssQ0FBQ29CLFNBQVMsQ0FBQztFQUNqQixPQUNFLENBQUMsUUFBUSxDQUNQLFdBQVcsQ0FBQyxDQUFDVCxXQUFXLENBQUMsQ0FDekIsUUFBUSxDQUFDLENBQUNFLFFBQVEsQ0FBQyxDQUNuQixrQkFBa0IsQ0FBQyxDQUFDQyxrQkFBa0IsQ0FBQyxDQUN2QyxNQUFNLENBQUMsQ0FBQ1AsTUFBTSxDQUFDLENBQ2YsZUFBZSxDQUFDLENBQUNRLGVBQWUsQ0FBQyxHQUNqQztBQUVOO0FBRUEsT0FBTyxlQUFlTSxJQUFJQSxDQUN4QmQsTUFBTSxFQUFFSCxxQkFBcUIsRUFDN0JrQixPQUFPLEVBQUVwQixzQkFBc0IsRUFDL0JxQixJQUFhLENBQVIsRUFBRSxNQUFNLENBQ2QsRUFBRUMsT0FBTyxDQUFDeEIsS0FBSyxDQUFDb0IsU0FBUyxDQUFDLENBQUM7RUFDMUIsTUFBTU4sa0JBQWtCLEdBQUdTLElBQUksSUFBSSxFQUFFO0VBQ3JDLE9BQU9qQix1QkFBdUIsQ0FDNUJDLE1BQU0sRUFDTmUsT0FBTyxDQUFDRyxlQUFlLENBQUNDLE1BQU0sRUFDOUJKLE9BQU8sQ0FBQ1QsUUFBUSxFQUNoQkMsa0JBQ0YsQ0FBQztBQUNIIiwiaWdub3JlTGlzdCI6W119 \ No newline at end of file diff --git a/src/commands/feedback/index.ts b/src/commands/feedback/index.ts new file mode 100644 index 0000000..ec092c8 --- /dev/null +++ b/src/commands/feedback/index.ts @@ -0,0 +1,26 @@ +import type { Command } from '../../commands.js' +import { isPolicyAllowed } from '../../services/policyLimits/index.js' +import { isEnvTruthy } from '../../utils/envUtils.js' +import { isEssentialTrafficOnly } from '../../utils/privacyLevel.js' + +const feedback = { + aliases: ['bug'], + type: 'local-jsx', + name: 'feedback', + description: `Submit feedback about Claude Code`, + argumentHint: '[report]', + isEnabled: () => + !( + isEnvTruthy(process.env.CLAUDE_CODE_USE_BEDROCK) || + isEnvTruthy(process.env.CLAUDE_CODE_USE_VERTEX) || + isEnvTruthy(process.env.CLAUDE_CODE_USE_FOUNDRY) || + isEnvTruthy(process.env.DISABLE_FEEDBACK_COMMAND) || + isEnvTruthy(process.env.DISABLE_BUG_COMMAND) || + isEssentialTrafficOnly() || + process.env.USER_TYPE === 'ant' || + !isPolicyAllowed('allow_product_feedback') + ), + load: () => import('./feedback.js'), +} satisfies Command + +export default feedback diff --git a/src/commands/files/files.ts b/src/commands/files/files.ts new file mode 100644 index 0000000..6da238b --- /dev/null +++ b/src/commands/files/files.ts @@ -0,0 +1,19 @@ +import { relative } from 'path' +import type { ToolUseContext } from '../../Tool.js' +import type { LocalCommandResult } from '../../types/command.js' +import { getCwd } from '../../utils/cwd.js' +import { cacheKeys } from '../../utils/fileStateCache.js' + +export async function call( + _args: string, + context: ToolUseContext, +): Promise { + const files = context.readFileState ? cacheKeys(context.readFileState) : [] + + if (files.length === 0) { + return { type: 'text' as const, value: 'No files in context' } + } + + const fileList = files.map(file => relative(getCwd(), file)).join('\n') + return { type: 'text' as const, value: `Files in context:\n${fileList}` } +} diff --git a/src/commands/files/index.ts b/src/commands/files/index.ts new file mode 100644 index 0000000..984b2d3 --- /dev/null +++ b/src/commands/files/index.ts @@ -0,0 +1,12 @@ +import type { Command } from '../../commands.js' + +const files = { + type: 'local', + name: 'files', + description: 'List all files currently in context', + isEnabled: () => process.env.USER_TYPE === 'ant', + supportsNonInteractive: true, + load: () => import('./files.js'), +} satisfies Command + +export default files diff --git a/src/commands/good-claude/index.js b/src/commands/good-claude/index.js new file mode 100644 index 0000000..e1a619d --- /dev/null +++ b/src/commands/good-claude/index.js @@ -0,0 +1 @@ +export default { isEnabled: () => false, isHidden: true, name: 'stub' }; diff --git a/src/commands/heapdump/heapdump.ts b/src/commands/heapdump/heapdump.ts new file mode 100644 index 0000000..75dd90e --- /dev/null +++ b/src/commands/heapdump/heapdump.ts @@ -0,0 +1,17 @@ +import { performHeapDump } from '../../utils/heapDumpService.js' + +export async function call(): Promise<{ type: 'text'; value: string }> { + const result = await performHeapDump() + + if (!result.success) { + return { + type: 'text', + value: `Failed to create heap dump: ${result.error}`, + } + } + + return { + type: 'text', + value: `${result.heapPath}\n${result.diagPath}`, + } +} diff --git a/src/commands/heapdump/index.ts b/src/commands/heapdump/index.ts new file mode 100644 index 0000000..11628ae --- /dev/null +++ b/src/commands/heapdump/index.ts @@ -0,0 +1,12 @@ +import type { Command } from '../../commands.js' + +const heapDump = { + type: 'local', + name: 'heapdump', + description: 'Dump the JS heap to ~/Desktop', + isHidden: true, + supportsNonInteractive: true, + load: () => import('./heapdump.js'), +} satisfies Command + +export default heapDump diff --git a/src/commands/help/help.tsx b/src/commands/help/help.tsx new file mode 100644 index 0000000..2d86e71 --- /dev/null +++ b/src/commands/help/help.tsx @@ -0,0 +1,11 @@ +import * as React from 'react'; +import { HelpV2 } from '../../components/HelpV2/HelpV2.js'; +import type { LocalJSXCommandCall } from '../../types/command.js'; +export const call: LocalJSXCommandCall = async (onDone, { + options: { + commands + } +}) => { + return ; +}; +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkhlbHBWMiIsIkxvY2FsSlNYQ29tbWFuZENhbGwiLCJjYWxsIiwib25Eb25lIiwib3B0aW9ucyIsImNvbW1hbmRzIl0sInNvdXJjZXMiOlsiaGVscC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBIZWxwVjIgfSBmcm9tICcuLi8uLi9jb21wb25lbnRzL0hlbHBWMi9IZWxwVjIuanMnXG5pbXBvcnQgdHlwZSB7IExvY2FsSlNYQ29tbWFuZENhbGwgfSBmcm9tICcuLi8uLi90eXBlcy9jb21tYW5kLmpzJ1xuXG5leHBvcnQgY29uc3QgY2FsbDogTG9jYWxKU1hDb21tYW5kQ2FsbCA9IGFzeW5jIChcbiAgb25Eb25lLFxuICB7IG9wdGlvbnM6IHsgY29tbWFuZHMgfSB9LFxuKSA9PiB7XG4gIHJldHVybiA8SGVscFYyIGNvbW1hbmRzPXtjb21tYW5kc30gb25DbG9zZT17b25Eb25lfSAvPlxufVxuIl0sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUtBLEtBQUssTUFBTSxPQUFPO0FBQzlCLFNBQVNDLE1BQU0sUUFBUSxtQ0FBbUM7QUFDMUQsY0FBY0MsbUJBQW1CLFFBQVEsd0JBQXdCO0FBRWpFLE9BQU8sTUFBTUMsSUFBSSxFQUFFRCxtQkFBbUIsR0FBRyxNQUFBQyxDQUN2Q0MsTUFBTSxFQUNOO0VBQUVDLE9BQU8sRUFBRTtJQUFFQztFQUFTO0FBQUUsQ0FBQyxLQUN0QjtFQUNILE9BQU8sQ0FBQyxNQUFNLENBQUMsUUFBUSxDQUFDLENBQUNBLFFBQVEsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxDQUFDRixNQUFNLENBQUMsR0FBRztBQUN4RCxDQUFDIiwiaWdub3JlTGlzdCI6W119 \ No newline at end of file diff --git a/src/commands/help/index.ts b/src/commands/help/index.ts new file mode 100644 index 0000000..31f465d --- /dev/null +++ b/src/commands/help/index.ts @@ -0,0 +1,10 @@ +import type { Command } from '../../commands.js' + +const help = { + type: 'local-jsx', + name: 'help', + description: 'Show help and available commands', + load: () => import('./help.js'), +} satisfies Command + +export default help diff --git a/src/commands/hooks/hooks.tsx b/src/commands/hooks/hooks.tsx new file mode 100644 index 0000000..c399454 --- /dev/null +++ b/src/commands/hooks/hooks.tsx @@ -0,0 +1,13 @@ +import * as React from 'react'; +import { HooksConfigMenu } from '../../components/hooks/HooksConfigMenu.js'; +import { logEvent } from '../../services/analytics/index.js'; +import { getTools } from '../../tools.js'; +import type { LocalJSXCommandCall } from '../../types/command.js'; +export const call: LocalJSXCommandCall = async (onDone, context) => { + logEvent('tengu_hooks_command', {}); + const appState = context.getAppState(); + const permissionContext = appState.toolPermissionContext; + const toolNames = getTools(permissionContext).map(tool => tool.name); + return ; +}; +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkhvb2tzQ29uZmlnTWVudSIsImxvZ0V2ZW50IiwiZ2V0VG9vbHMiLCJMb2NhbEpTWENvbW1hbmRDYWxsIiwiY2FsbCIsIm9uRG9uZSIsImNvbnRleHQiLCJhcHBTdGF0ZSIsImdldEFwcFN0YXRlIiwicGVybWlzc2lvbkNvbnRleHQiLCJ0b29sUGVybWlzc2lvbkNvbnRleHQiLCJ0b29sTmFtZXMiLCJtYXAiLCJ0b29sIiwibmFtZSJdLCJzb3VyY2VzIjpbImhvb2tzLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB7IEhvb2tzQ29uZmlnTWVudSB9IGZyb20gJy4uLy4uL2NvbXBvbmVudHMvaG9va3MvSG9va3NDb25maWdNZW51LmpzJ1xuaW1wb3J0IHsgbG9nRXZlbnQgfSBmcm9tICcuLi8uLi9zZXJ2aWNlcy9hbmFseXRpY3MvaW5kZXguanMnXG5pbXBvcnQgeyBnZXRUb29scyB9IGZyb20gJy4uLy4uL3Rvb2xzLmpzJ1xuaW1wb3J0IHR5cGUgeyBMb2NhbEpTWENvbW1hbmRDYWxsIH0gZnJvbSAnLi4vLi4vdHlwZXMvY29tbWFuZC5qcydcblxuZXhwb3J0IGNvbnN0IGNhbGw6IExvY2FsSlNYQ29tbWFuZENhbGwgPSBhc3luYyAob25Eb25lLCBjb250ZXh0KSA9PiB7XG4gIGxvZ0V2ZW50KCd0ZW5ndV9ob29rc19jb21tYW5kJywge30pXG4gIGNvbnN0IGFwcFN0YXRlID0gY29udGV4dC5nZXRBcHBTdGF0ZSgpXG4gIGNvbnN0IHBlcm1pc3Npb25Db250ZXh0ID0gYXBwU3RhdGUudG9vbFBlcm1pc3Npb25Db250ZXh0XG4gIGNvbnN0IHRvb2xOYW1lcyA9IGdldFRvb2xzKHBlcm1pc3Npb25Db250ZXh0KS5tYXAodG9vbCA9PiB0b29sLm5hbWUpXG4gIHJldHVybiA8SG9va3NDb25maWdNZW51IHRvb2xOYW1lcz17dG9vbE5hbWVzfSBvbkV4aXQ9e29uRG9uZX0gLz5cbn1cbiJdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLQSxLQUFLLE1BQU0sT0FBTztBQUM5QixTQUFTQyxlQUFlLFFBQVEsMkNBQTJDO0FBQzNFLFNBQVNDLFFBQVEsUUFBUSxtQ0FBbUM7QUFDNUQsU0FBU0MsUUFBUSxRQUFRLGdCQUFnQjtBQUN6QyxjQUFjQyxtQkFBbUIsUUFBUSx3QkFBd0I7QUFFakUsT0FBTyxNQUFNQyxJQUFJLEVBQUVELG1CQUFtQixHQUFHLE1BQUFDLENBQU9DLE1BQU0sRUFBRUMsT0FBTyxLQUFLO0VBQ2xFTCxRQUFRLENBQUMscUJBQXFCLEVBQUUsQ0FBQyxDQUFDLENBQUM7RUFDbkMsTUFBTU0sUUFBUSxHQUFHRCxPQUFPLENBQUNFLFdBQVcsQ0FBQyxDQUFDO0VBQ3RDLE1BQU1DLGlCQUFpQixHQUFHRixRQUFRLENBQUNHLHFCQUFxQjtFQUN4RCxNQUFNQyxTQUFTLEdBQUdULFFBQVEsQ0FBQ08saUJBQWlCLENBQUMsQ0FBQ0csR0FBRyxDQUFDQyxJQUFJLElBQUlBLElBQUksQ0FBQ0MsSUFBSSxDQUFDO0VBQ3BFLE9BQU8sQ0FBQyxlQUFlLENBQUMsU0FBUyxDQUFDLENBQUNILFNBQVMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDTixNQUFNLENBQUMsR0FBRztBQUNsRSxDQUFDIiwiaWdub3JlTGlzdCI6W119 \ No newline at end of file diff --git a/src/commands/hooks/index.ts b/src/commands/hooks/index.ts new file mode 100644 index 0000000..4567dbf --- /dev/null +++ b/src/commands/hooks/index.ts @@ -0,0 +1,11 @@ +import type { Command } from '../../commands.js' + +const hooks = { + type: 'local-jsx', + name: 'hooks', + description: 'View hook configurations for tool events', + immediate: true, + load: () => import('./hooks.js'), +} satisfies Command + +export default hooks diff --git a/src/commands/ide/ide.tsx b/src/commands/ide/ide.tsx new file mode 100644 index 0000000..0a41b97 --- /dev/null +++ b/src/commands/ide/ide.tsx @@ -0,0 +1,646 @@ +import { c as _c } from "react/compiler-runtime"; +import chalk from 'chalk'; +import * as path from 'path'; +import React, { useCallback, useEffect, useRef, useState } from 'react'; +import { logEvent } from 'src/services/analytics/index.js'; +import type { CommandResultDisplay, LocalJSXCommandContext } from '../../commands.js'; +import { Select } from '../../components/CustomSelect/index.js'; +import { Dialog } from '../../components/design-system/Dialog.js'; +import { IdeAutoConnectDialog, IdeDisableAutoConnectDialog, shouldShowAutoConnectDialog, shouldShowDisableAutoConnectDialog } from '../../components/IdeAutoConnectDialog.js'; +import { Box, Text } from '../../ink.js'; +import { clearServerCache } from '../../services/mcp/client.js'; +import type { ScopedMcpServerConfig } from '../../services/mcp/types.js'; +import { useAppState, useSetAppState } from '../../state/AppState.js'; +import { getCwd } from '../../utils/cwd.js'; +import { execFileNoThrow } from '../../utils/execFileNoThrow.js'; +import { type DetectedIDEInfo, detectIDEs, detectRunningIDEs, type IdeType, isJetBrainsIde, isSupportedJetBrainsTerminal, isSupportedTerminal, toIDEDisplayName } from '../../utils/ide.js'; +import { getCurrentWorktreeSession } from '../../utils/worktree.js'; +type IDEScreenProps = { + availableIDEs: DetectedIDEInfo[]; + unavailableIDEs: DetectedIDEInfo[]; + selectedIDE?: DetectedIDEInfo | null; + onClose: () => void; + onSelect: (ide?: DetectedIDEInfo) => void; +}; +function IDEScreen(t0) { + const $ = _c(39); + const { + availableIDEs, + unavailableIDEs, + selectedIDE, + onClose, + onSelect + } = t0; + let t1; + if ($[0] !== selectedIDE?.port) { + t1 = selectedIDE?.port?.toString() ?? "None"; + $[0] = selectedIDE?.port; + $[1] = t1; + } else { + t1 = $[1]; + } + const [selectedValue, setSelectedValue] = useState(t1); + const [showAutoConnectDialog, setShowAutoConnectDialog] = useState(false); + const [showDisableAutoConnectDialog, setShowDisableAutoConnectDialog] = useState(false); + let t2; + if ($[2] !== availableIDEs || $[3] !== onSelect) { + t2 = value => { + if (value !== "None" && shouldShowAutoConnectDialog()) { + setShowAutoConnectDialog(true); + } else { + if (value === "None" && shouldShowDisableAutoConnectDialog()) { + setShowDisableAutoConnectDialog(true); + } else { + onSelect(availableIDEs.find(ide => ide.port === parseInt(value))); + } + } + }; + $[2] = availableIDEs; + $[3] = onSelect; + $[4] = t2; + } else { + t2 = $[4]; + } + const handleSelectIDE = t2; + let t3; + if ($[5] !== availableIDEs) { + t3 = availableIDEs.reduce(_temp, {}); + $[5] = availableIDEs; + $[6] = t3; + } else { + t3 = $[6]; + } + const ideCounts = t3; + let t4; + if ($[7] !== availableIDEs || $[8] !== ideCounts) { + let t5; + if ($[10] !== ideCounts) { + t5 = ide_1 => { + const hasMultipleInstances = (ideCounts[ide_1.name] || 0) > 1; + const showWorkspace = hasMultipleInstances && ide_1.workspaceFolders.length > 0; + return { + label: ide_1.name, + value: ide_1.port.toString(), + description: showWorkspace ? formatWorkspaceFolders(ide_1.workspaceFolders) : undefined + }; + }; + $[10] = ideCounts; + $[11] = t5; + } else { + t5 = $[11]; + } + t4 = availableIDEs.map(t5).concat([{ + label: "None", + value: "None", + description: undefined + }]); + $[7] = availableIDEs; + $[8] = ideCounts; + $[9] = t4; + } else { + t4 = $[9]; + } + const options = t4; + if (showAutoConnectDialog) { + let t5; + if ($[12] !== handleSelectIDE || $[13] !== selectedValue) { + t5 = handleSelectIDE(selectedValue)} />; + $[12] = handleSelectIDE; + $[13] = selectedValue; + $[14] = t5; + } else { + t5 = $[14]; + } + return t5; + } + if (showDisableAutoConnectDialog) { + let t5; + if ($[15] !== onSelect) { + t5 = { + onSelect(undefined); + }} />; + $[15] = onSelect; + $[16] = t5; + } else { + t5 = $[16]; + } + return t5; + } + let t5; + if ($[17] !== availableIDEs.length) { + t5 = availableIDEs.length === 0 && {isSupportedJetBrainsTerminal() ? "No available IDEs detected. Please install the plugin and restart your IDE:\nhttps://docs.claude.com/s/claude-code-jetbrains" : "No available IDEs detected. Make sure your IDE has the Claude Code extension or plugin installed and is running."}; + $[17] = availableIDEs.length; + $[18] = t5; + } else { + t5 = $[18]; + } + let t6; + if ($[19] !== availableIDEs.length || $[20] !== handleSelectIDE || $[21] !== options || $[22] !== selectedValue) { + t6 = availableIDEs.length !== 0 && ; + $[11] = options; + $[12] = selectedValue; + $[13] = t5; + $[14] = t6; + } else { + t6 = $[14]; + } + let t7; + if ($[15] !== handleCancel || $[16] !== t6) { + t7 = {t6}; + $[15] = handleCancel; + $[16] = t6; + $[17] = t7; + } else { + t7 = $[17]; + } + return t7; +} +function _temp4(ide_0) { + return { + label: ide_0.name, + value: ide_0.port.toString() + }; +} +function RunningIDESelector(t0) { + const $ = _c(15); + const { + runningIDEs, + onSelectIDE, + onDone + } = t0; + const [selectedValue, setSelectedValue] = useState(runningIDEs[0] ?? ""); + let t1; + if ($[0] !== onSelectIDE) { + t1 = value => { + onSelectIDE(value as IdeType); + }; + $[0] = onSelectIDE; + $[1] = t1; + } else { + t1 = $[1]; + } + const handleSelectIDE = t1; + let t2; + if ($[2] !== runningIDEs) { + t2 = runningIDEs.map(_temp5); + $[2] = runningIDEs; + $[3] = t2; + } else { + t2 = $[3]; + } + const options = t2; + let t3; + if ($[4] !== onDone) { + t3 = function handleCancel() { + onDone("IDE selection cancelled", { + display: "system" + }); + }; + $[4] = onDone; + $[5] = t3; + } else { + t3 = $[5]; + } + const handleCancel = t3; + let t4; + if ($[6] !== handleSelectIDE) { + t4 = value_0 => { + setSelectedValue(value_0); + handleSelectIDE(value_0); + }; + $[6] = handleSelectIDE; + $[7] = t4; + } else { + t4 = $[7]; + } + let t5; + if ($[8] !== options || $[9] !== selectedValue || $[10] !== t4) { + t5 = + +
${escapeHtml(add.why)}
+ + `, + ) + .join('')} + + ` + : '' + } + ${ + suggestions.features_to_try && suggestions.features_to_try.length > 0 + ? ` +

Just copy this into Claude Code and it'll set it up for you.

+
+ ${suggestions.features_to_try + .map( + feat => ` +
+
${escapeHtml(feat.feature || '')}
+
${escapeHtml(feat.one_liner || '')}
+
Why for you: ${escapeHtml(feat.why_for_you || '')}
+ ${ + feat.example_code + ? ` +
+
+
+ ${escapeHtml(feat.example_code)} + +
+
+
+ ` + : '' + } +
+ `, + ) + .join('')} +
+ ` + : '' + } + ${ + suggestions.usage_patterns && suggestions.usage_patterns.length > 0 + ? ` +

New Ways to Use Claude Code

+

Just copy this into Claude Code and it'll walk you through it.

+
+ ${suggestions.usage_patterns + .map( + pat => ` +
+
${escapeHtml(pat.title || '')}
+
${escapeHtml(pat.suggestion || '')}
+ ${pat.detail ? `
${escapeHtml(pat.detail)}
` : ''} + ${ + pat.copyable_prompt + ? ` +
+
Paste into Claude Code:
+
+ ${escapeHtml(pat.copyable_prompt)} + +
+
+ ` + : '' + } +
+ `, + ) + .join('')} +
+ ` + : '' + } + ` + : '' + + // Build On the Horizon section + const horizonData = insights.on_the_horizon + const horizonHtml = + horizonData?.opportunities && horizonData.opportunities.length > 0 + ? ` +

On the Horizon

+ ${horizonData.intro ? `

${escapeHtml(horizonData.intro)}

` : ''} +
+ ${horizonData.opportunities + .map( + opp => ` +
+
${escapeHtml(opp.title || '')}
+
${escapeHtml(opp.whats_possible || '')}
+ ${opp.how_to_try ? `
Getting started: ${escapeHtml(opp.how_to_try)}
` : ''} + ${opp.copyable_prompt ? `
Paste into Claude Code:
${escapeHtml(opp.copyable_prompt)}
` : ''} +
+ `, + ) + .join('')} +
+ ` + : '' + + // Build Team Feedback section (collapsible, ant-only) + const ccImprovements = + process.env.USER_TYPE === 'ant' + ? insights.cc_team_improvements?.improvements || [] + : [] + const modelImprovements = + process.env.USER_TYPE === 'ant' + ? insights.model_behavior_improvements?.improvements || [] + : [] + const teamFeedbackHtml = + ccImprovements.length > 0 || modelImprovements.length > 0 + ? ` + + + ${ + ccImprovements.length > 0 + ? ` +
+
+ +

Product Improvements for CC Team

+
+
+
+ ${ccImprovements + .map( + imp => ` + + `, + ) + .join('')} +
+
+
+ ` + : '' + } + ${ + modelImprovements.length > 0 + ? ` +
+
+ +

Model Behavior Improvements

+
+
+
+ ${modelImprovements + .map( + imp => ` + + `, + ) + .join('')} +
+
+
+ ` + : '' + } + ` + : '' + + // Build Fun Ending section + const funEnding = insights.fun_ending + const funEndingHtml = funEnding?.headline + ? ` +
+
"${escapeHtml(funEnding.headline)}"
+ ${funEnding.detail ? `
${escapeHtml(funEnding.detail)}
` : ''} +
+ ` + : '' + + const css = ` + * { box-sizing: border-box; margin: 0; padding: 0; } + body { font-family: 'Inter', -apple-system, BlinkMacSystemFont, sans-serif; background: #f8fafc; color: #334155; line-height: 1.65; padding: 48px 24px; } + .container { max-width: 800px; margin: 0 auto; } + h1 { font-size: 32px; font-weight: 700; color: #0f172a; margin-bottom: 8px; } + h2 { font-size: 20px; font-weight: 600; color: #0f172a; margin-top: 48px; margin-bottom: 16px; } + .subtitle { color: #64748b; font-size: 15px; margin-bottom: 32px; } + .nav-toc { display: flex; flex-wrap: wrap; gap: 8px; margin: 24px 0 32px 0; padding: 16px; background: white; border-radius: 8px; border: 1px solid #e2e8f0; } + .nav-toc a { font-size: 12px; color: #64748b; text-decoration: none; padding: 6px 12px; border-radius: 6px; background: #f1f5f9; transition: all 0.15s; } + .nav-toc a:hover { background: #e2e8f0; color: #334155; } + .stats-row { display: flex; gap: 24px; margin-bottom: 40px; padding: 20px 0; border-top: 1px solid #e2e8f0; border-bottom: 1px solid #e2e8f0; flex-wrap: wrap; } + .stat { text-align: center; } + .stat-value { font-size: 24px; font-weight: 700; color: #0f172a; } + .stat-label { font-size: 11px; color: #64748b; text-transform: uppercase; } + .at-a-glance { background: linear-gradient(135deg, #fef3c7 0%, #fde68a 100%); border: 1px solid #f59e0b; border-radius: 12px; padding: 20px 24px; margin-bottom: 32px; } + .glance-title { font-size: 16px; font-weight: 700; color: #92400e; margin-bottom: 16px; } + .glance-sections { display: flex; flex-direction: column; gap: 12px; } + .glance-section { font-size: 14px; color: #78350f; line-height: 1.6; } + .glance-section strong { color: #92400e; } + .see-more { color: #b45309; text-decoration: none; font-size: 13px; white-space: nowrap; } + .see-more:hover { text-decoration: underline; } + .project-areas { display: flex; flex-direction: column; gap: 12px; margin-bottom: 32px; } + .project-area { background: white; border: 1px solid #e2e8f0; border-radius: 8px; padding: 16px; } + .area-header { display: flex; justify-content: space-between; align-items: center; margin-bottom: 8px; } + .area-name { font-weight: 600; font-size: 15px; color: #0f172a; } + .area-count { font-size: 12px; color: #64748b; background: #f1f5f9; padding: 2px 8px; border-radius: 4px; } + .area-desc { font-size: 14px; color: #475569; line-height: 1.5; } + .narrative { background: white; border: 1px solid #e2e8f0; border-radius: 8px; padding: 20px; margin-bottom: 24px; } + .narrative p { margin-bottom: 12px; font-size: 14px; color: #475569; line-height: 1.7; } + .key-insight { background: #f0fdf4; border: 1px solid #bbf7d0; border-radius: 8px; padding: 12px 16px; margin-top: 12px; font-size: 14px; color: #166534; } + .section-intro { font-size: 14px; color: #64748b; margin-bottom: 16px; } + .big-wins { display: flex; flex-direction: column; gap: 12px; margin-bottom: 24px; } + .big-win { background: #f0fdf4; border: 1px solid #bbf7d0; border-radius: 8px; padding: 16px; } + .big-win-title { font-weight: 600; font-size: 15px; color: #166534; margin-bottom: 8px; } + .big-win-desc { font-size: 14px; color: #15803d; line-height: 1.5; } + .friction-categories { display: flex; flex-direction: column; gap: 16px; margin-bottom: 24px; } + .friction-category { background: #fef2f2; border: 1px solid #fca5a5; border-radius: 8px; padding: 16px; } + .friction-title { font-weight: 600; font-size: 15px; color: #991b1b; margin-bottom: 6px; } + .friction-desc { font-size: 13px; color: #7f1d1d; margin-bottom: 10px; } + .friction-examples { margin: 0 0 0 20px; font-size: 13px; color: #334155; } + .friction-examples li { margin-bottom: 4px; } + .claude-md-section { background: #eff6ff; border: 1px solid #bfdbfe; border-radius: 8px; padding: 16px; margin-bottom: 20px; } + .claude-md-section h3 { font-size: 14px; font-weight: 600; color: #1e40af; margin: 0 0 12px 0; } + .claude-md-actions { margin-bottom: 12px; padding-bottom: 12px; border-bottom: 1px solid #dbeafe; } + .copy-all-btn { background: #2563eb; color: white; border: none; border-radius: 4px; padding: 6px 12px; font-size: 12px; cursor: pointer; font-weight: 500; transition: all 0.2s; } + .copy-all-btn:hover { background: #1d4ed8; } + .copy-all-btn.copied { background: #16a34a; } + .claude-md-item { display: flex; flex-wrap: wrap; align-items: flex-start; gap: 8px; padding: 10px 0; border-bottom: 1px solid #dbeafe; } + .claude-md-item:last-child { border-bottom: none; } + .cmd-checkbox { margin-top: 2px; } + .cmd-code { background: white; padding: 8px 12px; border-radius: 4px; font-size: 12px; color: #1e40af; border: 1px solid #bfdbfe; font-family: monospace; display: block; white-space: pre-wrap; word-break: break-word; flex: 1; } + .cmd-why { font-size: 12px; color: #64748b; width: 100%; padding-left: 24px; margin-top: 4px; } + .features-section, .patterns-section { display: flex; flex-direction: column; gap: 12px; margin: 16px 0; } + .feature-card { background: #f0fdf4; border: 1px solid #86efac; border-radius: 8px; padding: 16px; } + .pattern-card { background: #f0f9ff; border: 1px solid #7dd3fc; border-radius: 8px; padding: 16px; } + .feature-title, .pattern-title { font-weight: 600; font-size: 15px; color: #0f172a; margin-bottom: 6px; } + .feature-oneliner { font-size: 14px; color: #475569; margin-bottom: 8px; } + .pattern-summary { font-size: 14px; color: #475569; margin-bottom: 8px; } + .feature-why, .pattern-detail { font-size: 13px; color: #334155; line-height: 1.5; } + .feature-examples { margin-top: 12px; } + .feature-example { padding: 8px 0; border-top: 1px solid #d1fae5; } + .feature-example:first-child { border-top: none; } + .example-desc { font-size: 13px; color: #334155; margin-bottom: 6px; } + .example-code-row { display: flex; align-items: flex-start; gap: 8px; } + .example-code { flex: 1; background: #f1f5f9; padding: 8px 12px; border-radius: 4px; font-family: monospace; font-size: 12px; color: #334155; overflow-x: auto; white-space: pre-wrap; } + .copyable-prompt-section { margin-top: 12px; padding-top: 12px; border-top: 1px solid #e2e8f0; } + .copyable-prompt-row { display: flex; align-items: flex-start; gap: 8px; } + .copyable-prompt { flex: 1; background: #f8fafc; padding: 10px 12px; border-radius: 4px; font-family: monospace; font-size: 12px; color: #334155; border: 1px solid #e2e8f0; white-space: pre-wrap; line-height: 1.5; } + .feature-code { background: #f8fafc; padding: 12px; border-radius: 6px; margin-top: 12px; border: 1px solid #e2e8f0; display: flex; align-items: flex-start; gap: 8px; } + .feature-code code { flex: 1; font-family: monospace; font-size: 12px; color: #334155; white-space: pre-wrap; } + .pattern-prompt { background: #f8fafc; padding: 12px; border-radius: 6px; margin-top: 12px; border: 1px solid #e2e8f0; } + .pattern-prompt code { font-family: monospace; font-size: 12px; color: #334155; display: block; white-space: pre-wrap; margin-bottom: 8px; } + .prompt-label { font-size: 11px; font-weight: 600; text-transform: uppercase; color: #64748b; margin-bottom: 6px; } + .copy-btn { background: #e2e8f0; border: none; border-radius: 4px; padding: 4px 8px; font-size: 11px; cursor: pointer; color: #475569; flex-shrink: 0; } + .copy-btn:hover { background: #cbd5e1; } + .charts-row { display: grid; grid-template-columns: 1fr 1fr; gap: 24px; margin: 24px 0; } + .chart-card { background: white; border: 1px solid #e2e8f0; border-radius: 8px; padding: 16px; } + .chart-title { font-size: 12px; font-weight: 600; color: #64748b; text-transform: uppercase; margin-bottom: 12px; } + .bar-row { display: flex; align-items: center; margin-bottom: 6px; } + .bar-label { width: 100px; font-size: 11px; color: #475569; flex-shrink: 0; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; } + .bar-track { flex: 1; height: 6px; background: #f1f5f9; border-radius: 3px; margin: 0 8px; } + .bar-fill { height: 100%; border-radius: 3px; } + .bar-value { width: 28px; font-size: 11px; font-weight: 500; color: #64748b; text-align: right; } + .empty { color: #94a3b8; font-size: 13px; } + .horizon-section { display: flex; flex-direction: column; gap: 16px; } + .horizon-card { background: linear-gradient(135deg, #faf5ff 0%, #f5f3ff 100%); border: 1px solid #c4b5fd; border-radius: 8px; padding: 16px; } + .horizon-title { font-weight: 600; font-size: 15px; color: #5b21b6; margin-bottom: 8px; } + .horizon-possible { font-size: 14px; color: #334155; margin-bottom: 10px; line-height: 1.5; } + .horizon-tip { font-size: 13px; color: #6b21a8; background: rgba(255,255,255,0.6); padding: 8px 12px; border-radius: 4px; } + .feedback-header { margin-top: 48px; color: #64748b; font-size: 16px; } + .feedback-intro { font-size: 13px; color: #94a3b8; margin-bottom: 16px; } + .feedback-section { margin-top: 16px; } + .feedback-section h3 { font-size: 14px; font-weight: 600; color: #475569; margin-bottom: 12px; } + .feedback-card { background: white; border: 1px solid #e2e8f0; border-radius: 8px; padding: 16px; margin-bottom: 12px; } + .feedback-card.team-card { background: #eff6ff; border-color: #bfdbfe; } + .feedback-card.model-card { background: #faf5ff; border-color: #e9d5ff; } + .feedback-title { font-weight: 600; font-size: 14px; color: #0f172a; margin-bottom: 6px; } + .feedback-detail { font-size: 13px; color: #475569; line-height: 1.5; } + .feedback-evidence { font-size: 12px; color: #64748b; margin-top: 8px; } + .fun-ending { background: linear-gradient(135deg, #fef3c7 0%, #fde68a 100%); border: 1px solid #fbbf24; border-radius: 12px; padding: 24px; margin-top: 40px; text-align: center; } + .fun-headline { font-size: 18px; font-weight: 600; color: #78350f; margin-bottom: 8px; } + .fun-detail { font-size: 14px; color: #92400e; } + .collapsible-section { margin-top: 16px; } + .collapsible-header { display: flex; align-items: center; gap: 8px; cursor: pointer; padding: 12px 0; border-bottom: 1px solid #e2e8f0; } + .collapsible-header h3 { margin: 0; font-size: 14px; font-weight: 600; color: #475569; } + .collapsible-arrow { font-size: 12px; color: #94a3b8; transition: transform 0.2s; } + .collapsible-content { display: none; padding-top: 16px; } + .collapsible-content.open { display: block; } + .collapsible-header.open .collapsible-arrow { transform: rotate(90deg); } + @media (max-width: 640px) { .charts-row { grid-template-columns: 1fr; } .stats-row { justify-content: center; } } + ` + + const hourCountsJson = getHourCountsJson(data.message_hours) + + const js = ` + function toggleCollapsible(header) { + header.classList.toggle('open'); + const content = header.nextElementSibling; + content.classList.toggle('open'); + } + function copyText(btn) { + const code = btn.previousElementSibling; + navigator.clipboard.writeText(code.textContent).then(() => { + btn.textContent = 'Copied!'; + setTimeout(() => { btn.textContent = 'Copy'; }, 2000); + }); + } + function copyCmdItem(idx) { + const checkbox = document.getElementById('cmd-' + idx); + if (checkbox) { + const text = checkbox.dataset.text; + navigator.clipboard.writeText(text).then(() => { + const btn = checkbox.nextElementSibling.querySelector('.copy-btn'); + if (btn) { btn.textContent = 'Copied!'; setTimeout(() => { btn.textContent = 'Copy'; }, 2000); } + }); + } + } + function copyAllCheckedClaudeMd() { + const checkboxes = document.querySelectorAll('.cmd-checkbox:checked'); + const texts = []; + checkboxes.forEach(cb => { + if (cb.dataset.text) { texts.push(cb.dataset.text); } + }); + const combined = texts.join('\\n'); + const btn = document.querySelector('.copy-all-btn'); + if (btn) { + navigator.clipboard.writeText(combined).then(() => { + btn.textContent = 'Copied ' + texts.length + ' items!'; + btn.classList.add('copied'); + setTimeout(() => { btn.textContent = 'Copy All Checked'; btn.classList.remove('copied'); }, 2000); + }); + } + } + // Timezone selector for time of day chart (data is from our own analytics, not user input) + const rawHourCounts = ${hourCountsJson}; + function updateHourHistogram(offsetFromPT) { + const periods = [ + { label: "Morning (6-12)", range: [6,7,8,9,10,11] }, + { label: "Afternoon (12-18)", range: [12,13,14,15,16,17] }, + { label: "Evening (18-24)", range: [18,19,20,21,22,23] }, + { label: "Night (0-6)", range: [0,1,2,3,4,5] } + ]; + const adjustedCounts = {}; + for (const [hour, count] of Object.entries(rawHourCounts)) { + const newHour = (parseInt(hour) + offsetFromPT + 24) % 24; + adjustedCounts[newHour] = (adjustedCounts[newHour] || 0) + count; + } + const periodCounts = periods.map(p => ({ + label: p.label, + count: p.range.reduce((sum, h) => sum + (adjustedCounts[h] || 0), 0) + })); + const maxCount = Math.max(...periodCounts.map(p => p.count)) || 1; + const container = document.getElementById('hour-histogram'); + container.textContent = ''; + periodCounts.forEach(p => { + const row = document.createElement('div'); + row.className = 'bar-row'; + const label = document.createElement('div'); + label.className = 'bar-label'; + label.textContent = p.label; + const track = document.createElement('div'); + track.className = 'bar-track'; + const fill = document.createElement('div'); + fill.className = 'bar-fill'; + fill.style.width = (p.count / maxCount) * 100 + '%'; + fill.style.background = '#8b5cf6'; + track.appendChild(fill); + const value = document.createElement('div'); + value.className = 'bar-value'; + value.textContent = p.count; + row.appendChild(label); + row.appendChild(track); + row.appendChild(value); + container.appendChild(row); + }); + } + document.getElementById('timezone-select').addEventListener('change', function() { + const customInput = document.getElementById('custom-offset'); + if (this.value === 'custom') { + customInput.style.display = 'inline-block'; + customInput.focus(); + } else { + customInput.style.display = 'none'; + updateHourHistogram(parseInt(this.value)); + } + }); + document.getElementById('custom-offset').addEventListener('change', function() { + const offset = parseInt(this.value) + 8; + updateHourHistogram(offset); + }); + ` + + return ` + + + + Claude Code Insights + + + + +
+

Claude Code Insights

+

${data.total_messages.toLocaleString()} messages across ${data.total_sessions} sessions${data.total_sessions_scanned && data.total_sessions_scanned > data.total_sessions ? ` (${data.total_sessions_scanned.toLocaleString()} total)` : ''} | ${data.date_range.start} to ${data.date_range.end}

+ + ${atAGlanceHtml} + + + +
+
${data.total_messages.toLocaleString()}
Messages
+
+${data.total_lines_added.toLocaleString()}/-${data.total_lines_removed.toLocaleString()}
Lines
+
${data.total_files_modified}
Files
+
${data.days_active}
Days
+
${data.messages_per_day}
Msgs/Day
+
+ + ${projectAreasHtml} + +
+
+
What You Wanted
+ ${generateBarChart(data.goal_categories, '#2563eb')} +
+
+
Top Tools Used
+ ${generateBarChart(data.tool_counts, '#0891b2')} +
+
+ +
+
+
Languages
+ ${generateBarChart(data.languages, '#10b981')} +
+
+
Session Types
+ ${generateBarChart(data.session_types || {}, '#8b5cf6')} +
+
+ + ${interactionHtml} + + +
+
User Response Time Distribution
+ ${generateResponseTimeHistogram(data.user_response_times)} +
+ Median: ${data.median_response_time.toFixed(1)}s • Average: ${data.avg_response_time.toFixed(1)}s +
+
+ + +
+
Multi-Clauding (Parallel Sessions)
+ ${ + data.multi_clauding.overlap_events === 0 + ? ` +

+ No parallel session usage detected. You typically work with one Claude Code session at a time. +

+ ` + : ` +
+
+
${data.multi_clauding.overlap_events}
+
Overlap Events
+
+
+
${data.multi_clauding.sessions_involved}
+
Sessions Involved
+
+
+
${data.total_messages > 0 ? Math.round((100 * data.multi_clauding.user_messages_during) / data.total_messages) : 0}%
+
Of Messages
+
+
+

+ You run multiple Claude Code sessions simultaneously. Multi-clauding is detected when sessions + overlap in time, suggesting parallel workflows. +

+ ` + } +
+ + +
+
+
+ User Messages by Time of Day + + +
+ ${generateTimeOfDayChart(data.message_hours)} +
+
+
Tool Errors Encountered
+ ${Object.keys(data.tool_error_categories).length > 0 ? generateBarChart(data.tool_error_categories, '#dc2626') : '

No tool errors

'} +
+
+ + ${whatWorksHtml} + +
+
+
What Helped Most (Claude's Capabilities)
+ ${generateBarChart(data.success, '#16a34a')} +
+
+
Outcomes
+ ${generateBarChart(data.outcomes, '#8b5cf6', 6, OUTCOME_ORDER)} +
+
+ + ${frictionHtml} + +
+
+
Primary Friction Types
+ ${generateBarChart(data.friction, '#dc2626')} +
+
+
Inferred Satisfaction (model-estimated)
+ ${generateBarChart(data.satisfaction, '#eab308', 6, SATISFACTION_ORDER)} +
+
+ + ${suggestionsHtml} + + ${horizonHtml} + + ${funEndingHtml} + + ${teamFeedbackHtml} +
+ + +` +} + +// ============================================================================ +// Export Types & Functions +// ============================================================================ + +/** + * Structured export format for claudescope consumption + */ +export type InsightsExport = { + metadata: { + username: string + generated_at: string + claude_code_version: string + date_range: { start: string; end: string } + session_count: number + remote_hosts_collected?: string[] + } + aggregated_data: AggregatedData + insights: InsightResults + facets_summary?: { + total: number + goal_categories: Record + outcomes: Record + satisfaction: Record + friction: Record + } +} + +/** + * Build export data from already-computed values. + * Used by background upload to S3. + */ +export function buildExportData( + data: AggregatedData, + insights: InsightResults, + facets: Map, + remoteStats?: { hosts: RemoteHostInfo[]; totalCopied: number }, +): InsightsExport { + const version = typeof MACRO !== 'undefined' ? MACRO.VERSION : 'unknown' + + const remote_hosts_collected = remoteStats?.hosts + .filter(h => h.sessionCount > 0) + .map(h => h.name) + + const facets_summary = { + total: facets.size, + goal_categories: {} as Record, + outcomes: {} as Record, + satisfaction: {} as Record, + friction: {} as Record, + } + for (const f of facets.values()) { + for (const [cat, count] of safeEntries(f.goal_categories)) { + if (count > 0) { + facets_summary.goal_categories[cat] = + (facets_summary.goal_categories[cat] || 0) + count + } + } + facets_summary.outcomes[f.outcome] = + (facets_summary.outcomes[f.outcome] || 0) + 1 + for (const [level, count] of safeEntries(f.user_satisfaction_counts)) { + if (count > 0) { + facets_summary.satisfaction[level] = + (facets_summary.satisfaction[level] || 0) + count + } + } + for (const [type, count] of safeEntries(f.friction_counts)) { + if (count > 0) { + facets_summary.friction[type] = + (facets_summary.friction[type] || 0) + count + } + } + } + + return { + metadata: { + username: process.env.SAFEUSER || process.env.USER || 'unknown', + generated_at: new Date().toISOString(), + claude_code_version: version, + date_range: data.date_range, + session_count: data.total_sessions, + ...(remote_hosts_collected && + remote_hosts_collected.length > 0 && { + remote_hosts_collected, + }), + }, + aggregated_data: data, + insights, + facets_summary, + } +} + +// ============================================================================ +// Lite Session Scanning +// ============================================================================ + +type LiteSessionInfo = { + sessionId: string + path: string + mtime: number + size: number +} + +/** + * Scans all project directories using filesystem metadata only (no JSONL parsing). + * Returns a list of session file info sorted by mtime descending. + * Yields to the event loop between project directories to keep the UI responsive. + */ +async function scanAllSessions(): Promise { + const projectsDir = getProjectsDir() + + let dirents: Awaited> + try { + dirents = await readdir(projectsDir, { withFileTypes: true }) + } catch { + return [] + } + + const projectDirs = dirents + .filter(dirent => dirent.isDirectory()) + .map(dirent => join(projectsDir, dirent.name)) + + const allSessions: LiteSessionInfo[] = [] + + for (let i = 0; i < projectDirs.length; i++) { + const sessionFiles = await getSessionFilesWithMtime(projectDirs[i]!) + for (const [sessionId, fileInfo] of sessionFiles) { + allSessions.push({ + sessionId, + path: fileInfo.path, + mtime: fileInfo.mtime, + size: fileInfo.size, + }) + } + // Yield to event loop every 10 project directories + if (i % 10 === 9) { + await new Promise(resolve => setImmediate(resolve)) + } + } + + // Sort by mtime descending (most recent first) + allSessions.sort((a, b) => b.mtime - a.mtime) + return allSessions +} + +// ============================================================================ +// Main Function +// ============================================================================ + +export async function generateUsageReport(options?: { + collectRemote?: boolean +}): Promise<{ + insights: InsightResults + htmlPath: string + data: AggregatedData + remoteStats?: { hosts: RemoteHostInfo[]; totalCopied: number } + facets: Map +}> { + let remoteStats: { hosts: RemoteHostInfo[]; totalCopied: number } | undefined + + // Optionally collect data from remote hosts first (ant-only) + if (process.env.USER_TYPE === 'ant' && options?.collectRemote) { + const destDir = join(getClaudeConfigHomeDir(), 'projects') + const { hosts, totalCopied } = await collectAllRemoteHostData(destDir) + remoteStats = { hosts, totalCopied } + } + + // Phase 1: Lite scan — filesystem metadata only (no JSONL parsing) + const allScannedSessions = await scanAllSessions() + const totalSessionsScanned = allScannedSessions.length + + // Phase 2: Load SessionMeta — use cache where available, parse only uncached + // Read cached metas in parallel batches to avoid blocking the event loop + const META_BATCH_SIZE = 50 + const MAX_SESSIONS_TO_LOAD = 200 + let allMetas: SessionMeta[] = [] + const uncachedSessions: LiteSessionInfo[] = [] + + for (let i = 0; i < allScannedSessions.length; i += META_BATCH_SIZE) { + const batch = allScannedSessions.slice(i, i + META_BATCH_SIZE) + const results = await Promise.all( + batch.map(async sessionInfo => ({ + sessionInfo, + cached: await loadCachedSessionMeta(sessionInfo.sessionId), + })), + ) + for (const { sessionInfo, cached } of results) { + if (cached) { + allMetas.push(cached) + } else if (uncachedSessions.length < MAX_SESSIONS_TO_LOAD) { + uncachedSessions.push(sessionInfo) + } + } + } + + // Load full message data only for uncached sessions and compute SessionMeta + const logsForFacets = new Map() + + // Filter out /insights meta-sessions (facet extraction API calls get logged as sessions) + const isMetaSession = (log: LogOption): boolean => { + for (const msg of log.messages.slice(0, 5)) { + if (msg.type === 'user' && msg.message) { + const content = msg.message.content + if (typeof content === 'string') { + if ( + content.includes('RESPOND WITH ONLY A VALID JSON OBJECT') || + content.includes('record_facets') + ) { + return true + } + } + } + } + return false + } + + // Load uncached sessions in batches to yield to event loop between batches + const LOAD_BATCH_SIZE = 10 + for (let i = 0; i < uncachedSessions.length; i += LOAD_BATCH_SIZE) { + const batch = uncachedSessions.slice(i, i + LOAD_BATCH_SIZE) + const batchResults = await Promise.all( + batch.map(async sessionInfo => { + try { + return await loadAllLogsFromSessionFile(sessionInfo.path) + } catch { + return [] + } + }), + ) + // Collect metas synchronously, then save them in parallel (independent writes) + const metasToSave: SessionMeta[] = [] + for (const logs of batchResults) { + for (const log of logs) { + if (isMetaSession(log) || !hasValidDates(log)) continue + const meta = logToSessionMeta(log) + allMetas.push(meta) + metasToSave.push(meta) + // Keep the log around for potential facet extraction + logsForFacets.set(meta.session_id, log) + } + } + await Promise.all(metasToSave.map(meta => saveSessionMeta(meta))) + } + + // Deduplicate session branches (keep the one with most user messages per session_id) + // This prevents inflated totals when a session has multiple conversation branches + const bestBySession = new Map() + for (const meta of allMetas) { + const existing = bestBySession.get(meta.session_id) + if ( + !existing || + meta.user_message_count > existing.user_message_count || + (meta.user_message_count === existing.user_message_count && + meta.duration_minutes > existing.duration_minutes) + ) { + bestBySession.set(meta.session_id, meta) + } + } + // Replace allMetas with deduplicated list and remove unused logs from logsForFacets + const keptSessionIds = new Set(bestBySession.keys()) + allMetas = [...bestBySession.values()] + for (const sessionId of logsForFacets.keys()) { + if (!keptSessionIds.has(sessionId)) { + logsForFacets.delete(sessionId) + } + } + + // Sort all metas by start_time descending (most recent first) + allMetas.sort((a, b) => b.start_time.localeCompare(a.start_time)) + + // Pre-filter obviously minimal sessions to save API calls + // (matching Python's substantive filtering concept) + const isSubstantiveSession = (meta: SessionMeta): boolean => { + // Skip sessions with very few user messages + if (meta.user_message_count < 2) return false + // Skip very short sessions (< 1 minute) + if (meta.duration_minutes < 1) return false + return true + } + + const substantiveMetas = allMetas.filter(isSubstantiveSession) + + // Phase 3: Facet extraction — only for sessions without cached facets + const facets = new Map() + const toExtract: Array<{ log: LogOption; sessionId: string }> = [] + const MAX_FACET_EXTRACTIONS = 50 + + // Load cached facets for all substantive sessions in parallel + const cachedFacetResults = await Promise.all( + substantiveMetas.map(async meta => ({ + sessionId: meta.session_id, + cached: await loadCachedFacets(meta.session_id), + })), + ) + for (const { sessionId, cached } of cachedFacetResults) { + if (cached) { + facets.set(sessionId, cached) + } else { + const log = logsForFacets.get(sessionId) + if (log && toExtract.length < MAX_FACET_EXTRACTIONS) { + toExtract.push({ log, sessionId }) + } + } + } + + // Extract facets for sessions that need them (50 concurrent) + const CONCURRENCY = 50 + for (let i = 0; i < toExtract.length; i += CONCURRENCY) { + const batch = toExtract.slice(i, i + CONCURRENCY) + const results = await Promise.all( + batch.map(async ({ log, sessionId }) => { + const newFacets = await extractFacetsFromAPI(log, sessionId) + return { sessionId, newFacets } + }), + ) + // Collect facets synchronously, save in parallel (independent writes) + const facetsToSave: SessionFacets[] = [] + for (const { sessionId, newFacets } of results) { + if (newFacets) { + facets.set(sessionId, newFacets) + facetsToSave.push(newFacets) + } + } + await Promise.all(facetsToSave.map(f => saveFacets(f))) + } + + // Filter out warmup/minimal sessions (matching Python's is_minimal) + // A session is minimal if warmup_minimal is the ONLY goal category + const isMinimalSession = (sessionId: string): boolean => { + const sessionFacets = facets.get(sessionId) + if (!sessionFacets) return false + const cats = sessionFacets.goal_categories + const catKeys = safeKeys(cats).filter(k => (cats[k] ?? 0) > 0) + return catKeys.length === 1 && catKeys[0] === 'warmup_minimal' + } + + const substantiveSessions = substantiveMetas.filter( + s => !isMinimalSession(s.session_id), + ) + + const substantiveFacets = new Map() + for (const [sessionId, f] of facets) { + if (!isMinimalSession(sessionId)) { + substantiveFacets.set(sessionId, f) + } + } + + const aggregated = aggregateData(substantiveSessions, substantiveFacets) + aggregated.total_sessions_scanned = totalSessionsScanned + + // Generate parallel insights from Claude (6 sections) + const insights = await generateParallelInsights(aggregated, facets) + + // Generate HTML report + const htmlReport = generateHtmlReport(aggregated, insights) + + // Save reports + try { + await mkdir(getDataDir(), { recursive: true }) + } catch { + // Directory may already exist + } + + const htmlPath = join(getDataDir(), 'report.html') + await writeFile(htmlPath, htmlReport, { + encoding: 'utf-8', + mode: 0o600, + }) + + return { + insights, + htmlPath, + data: aggregated, + remoteStats, + facets: substantiveFacets, + } +} + +function safeEntries( + obj: Record | undefined | null, +): [string, V][] { + return obj ? Object.entries(obj) : [] +} + +function safeKeys(obj: Record | undefined | null): string[] { + return obj ? Object.keys(obj) : [] +} + +// ============================================================================ +// Command Definition +// ============================================================================ + +const usageReport: Command = { + type: 'prompt', + name: 'insights', + description: 'Generate a report analyzing your Claude Code sessions', + contentLength: 0, // Dynamic content + progressMessage: 'analyzing your sessions', + source: 'builtin', + async getPromptForCommand(args) { + let collectRemote = false + let remoteHosts: string[] = [] + let hasRemoteHosts = false + + if (process.env.USER_TYPE === 'ant') { + // Parse --homespaces flag + collectRemote = args?.includes('--homespaces') ?? false + + // Check for available remote hosts + remoteHosts = await getRunningRemoteHosts() + hasRemoteHosts = remoteHosts.length > 0 + + // Show collection message if collecting + if (collectRemote && hasRemoteHosts) { + // biome-ignore lint/suspicious/noConsole: intentional + console.error( + `Collecting sessions from ${remoteHosts.length} homespace(s): ${remoteHosts.join(', ')}...`, + ) + } + } + + const { insights, htmlPath, data, remoteStats } = await generateUsageReport( + { collectRemote }, + ) + + let reportUrl = `file://${htmlPath}` + let uploadHint = '' + + if (process.env.USER_TYPE === 'ant') { + // Try to upload to S3 + const timestamp = new Date() + .toISOString() + .replace(/[-:]/g, '') + .replace('T', '_') + .slice(0, 15) + const username = process.env.SAFEUSER || process.env.USER || 'unknown' + const filename = `${username}_insights_${timestamp}.html` + const s3Path = `s3://anthropic-serve/atamkin/cc-user-reports/${filename}` + const s3Url = `https://s3-frontend.infra.ant.dev/anthropic-serve/atamkin/cc-user-reports/${filename}` + + reportUrl = s3Url + try { + execFileSync('ff', ['cp', htmlPath, s3Path], { + timeout: 60000, + stdio: 'pipe', // Suppress output + }) + } catch { + // Upload failed - fall back to local file and show upload command + reportUrl = `file://${htmlPath}` + uploadHint = `\nAutomatic upload failed. Are you on the boron namespace? Try \`use-bo\` and ensure you've run \`sso\`. +To share, run: ff cp ${htmlPath} ${s3Path} +Then access at: ${s3Url}` + } + } + + // Build header with stats + const sessionLabel = + data.total_sessions_scanned && + data.total_sessions_scanned > data.total_sessions + ? `${data.total_sessions_scanned.toLocaleString()} sessions total · ${data.total_sessions} analyzed` + : `${data.total_sessions} sessions` + const stats = [ + sessionLabel, + `${data.total_messages.toLocaleString()} messages`, + `${Math.round(data.total_duration_hours)}h`, + `${data.git_commits} commits`, + ].join(' · ') + + // Build remote host info (ant-only) + let remoteInfo = '' + if (process.env.USER_TYPE === 'ant') { + if (remoteStats && remoteStats.totalCopied > 0) { + const hsNames = remoteStats.hosts + .filter(h => h.sessionCount > 0) + .map(h => h.name) + .join(', ') + remoteInfo = `\n_Collected ${remoteStats.totalCopied} new sessions from: ${hsNames}_\n` + } else if (!collectRemote && hasRemoteHosts) { + // Suggest using --homespaces if they have remote hosts but didn't use the flag + remoteInfo = `\n_Tip: Run \`/insights --homespaces\` to include sessions from your ${remoteHosts.length} running homespace(s)_\n` + } + } + + // Build markdown summary from insights + const atAGlance = insights.at_a_glance + const summaryText = atAGlance + ? `## At a Glance + +${atAGlance.whats_working ? `**What's working:** ${atAGlance.whats_working} See _Impressive Things You Did_.` : ''} + +${atAGlance.whats_hindering ? `**What's hindering you:** ${atAGlance.whats_hindering} See _Where Things Go Wrong_.` : ''} + +${atAGlance.quick_wins ? `**Quick wins to try:** ${atAGlance.quick_wins} See _Features to Try_.` : ''} + +${atAGlance.ambitious_workflows ? `**Ambitious workflows:** ${atAGlance.ambitious_workflows} See _On the Horizon_.` : ''}` + : '_No insights generated_' + + const header = `# Claude Code Insights + +${stats} +${data.date_range.start} to ${data.date_range.end} +${remoteInfo} +` + + const userSummary = `${header}${summaryText} + +Your full shareable insights report is ready: ${reportUrl}${uploadHint}` + + // Return prompt for Claude to respond to + return [ + { + type: 'text', + text: `The user just ran /insights to generate a usage report analyzing their Claude Code sessions. + +Here is the full insights data: +${jsonStringify(insights, null, 2)} + +Report URL: ${reportUrl} +HTML file: ${htmlPath} +Facets directory: ${getFacetsDir()} + +Here is what the user sees: +${userSummary} + +Now output the following message exactly: + + +Your shareable insights report is ready: +${reportUrl}${uploadHint} + +Want to dig into any section or try one of the suggestions? +`, + }, + ] + }, +} + +function isValidSessionFacets(obj: unknown): obj is SessionFacets { + if (!obj || typeof obj !== 'object') return false + const o = obj as Record + return ( + typeof o.underlying_goal === 'string' && + typeof o.outcome === 'string' && + typeof o.brief_summary === 'string' && + o.goal_categories !== null && + typeof o.goal_categories === 'object' && + o.user_satisfaction_counts !== null && + typeof o.user_satisfaction_counts === 'object' && + o.friction_counts !== null && + typeof o.friction_counts === 'object' + ) +} + +export default usageReport diff --git a/src/commands/install-github-app/ApiKeyStep.tsx b/src/commands/install-github-app/ApiKeyStep.tsx new file mode 100644 index 0000000..2dcb312 --- /dev/null +++ b/src/commands/install-github-app/ApiKeyStep.tsx @@ -0,0 +1,231 @@ +import { c as _c } from "react/compiler-runtime"; +import React, { useCallback, useState } from 'react'; +import TextInput from '../../components/TextInput.js'; +import { useTerminalSize } from '../../hooks/useTerminalSize.js'; +import { Box, color, Text, useTheme } from '../../ink.js'; +import { useKeybindings } from '../../keybindings/useKeybinding.js'; +interface ApiKeyStepProps { + existingApiKey: string | null; + useExistingKey: boolean; + apiKeyOrOAuthToken: string; + onApiKeyChange: (value: string) => void; + onToggleUseExistingKey: (useExisting: boolean) => void; + onSubmit: () => void; + onCreateOAuthToken?: () => void; + selectedOption?: 'existing' | 'new' | 'oauth'; + onSelectOption?: (option: 'existing' | 'new' | 'oauth') => void; +} +export function ApiKeyStep(t0) { + const $ = _c(55); + const { + existingApiKey, + apiKeyOrOAuthToken, + onApiKeyChange, + onSubmit, + onToggleUseExistingKey, + onCreateOAuthToken, + selectedOption: t1, + onSelectOption + } = t0; + const selectedOption = t1 === undefined ? existingApiKey ? "existing" : onCreateOAuthToken ? "oauth" : "new" : t1; + const [cursorOffset, setCursorOffset] = useState(0); + const terminalSize = useTerminalSize(); + const [theme] = useTheme(); + let t2; + if ($[0] !== existingApiKey || $[1] !== onCreateOAuthToken || $[2] !== onSelectOption || $[3] !== onToggleUseExistingKey || $[4] !== selectedOption) { + t2 = () => { + if (selectedOption === "new" && onCreateOAuthToken) { + onSelectOption?.("oauth"); + } else { + if (selectedOption === "oauth" && existingApiKey) { + onSelectOption?.("existing"); + onToggleUseExistingKey(true); + } + } + }; + $[0] = existingApiKey; + $[1] = onCreateOAuthToken; + $[2] = onSelectOption; + $[3] = onToggleUseExistingKey; + $[4] = selectedOption; + $[5] = t2; + } else { + t2 = $[5]; + } + const handlePrevious = t2; + let t3; + if ($[6] !== onCreateOAuthToken || $[7] !== onSelectOption || $[8] !== onToggleUseExistingKey || $[9] !== selectedOption) { + t3 = () => { + if (selectedOption === "existing") { + onSelectOption?.(onCreateOAuthToken ? "oauth" : "new"); + onToggleUseExistingKey(false); + } else { + if (selectedOption === "oauth") { + onSelectOption?.("new"); + } + } + }; + $[6] = onCreateOAuthToken; + $[7] = onSelectOption; + $[8] = onToggleUseExistingKey; + $[9] = selectedOption; + $[10] = t3; + } else { + t3 = $[10]; + } + const handleNext = t3; + let t4; + if ($[11] !== onCreateOAuthToken || $[12] !== onSubmit || $[13] !== selectedOption) { + t4 = () => { + if (selectedOption === "oauth" && onCreateOAuthToken) { + onCreateOAuthToken(); + } else { + onSubmit(); + } + }; + $[11] = onCreateOAuthToken; + $[12] = onSubmit; + $[13] = selectedOption; + $[14] = t4; + } else { + t4 = $[14]; + } + const handleConfirm = t4; + const isTextInputVisible = selectedOption === "new"; + let t5; + if ($[15] !== handleConfirm || $[16] !== handleNext || $[17] !== handlePrevious) { + t5 = { + "confirm:previous": handlePrevious, + "confirm:next": handleNext, + "confirm:yes": handleConfirm + }; + $[15] = handleConfirm; + $[16] = handleNext; + $[17] = handlePrevious; + $[18] = t5; + } else { + t5 = $[18]; + } + const t6 = !isTextInputVisible; + let t7; + if ($[19] !== t6) { + t7 = { + context: "Confirmation", + isActive: t6 + }; + $[19] = t6; + $[20] = t7; + } else { + t7 = $[20]; + } + useKeybindings(t5, t7); + let t8; + if ($[21] !== handleNext || $[22] !== handlePrevious) { + t8 = { + "confirm:previous": handlePrevious, + "confirm:next": handleNext + }; + $[21] = handleNext; + $[22] = handlePrevious; + $[23] = t8; + } else { + t8 = $[23]; + } + let t9; + if ($[24] !== isTextInputVisible) { + t9 = { + context: "Confirmation", + isActive: isTextInputVisible + }; + $[24] = isTextInputVisible; + $[25] = t9; + } else { + t9 = $[25]; + } + useKeybindings(t8, t9); + let t10; + if ($[26] === Symbol.for("react.memo_cache_sentinel")) { + t10 = Install GitHub AppChoose API key; + $[26] = t10; + } else { + t10 = $[26]; + } + let t11; + if ($[27] !== existingApiKey || $[28] !== selectedOption || $[29] !== theme) { + t11 = existingApiKey && {selectedOption === "existing" ? color("success", theme)("> ") : " "}Use your existing Claude Code API key; + $[27] = existingApiKey; + $[28] = selectedOption; + $[29] = theme; + $[30] = t11; + } else { + t11 = $[30]; + } + let t12; + if ($[31] !== onCreateOAuthToken || $[32] !== selectedOption || $[33] !== theme) { + t12 = onCreateOAuthToken && {selectedOption === "oauth" ? color("success", theme)("> ") : " "}Create a long-lived token with your Claude subscription; + $[31] = onCreateOAuthToken; + $[32] = selectedOption; + $[33] = theme; + $[34] = t12; + } else { + t12 = $[34]; + } + let t13; + if ($[35] !== selectedOption || $[36] !== theme) { + t13 = selectedOption === "new" ? color("success", theme)("> ") : " "; + $[35] = selectedOption; + $[36] = theme; + $[37] = t13; + } else { + t13 = $[37]; + } + let t14; + if ($[38] !== t13) { + t14 = {t13}Enter a new API key; + $[38] = t13; + $[39] = t14; + } else { + t14 = $[39]; + } + let t15; + if ($[40] !== apiKeyOrOAuthToken || $[41] !== cursorOffset || $[42] !== onApiKeyChange || $[43] !== onSubmit || $[44] !== selectedOption || $[45] !== terminalSize) { + t15 = selectedOption === "new" && ; + $[40] = apiKeyOrOAuthToken; + $[41] = cursorOffset; + $[42] = onApiKeyChange; + $[43] = onSubmit; + $[44] = selectedOption; + $[45] = terminalSize; + $[46] = t15; + } else { + t15 = $[46]; + } + let t16; + if ($[47] !== t11 || $[48] !== t12 || $[49] !== t14 || $[50] !== t15) { + t16 = {t10}{t11}{t12}{t14}{t15}; + $[47] = t11; + $[48] = t12; + $[49] = t14; + $[50] = t15; + $[51] = t16; + } else { + t16 = $[51]; + } + let t17; + if ($[52] === Symbol.for("react.memo_cache_sentinel")) { + t17 = ↑/↓ to select · Enter to continue; + $[52] = t17; + } else { + t17 = $[52]; + } + let t18; + if ($[53] !== t16) { + t18 = <>{t16}{t17}; + $[53] = t16; + $[54] = t18; + } else { + t18 = $[54]; + } + return t18; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsInVzZUNhbGxiYWNrIiwidXNlU3RhdGUiLCJUZXh0SW5wdXQiLCJ1c2VUZXJtaW5hbFNpemUiLCJCb3giLCJjb2xvciIsIlRleHQiLCJ1c2VUaGVtZSIsInVzZUtleWJpbmRpbmdzIiwiQXBpS2V5U3RlcFByb3BzIiwiZXhpc3RpbmdBcGlLZXkiLCJ1c2VFeGlzdGluZ0tleSIsImFwaUtleU9yT0F1dGhUb2tlbiIsIm9uQXBpS2V5Q2hhbmdlIiwidmFsdWUiLCJvblRvZ2dsZVVzZUV4aXN0aW5nS2V5IiwidXNlRXhpc3RpbmciLCJvblN1Ym1pdCIsIm9uQ3JlYXRlT0F1dGhUb2tlbiIsInNlbGVjdGVkT3B0aW9uIiwib25TZWxlY3RPcHRpb24iLCJvcHRpb24iLCJBcGlLZXlTdGVwIiwidDAiLCIkIiwiX2MiLCJ0MSIsInVuZGVmaW5lZCIsImN1cnNvck9mZnNldCIsInNldEN1cnNvck9mZnNldCIsInRlcm1pbmFsU2l6ZSIsInRoZW1lIiwidDIiLCJoYW5kbGVQcmV2aW91cyIsInQzIiwiaGFuZGxlTmV4dCIsInQ0IiwiaGFuZGxlQ29uZmlybSIsImlzVGV4dElucHV0VmlzaWJsZSIsInQ1IiwidDYiLCJ0NyIsImNvbnRleHQiLCJpc0FjdGl2ZSIsInQ4IiwidDkiLCJ0MTAiLCJTeW1ib2wiLCJmb3IiLCJ0MTEiLCJ0MTIiLCJ0MTMiLCJ0MTQiLCJ0MTUiLCJjb2x1bW5zIiwidDE2IiwidDE3IiwidDE4Il0sInNvdXJjZXMiOlsiQXBpS2V5U3RlcC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFJlYWN0LCB7IHVzZUNhbGxiYWNrLCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IFRleHRJbnB1dCBmcm9tICcuLi8uLi9jb21wb25lbnRzL1RleHRJbnB1dC5qcydcbmltcG9ydCB7IHVzZVRlcm1pbmFsU2l6ZSB9IGZyb20gJy4uLy4uL2hvb2tzL3VzZVRlcm1pbmFsU2l6ZS5qcydcbmltcG9ydCB7IEJveCwgY29sb3IsIFRleHQsIHVzZVRoZW1lIH0gZnJvbSAnLi4vLi4vaW5rLmpzJ1xuaW1wb3J0IHsgdXNlS2V5YmluZGluZ3MgfSBmcm9tICcuLi8uLi9rZXliaW5kaW5ncy91c2VLZXliaW5kaW5nLmpzJ1xuXG5pbnRlcmZhY2UgQXBpS2V5U3RlcFByb3BzIHtcbiAgZXhpc3RpbmdBcGlLZXk6IHN0cmluZyB8IG51bGxcbiAgdXNlRXhpc3RpbmdLZXk6IGJvb2xlYW5cbiAgYXBpS2V5T3JPQXV0aFRva2VuOiBzdHJpbmdcbiAgb25BcGlLZXlDaGFuZ2U6ICh2YWx1ZTogc3RyaW5nKSA9PiB2b2lkXG4gIG9uVG9nZ2xlVXNlRXhpc3RpbmdLZXk6ICh1c2VFeGlzdGluZzogYm9vbGVhbikgPT4gdm9pZFxuICBvblN1Ym1pdDogKCkgPT4gdm9pZFxuICBvbkNyZWF0ZU9BdXRoVG9rZW4/OiAoKSA9PiB2b2lkXG4gIHNlbGVjdGVkT3B0aW9uPzogJ2V4aXN0aW5nJyB8ICduZXcnIHwgJ29hdXRoJ1xuICBvblNlbGVjdE9wdGlvbj86IChvcHRpb246ICdleGlzdGluZycgfCAnbmV3JyB8ICdvYXV0aCcpID0+IHZvaWRcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIEFwaUtleVN0ZXAoe1xuICBleGlzdGluZ0FwaUtleSxcbiAgYXBpS2V5T3JPQXV0aFRva2VuLFxuICBvbkFwaUtleUNoYW5nZSxcbiAgb25TdWJtaXQsXG4gIG9uVG9nZ2xlVXNlRXhpc3RpbmdLZXksXG4gIG9uQ3JlYXRlT0F1dGhUb2tlbixcbiAgc2VsZWN0ZWRPcHRpb24gPSBleGlzdGluZ0FwaUtleVxuICAgID8gJ2V4aXN0aW5nJ1xuICAgIDogb25DcmVhdGVPQXV0aFRva2VuXG4gICAgICA/ICdvYXV0aCdcbiAgICAgIDogJ25ldycsXG4gIG9uU2VsZWN0T3B0aW9uLFxufTogQXBpS2V5U3RlcFByb3BzKSB7XG4gIGNvbnN0IFtjdXJzb3JPZmZzZXQsIHNldEN1cnNvck9mZnNldF0gPSB1c2VTdGF0ZSgwKVxuICBjb25zdCB0ZXJtaW5hbFNpemUgPSB1c2VUZXJtaW5hbFNpemUoKVxuICBjb25zdCBbdGhlbWVdID0gdXNlVGhlbWUoKVxuXG4gIGNvbnN0IGhhbmRsZVByZXZpb3VzID0gdXNlQ2FsbGJhY2soKCkgPT4ge1xuICAgIGlmIChzZWxlY3RlZE9wdGlvbiA9PT0gJ25ldycgJiYgb25DcmVhdGVPQXV0aFRva2VuKSB7XG4gICAgICAvLyBGcm9tICduZXcnIGdvIHVwIHRvICdvYXV0aCdcbiAgICAgIG9uU2VsZWN0T3B0aW9uPy4oJ29hdXRoJylcbiAgICB9IGVsc2UgaWYgKHNlbGVjdGVkT3B0aW9uID09PSAnb2F1dGgnICYmIGV4aXN0aW5nQXBpS2V5KSB7XG4gICAgICAvLyBGcm9tICdvYXV0aCcgZ28gdXAgdG8gJ2V4aXN0aW5nJyAob25seSBpZiBpdCBleGlzdHMpXG4gICAgICBvblNlbGVjdE9wdGlvbj8uKCdleGlzdGluZycpXG4gICAgICBvblRvZ2dsZVVzZUV4aXN0aW5nS2V5KHRydWUpXG4gICAgfVxuICB9LCBbXG4gICAgc2VsZWN0ZWRPcHRpb24sXG4gICAgb25DcmVhdGVPQXV0aFRva2VuLFxuICAgIGV4aXN0aW5nQXBpS2V5LFxuICAgIG9uU2VsZWN0T3B0aW9uLFxuICAgIG9uVG9nZ2xlVXNlRXhpc3RpbmdLZXksXG4gIF0pXG5cbiAgY29uc3QgaGFuZGxlTmV4dCA9IHVzZUNhbGxiYWNrKCgpID0+IHtcbiAgICBpZiAoc2VsZWN0ZWRPcHRpb24gPT09ICdleGlzdGluZycpIHtcbiAgICAgIC8vIEZyb20gJ2V4aXN0aW5nJyBnbyBkb3duIHRvICdvYXV0aCcgKGlmIGF2YWlsYWJsZSkgb3IgJ25ldydcbiAgICAgIG9uU2VsZWN0T3B0aW9uPy4ob25DcmVhdGVPQXV0aFRva2VuID8gJ29hdXRoJyA6ICduZXcnKVxuICAgICAgb25Ub2dnbGVVc2VFeGlzdGluZ0tleShmYWxzZSlcbiAgICB9IGVsc2UgaWYgKHNlbGVjdGVkT3B0aW9uID09PSAnb2F1dGgnKSB7XG4gICAgICAvLyBGcm9tICdvYXV0aCcgZ28gZG93biB0byAnbmV3J1xuICAgICAgb25TZWxlY3RPcHRpb24/LignbmV3JylcbiAgICB9XG4gIH0sIFtcbiAgICBzZWxlY3RlZE9wdGlvbixcbiAgICBvbkNyZWF0ZU9BdXRoVG9rZW4sXG4gICAgb25TZWxlY3RPcHRpb24sXG4gICAgb25Ub2dnbGVVc2VFeGlzdGluZ0tleSxcbiAgXSlcblxuICBjb25zdCBoYW5kbGVDb25maXJtID0gdXNlQ2FsbGJhY2soKCkgPT4ge1xuICAgIGlmIChzZWxlY3RlZE9wdGlvbiA9PT0gJ29hdXRoJyAmJiBvbkNyZWF0ZU9BdXRoVG9rZW4pIHtcbiAgICAgIG9uQ3JlYXRlT0F1dGhUb2tlbigpXG4gICAgfSBlbHNlIHtcbiAgICAgIG9uU3VibWl0KClcbiAgICB9XG4gIH0sIFtzZWxlY3RlZE9wdGlvbiwgb25DcmVhdGVPQXV0aFRva2VuLCBvblN1Ym1pdF0pXG5cbiAgLy8gV2hlbiB0aGUgdGV4dCBpbnB1dCBpcyB2aXNpYmxlLCBvbWl0IGNvbmZpcm06eWVzIHNvIGJhcmUgJ3knIHBhc3Nlc1xuICAvLyB0aHJvdWdoIHRvIHRoZSBpbnB1dCBpbnN0ZWFkIG9mIHN1Ym1pdHRpbmcuIFRleHRJbnB1dCdzIG9uU3VibWl0IGhhbmRsZXNcbiAgLy8gRW50ZXIuIEtlZXAgdGhlIENvbmZpcm1hdGlvbiBjb250ZXh0IChub3QgU2V0dGluZ3MpIHRvIGF2b2lkIGovayBiaW5kaW5ncy5cbiAgY29uc3QgaXNUZXh0SW5wdXRWaXNpYmxlID0gc2VsZWN0ZWRPcHRpb24gPT09ICduZXcnXG4gIHVzZUtleWJpbmRpbmdzKFxuICAgIHtcbiAgICAgICdjb25maXJtOnByZXZpb3VzJzogaGFuZGxlUHJldmlvdXMsXG4gICAgICAnY29uZmlybTpuZXh0JzogaGFuZGxlTmV4dCxcbiAgICAgICdjb25maXJtOnllcyc6IGhhbmRsZUNvbmZpcm0sXG4gICAgfSxcbiAgICB7IGNvbnRleHQ6ICdDb25maXJtYXRpb24nLCBpc0FjdGl2ZTogIWlzVGV4dElucHV0VmlzaWJsZSB9LFxuICApXG4gIHVzZUtleWJpbmRpbmdzKFxuICAgIHtcbiAgICAgICdjb25maXJtOnByZXZpb3VzJzogaGFuZGxlUHJldmlvdXMsXG4gICAgICAnY29uZmlybTpuZXh0JzogaGFuZGxlTmV4dCxcbiAgICB9LFxuICAgIHsgY29udGV4dDogJ0NvbmZpcm1hdGlvbicsIGlzQWN0aXZlOiBpc1RleHRJbnB1dFZpc2libGUgfSxcbiAgKVxuXG4gIHJldHVybiAoXG4gICAgPD5cbiAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiIGJvcmRlclN0eWxlPVwicm91bmRcIiBwYWRkaW5nWD17MX0+XG4gICAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiIG1hcmdpbkJvdHRvbT17MX0+XG4gICAgICAgICAgPFRleHQgYm9sZD5JbnN0YWxsIEdpdEh1YiBBcHA8L1RleHQ+XG4gICAgICAgICAgPFRleHQgZGltQ29sb3I+Q2hvb3NlIEFQSSBrZXk8L1RleHQ+XG4gICAgICAgIDwvQm94PlxuICAgICAgICB7ZXhpc3RpbmdBcGlLZXkgJiYgKFxuICAgICAgICAgIDxCb3ggbWFyZ2luQm90dG9tPXsxfT5cbiAgICAgICAgICAgIDxUZXh0PlxuICAgICAgICAgICAgICB7c2VsZWN0ZWRPcHRpb24gPT09ICdleGlzdGluZydcbiAgICAgICAgICAgICAgICA/IGNvbG9yKCdzdWNjZXNzJywgdGhlbWUpKCc+ICcpXG4gICAgICAgICAgICAgICAgOiAnICAnfVxuICAgICAgICAgICAgICBVc2UgeW91ciBleGlzdGluZyBDbGF1ZGUgQ29kZSBBUEkga2V5XG4gICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgPC9Cb3g+XG4gICAgICAgICl9XG4gICAgICAgIHtvbkNyZWF0ZU9BdXRoVG9rZW4gJiYgKFxuICAgICAgICAgIDxCb3ggbWFyZ2luQm90dG9tPXsxfT5cbiAgICAgICAgICAgIDxUZXh0PlxuICAgICAgICAgICAgICB7c2VsZWN0ZWRPcHRpb24gPT09ICdvYXV0aCdcbiAgICAgICAgICAgICAgICA/IGNvbG9yKCdzdWNjZXNzJywgdGhlbWUpKCc+ICcpXG4gICAgICAgICAgICAgICAgOiAnICAnfVxuICAgICAgICAgICAgICBDcmVhdGUgYSBsb25nLWxpdmVkIHRva2VuIHdpdGggeW91ciBDbGF1ZGUgc3Vic2NyaXB0aW9uXG4gICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgPC9Cb3g+XG4gICAgICAgICl9XG4gICAgICAgIDxCb3ggbWFyZ2luQm90dG9tPXsxfT5cbiAgICAgICAgICA8VGV4dD5cbiAgICAgICAgICAgIHtzZWxlY3RlZE9wdGlvbiA9PT0gJ25ldycgPyBjb2xvcignc3VjY2VzcycsIHRoZW1lKSgnPiAnKSA6ICcgICd9XG4gICAgICAgICAgICBFbnRlciBhIG5ldyBBUEkga2V5XG4gICAgICAgICAgPC9UZXh0PlxuICAgICAgICA8L0JveD5cbiAgICAgICAge3NlbGVjdGVkT3B0aW9uID09PSAnbmV3JyAmJiAoXG4gICAgICAgICAgPFRleHRJbnB1dFxuICAgICAgICAgICAgdmFsdWU9e2FwaUtleU9yT0F1dGhUb2tlbn1cbiAgICAgICAgICAgIG9uQ2hhbmdlPXtvbkFwaUtleUNoYW5nZX1cbiAgICAgICAgICAgIG9uU3VibWl0PXtvblN1Ym1pdH1cbiAgICAgICAgICAgIG9uUGFzdGU9e29uQXBpS2V5Q2hhbmdlfVxuICAgICAgICAgICAgZm9jdXM9e3RydWV9XG4gICAgICAgICAgICBwbGFjZWhvbGRlcj1cInNrLWFudOKApiAoQ3JlYXRlIGEgbmV3IGtleSBhdCBodHRwczovL3BsYXRmb3JtLmNsYXVkZS5jb20vc2V0dGluZ3Mva2V5cylcIlxuICAgICAgICAgICAgbWFzaz1cIipcIlxuICAgICAgICAgICAgY29sdW1ucz17dGVybWluYWxTaXplLmNvbHVtbnN9XG4gICAgICAgICAgICBjdXJzb3JPZmZzZXQ9e2N1cnNvck9mZnNldH1cbiAgICAgICAgICAgIG9uQ2hhbmdlQ3Vyc29yT2Zmc2V0PXtzZXRDdXJzb3JPZmZzZXR9XG4gICAgICAgICAgICBzaG93Q3Vyc29yPXt0cnVlfVxuICAgICAgICAgIC8+XG4gICAgICAgICl9XG4gICAgICA8L0JveD5cbiAgICAgIDxCb3ggbWFyZ2luTGVmdD17M30+XG4gICAgICAgIDxUZXh0IGRpbUNvbG9yPuKGkS/ihpMgdG8gc2VsZWN0IMK3IEVudGVyIHRvIGNvbnRpbnVlPC9UZXh0PlxuICAgICAgPC9Cb3g+XG4gICAgPC8+XG4gIClcbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU9BLEtBQUssSUFBSUMsV0FBVyxFQUFFQyxRQUFRLFFBQVEsT0FBTztBQUNwRCxPQUFPQyxTQUFTLE1BQU0sK0JBQStCO0FBQ3JELFNBQVNDLGVBQWUsUUFBUSxnQ0FBZ0M7QUFDaEUsU0FBU0MsR0FBRyxFQUFFQyxLQUFLLEVBQUVDLElBQUksRUFBRUMsUUFBUSxRQUFRLGNBQWM7QUFDekQsU0FBU0MsY0FBYyxRQUFRLG9DQUFvQztBQUVuRSxVQUFVQyxlQUFlLENBQUM7RUFDeEJDLGNBQWMsRUFBRSxNQUFNLEdBQUcsSUFBSTtFQUM3QkMsY0FBYyxFQUFFLE9BQU87RUFDdkJDLGtCQUFrQixFQUFFLE1BQU07RUFDMUJDLGNBQWMsRUFBRSxDQUFDQyxLQUFLLEVBQUUsTUFBTSxFQUFFLEdBQUcsSUFBSTtFQUN2Q0Msc0JBQXNCLEVBQUUsQ0FBQ0MsV0FBVyxFQUFFLE9BQU8sRUFBRSxHQUFHLElBQUk7RUFDdERDLFFBQVEsRUFBRSxHQUFHLEdBQUcsSUFBSTtFQUNwQkMsa0JBQWtCLENBQUMsRUFBRSxHQUFHLEdBQUcsSUFBSTtFQUMvQkMsY0FBYyxDQUFDLEVBQUUsVUFBVSxHQUFHLEtBQUssR0FBRyxPQUFPO0VBQzdDQyxjQUFjLENBQUMsRUFBRSxDQUFDQyxNQUFNLEVBQUUsVUFBVSxHQUFHLEtBQUssR0FBRyxPQUFPLEVBQUUsR0FBRyxJQUFJO0FBQ2pFO0FBRUEsT0FBTyxTQUFBQyxXQUFBQyxFQUFBO0VBQUEsTUFBQUMsQ0FBQSxHQUFBQyxFQUFBO0VBQW9CO0lBQUFmLGNBQUE7SUFBQUUsa0JBQUE7SUFBQUMsY0FBQTtJQUFBSSxRQUFBO0lBQUFGLHNCQUFB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 \ No newline at end of file diff --git a/src/commands/install-github-app/CheckExistingSecretStep.tsx b/src/commands/install-github-app/CheckExistingSecretStep.tsx new file mode 100644 index 0000000..ff2bf45 --- /dev/null +++ b/src/commands/install-github-app/CheckExistingSecretStep.tsx @@ -0,0 +1,190 @@ +import { c as _c } from "react/compiler-runtime"; +import React, { useCallback, useState } from 'react'; +import TextInput from '../../components/TextInput.js'; +import { useTerminalSize } from '../../hooks/useTerminalSize.js'; +import { Box, color, Text, useTheme } from '../../ink.js'; +import { useKeybindings } from '../../keybindings/useKeybinding.js'; +interface CheckExistingSecretStepProps { + useExistingSecret: boolean; + secretName: string; + onToggleUseExistingSecret: (useExisting: boolean) => void; + onSecretNameChange: (value: string) => void; + onSubmit: () => void; +} +export function CheckExistingSecretStep(t0) { + const $ = _c(42); + const { + useExistingSecret, + secretName, + onToggleUseExistingSecret, + onSecretNameChange, + onSubmit + } = t0; + const [cursorOffset, setCursorOffset] = useState(0); + const terminalSize = useTerminalSize(); + const [theme] = useTheme(); + let t1; + if ($[0] !== onToggleUseExistingSecret) { + t1 = () => onToggleUseExistingSecret(true); + $[0] = onToggleUseExistingSecret; + $[1] = t1; + } else { + t1 = $[1]; + } + const handlePrevious = t1; + let t2; + if ($[2] !== onToggleUseExistingSecret) { + t2 = () => onToggleUseExistingSecret(false); + $[2] = onToggleUseExistingSecret; + $[3] = t2; + } else { + t2 = $[3]; + } + const handleNext = t2; + let t3; + if ($[4] !== handleNext || $[5] !== handlePrevious || $[6] !== onSubmit) { + t3 = { + "confirm:previous": handlePrevious, + "confirm:next": handleNext, + "confirm:yes": onSubmit + }; + $[4] = handleNext; + $[5] = handlePrevious; + $[6] = onSubmit; + $[7] = t3; + } else { + t3 = $[7]; + } + let t4; + if ($[8] !== useExistingSecret) { + t4 = { + context: "Confirmation", + isActive: useExistingSecret + }; + $[8] = useExistingSecret; + $[9] = t4; + } else { + t4 = $[9]; + } + useKeybindings(t3, t4); + let t5; + if ($[10] !== handleNext || $[11] !== handlePrevious) { + t5 = { + "confirm:previous": handlePrevious, + "confirm:next": handleNext + }; + $[10] = handleNext; + $[11] = handlePrevious; + $[12] = t5; + } else { + t5 = $[12]; + } + const t6 = !useExistingSecret; + let t7; + if ($[13] !== t6) { + t7 = { + context: "Confirmation", + isActive: t6 + }; + $[13] = t6; + $[14] = t7; + } else { + t7 = $[14]; + } + useKeybindings(t5, t7); + let t8; + if ($[15] === Symbol.for("react.memo_cache_sentinel")) { + t8 = Install GitHub AppSetup API key secret; + $[15] = t8; + } else { + t8 = $[15]; + } + let t9; + if ($[16] === Symbol.for("react.memo_cache_sentinel")) { + t9 = ANTHROPIC_API_KEY already exists in repository secrets!; + $[16] = t9; + } else { + t9 = $[16]; + } + let t10; + if ($[17] === Symbol.for("react.memo_cache_sentinel")) { + t10 = Would you like to:; + $[17] = t10; + } else { + t10 = $[17]; + } + let t11; + if ($[18] !== theme || $[19] !== useExistingSecret) { + t11 = useExistingSecret ? color("success", theme)("> ") : " "; + $[18] = theme; + $[19] = useExistingSecret; + $[20] = t11; + } else { + t11 = $[20]; + } + let t12; + if ($[21] !== t11) { + t12 = {t11}Use the existing API key; + $[21] = t11; + $[22] = t12; + } else { + t12 = $[22]; + } + let t13; + if ($[23] !== theme || $[24] !== useExistingSecret) { + t13 = !useExistingSecret ? color("success", theme)("> ") : " "; + $[23] = theme; + $[24] = useExistingSecret; + $[25] = t13; + } else { + t13 = $[25]; + } + let t14; + if ($[26] !== t13) { + t14 = {t13}Create a new secret with a different name; + $[26] = t13; + $[27] = t14; + } else { + t14 = $[27]; + } + let t15; + if ($[28] !== cursorOffset || $[29] !== onSecretNameChange || $[30] !== onSubmit || $[31] !== secretName || $[32] !== terminalSize || $[33] !== useExistingSecret) { + t15 = !useExistingSecret && <>Enter new secret name (alphanumeric with underscores):; + $[28] = cursorOffset; + $[29] = onSecretNameChange; + $[30] = onSubmit; + $[31] = secretName; + $[32] = terminalSize; + $[33] = useExistingSecret; + $[34] = t15; + } else { + t15 = $[34]; + } + let t16; + if ($[35] !== t12 || $[36] !== t14 || $[37] !== t15) { + t16 = {t8}{t9}{t10}{t12}{t14}{t15}; + $[35] = t12; + $[36] = t14; + $[37] = t15; + $[38] = t16; + } else { + t16 = $[38]; + } + let t17; + if ($[39] === Symbol.for("react.memo_cache_sentinel")) { + t17 = ↑/↓ to select · Enter to continue; + $[39] = t17; + } else { + t17 = $[39]; + } + let t18; + if ($[40] !== t16) { + t18 = <>{t16}{t17}; + $[40] = t16; + $[41] = t18; + } else { + t18 = $[41]; + } + return t18; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsInVzZUNhbGxiYWNrIiwidXNlU3RhdGUiLCJUZXh0SW5wdXQiLCJ1c2VUZXJtaW5hbFNpemUiLCJCb3giLCJjb2xvciIsIlRleHQiLCJ1c2VUaGVtZSIsInVzZUtleWJpbmRpbmdzIiwiQ2hlY2tFeGlzdGluZ1NlY3JldFN0ZXBQcm9wcyIsInVzZUV4aXN0aW5nU2VjcmV0Iiwic2VjcmV0TmFtZSIsIm9uVG9nZ2xlVXNlRXhpc3RpbmdTZWNyZXQiLCJ1c2VFeGlzdGluZyIsIm9uU2VjcmV0TmFtZUNoYW5nZSIsInZhbHVlIiwib25TdWJtaXQiLCJDaGVja0V4aXN0aW5nU2VjcmV0U3RlcCIsInQwIiwiJCIsIl9jIiwiY3Vyc29yT2Zmc2V0Iiwic2V0Q3Vyc29yT2Zmc2V0IiwidGVybWluYWxTaXplIiwidGhlbWUiLCJ0MSIsImhhbmRsZVByZXZpb3VzIiwidDIiLCJoYW5kbGVOZXh0IiwidDMiLCJ0NCIsImNvbnRleHQiLCJpc0FjdGl2ZSIsInQ1IiwidDYiLCJ0NyIsInQ4IiwiU3ltYm9sIiwiZm9yIiwidDkiLCJ0MTAiLCJ0MTEiLCJ0MTIiLCJ0MTMiLCJ0MTQiLCJ0MTUiLCJjb2x1bW5zIiwidDE2IiwidDE3IiwidDE4Il0sInNvdXJjZXMiOlsiQ2hlY2tFeGlzdGluZ1NlY3JldFN0ZXAudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBSZWFjdCwgeyB1c2VDYWxsYmFjaywgdXNlU3RhdGUgfSBmcm9tICdyZWFjdCdcbmltcG9ydCBUZXh0SW5wdXQgZnJvbSAnLi4vLi4vY29tcG9uZW50cy9UZXh0SW5wdXQuanMnXG5pbXBvcnQgeyB1c2VUZXJtaW5hbFNpemUgfSBmcm9tICcuLi8uLi9ob29rcy91c2VUZXJtaW5hbFNpemUuanMnXG5pbXBvcnQgeyBCb3gsIGNvbG9yLCBUZXh0LCB1c2VUaGVtZSB9IGZyb20gJy4uLy4uL2luay5qcydcbmltcG9ydCB7IHVzZUtleWJpbmRpbmdzIH0gZnJvbSAnLi4vLi4va2V5YmluZGluZ3MvdXNlS2V5YmluZGluZy5qcydcblxuaW50ZXJmYWNlIENoZWNrRXhpc3RpbmdTZWNyZXRTdGVwUHJvcHMge1xuICB1c2VFeGlzdGluZ1NlY3JldDogYm9vbGVhblxuICBzZWNyZXROYW1lOiBzdHJpbmdcbiAgb25Ub2dnbGVVc2VFeGlzdGluZ1NlY3JldDogKHVzZUV4aXN0aW5nOiBib29sZWFuKSA9PiB2b2lkXG4gIG9uU2VjcmV0TmFtZUNoYW5nZTogKHZhbHVlOiBzdHJpbmcpID0+IHZvaWRcbiAgb25TdWJtaXQ6ICgpID0+IHZvaWRcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIENoZWNrRXhpc3RpbmdTZWNyZXRTdGVwKHtcbiAgdXNlRXhpc3RpbmdTZWNyZXQsXG4gIHNlY3JldE5hbWUsXG4gIG9uVG9nZ2xlVXNlRXhpc3RpbmdTZWNyZXQsXG4gIG9uU2VjcmV0TmFtZUNoYW5nZSxcbiAgb25TdWJtaXQsXG59OiBDaGVja0V4aXN0aW5nU2VjcmV0U3RlcFByb3BzKSB7XG4gIGNvbnN0IFtjdXJzb3JPZmZzZXQsIHNldEN1cnNvck9mZnNldF0gPSB1c2VTdGF0ZSgwKVxuICBjb25zdCB0ZXJtaW5hbFNpemUgPSB1c2VUZXJtaW5hbFNpemUoKVxuICBjb25zdCBbdGhlbWVdID0gdXNlVGhlbWUoKVxuXG4gIC8vIFdoZW4gdGhlIHRleHQgaW5wdXQgaXMgdmlzaWJsZSwgb21pdCBjb25maXJtOnllcyBzbyBiYXJlICd5JyBwYXNzZXNcbiAgLy8gdGhyb3VnaCB0byB0aGUgaW5wdXQgaW5zdGVhZCBvZiBzdWJtaXR0aW5nLiBUZXh0SW5wdXQncyBvblN1Ym1pdCBoYW5kbGVzXG4gIC8vIEVudGVyLiBLZWVwIHRoZSBDb25maXJtYXRpb24gY29udGV4dCAobm90IFNldHRpbmdzKSB0byBhdm9pZCBqL2sgYmluZGluZ3MuXG4gIGNvbnN0IGhhbmRsZVByZXZpb3VzID0gdXNlQ2FsbGJhY2soXG4gICAgKCkgPT4gb25Ub2dnbGVVc2VFeGlzdGluZ1NlY3JldCh0cnVlKSxcbiAgICBbb25Ub2dnbGVVc2VFeGlzdGluZ1NlY3JldF0sXG4gIClcbiAgY29uc3QgaGFuZGxlTmV4dCA9IHVzZUNhbGxiYWNrKFxuICAgICgpID0+IG9uVG9nZ2xlVXNlRXhpc3RpbmdTZWNyZXQoZmFsc2UpLFxuICAgIFtvblRvZ2dsZVVzZUV4aXN0aW5nU2VjcmV0XSxcbiAgKVxuICB1c2VLZXliaW5kaW5ncyhcbiAgICB7XG4gICAgICAnY29uZmlybTpwcmV2aW91cyc6IGhhbmRsZVByZXZpb3VzLFxuICAgICAgJ2NvbmZpcm06bmV4dCc6IGhhbmRsZU5leHQsXG4gICAgICAnY29uZmlybTp5ZXMnOiBvblN1Ym1pdCxcbiAgICB9LFxuICAgIHsgY29udGV4dDogJ0NvbmZpcm1hdGlvbicsIGlzQWN0aXZlOiB1c2VFeGlzdGluZ1NlY3JldCB9LFxuICApXG4gIHVzZUtleWJpbmRpbmdzKFxuICAgIHtcbiAgICAgICdjb25maXJtOnByZXZpb3VzJzogaGFuZGxlUHJldmlvdXMsXG4gICAgICAnY29uZmlybTpuZXh0JzogaGFuZGxlTmV4dCxcbiAgICB9LFxuICAgIHsgY29udGV4dDogJ0NvbmZpcm1hdGlvbicsIGlzQWN0aXZlOiAhdXNlRXhpc3RpbmdTZWNyZXQgfSxcbiAgKVxuXG4gIHJldHVybiAoXG4gICAgPD5cbiAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiIGJvcmRlclN0eWxlPVwicm91bmRcIiBwYWRkaW5nWD17MX0+XG4gICAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiIG1hcmdpbkJvdHRvbT17MX0+XG4gICAgICAgICAgPFRleHQgYm9sZD5JbnN0YWxsIEdpdEh1YiBBcHA8L1RleHQ+XG4gICAgICAgICAgPFRleHQgZGltQ29sb3I+U2V0dXAgQVBJIGtleSBzZWNyZXQ8L1RleHQ+XG4gICAgICAgIDwvQm94PlxuICAgICAgICA8Qm94IG1hcmdpbkJvdHRvbT17MX0+XG4gICAgICAgICAgPFRleHQgY29sb3I9XCJ3YXJuaW5nXCI+XG4gICAgICAgICAgICBBTlRIUk9QSUNfQVBJX0tFWSBhbHJlYWR5IGV4aXN0cyBpbiByZXBvc2l0b3J5IHNlY3JldHMhXG4gICAgICAgICAgPC9UZXh0PlxuICAgICAgICA8L0JveD5cbiAgICAgICAgPEJveCBtYXJnaW5Cb3R0b209ezF9PlxuICAgICAgICAgIDxUZXh0PldvdWxkIHlvdSBsaWtlIHRvOjwvVGV4dD5cbiAgICAgICAgPC9Cb3g+XG4gICAgICAgIDxCb3ggbWFyZ2luQm90dG9tPXsxfT5cbiAgICAgICAgICA8VGV4dD5cbiAgICAgICAgICAgIHt1c2VFeGlzdGluZ1NlY3JldCA/IGNvbG9yKCdzdWNjZXNzJywgdGhlbWUpKCc+ICcpIDogJyAgJ31cbiAgICAgICAgICAgIFVzZSB0aGUgZXhpc3RpbmcgQVBJIGtleVxuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgPC9Cb3g+XG4gICAgICAgIDxCb3ggbWFyZ2luQm90dG9tPXsxfT5cbiAgICAgICAgICA8VGV4dD5cbiAgICAgICAgICAgIHshdXNlRXhpc3RpbmdTZWNyZXQgPyBjb2xvcignc3VjY2VzcycsIHRoZW1lKSgnPiAnKSA6ICcgICd9XG4gICAgICAgICAgICBDcmVhdGUgYSBuZXcgc2VjcmV0IHdpdGggYSBkaWZmZXJlbnQgbmFtZVxuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgPC9Cb3g+XG4gICAgICAgIHshdXNlRXhpc3RpbmdTZWNyZXQgJiYgKFxuICAgICAgICAgIDw+XG4gICAgICAgICAgICA8Qm94IG1hcmdpbkJvdHRvbT17MX0+XG4gICAgICAgICAgICAgIDxUZXh0PlxuICAgICAgICAgICAgICAgIEVudGVyIG5ldyBzZWNyZXQgbmFtZSAoYWxwaGFudW1lcmljIHdpdGggdW5kZXJzY29yZXMpOlxuICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICA8L0JveD5cbiAgICAgICAgICAgIDxUZXh0SW5wdXRcbiAgICAgICAgICAgICAgdmFsdWU9e3NlY3JldE5hbWV9XG4gICAgICAgICAgICAgIG9uQ2hhbmdlPXtvblNlY3JldE5hbWVDaGFuZ2V9XG4gICAgICAgICAgICAgIG9uU3VibWl0PXtvblN1Ym1pdH1cbiAgICAgICAgICAgICAgZm9jdXM9e3RydWV9XG4gICAgICAgICAgICAgIHBsYWNlaG9sZGVyPVwiZS5nLiwgQ0xBVURFX0FQSV9LRVlcIlxuICAgICAgICAgICAgICBjb2x1bW5zPXt0ZXJtaW5hbFNpemUuY29sdW1uc31cbiAgICAgICAgICAgICAgY3Vyc29yT2Zmc2V0PXtjdXJzb3JPZmZzZXR9XG4gICAgICAgICAgICAgIG9uQ2hhbmdlQ3Vyc29yT2Zmc2V0PXtzZXRDdXJzb3JPZmZzZXR9XG4gICAgICAgICAgICAgIHNob3dDdXJzb3I9e3RydWV9XG4gICAgICAgICAgICAvPlxuICAgICAgICAgIDwvPlxuICAgICAgICApfVxuICAgICAgPC9Cb3g+XG4gICAgICA8Qm94IG1hcmdpbkxlZnQ9ezN9PlxuICAgICAgICA8VGV4dCBkaW1Db2xvcj7ihpEv4oaTIHRvIHNlbGVjdCDCtyBFbnRlciB0byBjb250aW51ZTwvVGV4dD5cbiAgICAgIDwvQm94PlxuICAgIDwvPlxuICApXG59XG4iXSwibWFwcGluZ3MiOiI7QUFBQSxPQUFPQSxLQUFLLElBQUlDLFdBQVcsRUFBRUMsUUFBUSxRQUFRLE9BQU87QUFDcEQsT0FBT0MsU0FBUyxNQUFNLCtCQUErQjtBQUNyRCxTQUFTQyxlQUFlLFFBQVEsZ0NBQWdDO0FBQ2hFLFNBQVNDLEdBQUcsRUFBRUMsS0FBSyxFQUFFQyxJQUFJLEVBQUVDLFFBQVEsUUFBUSxjQUFjO0FBQ3pELFNBQVNDLGNBQWMsUUFBUSxvQ0FBb0M7QUFFbkUsVUFBVUMsNEJBQTRCLENBQUM7RUFDckNDLGlCQUFpQixFQUFFLE9BQU87RUFDMUJDLFVBQVUsRUFBRSxNQUFNO0VBQ2xCQyx5QkFBeUIsRUFBRSxDQUFDQyxXQUFXLEVBQUUsT0FBTyxFQUFFLEdBQUcsSUFBSTtFQUN6REMsa0JBQWtCLEVBQUUsQ0FBQ0MsS0FBSyxFQUFFLE1BQU0sRUFBRSxHQUFHLElBQUk7RUFDM0NDLFFBQVEsRUFBRSxHQUFHLEdBQUcsSUFBSTtBQUN0QjtB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 \ No newline at end of file diff --git a/src/commands/install-github-app/CheckGitHubStep.tsx b/src/commands/install-github-app/CheckGitHubStep.tsx new file mode 100644 index 0000000..5bf1d8f --- /dev/null +++ b/src/commands/install-github-app/CheckGitHubStep.tsx @@ -0,0 +1,15 @@ +import { c as _c } from "react/compiler-runtime"; +import React from 'react'; +import { Text } from '../../ink.js'; +export function CheckGitHubStep() { + const $ = _c(1); + let t0; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t0 = Checking GitHub CLI installation…; + $[0] = t0; + } else { + t0 = $[0]; + } + return t0; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIlRleHQiLCJDaGVja0dpdEh1YlN0ZXAiLCIkIiwiX2MiLCJ0MCIsIlN5bWJvbCIsImZvciJdLCJzb3VyY2VzIjpbIkNoZWNrR2l0SHViU3RlcC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgVGV4dCB9IGZyb20gJy4uLy4uL2luay5qcydcblxuZXhwb3J0IGZ1bmN0aW9uIENoZWNrR2l0SHViU3RlcCgpIHtcbiAgcmV0dXJuIDxUZXh0PkNoZWNraW5nIEdpdEh1YiBDTEkgaW5zdGFsbGF0aW9u4oCmPC9UZXh0PlxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsT0FBT0EsS0FBSyxNQUFNLE9BQU87QUFDekIsU0FBU0MsSUFBSSxRQUFRLGNBQWM7QUFFbkMsT0FBTyxTQUFBQyxnQkFBQTtFQUFBLE1BQUFDLENBQUEsR0FBQUMsRUFBQTtFQUFBLElBQUFDLEVBQUE7RUFBQSxJQUFBRixDQUFBLFFBQUFHLE1BQUEsQ0FBQUMsR0FBQTtJQUNFRixFQUFBLElBQUMsSUFBSSxDQUFDLGlDQUFpQyxFQUF0QyxJQUFJLENBQXlDO0lBQUFGLENBQUEsTUFBQUUsRUFBQTtFQUFBO0lBQUFBLEVBQUEsR0FBQUYsQ0FBQTtFQUFBO0VBQUEsT0FBOUNFLEVBQThDO0FBQUEiLCJpZ25vcmVMaXN0IjpbXX0= \ No newline at end of file diff --git a/src/commands/install-github-app/ChooseRepoStep.tsx b/src/commands/install-github-app/ChooseRepoStep.tsx new file mode 100644 index 0000000..04d1a6b --- /dev/null +++ b/src/commands/install-github-app/ChooseRepoStep.tsx @@ -0,0 +1,211 @@ +import { c as _c } from "react/compiler-runtime"; +import React, { useCallback, useState } from 'react'; +import TextInput from '../../components/TextInput.js'; +import { useTerminalSize } from '../../hooks/useTerminalSize.js'; +import { Box, Text } from '../../ink.js'; +import { useKeybindings } from '../../keybindings/useKeybinding.js'; +interface ChooseRepoStepProps { + currentRepo: string | null; + useCurrentRepo: boolean; + repoUrl: string; + onRepoUrlChange: (value: string) => void; + onToggleUseCurrentRepo: (useCurrentRepo: boolean) => void; + onSubmit: () => void; +} +export function ChooseRepoStep(t0) { + const $ = _c(49); + const { + currentRepo, + useCurrentRepo, + repoUrl, + onRepoUrlChange, + onSubmit, + onToggleUseCurrentRepo + } = t0; + const [cursorOffset, setCursorOffset] = useState(0); + const [showEmptyError, setShowEmptyError] = useState(false); + const terminalSize = useTerminalSize(); + const textInputColumns = terminalSize.columns; + let t1; + if ($[0] !== currentRepo || $[1] !== onSubmit || $[2] !== repoUrl || $[3] !== useCurrentRepo) { + t1 = () => { + const repoName = useCurrentRepo ? currentRepo : repoUrl; + if (!repoName?.trim()) { + setShowEmptyError(true); + return; + } + onSubmit(); + }; + $[0] = currentRepo; + $[1] = onSubmit; + $[2] = repoUrl; + $[3] = useCurrentRepo; + $[4] = t1; + } else { + t1 = $[4]; + } + const handleSubmit = t1; + const isTextInputVisible = !useCurrentRepo || !currentRepo; + let t2; + if ($[5] !== onToggleUseCurrentRepo) { + t2 = () => { + onToggleUseCurrentRepo(true); + setShowEmptyError(false); + }; + $[5] = onToggleUseCurrentRepo; + $[6] = t2; + } else { + t2 = $[6]; + } + const handlePrevious = t2; + let t3; + if ($[7] !== onToggleUseCurrentRepo) { + t3 = () => { + onToggleUseCurrentRepo(false); + setShowEmptyError(false); + }; + $[7] = onToggleUseCurrentRepo; + $[8] = t3; + } else { + t3 = $[8]; + } + const handleNext = t3; + let t4; + if ($[9] !== handleNext || $[10] !== handlePrevious || $[11] !== handleSubmit) { + t4 = { + "confirm:previous": handlePrevious, + "confirm:next": handleNext, + "confirm:yes": handleSubmit + }; + $[9] = handleNext; + $[10] = handlePrevious; + $[11] = handleSubmit; + $[12] = t4; + } else { + t4 = $[12]; + } + const t5 = !isTextInputVisible; + let t6; + if ($[13] !== t5) { + t6 = { + context: "Confirmation", + isActive: t5 + }; + $[13] = t5; + $[14] = t6; + } else { + t6 = $[14]; + } + useKeybindings(t4, t6); + let t7; + if ($[15] !== handleNext || $[16] !== handlePrevious) { + t7 = { + "confirm:previous": handlePrevious, + "confirm:next": handleNext + }; + $[15] = handleNext; + $[16] = handlePrevious; + $[17] = t7; + } else { + t7 = $[17]; + } + let t8; + if ($[18] !== isTextInputVisible) { + t8 = { + context: "Confirmation", + isActive: isTextInputVisible + }; + $[18] = isTextInputVisible; + $[19] = t8; + } else { + t8 = $[19]; + } + useKeybindings(t7, t8); + let t9; + if ($[20] === Symbol.for("react.memo_cache_sentinel")) { + t9 = Install GitHub AppSelect GitHub repository; + $[20] = t9; + } else { + t9 = $[20]; + } + let t10; + if ($[21] !== currentRepo || $[22] !== useCurrentRepo) { + t10 = currentRepo && {useCurrentRepo ? "> " : " "}Use current repository: {currentRepo}; + $[21] = currentRepo; + $[22] = useCurrentRepo; + $[23] = t10; + } else { + t10 = $[23]; + } + const t11 = !useCurrentRepo || !currentRepo; + const t12 = !useCurrentRepo || !currentRepo ? "permission" : undefined; + const t13 = !useCurrentRepo || !currentRepo ? "> " : " "; + const t14 = currentRepo ? "Enter a different repository" : "Enter repository"; + let t15; + if ($[24] !== t11 || $[25] !== t12 || $[26] !== t13 || $[27] !== t14) { + t15 = {t13}{t14}; + $[24] = t11; + $[25] = t12; + $[26] = t13; + $[27] = t14; + $[28] = t15; + } else { + t15 = $[28]; + } + let t16; + if ($[29] !== currentRepo || $[30] !== cursorOffset || $[31] !== handleSubmit || $[32] !== onRepoUrlChange || $[33] !== repoUrl || $[34] !== textInputColumns || $[35] !== useCurrentRepo) { + t16 = (!useCurrentRepo || !currentRepo) && { + onRepoUrlChange(value); + setShowEmptyError(false); + }} onSubmit={handleSubmit} focus={true} placeholder={"Enter a repo as owner/repo or https://github.com/owner/repo\u2026"} columns={textInputColumns} cursorOffset={cursorOffset} onChangeCursorOffset={setCursorOffset} showCursor={true} />; + $[29] = currentRepo; + $[30] = cursorOffset; + $[31] = handleSubmit; + $[32] = onRepoUrlChange; + $[33] = repoUrl; + $[34] = textInputColumns; + $[35] = useCurrentRepo; + $[36] = t16; + } else { + t16 = $[36]; + } + let t17; + if ($[37] !== t10 || $[38] !== t15 || $[39] !== t16) { + t17 = {t9}{t10}{t15}{t16}; + $[37] = t10; + $[38] = t15; + $[39] = t16; + $[40] = t17; + } else { + t17 = $[40]; + } + let t18; + if ($[41] !== showEmptyError) { + t18 = showEmptyError && Please enter a repository name to continue; + $[41] = showEmptyError; + $[42] = t18; + } else { + t18 = $[42]; + } + const t19 = currentRepo ? "\u2191/\u2193 to select \xB7 " : ""; + let t20; + if ($[43] !== t19) { + t20 = {t19}Enter to continue; + $[43] = t19; + $[44] = t20; + } else { + t20 = $[44]; + } + let t21; + if ($[45] !== t17 || $[46] !== t18 || $[47] !== t20) { + t21 = <>{t17}{t18}{t20}; + $[45] = t17; + $[46] = t18; + $[47] = t20; + $[48] = t21; + } else { + t21 = $[48]; + } + return t21; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsInVzZUNhbGxiYWNrIiwidXNlU3RhdGUiLCJUZXh0SW5wdXQiLCJ1c2VUZXJtaW5hbFNpemUiLCJCb3giLCJUZXh0IiwidXNlS2V5YmluZGluZ3MiLCJDaG9vc2VSZXBvU3RlcFByb3BzIiwiY3VycmVudFJlcG8iLCJ1c2VDdXJyZW50UmVwbyIsInJlcG9VcmwiLCJvblJlcG9VcmxDaGFuZ2UiLCJ2YWx1ZSIsIm9uVG9nZ2xlVXNlQ3VycmVudFJlcG8iLCJvblN1Ym1pdCIsIkNob29zZVJlcG9TdGVwIiwidDAiLCIkIiwiX2MiLCJjdXJzb3JPZmZzZXQiLCJzZXRDdXJzb3JPZmZzZXQiLCJzaG93RW1wdHlFcnJvciIsInNldFNob3dFbXB0eUVycm9yIiwidGVybWluYWxTaXplIiwidGV4dElucHV0Q29sdW1ucyIsImNvbHVtbnMiLCJ0MSIsInJlcG9OYW1lIiwidHJpbSIsImhhbmRsZVN1Ym1pdCIsImlzVGV4dElucHV0VmlzaWJsZSIsInQyIiwiaGFuZGxlUHJldmlvdXMiLCJ0MyIsImhhbmRsZU5leHQiLCJ0NCIsInQ1IiwidDYiLCJjb250ZXh0IiwiaXNBY3RpdmUiLCJ0NyIsInQ4IiwidDkiLCJTeW1ib2wiLCJmb3IiLCJ0MTAiLCJ1bmRlZmluZWQiLCJ0MTEiLCJ0MTIiLCJ0MTMiLCJ0MTQiLCJ0MTUiLCJ0MTYiLCJ0MTciLCJ0MTgiLCJ0MTkiLCJ0MjAiLCJ0MjEiXSwic291cmNlcyI6WyJDaG9vc2VSZXBvU3RlcC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFJlYWN0LCB7IHVzZUNhbGxiYWNrLCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IFRleHRJbnB1dCBmcm9tICcuLi8uLi9jb21wb25lbnRzL1RleHRJbnB1dC5qcydcbmltcG9ydCB7IHVzZVRlcm1pbmFsU2l6ZSB9IGZyb20gJy4uLy4uL2hvb2tzL3VzZVRlcm1pbmFsU2l6ZS5qcydcbmltcG9ydCB7IEJveCwgVGV4dCB9IGZyb20gJy4uLy4uL2luay5qcydcbmltcG9ydCB7IHVzZUtleWJpbmRpbmdzIH0gZnJvbSAnLi4vLi4va2V5YmluZGluZ3MvdXNlS2V5YmluZGluZy5qcydcblxuaW50ZXJmYWNlIENob29zZVJlcG9TdGVwUHJvcHMge1xuICBjdXJyZW50UmVwbzogc3RyaW5nIHwgbnVsbFxuICB1c2VDdXJyZW50UmVwbzogYm9vbGVhblxuICByZXBvVXJsOiBzdHJpbmdcbiAgb25SZXBvVXJsQ2hhbmdlOiAodmFsdWU6IHN0cmluZykgPT4gdm9pZFxuICBvblRvZ2dsZVVzZUN1cnJlbnRSZXBvOiAodXNlQ3VycmVudFJlcG86IGJvb2xlYW4pID0+IHZvaWRcbiAgb25TdWJtaXQ6ICgpID0+IHZvaWRcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIENob29zZVJlcG9TdGVwKHtcbiAgY3VycmVudFJlcG8sXG4gIHVzZUN1cnJlbnRSZXBvLFxuICByZXBvVXJsLFxuICBvblJlcG9VcmxDaGFuZ2UsXG4gIG9uU3VibWl0LFxuICBvblRvZ2dsZVVzZUN1cnJlbnRSZXBvLFxufTogQ2hvb3NlUmVwb1N0ZXBQcm9wcykge1xuICBjb25zdCBbY3Vyc29yT2Zmc2V0LCBzZXRDdXJzb3JPZmZzZXRdID0gdXNlU3RhdGUoMClcbiAgY29uc3QgW3Nob3dFbXB0eUVycm9yLCBzZXRTaG93RW1wdHlFcnJvcl0gPSB1c2VTdGF0ZShmYWxzZSlcbiAgY29uc3QgdGVybWluYWxTaXplID0gdXNlVGVybWluYWxTaXplKClcbiAgY29uc3QgdGV4dElucHV0Q29sdW1ucyA9IHRlcm1pbmFsU2l6ZS5jb2x1bW5zXG5cbiAgY29uc3QgaGFuZGxlU3VibWl0ID0gdXNlQ2FsbGJhY2soKCkgPT4ge1xuICAgIGNvbnN0IHJlcG9OYW1lID0gdXNlQ3VycmVudFJlcG8gPyBjdXJyZW50UmVwbyA6IHJlcG9VcmxcbiAgICBpZiAoIXJlcG9OYW1lPy50cmltKCkpIHtcbiAgICAgIHNldFNob3dFbXB0eUVycm9yKHRydWUpXG4gICAgICByZXR1cm5cbiAgICB9XG4gICAgb25TdWJtaXQoKVxuICB9LCBbdXNlQ3VycmVudFJlcG8sIGN1cnJlbnRSZXBvLCByZXBvVXJsLCBvblN1Ym1pdF0pXG5cbiAgLy8gV2hlbiB0aGUgdGV4dCBpbnB1dCBpcyB2aXNpYmxlLCBvbWl0IGNvbmZpcm06eWVzIHNvIGJhcmUgJ3knIHBhc3Nlc1xuICAvLyB0aHJvdWdoIHRvIHRoZSBpbnB1dCBpbnN0ZWFkIG9mIHN1Ym1pdHRpbmcuIFRleHRJbnB1dCdzIG9uU3VibWl0IGhhbmRsZXNcbiAgLy8gRW50ZXIuIEtlZXAgdGhlIENvbmZpcm1hdGlvbiBjb250ZXh0IChub3QgU2V0dGluZ3MpIHRvIGF2b2lkIGovayBiaW5kaW5ncy5cbiAgY29uc3QgaXNUZXh0SW5wdXRWaXNpYmxlID0gIXVzZUN1cnJlbnRSZXBvIHx8ICFjdXJyZW50UmVwb1xuICBjb25zdCBoYW5kbGVQcmV2aW91cyA9IHVzZUNhbGxiYWNrKCgpID0+IHtcbiAgICBvblRvZ2dsZVVzZUN1cnJlbnRSZXBvKHRydWUpXG4gICAgc2V0U2hvd0VtcHR5RXJyb3IoZmFsc2UpXG4gIH0sIFtvblRvZ2dsZVVzZUN1cnJlbnRSZXBvXSlcbiAgY29uc3QgaGFuZGxlTmV4dCA9IHVzZUNhbGxiYWNrKCgpID0+IHtcbiAgICBvblRvZ2dsZVVzZUN1cnJlbnRSZXBvKGZhbHNlKVxuICAgIHNldFNob3dFbXB0eUVycm9yKGZhbHNlKVxuICB9LCBbb25Ub2dnbGVVc2VDdXJyZW50UmVwb10pXG5cbiAgdXNlS2V5YmluZGluZ3MoXG4gICAge1xuICAgICAgJ2NvbmZpcm06cHJldmlvdXMnOiBoYW5kbGVQcmV2aW91cyxcbiAgICAgICdjb25maXJtOm5leHQnOiBoYW5kbGVOZXh0LFxuICAgICAgJ2NvbmZpcm06eWVzJzogaGFuZGxlU3VibWl0LFxuICAgIH0sXG4gICAgeyBjb250ZXh0OiAnQ29uZmlybWF0aW9uJywgaXNBY3RpdmU6ICFpc1RleHRJbnB1dFZpc2libGUgfSxcbiAgKVxuICB1c2VLZXliaW5kaW5ncyhcbiAgICB7XG4gICAgICAnY29uZmlybTpwcmV2aW91cyc6IGhhbmRsZVByZXZpb3VzLFxuICAgICAgJ2NvbmZpcm06bmV4dCc6IGhhbmRsZU5leHQsXG4gICAgfSxcbiAgICB7IGNvbnRleHQ6ICdDb25maXJtYXRpb24nLCBpc0FjdGl2ZTogaXNUZXh0SW5wdXRWaXNpYmxlIH0sXG4gIClcblxuICByZXR1cm4gKFxuICAgIDw+XG4gICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIiBib3JkZXJTdHlsZT1cInJvdW5kXCIgcGFkZGluZ1g9ezF9PlxuICAgICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIiBtYXJnaW5Cb3R0b209ezF9PlxuICAgICAgICAgIDxUZXh0IGJvbGQ+SW5zdGFsbCBHaXRIdWIgQXBwPC9UZXh0PlxuICAgICAgICAgIDxUZXh0IGRpbUNvbG9yPlNlbGVjdCBHaXRIdWIgcmVwb3NpdG9yeTwvVGV4dD5cbiAgICAgICAgPC9Cb3g+XG4gICAgICAgIHtjdXJyZW50UmVwbyAmJiAoXG4gICAgICAgICAgPEJveCBtYXJnaW5Cb3R0b209ezF9PlxuICAgICAgICAgICAgPFRleHRcbiAgICAgICAgICAgICAgYm9sZD17dXNlQ3VycmVudFJlcG99XG4gICAgICAgICAgICAgIGNvbG9yPXt1c2VDdXJyZW50UmVwbyA/ICdwZXJtaXNzaW9uJyA6IHVuZGVmaW5lZH1cbiAgICAgICAgICAgID5cbiAgICAgICAgICAgICAge3VzZUN1cnJlbnRSZXBvID8gJz4gJyA6ICcgICd9XG4gICAgICAgICAgICAgIFVzZSBjdXJyZW50IHJlcG9zaXRvcnk6IHtjdXJyZW50UmVwb31cbiAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICA8L0JveD5cbiAgICAgICAgKX1cbiAgICAgICAgPEJveCBtYXJnaW5Cb3R0b209ezF9PlxuICAgICAgICAgIDxUZXh0XG4gICAgICAgICAgICBib2xkPXshdXNlQ3VycmVudFJlcG8gfHwgIWN1cnJlbnRSZXBvfVxuICAgICAgICAgICAgY29sb3I9eyF1c2VDdXJyZW50UmVwbyB8fCAhY3VycmVudFJlcG8gPyAncGVybWlzc2lvbicgOiB1bmRlZmluZWR9XG4gICAgICAgICAgPlxuICAgICAgICAgICAgeyF1c2VDdXJyZW50UmVwbyB8fCAhY3VycmVudFJlcG8gPyAnPiAnIDogJyAgJ31cbiAgICAgICAgICAgIHtjdXJyZW50UmVwbyA/ICdFbnRlciBhIGRpZmZlcmVudCByZXBvc2l0b3J5JyA6ICdFbnRlciByZXBvc2l0b3J5J31cbiAgICAgICAgICA8L1RleHQ+XG4gICAgICAgIDwvQm94PlxuICAgICAgICB7KCF1c2VDdXJyZW50UmVwbyB8fCAhY3VycmVudFJlcG8pICYmIChcbiAgICAgICAgICA8Qm94IG1hcmdpbkxlZnQ9ezJ9IG1hcmdpbkJvdHRvbT17MX0+XG4gICAgICAgICAgICA8VGV4dElucHV0XG4gICAgICAgICAgICAgIHZhbHVlPXtyZXBvVXJsfVxuICAgICAgICAgICAgICBvbkNoYW5nZT17dmFsdWUgPT4ge1xuICAgICAgICAgICAgICAgIG9uUmVwb1VybENoYW5nZSh2YWx1ZSlcbiAgICAgICAgICAgICAgICBzZXRTaG93RW1wdHlFcnJvcihmYWxzZSlcbiAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgb25TdWJtaXQ9e2hhbmRsZVN1Ym1pdH1cbiAgICAgICAgICAgICAgZm9jdXM9e3RydWV9XG4gICAgICAgICAgICAgIHBsYWNlaG9sZGVyPVwiRW50ZXIgYSByZXBvIGFzIG93bmVyL3JlcG8gb3IgaHR0cHM6Ly9naXRodWIuY29tL293bmVyL3JlcG/igKZcIlxuICAgICAgICAgICAgICBjb2x1bW5zPXt0ZXh0SW5wdXRDb2x1bW5zfVxuICAgICAgICAgICAgICBjdXJzb3JPZmZzZXQ9e2N1cnNvck9mZnNldH1cbiAgICAgICAgICAgICAgb25DaGFuZ2VDdXJzb3JPZmZzZXQ9e3NldEN1cnNvck9mZnNldH1cbiAgICAgICAgICAgICAgc2hvd0N1cnNvcj17dHJ1ZX1cbiAgICAgICAgICAgIC8+XG4gICAgICAgICAgPC9Cb3g+XG4gICAgICAgICl9XG4gICAgICA8L0JveD5cbiAgICAgIHtzaG93RW1wdHlFcnJvciAmJiAoXG4gICAgICAgIDxCb3ggbWFyZ2luTGVmdD17M30gbWFyZ2luQm90dG9tPXsxfT5cbiAgICAgICAgICA8VGV4dCBjb2xvcj1cImVycm9yXCI+UGxlYXNlIGVudGVyIGEgcmVwb3NpdG9yeSBuYW1lIHRvIGNvbnRpbnVlPC9UZXh0PlxuICAgICAgICA8L0JveD5cbiAgICAgICl9XG4gICAgICA8Qm94IG1hcmdpbkxlZnQ9ezN9PlxuICAgICAgICA8VGV4dCBkaW1Db2xvcj5cbiAgICAgICAgICB7Y3VycmVudFJlcG8gPyAn4oaRL+KGkyB0byBzZWxlY3QgwrcgJyA6ICcnfUVudGVyIHRvIGNvbnRpbnVlXG4gICAgICAgIDwvVGV4dD5cbiAgICAgIDwvQm94PlxuICAgIDwvPlxuICApXG59XG4iXSwibWFwcGluZ3MiOiI7QUFBQSxPQUFPQSxLQUFLLElBQUlDLFdBQVcsRUFBRUMsUUFBUSxRQUFRLE9BQU87QUFDcEQsT0FBT0MsU0FBUyxNQUFNLCtCQUErQjtBQUNyRCxTQUFTQyxlQUFlLFFBQVEsZ0NBQWdDO0FBQ2hFLFNBQVNDLEdBQUcsRUFBRUMsSUFBSSxRQUFRLGNBQWM7QUFDeEMsU0FBU0MsY0FBYyxRQUFRLG9DQUFvQztBQUVuRSxVQUFVQyxtQkFBbUIsQ0FBQztFQUM1QkMsV0FBVyxFQUFFLE1BQU0sR0FBRyxJQUFJO0VBQzFCQyxjQUFjLEVBQUUsT0FBTztFQUN2QkMsT0FBTyxFQUFFLE1BQU07RUFDZkMsZUFBZSxFQUFFLENBQUNDLEtBQUssRUFBRSxNQUFNLEVBQUUsR0FBRyxJQUFJO0VBQ3hDQyxzQkFBc0IsRUFBRSxDQUFDSixjQUFjLEVBQUUsT0FBTyxFQUFFLEdBQUcsSUFBSTtFQUN6REssUUFBUSxFQUFFLEdBQUcsR0FBRyxJQUFJO0FBQ3RCO0FBRUEsT0FBTyxTQUFBQyxlQUFBQyxFQUFBO0VBQUEsTUFBQUMsQ0FBQSxHQUFBQyxFQUFBO0VBQXdCO0lBQUFWLFdBQUE7SUFBQUMsY0FBQTtJQUFBQyxPQUFBO0lBQUFD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 \ No newline at end of file diff --git a/src/commands/install-github-app/CreatingStep.tsx b/src/commands/install-github-app/CreatingStep.tsx new file mode 100644 index 0000000..ef59787 --- /dev/null +++ b/src/commands/install-github-app/CreatingStep.tsx @@ -0,0 +1,65 @@ +import { c as _c } from "react/compiler-runtime"; +import React from 'react'; +import { Box, Text } from '../../ink.js'; +import type { Workflow } from './types.js'; +interface CreatingStepProps { + currentWorkflowInstallStep: number; + secretExists: boolean; + useExistingSecret: boolean; + secretName: string; + skipWorkflow?: boolean; + selectedWorkflows: Workflow[]; +} +export function CreatingStep(t0) { + const $ = _c(10); + const { + currentWorkflowInstallStep, + secretExists, + useExistingSecret, + secretName, + skipWorkflow: t1, + selectedWorkflows + } = t0; + const skipWorkflow = t1 === undefined ? false : t1; + let t2; + if ($[0] !== secretExists || $[1] !== secretName || $[2] !== selectedWorkflows || $[3] !== skipWorkflow || $[4] !== useExistingSecret) { + t2 = skipWorkflow ? ["Getting repository information", secretExists && useExistingSecret ? "Using existing API key secret" : `Setting up ${secretName} secret`] : ["Getting repository information", "Creating branch", selectedWorkflows.length > 1 ? "Creating workflow files" : "Creating workflow file", secretExists && useExistingSecret ? "Using existing API key secret" : `Setting up ${secretName} secret`, "Opening pull request page"]; + $[0] = secretExists; + $[1] = secretName; + $[2] = selectedWorkflows; + $[3] = skipWorkflow; + $[4] = useExistingSecret; + $[5] = t2; + } else { + t2 = $[5]; + } + const progressSteps = t2; + let t3; + if ($[6] === Symbol.for("react.memo_cache_sentinel")) { + t3 = Install GitHub AppCreate GitHub Actions workflow; + $[6] = t3; + } else { + t3 = $[6]; + } + let t4; + if ($[7] !== currentWorkflowInstallStep || $[8] !== progressSteps) { + t4 = <>{t3}{progressSteps.map((stepText, index) => { + let status = "pending"; + if (index < currentWorkflowInstallStep) { + status = "completed"; + } else { + if (index === currentWorkflowInstallStep) { + status = "in-progress"; + } + } + return {status === "completed" ? "\u2713 " : ""}{stepText}{status === "in-progress" ? "\u2026" : ""}; + })}; + $[7] = currentWorkflowInstallStep; + $[8] = progressSteps; + $[9] = t4; + } else { + t4 = $[9]; + } + return t4; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkJveCIsIlRleHQiLCJXb3JrZmxvdyIsIkNyZWF0aW5nU3RlcFByb3BzIiwiY3VycmVudFdvcmtmbG93SW5zdGFsbFN0ZXAiLCJzZWNyZXRFeGlzdHMiLCJ1c2VFeGlzdGluZ1NlY3JldCIsInNlY3JldE5hbWUiLCJza2lwV29ya2Zsb3ciLCJzZWxlY3RlZFdvcmtmbG93cyIsIkNyZWF0aW5nU3RlcCIsInQwIiwiJCIsIl9jIiwidDEiLCJ1bmRlZmluZWQiLCJ0MiIsImxlbmd0aCIsInByb2dyZXNzU3RlcHMiLCJ0MyIsIlN5bWJvbCIsImZvciIsInQ0IiwibWFwIiwic3RlcFRleHQiLCJpbmRleCIsInN0YXR1cyJdLCJzb3VyY2VzIjpbIkNyZWF0aW5nU3RlcC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgQm94LCBUZXh0IH0gZnJvbSAnLi4vLi4vaW5rLmpzJ1xuaW1wb3J0IHR5cGUgeyBXb3JrZmxvdyB9IGZyb20gJy4vdHlwZXMuanMnXG5cbmludGVyZmFjZSBDcmVhdGluZ1N0ZXBQcm9wcyB7XG4gIGN1cnJlbnRXb3JrZmxvd0luc3RhbGxTdGVwOiBudW1iZXJcbiAgc2VjcmV0RXhpc3RzOiBib29sZWFuXG4gIHVzZUV4aXN0aW5nU2VjcmV0OiBib29sZWFuXG4gIHNlY3JldE5hbWU6IHN0cmluZ1xuICBza2lwV29ya2Zsb3c/OiBib29sZWFuXG4gIHNlbGVjdGVkV29ya2Zsb3dzOiBXb3JrZmxvd1tdXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBDcmVhdGluZ1N0ZXAoe1xuICBjdXJyZW50V29ya2Zsb3dJbnN0YWxsU3RlcCxcbiAgc2VjcmV0RXhpc3RzLFxuICB1c2VFeGlzdGluZ1NlY3JldCxcbiAgc2VjcmV0TmFtZSxcbiAgc2tpcFdvcmtmbG93ID0gZmFsc2UsXG4gIHNlbGVjdGVkV29ya2Zsb3dzLFxufTogQ3JlYXRpbmdTdGVwUHJvcHMpIHtcbiAgY29uc3QgcHJvZ3Jlc3NTdGVwcyA9IHNraXBXb3JrZmxvd1xuICAgID8gW1xuICAgICAgICAnR2V0dGluZyByZXBvc2l0b3J5IGluZm9ybWF0aW9uJyxcbiAgICAgICAgc2VjcmV0RXhpc3RzICYmIHVzZUV4aXN0aW5nU2VjcmV0XG4gICAgICAgICAgPyAnVXNpbmcgZXhpc3RpbmcgQVBJIGtleSBzZWNyZXQnXG4gICAgICAgICAgOiBgU2V0dGluZyB1cCAke3NlY3JldE5hbWV9IHNlY3JldGAsXG4gICAgICBdXG4gICAgOiBbXG4gICAgICAgICdHZXR0aW5nIHJlcG9zaXRvcnkgaW5mb3JtYXRpb24nLFxuICAgICAgICAnQ3JlYXRpbmcgYnJhbmNoJyxcbiAgICAgICAgc2VsZWN0ZWRXb3JrZmxvd3MubGVuZ3RoID4gMVxuICAgICAgICAgID8gJ0NyZWF0aW5nIHdvcmtmbG93IGZpbGVzJ1xuICAgICAgICAgIDogJ0NyZWF0aW5nIHdvcmtmbG93IGZpbGUnLFxuICAgICAgICBzZWNyZXRFeGlzdHMgJiYgdXNlRXhpc3RpbmdTZWNyZXRcbiAgICAgICAgICA/ICdVc2luZyBleGlzdGluZyBBUEkga2V5IHNlY3JldCdcbiAgICAgICAgICA6IGBTZXR0aW5nIHVwICR7c2VjcmV0TmFtZX0gc2VjcmV0YCxcbiAgICAgICAgJ09wZW5pbmcgcHVsbCByZXF1ZXN0IHBhZ2UnLFxuICAgICAgXVxuXG4gIHJldHVybiAoXG4gICAgPD5cbiAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiIGJvcmRlclN0eWxlPVwicm91bmRcIiBwYWRkaW5nWD17MX0+XG4gICAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiIG1hcmdpbkJvdHRvbT17MX0+XG4gICAgICAgICAgPFRleHQgYm9sZD5JbnN0YWxsIEdpdEh1YiBBcHA8L1RleHQ+XG4gICAgICAgICAgPFRleHQgZGltQ29sb3I+Q3JlYXRlIEdpdEh1YiBBY3Rpb25zIHdvcmtmbG93PC9UZXh0PlxuICAgICAgICA8L0JveD5cbiAgICAgICAge3Byb2dyZXNzU3RlcHMubWFwKChzdGVwVGV4dCwgaW5kZXgpID0+IHtcbiAgICAgICAgICBsZXQgc3RhdHVzOiAnY29tcGxldGVkJyB8ICdpbi1wcm9ncmVzcycgfCAncGVuZGluZycgPSAncGVuZGluZydcblxuICAgICAgICAgIGlmIChpbmRleCA8IGN1cnJlbnRXb3JrZmxvd0luc3RhbGxTdGVwKSB7XG4gICAgICAgICAgICBzdGF0dXMgPSAnY29tcGxldGVkJ1xuICAgICAgICAgIH0gZWxzZSBpZiAoaW5kZXggPT09IGN1cnJlbnRXb3JrZmxvd0luc3RhbGxTdGVwKSB7XG4gICAgICAgICAgICBzdGF0dXMgPSAnaW4tcHJvZ3Jlc3MnXG4gICAgICAgICAgfVxuXG4gICAgICAgICAgcmV0dXJuIChcbiAgICAgICAgICAgIDxCb3gga2V5PXtpbmRleH0+XG4gICAgICAgICAgICAgIDxUZXh0XG4gICAgICAgICAgICAgICAgY29sb3I9e1xuICAgICAgICAgICAgICAgICAgc3RhdHVzID09PSAnY29tcGxldGVkJ1xuICAgICAgICAgICAgICAgICAgICA/ICdzdWNjZXNzJ1xuICAgICAgICAgICAgICAgICAgICA6IHN0YXR1cyA9PT0gJ2luLXByb2dyZXNzJ1xuICAgICAgICAgICAgICAgICAgICAgID8gJ3dhcm5pbmcnXG4gICAgICAgICAgICAgICAgICAgICAgOiB1bmRlZmluZWRcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICB7c3RhdHVzID09PSAnY29tcGxldGVkJyA/ICfinJMgJyA6ICcnfVxuICAgICAgICAgICAgICAgIHtzdGVwVGV4dH1cbiAgICAgICAgICAgICAgICB7c3RhdHVzID09PSAnaW4tcHJvZ3Jlc3MnID8gJ+KApicgOiAnJ31cbiAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgPC9Cb3g+XG4gICAgICAgICAgKVxuICAgICAgICB9KX1cbiAgICAgIDwvQm94PlxuICAgIDwvPlxuICApXG59XG4iXSwibWFwcGluZ3MiOiI7QUFBQSxPQUFP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 \ No newline at end of file diff --git a/src/commands/install-github-app/ErrorStep.tsx b/src/commands/install-github-app/ErrorStep.tsx new file mode 100644 index 0000000..6fad7af --- /dev/null +++ b/src/commands/install-github-app/ErrorStep.tsx @@ -0,0 +1,85 @@ +import { c as _c } from "react/compiler-runtime"; +import React from 'react'; +import { GITHUB_ACTION_SETUP_DOCS_URL } from '../../constants/github-app.js'; +import { Box, Text } from '../../ink.js'; +interface ErrorStepProps { + error: string | undefined; + errorReason?: string; + errorInstructions?: string[]; +} +export function ErrorStep(t0) { + const $ = _c(15); + const { + error, + errorReason, + errorInstructions + } = t0; + let t1; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t1 = Install GitHub App; + $[0] = t1; + } else { + t1 = $[0]; + } + let t2; + if ($[1] !== error) { + t2 = Error: {error}; + $[1] = error; + $[2] = t2; + } else { + t2 = $[2]; + } + let t3; + if ($[3] !== errorReason) { + t3 = errorReason && Reason: {errorReason}; + $[3] = errorReason; + $[4] = t3; + } else { + t3 = $[4]; + } + let t4; + if ($[5] !== errorInstructions) { + t4 = errorInstructions && errorInstructions.length > 0 && How to fix:{errorInstructions.map(_temp)}; + $[5] = errorInstructions; + $[6] = t4; + } else { + t4 = $[6]; + } + let t5; + if ($[7] === Symbol.for("react.memo_cache_sentinel")) { + t5 = For manual setup instructions, see:{" "}{GITHUB_ACTION_SETUP_DOCS_URL}; + $[7] = t5; + } else { + t5 = $[7]; + } + let t6; + if ($[8] !== t2 || $[9] !== t3 || $[10] !== t4) { + t6 = {t1}{t2}{t3}{t4}{t5}; + $[8] = t2; + $[9] = t3; + $[10] = t4; + $[11] = t6; + } else { + t6 = $[11]; + } + let t7; + if ($[12] === Symbol.for("react.memo_cache_sentinel")) { + t7 = Press any key to exit; + $[12] = t7; + } else { + t7 = $[12]; + } + let t8; + if ($[13] !== t6) { + t8 = <>{t6}{t7}; + $[13] = t6; + $[14] = t8; + } else { + t8 = $[14]; + } + return t8; +} +function _temp(instruction, index) { + return {instruction}; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkdJVEhVQl9BQ1RJT05fU0VUVVBfRE9DU19VUkwiLCJCb3giLCJUZXh0IiwiRXJyb3JTdGVwUHJvcHMiLCJlcnJvciIsImVycm9yUmVhc29uIiwiZXJyb3JJbnN0cnVjdGlvbnMiLCJFcnJvclN0ZXAiLCJ0MCIsIiQiLCJfYyIsInQxIiwiU3ltYm9sIiwiZm9yIiwidDIiLCJ0MyIsInQ0IiwibGVuZ3RoIiwibWFwIiwiX3RlbXAiLCJ0NSIsInQ2IiwidDciLCJ0OCIsImluc3RydWN0aW9uIiwiaW5kZXgiXSwic291cmNlcyI6WyJFcnJvclN0ZXAudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB7IEdJVEhVQl9BQ1RJT05fU0VUVVBfRE9DU19VUkwgfSBmcm9tICcuLi8uLi9jb25zdGFudHMvZ2l0aHViLWFwcC5qcydcbmltcG9ydCB7IEJveCwgVGV4dCB9IGZyb20gJy4uLy4uL2luay5qcydcblxuaW50ZXJmYWNlIEVycm9yU3RlcFByb3BzIHtcbiAgZXJyb3I6IHN0cmluZyB8IHVuZGVmaW5lZFxuICBlcnJvclJlYXNvbj86IHN0cmluZ1xuICBlcnJvckluc3RydWN0aW9ucz86IHN0cmluZ1tdXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBFcnJvclN0ZXAoe1xuICBlcnJvcixcbiAgZXJyb3JSZWFzb24sXG4gIGVycm9ySW5zdHJ1Y3Rpb25zLFxufTogRXJyb3JTdGVwUHJvcHMpIHtcbiAgcmV0dXJuIChcbiAgICA8PlxuICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCIgYm9yZGVyU3R5bGU9XCJyb3VuZFwiIHBhZGRpbmdYPXsxfT5cbiAgICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCIgbWFyZ2luQm90dG9tPXsxfT5cbiAgICAgICAgICA8VGV4dCBib2xkPkluc3RhbGwgR2l0SHViIEFwcDwvVGV4dD5cbiAgICAgICAgPC9Cb3g+XG4gICAgICAgIDxUZXh0IGNvbG9yPVwiZXJyb3JcIj5FcnJvcjoge2Vycm9yfTwvVGV4dD5cbiAgICAgICAge2Vycm9yUmVhc29uICYmIChcbiAgICAgICAgICA8Qm94IG1hcmdpblRvcD17MX0+XG4gICAgICAgICAgICA8VGV4dCBkaW1Db2xvcj5SZWFzb246IHtlcnJvclJlYXNvbn08L1RleHQ+XG4gICAgICAgICAgPC9Cb3g+XG4gICAgICAgICl9XG4gICAgICAgIHtlcnJvckluc3RydWN0aW9ucyAmJiBlcnJvckluc3RydWN0aW9ucy5sZW5ndGggPiAwICYmIChcbiAgICAgICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIiBtYXJnaW5Ub3A9ezF9PlxuICAgICAgICAgICAgPFRleHQgZGltQ29sb3I+SG93IHRvIGZpeDo8L1RleHQ+XG4gICAgICAgICAgICB7ZXJyb3JJbnN0cnVjdGlvbnMubWFwKChpbnN0cnVjdGlvbiwgaW5kZXgpID0+IChcbiAgICAgICAgICAgICAgPEJveCBrZXk9e2luZGV4fSBtYXJnaW5MZWZ0PXsyfT5cbiAgICAgICAgICAgICAgICA8VGV4dCBkaW1Db2xvcj7igKIgPC9UZXh0PlxuICAgICAgICAgICAgICAgIDxUZXh0PntpbnN0cnVjdGlvbn08L1RleHQ+XG4gICAgICAgICAgICAgIDwvQm94PlxuICAgICAgICAgICAgKSl9XG4gICAgICAgICAgPC9Cb3g+XG4gICAgICAgICl9XG4gICAgICAgIDxCb3ggbWFyZ2luVG9wPXsxfT5cbiAgICAgICAgICA8VGV4dCBkaW1Db2xvcj5cbiAgICAgICAgICAgIEZvciBtYW51YWwgc2V0dXAgaW5zdHJ1Y3Rpb25zLCBzZWU6eycgJ31cbiAgICAgICAgICAgIDxUZXh0IGNvbG9yPVwiY2xhdWRlXCI+e0dJVEhVQl9BQ1RJT05fU0VUVVBfRE9DU19VUkx9PC9UZXh0PlxuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgPC9Cb3g+XG4gICAgICA8L0JveD5cbiAgICAgIDxCb3ggbWFyZ2luTGVmdD17M30+XG4gICAgICAgIDxUZXh0IGRpbUNvbG9yPlByZXNzIGFueSBrZXkgdG8gZXhpdDwvVGV4dD5cbiAgICAgIDwvQm94PlxuICAgIDwvPlxuICApXG59XG4iXSwibWFwcGluZ3MiOiI7QUFBQSxPQUFPQSxLQUFLLE1BQU0sT0FBTztBQUN6QixTQUFTQyw0QkFBNEIsUUFBUSwrQkFBK0I7QUFDNUUsU0FBU0MsR0FBRyxFQUFFQyxJQUFJLFFBQVEsY0FBYztBQUV4QyxVQUFVQyxjQUFjLENBQUM7RUFDdkJDLEtBQUssRUFBRSxNQUFNLEdBQUcsU0FBUztFQUN6QkMsV0FBVyxDQUFDLEVBQUUsTUFBTTtFQUNwQkMsaUJBQWlCLENBQUMsRUFBRSxNQUFNLEVBQUU7QUFDOUI7QUFFQSxPQUFPLFNBQUFDLFVBQUFDLEVBQUE7RUFBQSxNQUFBQyxDQUFBLEdBQUFDLEVBQUE7RUFBbUI7SUFBQU4sS0FBQTtJQUFBQyxXQUFBO0lBQUFDO0VBQUEsSUFBQUUsRUFJVDtFQUFBLElBQUFHLEVBQUE7RUFBQSxJQUFBRixDQUFBLFFBQUFHLE1BQUEsQ0FBQUMsR0FBQTtJQUlURixFQUFBLElBQUMsR0FBRyxDQUFlLGFBQVEsQ0FBUixRQUFRLENBQWUsWUFBQyxDQUFELEdBQUMsQ0FDekMsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFKLEtBQUcsQ0FBQyxDQUFDLGtCQUFrQixFQUE1QixJQUFJLENBQ1AsRUFGQyxHQUFHLENBRUU7SUFBQUYsQ0FBQSxNQUFBRSxFQUFBO0VBQUE7SUFBQUEsRUFBQSxHQUFBRixDQUFBO0VBQUE7RUFBQSxJQUFBSyxFQUFBO0VBQUEsSUFBQUwsQ0FBQSxRQUFBTCxLQUFBO0lBQ05VLEVBQUEsSUFBQyxJQUFJLENBQU8sS0FBTyxDQUFQLE9BQU8sQ0FBQyxPQUFRVixNQUFJLENBQUUsRUFBakMsSUFBSSxDQUFvQztJQUFBSyxDQUFBLE1BQUFMLEtBQUE7SUFBQUssQ0FBQSxNQUFBSyxFQUFBO0VBQUE7SUFBQUEsRUFBQSxHQUFBTCxDQUFBO0VBQUE7RUFBQSxJQUFBTSxFQUFBO0VBQUEsSUFBQU4sQ0FBQSxRQUFBSixXQUFBO0lBQ3hDVSxFQUFBLEdBQUFWLFdBSUEsSUFIQyxDQUFDLEdBQUcsQ0FBWSxTQUFDLENBQUQsR0FBQyxDQUNmLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBUixLQUFPLENBQUMsQ0FBQyxRQUFTQSxZQUFVLENBQUUsRUFBbkMsSUFBSSxDQUNQLEVBRkMsR0FBRyxDQUdMO0lBQUFJLENBQUEsTUFBQUosV0FBQTtJQUFBSSxDQUFBLE1BQUFNLEVBQUE7RUFBQTtJQUFBQSxFQUFBLEdBQUFOLENBQUE7RUFBQTtFQUFBLElBQUFPLEVBQUE7RUFBQSxJQUFBUCxDQUFBLFFBQUFILGlCQUFBO0lBQ0FVLEVBQUEsR0FBQVYsaUJBQWlELElBQTVCQSxpQkFBaUIsQ0FBQVcsTUFBTyxHQUFHLENBVWhELElBVEMsQ0FBQyxHQUFHLENBQWUsYUFBUSxDQUFSLFFBQVEsQ0FBWSxTQUFDLENBQUQsR0FBQyxDQUN0QyxDQUFDLElBQUksQ0FBQyxRQUFRLENBQVIsS0FBTyxDQUFDLENBQUMsV0FBVyxFQUF6QixJQUFJLENBQ0osQ0FBQVgsaUJBQWlCLENBQUFZLEdBQUksQ0FBQ0MsS0FLdEIsRUFDSCxFQVJDLEdBQUcsQ0FTTDtJQUFBVixDQUFBLE1BQUFILGlCQUFBO0lBQUFHLENBQUEsTUFBQU8sRUFBQTtFQUFBO0lBQUFBLEVBQUEsR0FBQVAsQ0FBQTtFQUFBO0VBQUEsSUFBQVcsRUFBQTtFQUFBLElBQUFYLENBQUEsUUFBQUcsTUFBQSxDQUFBQyxHQUFBO0lBQ0RPLEVBQUEsSUFBQyxHQUFHLENBQVksU0FBQyxDQUFELEdBQUMsQ0FDZixDQUFDLElBQUksQ0FBQyxRQUFRLENBQVIsS0FBTyxDQUFDLENBQUMsbUNBQ3VCLElBQUUsQ0FDdEMsQ0FBQyxJQUFJLENBQU8sS0FBUSxDQUFSLFFBQVEsQ0FBRXBCLDZCQUEyQixDQUFFLEVBQWxELElBQUksQ0FDUCxFQUhDLElBQUksQ0FJUCxFQUxDLEdBQUcsQ0FLRTtJQUFBUyxDQUFBLE1BQUFXLEVBQUE7RUFBQTtJQUFBQSxFQUFBLEdBQUFYLENBQUE7RUFBQTtFQUFBLElBQUFZLEVBQUE7RUFBQSxJQUFBWixDQUFBLFFBQUFLLEVBQUEsSUFBQUwsQ0FBQSxRQUFBTSxFQUFBLElBQUFOLENBQUEsU0FBQU8sRUFBQTtJQTFCUkssRUFBQSxJQUFDLEdBQUcsQ0FBZSxhQUFRLENBQVIsUUFBUSxDQUFhLFdBQU8sQ0FBUCxPQUFPLENBQVcsUUFBQyxDQUFELEdBQUMsQ0FDekQsQ0FBQVYsRUFFSyxDQUNMLENBQUFHLEVBQXdDLENBQ3ZDLENBQUFDLEVBSUQsQ0FDQyxDQUFBQyxFQVVELENBQ0EsQ0FBQUksRUFLSyxDQUNQLEVBM0JDLEdBQUcsQ0EyQkU7SUFBQVgsQ0FBQSxNQUFBSyxFQUFBO0lBQUFMLENBQUEsTUFBQU0sRUFBQTtJQUFBTixDQUFBLE9BQUFPLEVBQUE7SUFBQVAsQ0FBQSxPQUFBWSxFQUFBO0VBQUE7SUFBQUEsRUFBQSxHQUFBWixDQUFBO0VBQUE7RUFBQSxJQUFBYSxFQUFBO0VBQUEsSUFBQWIsQ0FBQSxTQUFBRyxNQUFBLENBQUFDLEdBQUE7SUFDTlMsRUFBQSxJQUFDLEdBQUcsQ0FBYSxVQUFDLENBQUQsR0FBQyxDQUNoQixDQUFDLElBQUksQ0FBQyxRQUFRLENBQVIsS0FBTyxDQUFDLENBQUMscUJBQXFCLEVBQW5DLElBQUksQ0FDUCxFQUZDLEdBQUcsQ0FFRTtJQUFBYixDQUFBLE9BQUFhLEVBQUE7RUFBQTtJQUFBQSxFQUFBLEdBQUFiLENBQUE7RUFBQTtFQUFBLElBQUFjLEVBQUE7RUFBQSxJQUFBZCxDQUFBLFNBQUFZLEVBQUE7SUEvQlJFLEVBQUEsS0FDRSxDQUFBRixFQTJCSyxDQUNMLENBQUFDLEVBRUssQ0FBQyxHQUNMO0lBQUFiLENBQUEsT0FBQVksRUFBQTtJQUFBWixDQUFBLE9BQUFjLEVBQUE7RUFBQTtJQUFBQSxFQUFBLEdBQUFkLENBQUE7RUFBQTtFQUFBLE9BaENIYyxFQWdDRztBQUFBO0FBdENBLFNBQUFKLE1BQUFLLFdBQUEsRUFBQUMsS0FBQTtFQUFBLE9BcUJPLENBQUMsR0FBRyxDQUFNQSxHQUFLLENBQUxBLE1BQUksQ0FBQyxDQUFjLFVBQUMsQ0FBRCxHQUFDLENBQzVCLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBUixLQUFPLENBQUMsQ0FBQyxFQUFFLEVBQWhCLElBQUksQ0FDTCxDQUFDLElBQUksQ0FBRUQsWUFBVSxDQUFFLEVBQWxCLElBQUksQ0FDUCxFQUhDLEdBQUcsQ0FHRTtBQUFBIiwiaWdub3JlTGlzdCI6W119 \ No newline at end of file diff --git a/src/commands/install-github-app/ExistingWorkflowStep.tsx b/src/commands/install-github-app/ExistingWorkflowStep.tsx new file mode 100644 index 0000000..3efff6f --- /dev/null +++ b/src/commands/install-github-app/ExistingWorkflowStep.tsx @@ -0,0 +1,103 @@ +import { c as _c } from "react/compiler-runtime"; +import React from 'react'; +import { Select } from 'src/components/CustomSelect/index.js'; +import { Box, Text } from '../../ink.js'; +interface ExistingWorkflowStepProps { + repoName: string; + onSelectAction: (action: 'update' | 'skip' | 'exit') => void; +} +export function ExistingWorkflowStep(t0) { + const $ = _c(16); + const { + repoName, + onSelectAction + } = t0; + let t1; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t1 = [{ + label: "Update workflow file with latest version", + value: "update" + }, { + label: "Skip workflow update (configure secrets only)", + value: "skip" + }, { + label: "Exit without making changes", + value: "exit" + }]; + $[0] = t1; + } else { + t1 = $[0]; + } + const options = t1; + let t2; + if ($[1] !== onSelectAction) { + t2 = value => { + onSelectAction(value as 'update' | 'skip' | 'exit'); + }; + $[1] = onSelectAction; + $[2] = t2; + } else { + t2 = $[2]; + } + const handleSelect = t2; + let t3; + if ($[3] !== onSelectAction) { + t3 = () => { + onSelectAction("exit"); + }; + $[3] = onSelectAction; + $[4] = t3; + } else { + t3 = $[4]; + } + const handleCancel = t3; + let t4; + if ($[5] === Symbol.for("react.memo_cache_sentinel")) { + t4 = Existing Workflow Found; + $[5] = t4; + } else { + t4 = $[5]; + } + let t5; + if ($[6] !== repoName) { + t5 = {t4}Repository: {repoName}; + $[6] = repoName; + $[7] = t5; + } else { + t5 = $[7]; + } + let t6; + if ($[8] === Symbol.for("react.memo_cache_sentinel")) { + t6 = A Claude workflow file already exists at{" "}.github/workflows/claude.ymlWhat would you like to do?; + $[8] = t6; + } else { + t6 = $[8]; + } + let t7; + if ($[9] !== handleCancel || $[10] !== handleSelect) { + t7 = ; + $[19] = handleSelect; + $[20] = options; + $[21] = t6; + } else { + t6 = $[21]; + } + let t7; + if ($[22] !== handleCancel || $[23] !== t6) { + t7 = {t6}; + $[22] = handleCancel; + $[23] = t6; + $[24] = t7; + } else { + t7 = $[24]; + } + return t7; +} +export async function call(onDone: LocalJSXCommandOnDone, context: ToolUseContext & LocalJSXCommandContext): Promise { + return ; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsInVzZU1lbW8iLCJ1c2VTdGF0ZSIsIkNvbW1hbmRSZXN1bHREaXNwbGF5IiwiTG9jYWxKU1hDb21tYW5kQ29udGV4dCIsIk9wdGlvbldpdGhEZXNjcmlwdGlvbiIsIlNlbGVjdCIsIkRpYWxvZyIsImdldEZlYXR1cmVWYWx1ZV9DQUNIRURfTUFZX0JFX1NUQUxFIiwibG9nRXZlbnQiLCJ1c2VDbGF1ZGVBaUxpbWl0cyIsIlRvb2xVc2VDb250ZXh0IiwiTG9jYWxKU1hDb21tYW5kT25Eb25lIiwiZ2V0T2F1dGhBY2NvdW50SW5mbyIsImdldFJhdGVMaW1pdFRpZXIiLCJnZXRTdWJzY3JpcHRpb25UeXBlIiwiaGFzQ2xhdWRlQWlCaWxsaW5nQWNjZXNzIiwiY2FsbCIsImV4dHJhVXNhZ2VDYWxsIiwiZXh0cmFVc2FnZSIsInVwZ3JhZGUiLCJ1cGdyYWRlQ2FsbCIsIlJhdGVMaW1pdE9wdGlvbnNNZW51T3B0aW9uVHlwZSIsIlJhdGVMaW1pdE9wdGlvbnNNZW51UHJvcHMiLCJvbkRvbmUiLCJyZXN1bHQiLCJvcHRpb25zIiwiZGlzcGxheSIsImNvbnRleHQiLCJSYXRlTGltaXRPcHRpb25zTWVudSIsInQwIiwiJCIsIl9jIiwic3ViQ29tbWFuZEpTWCIsInNldFN1YkNvbW1hbmRKU1giLCJjbGF1ZGVBaUxpbWl0cyIsInQxIiwiU3ltYm9sIiwiZm9yIiwic3Vic2NyaXB0aW9uVHlwZSIsInQyIiwicmF0ZUxpbWl0VGllciIsImhhc0V4dHJhVXNhZ2VFbmFibGVkIiwiaXNNYXgiLCJpc01heDIweCIsImlzVGVhbU9yRW50ZXJwcmlzZSIsImJ1eUZpcnN0IiwidDMiLCJiYjAiLCJhY3Rpb25PcHRpb25zIiwib3ZlcmFnZURpc2FibGVkUmVhc29uIiwib3ZlcmFnZVN0YXR1cyIsImlzRW5hYmxlZCIsImhhc0JpbGxpbmdBY2Nlc3MiLCJuZWVkc1RvUmVxdWVzdEZyb21BZG1pbiIsImlzT3JnU3BlbmRDYXBEZXBsZXRlZCIsImlzT3ZlcmFnZVN0YXRlIiwibGFiZWwiLCJ0NCIsInZhbHVlIiwicHVzaCIsImNhbmNlbE9wdGlvbiIsInQ1IiwiaGFuZGxlQ2FuY2VsIiwidW5kZWZpbmVkIiwiaGFuZGxlU2VsZWN0IiwidGhlbiIsImpzeCIsImpzeF8wIiwidDYiLCJsZW5ndGgiLCJ0NyIsIlByb21pc2UiLCJSZWFjdE5vZGUiXSwic291cmNlcyI6WyJyYXRlLWxpbWl0LW9wdGlvbnMudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBSZWFjdCwgeyB1c2VNZW1vLCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHR5cGUge1xuICBDb21tYW5kUmVzdWx0RGlzcGxheSxcbiAgTG9jYWxKU1hDb21tYW5kQ29udGV4dCxcbn0gZnJvbSAnLi4vLi4vY29tbWFuZHMuanMnXG5pbXBvcnQge1xuICB0eXBlIE9wdGlvbldpdGhEZXNjcmlwdGlvbixcbiAgU2VsZWN0LFxufSBmcm9tICcuLi8uLi9jb21wb25lbnRzL0N1c3RvbVNlbGVjdC9zZWxlY3QuanMnXG5pbXBvcnQgeyBEaWFsb2cgfSBmcm9tICcuLi8uLi9jb21wb25lbnRzL2Rlc2lnbi1zeXN0ZW0vRGlhbG9nLmpzJ1xuaW1wb3J0IHsgZ2V0RmVhdHVyZVZhbHVlX0NBQ0hFRF9NQVlfQkVfU1RBTEUgfSBmcm9tICcuLi8uLi9zZXJ2aWNlcy9hbmFseXRpY3MvZ3Jvd3RoYm9vay5qcydcbmltcG9ydCB7IGxvZ0V2ZW50IH0gZnJvbSAnLi4vLi4vc2VydmljZXMvYW5hbHl0aWNzL2luZGV4LmpzJ1xuaW1wb3J0IHsgdXNlQ2xhdWRlQWlMaW1pdHMgfSBmcm9tICcuLi8uLi9zZXJ2aWNlcy9jbGF1ZGVBaUxpbWl0c0hvb2suanMnXG5pbXBvcnQgdHlwZSB7IFRvb2xVc2VDb250ZXh0IH0gZnJvbSAnLi4vLi4vVG9vbC5qcydcbmltcG9ydCB0eXBlIHsgTG9jYWxKU1hDb21tYW5kT25Eb25lIH0gZnJvbSAnLi4vLi4vdHlwZXMvY29tbWFuZC5qcydcbmltcG9ydCB7XG4gIGdldE9hdXRoQWNjb3VudEluZm8sXG4gIGdldFJhdGVMaW1pdFRpZXIsXG4gIGdldFN1YnNjcmlwdGlvblR5cGUsXG59IGZyb20gJy4uLy4uL3V0aWxzL2F1dGguanMnXG5pbXBvcnQgeyBoYXNDbGF1ZGVBaUJpbGxpbmdBY2Nlc3MgfSBmcm9tICcuLi8uLi91dGlscy9iaWxsaW5nLmpzJ1xuaW1wb3J0IHsgY2FsbCBhcyBleHRyYVVzYWdlQ2FsbCB9IGZyb20gJy4uL2V4dHJhLXVzYWdlL2V4dHJhLXVzYWdlLmpzJ1xuaW1wb3J0IHsgZXh0cmFVc2FnZSB9IGZyb20gJy4uL2V4dHJhLXVzYWdlL2luZGV4LmpzJ1xuaW1wb3J0IHVwZ3JhZGUgZnJvbSAnLi4vdXBncmFkZS9pbmRleC5qcydcbmltcG9ydCB7IGNhbGwgYXMgdXBncmFkZUNhbGwgfSBmcm9tICcuLi91cGdyYWRlL3VwZ3JhZGUuanMnXG5cbnR5cGUgUmF0ZUxpbWl0T3B0aW9uc01lbnVPcHRpb25UeXBlID0gJ3VwZ3JhZGUnIHwgJ2V4dHJhLXVzYWdlJyB8ICdjYW5jZWwnXG5cbnR5cGUgUmF0ZUxpbWl0T3B0aW9uc01lbnVQcm9wcyA9IHtcbiAgb25Eb25lOiAoXG4gICAgcmVzdWx0Pzogc3RyaW5nLFxuICAgIG9wdGlvbnM/OlxuICAgICAgfCB7XG4gICAgICAgICAgZGlzcGxheT86IENvbW1hbmRSZXN1bHREaXNwbGF5IHwgdW5kZWZpbmVkXG4gICAgICAgIH1cbiAgICAgIHwgdW5kZWZpbmVkLFxuICApID0+IHZvaWRcbiAgY29udGV4dDogVG9vbFVzZUNvbnRleHQgJiBMb2NhbEpTWENvbW1hbmRDb250ZXh0XG59XG5cbmZ1bmN0aW9uIFJhdGVMaW1pdE9wdGlvbnNNZW51KHtcbiAgb25Eb25lLFxuICBjb250ZXh0LFxufTogUmF0ZUxpbWl0T3B0aW9uc01lbnVQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IFtzdWJDb21tYW5kSlNYLCBzZXRTdWJDb21tYW5kSlNYXSA9IHVzZVN0YXRlPFJlYWN0LlJlYWN0Tm9kZT4obnVsbClcbiAgY29uc3QgY2xhdWRlQWlMaW1pdHMgPSB1c2VDbGF1ZGVBaUxpbWl0cygpXG4gIGNvbnN0IHN1YnNjcmlwdGlvblR5cGUgPSBnZXRTdWJzY3JpcHRpb25UeXBlKClcbiAgY29uc3QgcmF0ZUxpbWl0VGllciA9IGdldFJhdGVMaW1pdFRpZXIoKVxuICBjb25zdCBoYXNFeHRyYVVzYWdlRW5hYmxlZCA9XG4gICAgZ2V0T2F1dGhBY2NvdW50SW5mbygpPy5oYXNFeHRyYVVzYWdlRW5hYmxlZCA9PT0gdHJ1ZVxuICBjb25zdCBpc01heCA9IHN1YnNjcmlwdGlvblR5cGUgPT09ICdtYXgnXG4gIGNvbnN0IGlzTWF4MjB4ID0gaXNNYXggJiYgcmF0ZUxpbWl0VGllciA9PT0gJ2RlZmF1bHRfY2xhdWRlX21heF8yMHgnXG4gIGNvbnN0IGlzVGVhbU9yRW50ZXJwcmlzZSA9XG4gICAgc3Vic2NyaXB0aW9uVHlwZSA9PT0gJ3RlYW0nIHx8IHN1YnNjcmlwdGlvblR5cGUgPT09ICdlbnRlcnByaXNlJ1xuICBjb25zdCBidXlGaXJzdCA9IGdldEZlYXR1cmVWYWx1ZV9DQUNIRURfTUFZX0JFX1NUQUxFKFxuICAgICd0ZW5ndV9qYWRlX2FudmlsXzQnLFxuICAgIGZhbHNlLFxuICApXG5cbiAgY29uc3Qgb3B0aW9ucyA9IHVzZU1lbW88XG4gICAgT3B0aW9uV2l0aERlc2NyaXB0aW9uPFJhdGVMaW1pdE9wdGlvbnNNZW51T3B0aW9uVHlwZT5bXVxuICA+KCgpID0+IHtcbiAgICBjb25zdCBhY3Rpb25PcHRpb25zOiBPcHRpb25XaXRoRGVzY3JpcHRpb248UmF0ZUxpbWl0T3B0aW9uc01lbnVPcHRpb25UeXBlPltdID1cbiAgICAgIFtdXG5cbiAgICBpZiAoZXh0cmFVc2FnZS5pc0VuYWJsZWQoKSkge1xuICAgICAgY29uc3QgaGFzQmlsbGluZ0FjY2VzcyA9IGhhc0NsYXVkZUFpQmlsbGluZ0FjY2VzcygpXG4gICAgICBjb25zdCBuZWVkc1RvUmVxdWVzdEZyb21BZG1pbiA9IGlzVGVhbU9yRW50ZXJwcmlzZSAmJiAhaGFzQmlsbGluZ0FjY2Vzc1xuICAgICAgLy8gT3JnIHNwZW5kIGNhcCBkZXBsZXRlZCAtIG5vbi1hZG1pbnMgY2FuJ3QgcmVxdWVzdCBtb3JlIHNpbmNlIHRoZXJlJ3Mgbm90aGluZyB0byBhbGxvY2F0ZVxuICAgICAgLy8gLSBvdXRfb2ZfY3JlZGl0czogd2FsbGV0IGVtcHR5XG4gICAgICAvLyAtIG9yZ19sZXZlbF9kaXNhYmxlZF91bnRpbDogb3JnIHNwZW5kIGNhcCBoaXQgZm9yIHRoZSBtb250aFxuICAgICAgLy8gLSBvcmdfc2VydmljZV96ZXJvX2NyZWRpdF9saW1pdDogb3JnIHNlcnZpY2UgaGFzIHplcm8gY3JlZGl0IGxpbWl0XG4gICAgICBjb25zdCBpc09yZ1NwZW5kQ2FwRGVwbGV0ZWQgPVxuICAgICAgICBjbGF1ZGVBaUxpbWl0cy5vdmVyYWdlRGlzYWJsZWRSZWFzb24gPT09ICdvdXRfb2ZfY3JlZGl0cycgfHxcbiAgICAgICAgY2xhdWRlQWlMaW1pdHMub3ZlcmFnZURpc2FibGVkUmVhc29uID09PSAnb3JnX2xldmVsX2Rpc2FibGVkX3VudGlsJyB8fFxuICAgICAgICBjbGF1ZGVBaUxpbWl0cy5vdmVyYWdlRGlzYWJsZWRSZWFzb24gPT09ICdvcmdfc2VydmljZV96ZXJvX2NyZWRpdF9saW1pdCdcblxuICAgICAgLy8gSGlkZSBmb3Igbm9uLWFkbWluIFRlYW0vRW50ZXJwcmlzZSB1c2VycyB3aGVuIG9yZyBzcGVuZCBjYXAgaXMgZGVwbGV0ZWRcbiAgICAgIGlmIChuZWVkc1RvUmVxdWVzdEZyb21BZG1pbiAmJiBpc09yZ1NwZW5kQ2FwRGVwbGV0ZWQpIHtcbiAgICAgICAgLy8gRG9uJ3Qgc2hvdyBleHRyYS11c2FnZSBvcHRpb25cbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGNvbnN0IGlzT3ZlcmFnZVN0YXRlID1cbiAgICAgICAgICBjbGF1ZGVBaUxpbWl0cy5vdmVyYWdlU3RhdHVzID09PSAncmVqZWN0ZWQnIHx8XG4gICAgICAgICAgY2xhdWRlQWlMaW1pdHMub3ZlcmFnZVN0YXR1cyA9PT0gJ2FsbG93ZWRfd2FybmluZydcblxuICAgICAgICBsZXQgbGFiZWw6IHN0cmluZ1xuICAgICAgICBpZiAobmVlZHNUb1JlcXVlc3RGcm9tQWRtaW4pIHtcbiAgICAgICAgICBsYWJlbCA9IGlzT3ZlcmFnZVN0YXRlID8gJ1JlcXVlc3QgbW9yZScgOiAnUmVxdWVzdCBleHRyYSB1c2FnZSdcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBsYWJlbCA9IGhhc0V4dHJhVXNhZ2VFbmFibGVkXG4gICAgICAgICAgICA/ICdBZGQgZnVuZHMgdG8gY29udGludWUgd2l0aCBleHRyYSB1c2FnZSdcbiAgICAgICAgICAgIDogJ1N3aXRjaCB0byBleHRyYSB1c2FnZSdcbiAgICAgICAgfVxuXG4gICAgICAgIGFjdGlvbk9wdGlvbnMucHVzaCh7XG4gICAgICAgICAgbGFiZWwsXG4gICAgICAgICAgdmFsdWU6ICdleHRyYS11c2FnZScsXG4gICAgICAgIH0pXG4gICAgICB9XG4gICAgfVxuXG4gICAgaWYgKCFpc01heDIweCAmJiAhaXNUZWFtT3JFbnRlcnByaXNlICYmIHVwZ3JhZGUuaXNFbmFibGVkKCkpIHtcbiAgICAgIGFjdGlvbk9wdGlvbnMucHVzaCh7XG4gICAgICAgIGxhYmVsOiAnVXBncmFkZSB5b3VyIHBsYW4nLFxuICAgICAgICB2YWx1ZTogJ3VwZ3JhZGUnLFxuICAgICAgfSlcbiAgICB9XG5cbiAgICBjb25zdCBjYW5jZWxPcHRpb246IE9wdGlvbldpdGhEZXNjcmlwdGlvbjxSYXRlTGltaXRPcHRpb25zTWVudU9wdGlvblR5cGU+ID1cbiAgICAgIHtcbiAgICAgICAgbGFiZWw6ICdTdG9wIGFuZCB3YWl0IGZvciBsaW1pdCB0byByZXNldCcsXG4gICAgICAgIHZhbHVlOiAnY2FuY2VsJyxcbiAgICAgIH1cblxuICAgIGlmIChidXlGaXJzdCkge1xuICAgICAgcmV0dXJuIFsuLi5hY3Rpb25PcHRpb25zLCBjYW5jZWxPcHRpb25dXG4gICAgfVxuICAgIHJldHVybiBbY2FuY2VsT3B0aW9uLCAuLi5hY3Rpb25PcHRpb25zXVxuICB9LCBbXG4gICAgYnV5Rmlyc3QsXG4gICAgaXNNYXgyMHgsXG4gICAgaXNUZWFtT3JFbnRlcnByaXNlLFxuICAgIGhhc0V4dHJhVXNhZ2VFbmFibGVkLFxuICAgIGNsYXVkZUFpTGltaXRzLm92ZXJhZ2VTdGF0dXMsXG4gICAgY2xhdWRlQWlMaW1pdHMub3ZlcmFnZURpc2FibGVkUmVhc29uLFxuICBdKVxuXG4gIGZ1bmN0aW9uIGhhbmRsZUNhbmNlbCgpOiB2b2lkIHtcbiAgICBsb2dFdmVudCgndGVuZ3VfcmF0ZV9saW1pdF9vcHRpb25zX21lbnVfY2FuY2VsJywge30pXG4gICAgb25Eb25lKHVuZGVmaW5lZCwgeyBkaXNwbGF5OiAnc2tpcCcgfSlcbiAgfVxuXG4gIGZ1bmN0aW9uIGhhbmRsZVNlbGVjdCh2YWx1ZTogUmF0ZUxpbWl0T3B0aW9uc01lbnVPcHRpb25UeXBlKTogdm9pZCB7XG4gICAgaWYgKHZhbHVlID09PSAndXBncmFkZScpIHtcbiAgICAgIGxvZ0V2ZW50KCd0ZW5ndV9yYXRlX2xpbWl0X29wdGlvbnNfbWVudV9zZWxlY3RfdXBncmFkZScsIHt9KVxuICAgICAgdm9pZCB1cGdyYWRlQ2FsbChvbkRvbmUsIGNvbnRleHQpLnRoZW4oanN4ID0+IHtcbiAgICAgICAgaWYgKGpzeCkge1xuICAgICAgICAgIHNldFN1YkNvbW1hbmRKU1goanN4KVxuICAgICAgICB9XG4gICAgICB9KVxuICAgIH0gZWxzZSBpZiAodmFsdWUgPT09ICdleHRyYS11c2FnZScpIHtcbiAgICAgIGxvZ0V2ZW50KCd0ZW5ndV9yYXRlX2xpbWl0X29wdGlvbnNfbWVudV9zZWxlY3RfZXh0cmFfdXNhZ2UnLCB7fSlcbiAgICAgIHZvaWQgZXh0cmFVc2FnZUNhbGwob25Eb25lLCBjb250ZXh0KS50aGVuKGpzeCA9PiB7XG4gICAgICAgIGlmIChqc3gpIHtcbiAgICAgICAgICBzZXRTdWJDb21tYW5kSlNYKGpzeClcbiAgICAgICAgfVxuICAgICAgfSlcbiAgICB9IGVsc2UgaWYgKHZhbHVlID09PSAnY2FuY2VsJykge1xuICAgICAgaGFuZGxlQ2FuY2VsKClcbiAgICB9XG4gIH1cblxuICBpZiAoc3ViQ29tbWFuZEpTWCkge1xuICAgIHJldHVybiBzdWJDb21tYW5kSlNYXG4gIH1cblxuICByZXR1cm4gKFxuICAgIDxEaWFsb2dcbiAgICAgIHRpdGxlPVwiV2hhdCBkbyB5b3Ugd2FudCB0byBkbz9cIlxuICAgICAgb25DYW5jZWw9e2hhbmRsZUNhbmNlbH1cbiAgICAgIGNvbG9yPVwic3VnZ2VzdGlvblwiXG4gICAgPlxuICAgICAgPFNlbGVjdDxSYXRlTGltaXRPcHRpb25zTWVudU9wdGlvblR5cGU+XG4gICAgICAgIG9wdGlvbnM9e29wdGlvbnN9XG4gICAgICAgIG9uQ2hhbmdlPXtoYW5kbGVTZWxlY3R9XG4gICAgICAgIHZpc2libGVPcHRpb25Db3VudD17b3B0aW9ucy5sZW5ndGh9XG4gICAgICAvPlxuICAgIDwvRGlhbG9nPlxuICApXG59XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBjYWxsKFxuICBvbkRvbmU6IExvY2FsSlNYQ29tbWFuZE9uRG9uZSxcbiAgY29udGV4dDogVG9vbFVzZUNvbnRleHQgJiBMb2NhbEpTWENvbW1hbmRDb250ZXh0LFxuKTogUHJvbWlzZTxSZWFjdC5SZWFjdE5vZGU+IHtcbiAgcmV0dXJuIDxSYXRlTGltaXRPcHRpb25zTWVudSBvbkRvbmU9e29uRG9uZX0gY29udGV4dD17Y29udGV4dH0gLz5cbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU9BLEtBQUssSUFBSUMsT0FBTyxFQUFFQyxRQUFRLFFBQVEsT0FBTztBQUNoRCxjQUNFQyxvQkFBb0IsRUFDcEJDLHNCQUFzQixRQUNqQixtQkFBbUI7QUFDMUIsU0FDRSxLQUFLQyxxQkFBcUIsRUFDMUJDLE1BQU0sUUFDRCx5Q0FBeUM7QUFDaEQsU0FBU0MsTUFBTSxRQUFRLDBDQUEwQztBQUNqRSxTQUFTQyxtQ0FBbUMsUUFBUSx3Q0FBd0M7QUFDNUYsU0FBU0MsUUFBUSxR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 \ No newline at end of file diff --git a/src/commands/release-notes/index.ts b/src/commands/release-notes/index.ts new file mode 100644 index 0000000..75413de --- /dev/null +++ b/src/commands/release-notes/index.ts @@ -0,0 +1,11 @@ +import type { Command } from '../../commands.js' + +const releaseNotes: Command = { + description: 'View release notes', + name: 'release-notes', + type: 'local', + supportsNonInteractive: true, + load: () => import('./release-notes.js'), +} + +export default releaseNotes diff --git a/src/commands/release-notes/release-notes.ts b/src/commands/release-notes/release-notes.ts new file mode 100644 index 0000000..dfd7aec --- /dev/null +++ b/src/commands/release-notes/release-notes.ts @@ -0,0 +1,50 @@ +import type { LocalCommandResult } from '../../types/command.js' +import { + CHANGELOG_URL, + fetchAndStoreChangelog, + getAllReleaseNotes, + getStoredChangelog, +} from '../../utils/releaseNotes.js' + +function formatReleaseNotes(notes: Array<[string, string[]]>): string { + return notes + .map(([version, notes]) => { + const header = `Version ${version}:` + const bulletPoints = notes.map(note => `· ${note}`).join('\n') + return `${header}\n${bulletPoints}` + }) + .join('\n\n') +} + +export async function call(): Promise { + // Try to fetch the latest changelog with a 500ms timeout + let freshNotes: Array<[string, string[]]> = [] + + try { + const timeoutPromise = new Promise((_, reject) => { + setTimeout(rej => rej(new Error('Timeout')), 500, reject) + }) + + await Promise.race([fetchAndStoreChangelog(), timeoutPromise]) + freshNotes = getAllReleaseNotes(await getStoredChangelog()) + } catch { + // Either fetch failed or timed out - just use cached notes + } + + // If we have fresh notes from the quick fetch, use those + if (freshNotes.length > 0) { + return { type: 'text', value: formatReleaseNotes(freshNotes) } + } + + // Otherwise check cached notes + const cachedNotes = getAllReleaseNotes(await getStoredChangelog()) + if (cachedNotes.length > 0) { + return { type: 'text', value: formatReleaseNotes(cachedNotes) } + } + + // Nothing available, show link + return { + type: 'text', + value: `See the full changelog at: ${CHANGELOG_URL}`, + } +} diff --git a/src/commands/reload-plugins/index.ts b/src/commands/reload-plugins/index.ts new file mode 100644 index 0000000..5d7a163 --- /dev/null +++ b/src/commands/reload-plugins/index.ts @@ -0,0 +1,18 @@ +/** + * /reload-plugins — Layer-3 refresh. Applies pending plugin changes to the + * running session. Implementation lazy-loaded. + */ +import type { Command } from '../../commands.js' + +const reloadPlugins = { + type: 'local', + name: 'reload-plugins', + description: 'Activate pending plugin changes in the current session', + // SDK callers use query.reloadPlugins() (control request) instead of + // sending this as a text prompt — that returns structured data + // (commands, agents, plugins, mcpServers) for UI updates. + supportsNonInteractive: false, + load: () => import('./reload-plugins.js'), +} satisfies Command + +export default reloadPlugins diff --git a/src/commands/reload-plugins/reload-plugins.ts b/src/commands/reload-plugins/reload-plugins.ts new file mode 100644 index 0000000..0789be4 --- /dev/null +++ b/src/commands/reload-plugins/reload-plugins.ts @@ -0,0 +1,61 @@ +import { feature } from 'bun:bundle' +import { getIsRemoteMode } from '../../bootstrap/state.js' +import { redownloadUserSettings } from '../../services/settingsSync/index.js' +import type { LocalCommandCall } from '../../types/command.js' +import { isEnvTruthy } from '../../utils/envUtils.js' +import { refreshActivePlugins } from '../../utils/plugins/refresh.js' +import { settingsChangeDetector } from '../../utils/settings/changeDetector.js' +import { plural } from '../../utils/stringUtils.js' + +export const call: LocalCommandCall = async (_args, context) => { + // CCR: re-pull user settings before the cache sweep so enabledPlugins / + // extraKnownMarketplaces pushed from the user's local CLI (settingsSync) + // take effect. Non-CCR headless (e.g. vscode SDK subprocess) shares disk + // with whoever writes settings — the file watcher delivers changes, no + // re-pull needed there. + // + // Managed settings intentionally NOT re-fetched: it already polls hourly + // (POLLING_INTERVAL_MS), and policy enforcement is eventually-consistent + // by design (stale-cache fallback on fetch failure). Interactive + // /reload-plugins has never re-fetched it either. + // + // No retries: user-initiated command, one attempt + fail-open. The user + // can re-run /reload-plugins to retry. Startup path keeps its retries. + if ( + feature('DOWNLOAD_USER_SETTINGS') && + (isEnvTruthy(process.env.CLAUDE_CODE_REMOTE) || getIsRemoteMode()) + ) { + const applied = await redownloadUserSettings() + // applyRemoteEntriesToLocal uses markInternalWrite to suppress the + // file watcher (correct for startup, nothing listening yet); fire + // notifyChange here so mid-session applySettingsChange runs. + if (applied) { + settingsChangeDetector.notifyChange('userSettings') + } + } + + const r = await refreshActivePlugins(context.setAppState) + + const parts = [ + n(r.enabled_count, 'plugin'), + n(r.command_count, 'skill'), + n(r.agent_count, 'agent'), + n(r.hook_count, 'hook'), + // "plugin MCP/LSP" disambiguates from user-config/built-in servers, + // which /reload-plugins doesn't touch. Commands/hooks are plugin-only; + // agent_count is total agents (incl. built-ins). (gh-31321) + n(r.mcp_count, 'plugin MCP server'), + n(r.lsp_count, 'plugin LSP server'), + ] + let msg = `Reloaded: ${parts.join(' · ')}` + + if (r.error_count > 0) { + msg += `\n${n(r.error_count, 'error')} during load. Run /doctor for details.` + } + + return { type: 'text', value: msg } +} + +function n(count: number, noun: string): string { + return `${count} ${plural(count, noun)}` +} diff --git a/src/commands/remote-env/index.ts b/src/commands/remote-env/index.ts new file mode 100644 index 0000000..090cc60 --- /dev/null +++ b/src/commands/remote-env/index.ts @@ -0,0 +1,15 @@ +import type { Command } from '../../commands.js' +import { isPolicyAllowed } from '../../services/policyLimits/index.js' +import { isClaudeAISubscriber } from '../../utils/auth.js' + +export default { + type: 'local-jsx', + name: 'remote-env', + description: 'Configure the default remote environment for teleport sessions', + isEnabled: () => + isClaudeAISubscriber() && isPolicyAllowed('allow_remote_sessions'), + get isHidden() { + return !isClaudeAISubscriber() || !isPolicyAllowed('allow_remote_sessions') + }, + load: () => import('./remote-env.js'), +} satisfies Command diff --git a/src/commands/remote-env/remote-env.tsx b/src/commands/remote-env/remote-env.tsx new file mode 100644 index 0000000..dce659a --- /dev/null +++ b/src/commands/remote-env/remote-env.tsx @@ -0,0 +1,7 @@ +import * as React from 'react'; +import { RemoteEnvironmentDialog } from '../../components/RemoteEnvironmentDialog.js'; +import type { LocalJSXCommandOnDone } from '../../types/command.js'; +export async function call(onDone: LocalJSXCommandOnDone): Promise { + return ; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIlJlbW90ZUVudmlyb25tZW50RGlhbG9nIiwiTG9jYWxKU1hDb21tYW5kT25Eb25lIiwiY2FsbCIsIm9uRG9uZSIsIlByb21pc2UiLCJSZWFjdE5vZGUiXSwic291cmNlcyI6WyJyZW1vdGUtZW52LnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB7IFJlbW90ZUVudmlyb25tZW50RGlhbG9nIH0gZnJvbSAnLi4vLi4vY29tcG9uZW50cy9SZW1vdGVFbnZpcm9ubWVudERpYWxvZy5qcydcbmltcG9ydCB0eXBlIHsgTG9jYWxKU1hDb21tYW5kT25Eb25lIH0gZnJvbSAnLi4vLi4vdHlwZXMvY29tbWFuZC5qcydcblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGNhbGwoXG4gIG9uRG9uZTogTG9jYWxKU1hDb21tYW5kT25Eb25lLFxuKTogUHJvbWlzZTxSZWFjdC5SZWFjdE5vZGU+IHtcbiAgcmV0dXJuIDxSZW1vdGVFbnZpcm9ubWVudERpYWxvZyBvbkRvbmU9e29uRG9uZX0gLz5cbn1cbiJdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLQSxLQUFLLE1BQU0sT0FBTztBQUM5QixTQUFTQyx1QkFBdUIsUUFBUSw2Q0FBNkM7QUFDckYsY0FBY0MscUJBQXFCLFFBQVEsd0JBQXdCO0FBRW5FLE9BQU8sZUFBZUMsSUFBSUEsQ0FDeEJDLE1BQU0sRUFBRUYscUJBQXFCLENBQzlCLEVBQUVHLE9BQU8sQ0FBQ0wsS0FBSyxDQUFDTSxTQUFTLENBQUMsQ0FBQztFQUMxQixPQUFPLENBQUMsdUJBQXVCLENBQUMsTUFBTSxDQUFDLENBQUNGLE1BQU0sQ0FBQyxHQUFHO0FBQ3BEIiwiaWdub3JlTGlzdCI6W119 \ No newline at end of file diff --git a/src/commands/remote-setup/api.ts b/src/commands/remote-setup/api.ts new file mode 100644 index 0000000..d08659c --- /dev/null +++ b/src/commands/remote-setup/api.ts @@ -0,0 +1,182 @@ +import axios from 'axios' +import { getOauthConfig } from '../../constants/oauth.js' +import { logForDebugging } from '../../utils/debug.js' +import { getOAuthHeaders, prepareApiRequest } from '../../utils/teleport/api.js' +import { fetchEnvironments } from '../../utils/teleport/environments.js' + +const CCR_BYOC_BETA_HEADER = 'ccr-byoc-2025-07-29' + +/** + * Wraps a raw GitHub token so that its string representation is redacted. + * `String(token)`, template literals, `JSON.stringify(token)`, and any + * attached error messages will show `[REDACTED:gh-token]` instead of the + * token value. Call `.reveal()` only at the single point where the raw + * value is placed into an HTTP body. + */ +export class RedactedGithubToken { + readonly #value: string + constructor(raw: string) { + this.#value = raw + } + reveal(): string { + return this.#value + } + toString(): string { + return '[REDACTED:gh-token]' + } + toJSON(): string { + return '[REDACTED:gh-token]' + } + [Symbol.for('nodejs.util.inspect.custom')](): string { + return '[REDACTED:gh-token]' + } +} + +export type ImportTokenResult = { + github_username: string +} + +export type ImportTokenError = + | { kind: 'not_signed_in' } + | { kind: 'invalid_token' } + | { kind: 'server'; status: number } + | { kind: 'network' } + +/** + * POSTs a GitHub token to the CCR backend, which validates it against + * GitHub's /user endpoint and stores it Fernet-encrypted in sync_user_tokens. + * The stored token satisfies the same read paths as an OAuth token, so + * clone/push in claude.ai/code works immediately after this succeeds. + */ +export async function importGithubToken( + token: RedactedGithubToken, +): Promise< + | { ok: true; result: ImportTokenResult } + | { ok: false; error: ImportTokenError } +> { + let accessToken: string, orgUUID: string + try { + ;({ accessToken, orgUUID } = await prepareApiRequest()) + } catch { + return { ok: false, error: { kind: 'not_signed_in' } } + } + + const url = `${getOauthConfig().BASE_API_URL}/v1/code/github/import-token` + const headers = { + ...getOAuthHeaders(accessToken), + 'anthropic-beta': CCR_BYOC_BETA_HEADER, + 'x-organization-uuid': orgUUID, + } + + try { + const response = await axios.post( + url, + { token: token.reveal() }, + { headers, timeout: 15000, validateStatus: () => true }, + ) + if (response.status === 200) { + return { ok: true, result: response.data } + } + if (response.status === 400) { + return { ok: false, error: { kind: 'invalid_token' } } + } + if (response.status === 401) { + return { ok: false, error: { kind: 'not_signed_in' } } + } + logForDebugging(`import-token returned ${response.status}`, { + level: 'error', + }) + return { ok: false, error: { kind: 'server', status: response.status } } + } catch (err) { + if (axios.isAxiosError(err)) { + // err.config.data would contain the POST body with the raw token. + // Do not include it in any log. The error code alone is enough. + logForDebugging(`import-token network error: ${err.code ?? 'unknown'}`, { + level: 'error', + }) + } + return { ok: false, error: { kind: 'network' } } + } +} + +async function hasExistingEnvironment(): Promise { + try { + const envs = await fetchEnvironments() + return envs.length > 0 + } catch { + return false + } +} + +/** + * Best-effort default environment creation. Mirrors the web onboarding's + * DEFAULT_CLOUD_ENVIRONMENT_REQUEST so a first-time user lands on the + * composer instead of env-setup. Checks for existing environments first + * so re-running /web-setup doesn't pile up duplicates. Failures are + * non-fatal — the token import already succeeded, and the web state + * machine falls back to env-setup on next load. + */ +export async function createDefaultEnvironment(): Promise { + let accessToken: string, orgUUID: string + try { + ;({ accessToken, orgUUID } = await prepareApiRequest()) + } catch { + return false + } + + if (await hasExistingEnvironment()) { + return true + } + + // The /private/organizations/{org}/ path rejects CLI OAuth tokens (wrong + // auth dep). The public path uses build_flexible_auth — same path + // fetchEnvironments() uses. Org is passed via x-organization-uuid header. + const url = `${getOauthConfig().BASE_API_URL}/v1/environment_providers/cloud/create` + const headers = { + ...getOAuthHeaders(accessToken), + 'x-organization-uuid': orgUUID, + } + + try { + const response = await axios.post( + url, + { + name: 'Default', + kind: 'anthropic_cloud', + description: 'Default - trusted network access', + config: { + environment_type: 'anthropic', + cwd: '/home/user', + init_script: null, + environment: {}, + languages: [ + { name: 'python', version: '3.11' }, + { name: 'node', version: '20' }, + ], + network_config: { + allowed_hosts: [], + allow_default_hosts: true, + }, + }, + }, + { headers, timeout: 15000, validateStatus: () => true }, + ) + return response.status >= 200 && response.status < 300 + } catch { + return false + } +} + +/** Returns true when the user has valid Claude OAuth credentials. */ +export async function isSignedIn(): Promise { + try { + await prepareApiRequest() + return true + } catch { + return false + } +} + +export function getCodeWebUrl(): string { + return `${getOauthConfig().CLAUDE_AI_ORIGIN}/code` +} diff --git a/src/commands/remote-setup/index.ts b/src/commands/remote-setup/index.ts new file mode 100644 index 0000000..7b291df --- /dev/null +++ b/src/commands/remote-setup/index.ts @@ -0,0 +1,20 @@ +import type { Command } from '../../commands.js' +import { getFeatureValue_CACHED_MAY_BE_STALE } from '../../services/analytics/growthbook.js' +import { isPolicyAllowed } from '../../services/policyLimits/index.js' + +const web = { + type: 'local-jsx', + name: 'web-setup', + description: + 'Setup Claude Code on the web (requires connecting your GitHub account)', + availability: ['claude-ai'], + isEnabled: () => + getFeatureValue_CACHED_MAY_BE_STALE('tengu_cobalt_lantern', false) && + isPolicyAllowed('allow_remote_sessions'), + get isHidden() { + return !isPolicyAllowed('allow_remote_sessions') + }, + load: () => import('./remote-setup.js'), +} satisfies Command + +export default web diff --git a/src/commands/remote-setup/remote-setup.tsx b/src/commands/remote-setup/remote-setup.tsx new file mode 100644 index 0000000..0092879 --- /dev/null +++ b/src/commands/remote-setup/remote-setup.tsx @@ -0,0 +1,187 @@ +import { execa } from 'execa'; +import * as React from 'react'; +import { useEffect, useState } from 'react'; +import { Select } from '../../components/CustomSelect/index.js'; +import { Dialog } from '../../components/design-system/Dialog.js'; +import { LoadingState } from '../../components/design-system/LoadingState.js'; +import { Box, Text } from '../../ink.js'; +import { logEvent, type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS as SafeString } from '../../services/analytics/index.js'; +import type { LocalJSXCommandOnDone } from '../../types/command.js'; +import { openBrowser } from '../../utils/browser.js'; +import { getGhAuthStatus } from '../../utils/github/ghAuthStatus.js'; +import { createDefaultEnvironment, getCodeWebUrl, type ImportTokenError, importGithubToken, isSignedIn, RedactedGithubToken } from './api.js'; +type CheckResult = { + status: 'not_signed_in'; +} | { + status: 'has_gh_token'; + token: RedactedGithubToken; +} | { + status: 'gh_not_installed'; +} | { + status: 'gh_not_authenticated'; +}; +async function checkLoginState(): Promise { + if (!(await isSignedIn())) { + return { + status: 'not_signed_in' + }; + } + const ghStatus = await getGhAuthStatus(); + if (ghStatus === 'not_installed') { + return { + status: 'gh_not_installed' + }; + } + if (ghStatus === 'not_authenticated') { + return { + status: 'gh_not_authenticated' + }; + } + + // ghStatus === 'authenticated'. getGhAuthStatus spawns with stdout:'ignore' + // (telemetry-safe); spawn once more with stdout:'pipe' to read the token. + const { + stdout + } = await execa('gh', ['auth', 'token'], { + stdout: 'pipe', + stderr: 'ignore', + timeout: 5000, + reject: false + }); + const trimmed = stdout.trim(); + if (!trimmed) { + return { + status: 'gh_not_authenticated' + }; + } + return { + status: 'has_gh_token', + token: new RedactedGithubToken(trimmed) + }; +} +function errorMessage(err: ImportTokenError, codeUrl: string): string { + switch (err.kind) { + case 'not_signed_in': + return `Login failed. Please visit ${codeUrl} and login using the GitHub App`; + case 'invalid_token': + return 'GitHub rejected that token. Run `gh auth login` and try again.'; + case 'server': + return `Server error (${err.status}). Try again in a moment.`; + case 'network': + return "Couldn't reach the server. Check your connection."; + } +} +type Step = { + name: 'checking'; +} | { + name: 'confirm'; + token: RedactedGithubToken; +} | { + name: 'uploading'; +}; +function Web({ + onDone +}: { + onDone: LocalJSXCommandOnDone; +}) { + const [step, setStep] = useState({ + name: 'checking' + }); + useEffect(() => { + logEvent('tengu_remote_setup_started', {}); + void checkLoginState().then(async result => { + switch (result.status) { + case 'not_signed_in': + logEvent('tengu_remote_setup_result', { + result: 'not_signed_in' as SafeString + }); + onDone('Not signed in to Claude. Run /login first.'); + return; + case 'gh_not_installed': + case 'gh_not_authenticated': + { + const url = `${getCodeWebUrl()}/onboarding?step=alt-auth`; + await openBrowser(url); + logEvent('tengu_remote_setup_result', { + result: result.status as SafeString + }); + onDone(result.status === 'gh_not_installed' ? `GitHub CLI not found. Install it via https://cli.github.com/, then run \`gh auth login\`, or connect GitHub on the web: ${url}` : `GitHub CLI not authenticated. Run \`gh auth login\` and try again, or connect GitHub on the web: ${url}`); + return; + } + case 'has_gh_token': + setStep({ + name: 'confirm', + token: result.token + }); + } + }); + // onDone is stable across renders; intentionally not in deps. + // eslint-disable-next-line react-hooks/exhaustive-deps + }, []); + const handleCancel = () => { + logEvent('tengu_remote_setup_result', { + result: 'cancelled' as SafeString + }); + onDone(); + }; + const handleConfirm = async (token: RedactedGithubToken) => { + setStep({ + name: 'uploading' + }); + const result = await importGithubToken(token); + if (!result.ok) { + logEvent('tengu_remote_setup_result', { + result: 'import_failed' as SafeString, + error_kind: result.error.kind as SafeString + }); + onDone(errorMessage(result.error, getCodeWebUrl())); + return; + } + + // Token import succeeded. Environment creation is best-effort — if it + // fails, the web state machine routes to env-setup on landing, which is + // one extra click but still better than the OAuth dance. + await createDefaultEnvironment(); + const url = getCodeWebUrl(); + await openBrowser(url); + logEvent('tengu_remote_setup_result', { + result: 'success' as SafeString + }); + onDone(`Connected as ${result.result.github_username}. Opened ${url}`); + }; + if (step.name === 'checking') { + return ; + } + if (step.name === 'uploading') { + return ; + } + const token = step.token; + return + + + Claude on the web requires connecting to your GitHub account to clone + and push code on your behalf. + + + Your local credentials are used to authenticate with GitHub + + + }; + $[8] = handleCancel; + $[9] = handleSelect; + $[10] = isLaunching; + $[11] = t6; + } else { + t6 = $[11]; + } + let t7; + if ($[12] !== handleCancel || $[13] !== t6) { + t7 = {t6}; + $[12] = handleCancel; + $[13] = t6; + $[14] = t7; + } else { + t7 = $[14]; + } + return t7; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsInVzZUNhbGxiYWNrIiwidXNlUmVmIiwidXNlU3RhdGUiLCJTZWxlY3QiLCJEaWFsb2ciLCJCb3giLCJUZXh0IiwiUHJvcHMiLCJvblByb2NlZWQiLCJzaWduYWwiLCJBYm9ydFNpZ25hbCIsIlByb21pc2UiLCJvbkNhbmNlbCIsIlVsdHJhcmV2aWV3T3ZlcmFnZURpYWxvZyIsInQwIiwiJCIsIl9jIiwiaXNMYXVuY2hpbmciLCJzZXRJc0xhdW5jaGluZyIsInQxIiwiU3ltYm9sIiwiZm9yIiwiQWJvcnRDb250cm9sbGVyIiwiYWJvcnRDb250cm9sbGVyUmVmIiwidDIiLCJ2YWx1ZSIsImN1cnJlbnQiLCJjYXRjaCIsImhhbmRsZVNlbGVjdCIsInQzIiwiYWJvcnQiLCJoYW5kbGVDYW5jZWwiLCJ0NCIsImxhYmVsIiwib3B0aW9ucyIsInQ1IiwidDYiLCJ0NyJdLCJzb3VyY2VzIjpbIlVsdHJhcmV2aWV3T3ZlcmFnZURpYWxvZy50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFJlYWN0LCB7IHVzZUNhbGxiYWNrLCB1c2VSZWYsIHVzZVN0YXRlIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBTZWxlY3QgfSBmcm9tICcuLi8uLi9jb21wb25lbnRzL0N1c3RvbVNlbGVjdC9zZWxlY3QuanMnXG5pbXBvcnQgeyBEaWFsb2cgfSBmcm9tICcuLi8uLi9jb21wb25lbnRzL2Rlc2lnbi1zeXN0ZW0vRGlhbG9nLmpzJ1xuaW1wb3J0IHsgQm94LCBUZXh0IH0gZnJvbSAnLi4vLi4vaW5rLmpzJ1xuXG50eXBlIFByb3BzID0ge1xuICBvblByb2NlZWQ6IChzaWduYWw6IEFib3J0U2lnbmFsKSA9PiBQcm9taXNlPHZvaWQ+XG4gIG9uQ2FuY2VsOiAoKSA9PiB2b2lkXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBVbHRyYXJldmlld092ZXJhZ2VEaWFsb2coe1xuICBvblByb2NlZWQsXG4gIG9uQ2FuY2VsLFxufTogUHJvcHMpOiBSZWFjdC5SZWFjdE5vZGUge1xuICBjb25zdCBbaXNMYXVuY2hpbmcsIHNldElzTGF1bmNoaW5nXSA9IHVzZVN0YXRlKGZhbHNlKVxuICBjb25zdCBhYm9ydENvbnRyb2xsZXJSZWYgPSB1c2VSZWYobmV3IEFib3J0Q29udHJvbGxlcigpKVxuXG4gIGNvbnN0IGhhbmRsZVNlbGVjdCA9IHVzZUNhbGxiYWNrKFxuICAgICh2YWx1ZTogc3RyaW5nKSA9PiB7XG4gICAgICBpZiAodmFsdWUgPT09ICdwcm9jZWVkJykge1xuICAgICAgICBzZXRJc0xhdW5jaGluZyh0cnVlKVxuICAgICAgICAvLyBJZiBvblByb2NlZWQgcmVqZWN0cyAoZS5nLiBsYXVuY2hSZW1vdGVSZXZpZXcgdGhyb3dzKSwgb25Eb25lIGlzXG4gICAgICAgIC8vIG5ldmVyIGNhbGxlZCBhbmQgdGhlIGRpYWxvZyBzdGF5cyBtb3VudGVkIOKAlCByZXN0b3JlIHRoZSBTZWxlY3Qgc29cbiAgICAgICAgLy8gdGhlIHVzZXIgY2FuIHJldHJ5IG9yIGNhbmNlbCBpbnN0ZWFkIG9mIHN0YXJpbmcgYXQgXCJMYXVuY2hpbmfigKZcIi5cbiAgICAgICAgdm9pZCBvblByb2NlZWQoYWJvcnRDb250cm9sbGVyUmVmLmN1cnJlbnQuc2lnbmFsKS5jYXRjaCgoKSA9PlxuICAgICAgICAgIHNldElzTGF1bmNoaW5nKGZhbHNlKSxcbiAgICAgICAgKVxuICAgICAgfSBlbHNlIHtcbiAgICAgICAgb25DYW5jZWwoKVxuICAgICAgfVxuICAgIH0sXG4gICAgW29uUHJvY2VlZCwgb25DYW5jZWxdLFxuICApXG5cbiAgLy8gRXNjYXBlIGR1cmluZyBsYXVuY2ggYWJvcnRzIHRoZSBpbi1mbGlnaHQgb25Qcm9jZWVkIHZpYSBzaWduYWwgc28gdGhlXG4gIC8vIGNhbGxlciBjYW4gc2tpcCBzaWRlIGVmZmVjdHMgKGNvbmZpcm1PdmVyYWdlLCBvbkRvbmUpIOKAlCBvdGhlcndpc2UgYVxuICAvLyBmaXJlLWFuZC1mb3JnZXQgbGF1bmNoIHdvdWxkIGtlZXAgcnVubmluZyBhbmQgYmlsbCBkZXNwaXRlIFwiY2FuY2VsbGVkXCIuXG4gIGNvbnN0IGhhbmRsZUNhbmNlbCA9IHVzZUNhbGxiYWNrKCgpID0+IHtcbiAgICBhYm9ydENvbnRyb2xsZXJSZWYuY3VycmVudC5hYm9ydCgpXG4gICAgb25DYW5jZWwoKVxuICB9LCBbb25DYW5jZWxdKVxuXG4gIGNvbnN0IG9wdGlvbnMgPSBbXG4gICAgeyBsYWJlbDogJ1Byb2NlZWQgd2l0aCBFeHRyYSBVc2FnZSBiaWxsaW5nJywgdmFsdWU6ICdwcm9jZWVkJyB9LFxuICAgIHsgbGFiZWw6ICdDYW5jZWwnLCB2YWx1ZTogJ2NhbmNlbCcgfSxcbiAgXVxuXG4gIHJldHVybiAoXG4gICAgPERpYWxvZ1xuICAgICAgdGl0bGU9XCJVbHRyYXJldmlldyBiaWxsaW5nXCJcbiAgICAgIG9uQ2FuY2VsPXtoYW5kbGVDYW5jZWx9XG4gICAgICBjb2xvcj1cImJhY2tncm91bmRcIlxuICAgID5cbiAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiIGdhcD17MX0+XG4gICAgICAgIDxUZXh0PlxuICAgICAgICAgIFlvdXIgZnJlZSB1bHRyYXJldmlld3MgZm9yIHRoaXMgb3JnYW5pemF0aW9uIGFyZSB1c2VkLiBGdXJ0aGVyIHJldmlld3NcbiAgICAgICAgICBiaWxsIGFzIEV4dHJhIFVzYWdlIChwYXktcGVyLXVzZSkuXG4gICAgICAgIDwvVGV4dD5cbiAgICAgICAge2lzTGF1bmNoaW5nID8gKFxuICAgICAgICAgIDxUZXh0IGNvbG9yPVwiYmFja2dyb3VuZFwiPkxhdW5jaGluZ+KApjwvVGV4dD5cbiAgICAgICAgKSA6IChcbiAgICAgICAgICA8U2VsZWN0XG4gICAgICAgICAgICBvcHRpb25zPXtvcHRpb25zfVxuICAgICAgICAgICAgb25DaGFuZ2U9e2hhbmRsZVNlbGVjdH1cbiAgICAgICAgICAgIG9uQ2FuY2VsPXtoYW5kbGVDYW5jZWx9XG4gICAgICAgICAgLz5cbiAgICAgICAgKX1cbiAgICAgIDwvQm94PlxuICAgIDwvRGlhbG9nPlxuICAp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 \ No newline at end of file diff --git a/src/commands/review/reviewRemote.ts b/src/commands/review/reviewRemote.ts new file mode 100644 index 0000000..0b80e33 --- /dev/null +++ b/src/commands/review/reviewRemote.ts @@ -0,0 +1,316 @@ +/** + * Teleported /ultrareview execution. Creates a CCR session with the current repo, + * sends the review prompt as the initial message, and registers a + * RemoteAgentTask so the polling loop pipes results back into the local + * session via task-notification. Mirrors the /ultraplan → CCR flow. + * + * TODO(#22051): pass useBundleMode once landed so local-only / uncommitted + * repo state is captured. The GitHub-clone path (current) only works for + * pushed branches on repos with the Claude GitHub app installed. + */ + +import type { ContentBlockParam } from '@anthropic-ai/sdk/resources/messages.js' +import { getFeatureValue_CACHED_MAY_BE_STALE } from '../../services/analytics/growthbook.js' +import { + type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + logEvent, +} from '../../services/analytics/index.js' +import { fetchUltrareviewQuota } from '../../services/api/ultrareviewQuota.js' +import { fetchUtilization } from '../../services/api/usage.js' +import type { ToolUseContext } from '../../Tool.js' +import { + checkRemoteAgentEligibility, + formatPreconditionError, + getRemoteTaskSessionUrl, + registerRemoteAgentTask, +} from '../../tasks/RemoteAgentTask/RemoteAgentTask.js' +import { isEnterpriseSubscriber, isTeamSubscriber } from '../../utils/auth.js' +import { detectCurrentRepositoryWithHost } from '../../utils/detectRepository.js' +import { execFileNoThrow } from '../../utils/execFileNoThrow.js' +import { getDefaultBranch, gitExe } from '../../utils/git.js' +import { teleportToRemote } from '../../utils/teleport.js' + +// One-time session flag: once the user confirms overage billing via the +// dialog, all subsequent /ultrareview invocations in this session proceed +// without re-prompting. +let sessionOverageConfirmed = false + +export function confirmOverage(): void { + sessionOverageConfirmed = true +} + +export type OverageGate = + | { kind: 'proceed'; billingNote: string } + | { kind: 'not-enabled' } + | { kind: 'low-balance'; available: number } + | { kind: 'needs-confirm' } + +/** + * Determine whether the user can launch an ultrareview and under what + * billing terms. Fetches quota and utilization in parallel. + */ +export async function checkOverageGate(): Promise { + // Team and Enterprise plans include ultrareview — no free-review quota + // or Extra Usage dialog. The quota endpoint is scoped to consumer plans + // (pro/max); hitting it on team/ent would surface a confusing dialog. + if (isTeamSubscriber() || isEnterpriseSubscriber()) { + return { kind: 'proceed', billingNote: '' } + } + + const [quota, utilization] = await Promise.all([ + fetchUltrareviewQuota(), + fetchUtilization().catch(() => null), + ]) + + // No quota info (non-subscriber or endpoint down) — let it through, + // server-side billing will handle it. + if (!quota) { + return { kind: 'proceed', billingNote: '' } + } + + if (quota.reviews_remaining > 0) { + return { + kind: 'proceed', + billingNote: ` This is free ultrareview ${quota.reviews_used + 1} of ${quota.reviews_limit}.`, + } + } + + // Utilization fetch failed (transient network error, timeout, etc.) — + // let it through, same rationale as the quota fallback above. + if (!utilization) { + return { kind: 'proceed', billingNote: '' } + } + + // Free reviews exhausted — check Extra Usage setup. + const extraUsage = utilization.extra_usage + if (!extraUsage?.is_enabled) { + logEvent('tengu_review_overage_not_enabled', {}) + return { kind: 'not-enabled' } + } + + // Check available balance (null monthly_limit = unlimited). + const monthlyLimit = extraUsage.monthly_limit + const usedCredits = extraUsage.used_credits ?? 0 + const available = + monthlyLimit === null || monthlyLimit === undefined + ? Infinity + : monthlyLimit - usedCredits + + if (available < 10) { + logEvent('tengu_review_overage_low_balance', { available }) + return { kind: 'low-balance', available } + } + + if (!sessionOverageConfirmed) { + logEvent('tengu_review_overage_dialog_shown', {}) + return { kind: 'needs-confirm' } + } + + return { + kind: 'proceed', + billingNote: ' This review bills as Extra Usage.', + } +} + +/** + * Launch a teleported review session. Returns ContentBlockParam[] describing + * the launch outcome for injection into the local conversation (model is then + * queried with this content, so it can narrate the launch to the user). + * + * Returns ContentBlockParam[] with user-facing error messages on recoverable + * failures (missing merge-base, empty diff, bundle too large), or null on + * other failures so the caller falls through to the local-review prompt. + * Reason is captured in analytics. + * + * Caller must run checkOverageGate() BEFORE calling this function + * (ultrareviewCommand.tsx handles the dialog). + */ +export async function launchRemoteReview( + args: string, + context: ToolUseContext, + billingNote?: string, +): Promise { + const eligibility = await checkRemoteAgentEligibility() + // Synthetic DEFAULT_CODE_REVIEW_ENVIRONMENT_ID works without per-org CCR + // setup, so no_remote_environment isn't a blocker. Server-side quota + // consume at session creation routes billing: first N zero-rate, then + // anthropic:cccr org-service-key (overage-only). + if (!eligibility.eligible) { + const blockers = eligibility.errors.filter( + e => e.type !== 'no_remote_environment', + ) + if (blockers.length > 0) { + logEvent('tengu_review_remote_precondition_failed', { + precondition_errors: blockers + .map(e => e.type) + .join( + ',', + ) as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + const reasons = blockers.map(formatPreconditionError).join('\n') + return [ + { + type: 'text', + text: `Ultrareview cannot launch:\n${reasons}`, + }, + ] + } + } + + const resolvedBillingNote = billingNote ?? '' + + const prNumber = args.trim() + const isPrNumber = /^\d+$/.test(prNumber) + // Synthetic code_review env. Go taggedid.FromUUID(TagEnvironment, + // UUID{...,0x02}) encodes with version prefix '01' — NOT Python's + // legacy tagged_id() format. Verified in prod. + const CODE_REVIEW_ENV_ID = 'env_011111111111111111111113' + // Lite-review bypasses bughunter.go entirely, so it doesn't see the + // webhook's bug_hunter_config (different GB project). These env vars are + // the only tuning surface — without them, run_hunt.sh's bash defaults + // apply (60min, 120s agent timeout), and 120s kills verifiers mid-run + // which causes infinite respawn. + // + // total_wallclock must stay below RemoteAgentTask's 30min poll timeout + // with headroom for finalization (~3min synthesis). Per-field guards + // match autoDream.ts — GB cache can return stale wrong-type values. + const raw = getFeatureValue_CACHED_MAY_BE_STALE | null>('tengu_review_bughunter_config', null) + const posInt = (v: unknown, fallback: number, max?: number): number => { + if (typeof v !== 'number' || !Number.isFinite(v)) return fallback + const n = Math.floor(v) + if (n <= 0) return fallback + return max !== undefined && n > max ? fallback : n + } + // Upper bounds: 27min on wallclock leaves ~3min for finalization under + // RemoteAgentTask's 30min poll timeout. If GB is set above that, the + // hang we're fixing comes back — fall to the safe default instead. + const commonEnvVars = { + BUGHUNTER_DRY_RUN: '1', + BUGHUNTER_FLEET_SIZE: String(posInt(raw?.fleet_size, 5, 20)), + BUGHUNTER_MAX_DURATION: String(posInt(raw?.max_duration_minutes, 10, 25)), + BUGHUNTER_AGENT_TIMEOUT: String( + posInt(raw?.agent_timeout_seconds, 600, 1800), + ), + BUGHUNTER_TOTAL_WALLCLOCK: String( + posInt(raw?.total_wallclock_minutes, 22, 27), + ), + ...(process.env.BUGHUNTER_DEV_BUNDLE_B64 && { + BUGHUNTER_DEV_BUNDLE_B64: process.env.BUGHUNTER_DEV_BUNDLE_B64, + }), + } + + let session + let command + let target + if (isPrNumber) { + // PR mode: refs/pull/N/head via github.com. Orchestrator --pr N. + const repo = await detectCurrentRepositoryWithHost() + if (!repo || repo.host !== 'github.com') { + logEvent('tengu_review_remote_precondition_failed', {}) + return null + } + session = await teleportToRemote({ + initialMessage: null, + description: `ultrareview: ${repo.owner}/${repo.name}#${prNumber}`, + signal: context.abortController.signal, + branchName: `refs/pull/${prNumber}/head`, + environmentId: CODE_REVIEW_ENV_ID, + environmentVariables: { + BUGHUNTER_PR_NUMBER: prNumber, + BUGHUNTER_REPOSITORY: `${repo.owner}/${repo.name}`, + ...commonEnvVars, + }, + }) + command = `/ultrareview ${prNumber}` + target = `${repo.owner}/${repo.name}#${prNumber}` + } else { + // Branch mode: bundle the working tree, orchestrator diffs against + // the fork point. No PR, no existing comments, no dedup. + const baseBranch = (await getDefaultBranch()) || 'main' + // Env-manager's `git remote remove origin` after bundle-clone + // deletes refs/remotes/origin/* — the base branch name won't resolve + // in the container. Pass the merge-base SHA instead: it's reachable + // from HEAD's history so `git diff ` works without a named ref. + const { stdout: mbOut, code: mbCode } = await execFileNoThrow( + gitExe(), + ['merge-base', baseBranch, 'HEAD'], + { preserveOutputOnError: false }, + ) + const mergeBaseSha = mbOut.trim() + if (mbCode !== 0 || !mergeBaseSha) { + logEvent('tengu_review_remote_precondition_failed', {}) + return [ + { + type: 'text', + text: `Could not find merge-base with ${baseBranch}. Make sure you're in a git repo with a ${baseBranch} branch.`, + }, + ] + } + + // Bail early on empty diffs instead of launching a container that + // will just echo "no changes". + const { stdout: diffStat, code: diffCode } = await execFileNoThrow( + gitExe(), + ['diff', '--shortstat', mergeBaseSha], + { preserveOutputOnError: false }, + ) + if (diffCode === 0 && !diffStat.trim()) { + logEvent('tengu_review_remote_precondition_failed', {}) + return [ + { + type: 'text', + text: `No changes against the ${baseBranch} fork point. Make some commits or stage files first.`, + }, + ] + } + + session = await teleportToRemote({ + initialMessage: null, + description: `ultrareview: ${baseBranch}`, + signal: context.abortController.signal, + useBundle: true, + environmentId: CODE_REVIEW_ENV_ID, + environmentVariables: { + BUGHUNTER_BASE_BRANCH: mergeBaseSha, + ...commonEnvVars, + }, + }) + if (!session) { + logEvent('tengu_review_remote_teleport_failed', {}) + return [ + { + type: 'text', + text: 'Repo is too large. Push a PR and use `/ultrareview ` instead.', + }, + ] + } + command = '/ultrareview' + target = baseBranch + } + + if (!session) { + logEvent('tengu_review_remote_teleport_failed', {}) + return null + } + registerRemoteAgentTask({ + remoteTaskType: 'ultrareview', + session, + command, + context, + isRemoteReview: true, + }) + logEvent('tengu_review_remote_launched', {}) + const sessionUrl = getRemoteTaskSessionUrl(session.id) + // Concise — the tool-output block is visible to the user, so the model + // shouldn't echo the same info. Just enough for Claude to acknowledge the + // launch without restating the target/URL (both already printed above). + return [ + { + type: 'text', + text: `Ultrareview launched for ${target} (~10–20 min, runs in the cloud). Track: ${sessionUrl}${resolvedBillingNote} Findings arrive via task-notification. Briefly acknowledge the launch to the user without repeating the target or URL — both are already visible in the tool output above.`, + }, + ] +} diff --git a/src/commands/review/ultrareviewCommand.tsx b/src/commands/review/ultrareviewCommand.tsx new file mode 100644 index 0000000..3af5f5c --- /dev/null +++ b/src/commands/review/ultrareviewCommand.tsx @@ -0,0 +1,58 @@ +import type { ContentBlockParam } from '@anthropic-ai/sdk/resources/messages.js'; +import React from 'react'; +import type { LocalJSXCommandCall, LocalJSXCommandOnDone } from '../../types/command.js'; +import { checkOverageGate, confirmOverage, launchRemoteReview } from './reviewRemote.js'; +import { UltrareviewOverageDialog } from './UltrareviewOverageDialog.js'; +function contentBlocksToString(blocks: ContentBlockParam[]): string { + return blocks.map(b => b.type === 'text' ? b.text : '').filter(Boolean).join('\n'); +} +async function launchAndDone(args: string, context: Parameters[1], onDone: LocalJSXCommandOnDone, billingNote: string, signal?: AbortSignal): Promise { + const result = await launchRemoteReview(args, context, billingNote); + // User hit Escape during the ~5s launch — the dialog already showed + // "cancelled" and unmounted, so skip onDone (would write to a dead + // transcript slot) and let the caller skip confirmOverage. + if (signal?.aborted) return; + if (result) { + onDone(contentBlocksToString(result), { + shouldQuery: true + }); + } else { + // Precondition failures now return specific ContentBlockParam[] above. + // null only reaches here on teleport failure (PR mode) or non-github + // repo — both are CCR/repo connectivity issues. + onDone('Ultrareview failed to launch the remote session. Check that this is a GitHub repo and try again.', { + display: 'system' + }); + } +} +export const call: LocalJSXCommandCall = async (onDone, context, args) => { + const gate = await checkOverageGate(); + if (gate.kind === 'not-enabled') { + onDone('Free ultrareviews used. Enable Extra Usage at https://claude.ai/settings/billing to continue.', { + display: 'system' + }); + return null; + } + if (gate.kind === 'low-balance') { + onDone(`Balance too low to launch ultrareview ($${gate.available.toFixed(2)} available, $10 minimum). Top up at https://claude.ai/settings/billing`, { + display: 'system' + }); + return null; + } + if (gate.kind === 'needs-confirm') { + return { + await launchAndDone(args, context, onDone, ' This review bills as Extra Usage.', signal); + // Only persist the confirmation flag after a non-aborted launch — + // otherwise Escape-during-launch would leave the flag set and + // skip this dialog on the next attempt. + if (!signal.aborted) confirmOverage(); + }} onCancel={() => onDone('Ultrareview cancelled.', { + display: 'system' + })} />; + } + + // gate.kind === 'proceed' + await launchAndDone(args, context, onDone, gate.billingNote); + return null; +}; +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJDb250ZW50QmxvY2tQYXJhbSIsIlJlYWN0IiwiTG9jYWxKU1hDb21tYW5kQ2FsbCIsIkxvY2FsSlNYQ29tbWFuZE9uRG9uZSIsImNoZWNrT3ZlcmFnZUdhdGUiLCJjb25maXJtT3ZlcmFnZSIsImxhdW5jaFJlbW90ZVJldmlldyIsIlVsdHJhcmV2aWV3T3ZlcmFnZURpYWxvZyIsImNvbnRlbnRCbG9ja3NUb1N0cmluZyIsImJsb2NrcyIsIm1hcCIsImIiLCJ0eXBlIiwidGV4dCIsImZpbHRlciIsIkJvb2xlYW4iLCJqb2luIiwibGF1bmNoQW5kRG9uZSIsImFyZ3MiLCJjb250ZXh0IiwiUGFyYW1ldGVycyIsIm9uRG9uZSIsImJpbGxpbmdOb3RlIiwic2lnbmFsIiwiQWJvcnRTaWduYWwiLCJQcm9taXNlIiwicmVzdWx0IiwiYWJvcnRlZCIsInNob3VsZFF1ZXJ5IiwiZGlzcGxheSIsImNhbGwiLCJnYXRlIiwia2luZCIsImF2YWlsYWJsZSIsInRvRml4ZWQiXSwic291cmNlcyI6WyJ1bHRyYXJldmlld0NvbW1hbmQudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHsgQ29udGVudEJsb2NrUGFyYW0gfSBmcm9tICdAYW50aHJvcGljLWFpL3Nkay9yZXNvdXJjZXMvbWVzc2FnZXMuanMnXG5pbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgdHlwZSB7XG4gIExvY2FsSlNYQ29tbWFuZENhbGwsXG4gIExvY2FsSlNYQ29tbWFuZE9uRG9uZSxcbn0gZnJvbSAnLi4vLi4vdHlwZXMvY29tbWFuZC5qcydcbmltcG9ydCB7XG4gIGNoZWNrT3ZlcmFnZUdhdGUsXG4gIGNvbmZpcm1PdmVyYWdlLFxuICBsYXVuY2hSZW1vdGVSZXZpZXcsXG59IGZyb20gJy4vcmV2aWV3UmVtb3RlLmpzJ1xuaW1wb3J0IHsgVWx0cmFyZXZpZXdPdmVyYWdlRGlhbG9nIH0gZnJvbSAnLi9VbHRyYXJldmlld092ZXJhZ2VEaWFsb2cuanMnXG5cbmZ1bmN0aW9uIGNvbnRlbnRCbG9ja3NUb1N0cmluZyhibG9ja3M6IENvbnRlbnRCbG9ja1BhcmFtW10pOiBzdHJpbmcge1xuICByZXR1cm4gYmxvY2tzXG4gICAgLm1hcChiID0+IChiLnR5cGUgPT09ICd0ZXh0JyA/IGIudGV4dCA6ICcnKSlcbiAgICAuZmlsdGVyKEJvb2xlYW4pXG4gICAgLmpvaW4oJ1xcbicpXG59XG5cbmFzeW5jIGZ1bmN0aW9uIGxhdW5jaEFuZERvbmUoXG4gIGFyZ3M6IHN0cmluZyxcbiAgY29udGV4dDogUGFyYW1ldGVyczxMb2NhbEpTWENvbW1hbmRDYWxsPlsxXSxcbiAgb25Eb25lOiBMb2NhbEpTWENvbW1hbmRPbkRvbmUsXG4gIGJpbGxpbmdOb3RlOiBzdHJpbmcsXG4gIHNpZ25hbD86IEFib3J0U2lnbmFsLFxuKTogUHJvbWlzZTx2b2lkPiB7XG4gIGNvbnN0IHJlc3VsdCA9IGF3YWl0IGxhdW5jaFJlbW90ZVJldmlldyhhcmdzLCBjb250ZXh0LCBiaWxsaW5nTm90ZSlcbiAgLy8gVXNlciBoaXQgRXNjYXBlIGR1cmluZyB0aGUgfjVzIGxhdW5jaCDigJQgdGhlIGRpYWxvZyBhbHJlYWR5IHNob3dlZFxuICAvLyBcImNhbmNlbGxlZFwiIGFuZCB1bm1vdW50ZWQsIHNvIHNraXAgb25Eb25lICh3b3VsZCB3cml0ZSB0byBhIGRlYWRcbiAgLy8gdHJhbnNjcmlwdCBzbG90KSBhbmQgbGV0IHRoZSBjYWxsZXIgc2tpcCBjb25maXJtT3ZlcmFnZS5cbiAgaWYgKHNpZ25hbD8uYWJvcnRlZCkgcmV0dXJuXG4gIGlmIChyZXN1bHQpIHtcbiAgICBvbkRvbmUoY29udGVudEJsb2Nrc1RvU3RyaW5nKHJlc3VsdCksIHsgc2hvdWxkUXVlcnk6IHRydWUgfSlcbiAgfSBlbHNlIHtcbiAgICAvLyBQcmVjb25kaXRpb24gZmFpbHVyZXMgbm93IHJldHVybiBzcGVjaWZpYyBDb250ZW50QmxvY2tQYXJhbVtdIGFib3ZlLlxuICAgIC8vIG51bGwgb25seSByZWFjaGVzIGhlcmUgb24gdGVsZXBvcnQgZmFpbHVyZSAoUFIgbW9kZSkgb3Igbm9uLWdpdGh1YlxuICAgIC8vIHJlcG8g4oCUIGJvdGggYXJlIENDUi9yZXBvIGNvbm5lY3Rpdml0eSBpc3N1ZXMuXG4gICAgb25Eb25lKFxuICAgICAgJ1VsdHJhcmV2aWV3IGZhaWxlZCB0byBsYXVuY2ggdGhlIHJlbW90ZSBzZXNzaW9uLiBDaGVjayB0aGF0IHRoaXMgaXMgYSBHaXRIdWIgcmVwbyBhbmQgdHJ5IGFnYWluLicsXG4gICAgICB7IGRpc3BsYXk6ICdzeXN0ZW0nIH0sXG4gICAgKVxuICB9XG59XG5cbmV4cG9ydCBjb25zdCBjYWxsOiBMb2NhbEpTWENvbW1hbmRDYWxsID0gYXN5bmMgKG9uRG9uZSwgY29udGV4dCwgYXJncykgPT4ge1xuICBjb25zdCBnYXRlID0gYXdhaXQgY2hlY2tPdmVyYWdlR2F0ZSgpXG5cbiAgaWYgKGdhdGUua2luZCA9PT0gJ25vdC1lbmFibGVkJykge1xuICAgIG9uRG9uZShcbiAgICAgICdGcmVlIHVsdHJhcmV2aWV3cyB1c2VkLiBFbmFibGUgRXh0cmEgVXNhZ2UgYXQgaHR0cHM6Ly9jbGF1ZGUuYWkvc2V0dGluZ3MvYmlsbGluZyB0byBjb250aW51ZS4nLFxuICAgICAgeyBkaXNwbGF5OiAnc3lzdGVtJyB9LFxuICAgIClcbiAgICByZXR1cm4gbnVsbFxuICB9XG5cbiAgaWYgKGdhdGUua2luZCA9PT0gJ2xvdy1iYWxhbmNlJykge1xuICAgIG9uRG9uZShcbiAgICAgIGBCYWxhbmNlIHRvbyBsb3cgdG8gbGF1bmNoIHVsdHJhcmV2aWV3ICgkJHtnYXRlLmF2YWlsYWJsZS50b0ZpeGVkKDIpfSBhdmFpbGFibGUsICQxMCBtaW5pbXVtKS4gVG9wIHVwIGF0IGh0dHBzOi8vY2xhdWRlLmFpL3NldHRpbmdzL2JpbGxpbmdgLFxuICAgICAgeyBkaXNwbGF5OiAnc3lzdGVtJyB9LFxuICAgIClcbiAgICByZXR1cm4gbnVsbFxuICB9XG5cbiAgaWYgKGdhdGUua2luZCA9PT0gJ25lZWRzLWNvbmZpcm0nKSB7XG4gICAgcmV0dXJuIChcbiAgICAgIDxVbHRyYXJldmlld092ZXJhZ2VEaWFsb2dcbiAgICAgICAgb25Qcm9jZWVkPXthc3luYyBzaWduYWwgPT4ge1xuICAgICAgICAgIGF3YWl0IGxhdW5jaEFuZERvbmUoXG4gICAgICAgICAgICBhcmdzLFxuICAgICAgICAgICAgY29udGV4dCxcbiAgICAgICAgICAgIG9uRG9uZSxcbiAgICAgICAgICAgICcgVGhpcyByZXZpZXcgYmlsbHMgYXMgRXh0cmEgVXNhZ2UuJyxcbiAgICAgICAgICAgIHNpZ25hbCxcbiAgICAgICAgICApXG4gICAgICAgICAgLy8gT25seSBwZXJzaXN0IHRoZSBjb25maXJtYXRpb24gZmxhZyBhZnRlciBhIG5vbi1hYm9ydGVkIGxhdW5jaCDigJRcbiAgICAgICAgICAvLyBvdGhlcndpc2UgRXNjYXBlLWR1cmluZy1sYXVuY2ggd291bGQgbGVhdmUgdGhlIGZsYWcgc2V0IGFuZFxuICAgICAgICAgIC8vIHNraXAgdGhpcyBkaWFsb2cgb24gdGhlIG5leHQgYXR0ZW1wdC5cbiAgICAgICAgICBpZiAoIXNpZ25hbC5hYm9ydGVkKSBjb25maXJtT3ZlcmFnZSgpXG4gICAgICAgIH19XG4gICAgICAgIG9uQ2FuY2VsPXsoKSA9PiBvbkRvbmUoJ1VsdHJhcmV2aWV3IGNhbmNlbGxlZC4nLCB7IGRpc3BsYXk6ICdzeXN0ZW0nIH0pfVxuICAgICAgLz5cbiAgICApXG4gIH1cblxuICAvLyBnYXRlLmtpbmQgPT09ICdwcm9jZWVkJ1xuICBhd2FpdCBsYXVuY2hBbmREb25lKGFyZ3MsIGNvbnRleHQsIG9uRG9uZSwgZ2F0ZS5iaWxsaW5nTm90ZSlcbiAgcmV0dXJuIG51bGxcbn1cbiJdLCJtYXBwaW5ncyI6IkFBQUEsY0FBY0EsaUJBQWlCLFFBQVEseUNBQXlDO0FBQ2hGLE9BQU9DLEtBQUssTUFBTSxPQUFPO0FBQ3pCLGNBQ0VDLG1CQUFtQixFQUNuQkMscUJBQXFCLFFBQ2hCLHdCQUF3QjtBQUMvQixTQUNFQyxnQkFBZ0IsRUFDaEJDLGNBQWMsRUFDZEMsa0JB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 \ No newline at end of file diff --git a/src/commands/review/ultrareviewEnabled.ts b/src/commands/review/ultrareviewEnabled.ts new file mode 100644 index 0000000..d10e5f5 --- /dev/null +++ b/src/commands/review/ultrareviewEnabled.ts @@ -0,0 +1,14 @@ +import { getFeatureValue_CACHED_MAY_BE_STALE } from '../../services/analytics/growthbook.js' + +/** + * Runtime gate for /ultrareview. GB config's `enabled` field controls + * visibility — isEnabled() on the command filters it from getCommands() + * when false, so ungated users don't see the command at all. + */ +export function isUltrareviewEnabled(): boolean { + const cfg = getFeatureValue_CACHED_MAY_BE_STALE | null>('tengu_review_bughunter_config', null) + return cfg?.enabled === true +} diff --git a/src/commands/rewind/index.ts b/src/commands/rewind/index.ts new file mode 100644 index 0000000..cfce193 --- /dev/null +++ b/src/commands/rewind/index.ts @@ -0,0 +1,13 @@ +import type { Command } from '../../commands.js' + +const rewind = { + description: `Restore the code and/or conversation to a previous point`, + name: 'rewind', + aliases: ['checkpoint'], + argumentHint: '', + type: 'local', + supportsNonInteractive: false, + load: () => import('./rewind.js'), +} satisfies Command + +export default rewind diff --git a/src/commands/rewind/rewind.ts b/src/commands/rewind/rewind.ts new file mode 100644 index 0000000..4b48a99 --- /dev/null +++ b/src/commands/rewind/rewind.ts @@ -0,0 +1,13 @@ +import type { LocalCommandResult } from '../../commands.js' +import type { ToolUseContext } from '../../Tool.js' + +export async function call( + _args: string, + context: ToolUseContext, +): Promise { + if (context.openMessageSelector) { + context.openMessageSelector() + } + // Return a skip message to not append any messages. + return { type: 'skip' } +} diff --git a/src/commands/sandbox-toggle/index.ts b/src/commands/sandbox-toggle/index.ts new file mode 100644 index 0000000..f467394 --- /dev/null +++ b/src/commands/sandbox-toggle/index.ts @@ -0,0 +1,50 @@ +import figures from 'figures' +import type { Command } from '../../commands.js' +import { SandboxManager } from '../../utils/sandbox/sandbox-adapter.js' + +const command = { + name: 'sandbox', + get description() { + const currentlyEnabled = SandboxManager.isSandboxingEnabled() + const autoAllow = SandboxManager.isAutoAllowBashIfSandboxedEnabled() + const allowUnsandboxed = SandboxManager.areUnsandboxedCommandsAllowed() + const isLocked = SandboxManager.areSandboxSettingsLockedByPolicy() + const hasDeps = SandboxManager.checkDependencies().errors.length === 0 + + // Show warning icon if dependencies missing, otherwise enabled/disabled status + let icon: string + if (!hasDeps) { + icon = figures.warning + } else { + icon = currentlyEnabled ? figures.tick : figures.circle + } + + let statusText = 'sandbox disabled' + if (currentlyEnabled) { + statusText = autoAllow + ? 'sandbox enabled (auto-allow)' + : 'sandbox enabled' + + // Add unsandboxed fallback status + statusText += allowUnsandboxed ? ', fallback allowed' : '' + } + + if (isLocked) { + statusText += ' (managed)' + } + + return `${icon} ${statusText} (⏎ to configure)` + }, + argumentHint: 'exclude "command pattern"', + get isHidden() { + return ( + !SandboxManager.isSupportedPlatform() || + !SandboxManager.isPlatformInEnabledList() + ) + }, + immediate: true, + type: 'local-jsx', + load: () => import('./sandbox-toggle.js'), +} satisfies Command + +export default command diff --git a/src/commands/sandbox-toggle/sandbox-toggle.tsx b/src/commands/sandbox-toggle/sandbox-toggle.tsx new file mode 100644 index 0000000..f56503c --- /dev/null +++ b/src/commands/sandbox-toggle/sandbox-toggle.tsx @@ -0,0 +1,83 @@ +import { relative } from 'path'; +import React from 'react'; +import { getCwdState } from '../../bootstrap/state.js'; +import { SandboxSettings } from '../../components/sandbox/SandboxSettings.js'; +import { color } from '../../ink.js'; +import { getPlatform } from '../../utils/platform.js'; +import { addToExcludedCommands, SandboxManager } from '../../utils/sandbox/sandbox-adapter.js'; +import { getSettings_DEPRECATED, getSettingsFilePathForSource } from '../../utils/settings/settings.js'; +import type { ThemeName } from '../../utils/theme.js'; +export async function call(onDone: (result?: string) => void, _context: unknown, args?: string): Promise { + const settings = getSettings_DEPRECATED(); + const themeName: ThemeName = settings.theme as ThemeName || 'light'; + const platform = getPlatform(); + if (!SandboxManager.isSupportedPlatform()) { + // WSL1 users will see this since isSupportedPlatform returns false for WSL1 + const errorMessage = platform === 'wsl' ? 'Error: Sandboxing requires WSL2. WSL1 is not supported.' : 'Error: Sandboxing is currently only supported on macOS, Linux, and WSL2.'; + const message = color('error', themeName)(errorMessage); + onDone(message); + return null; + } + + // Check dependencies - get structured result with errors/warnings + const depCheck = SandboxManager.checkDependencies(); + + // Check if platform is in enabledPlatforms list (undocumented enterprise setting) + if (!SandboxManager.isPlatformInEnabledList()) { + const message = color('error', themeName)(`Error: Sandboxing is disabled for this platform (${platform}) via the enabledPlatforms setting.`); + onDone(message); + return null; + } + + // Check if sandbox settings are locked by higher-priority settings + if (SandboxManager.areSandboxSettingsLockedByPolicy()) { + const message = color('error', themeName)('Error: Sandbox settings are overridden by a higher-priority configuration and cannot be changed locally.'); + onDone(message); + return null; + } + + // Parse the arguments + const trimmedArgs = args?.trim() || ''; + + // If no args, show the interactive menu + if (!trimmedArgs) { + return ; + } + + // Handle subcommands + if (trimmedArgs) { + const parts = trimmedArgs.split(' '); + const subcommand = parts[0]; + if (subcommand === 'exclude') { + // Handle exclude subcommand + const commandPattern = trimmedArgs.slice('exclude '.length).trim(); + if (!commandPattern) { + const message = color('error', themeName)('Error: Please provide a command pattern to exclude (e.g., /sandbox exclude "npm run test:*")'); + onDone(message); + return null; + } + + // Remove quotes if present + const cleanPattern = commandPattern.replace(/^["']|["']$/g, ''); + + // Add to excludedCommands + addToExcludedCommands(cleanPattern); + + // Get the local settings path and make it relative to cwd + const localSettingsPath = getSettingsFilePathForSource('localSettings'); + const relativePath = localSettingsPath ? relative(getCwdState(), localSettingsPath) : '.claude/settings.local.json'; + const message = color('success', themeName)(`Added "${cleanPattern}" to excluded commands in ${relativePath}`); + onDone(message); + return null; + } else { + // Unknown subcommand + const message = color('error', themeName)(`Error: Unknown subcommand "${subcommand}". Available subcommand: exclude`); + onDone(message); + return null; + } + } + + // Should never reach here since we handle all cases above + return null; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJyZWxhdGl2ZSIsIlJlYWN0IiwiZ2V0Q3dkU3RhdGUiLCJTYW5kYm94U2V0dGluZ3MiLCJjb2xvciIsImdldFBsYXRmb3JtIiwiYWRkVG9FeGNsdWRlZENvbW1hbmRzIiwiU2FuZGJveE1hbmFnZXIiLCJnZXRTZXR0aW5nc19ERVBSRUNBVEVEIiwiZ2V0U2V0dGluZ3NGaWxlUGF0aEZvclNvdXJjZSIsIlRoZW1lTmFtZSIsImNhbGwiLCJvbkRvbmUiLCJyZXN1bHQiLCJfY29udGV4dCIsImFyZ3MiLCJQcm9taXNlIiwiUmVhY3ROb2RlIiwic2V0dGluZ3MiLCJ0aGVtZU5hbWUiLCJ0aGVtZSIsInBsYXRmb3JtIiwiaXNTdXBwb3J0ZWRQbGF0Zm9ybSIsImVycm9yTWVzc2FnZSIsIm1lc3NhZ2UiLCJkZXBDaGVjayIsImNoZWNrRGVwZW5kZW5jaWVzIiwiaXNQbGF0Zm9ybUluRW5hYmxlZExpc3QiLCJhcmVTYW5kYm94U2V0dGluZ3NMb2NrZWRCeVBvbGljeSIsInRyaW1tZWRBcmdzIiwidHJpbSIsInBhcnRzIiwic3BsaXQiLCJzdWJjb21tYW5kIiwiY29tbWFuZFBhdHRlcm4iLCJzbGljZSIsImxlbmd0aCIsImNsZWFuUGF0dGVybiIsInJlcGxhY2UiLCJsb2NhbFNldHRpbmdzUGF0aCIsInJlbGF0aXZlUGF0aCJdLCJzb3VyY2VzIjpbInNhbmRib3gtdG9nZ2xlLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyByZWxhdGl2ZSB9IGZyb20gJ3BhdGgnXG5pbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBnZXRDd2RTdGF0ZSB9IGZyb20gJy4uLy4uL2Jvb3RzdHJhcC9zdGF0ZS5qcydcbmltcG9ydCB7IFNhbmRib3hTZXR0aW5ncyB9IGZyb20gJy4uLy4uL2NvbXBvbmVudHMvc2FuZGJveC9TYW5kYm94U2V0dGluZ3MuanMnXG5pbXBvcnQgeyBjb2xvciB9IGZyb20gJy4uLy4uL2luay5qcydcbmltcG9ydCB7IGdldFBsYXRmb3JtIH0gZnJvbSAnLi4vLi4vdXRpbHMvcGxhdGZvcm0uanMnXG5pbXBvcnQge1xuICBhZGRUb0V4Y2x1ZGVkQ29tbWFuZHMsXG4gIFNhbmRib3hNYW5hZ2VyLFxufSBmcm9tICcuLi8uLi91dGlscy9zYW5kYm94L3NhbmRib3gtYWRhcHRlci5qcydcbmltcG9ydCB7XG4gIGdldFNldHRpbmdzX0RFUFJFQ0FURUQsXG4gIGdldFNldHRpbmdzRmlsZVBhdGhGb3JTb3VyY2UsXG59IGZyb20gJy4uLy4uL3V0aWxzL3NldHRpbmdzL3NldHRpbmdzLmpzJ1xuaW1wb3J0IHR5cGUgeyBUaGVtZU5hbWUgfSBmcm9tICcuLi8uLi91dGlscy90aGVtZS5qcydcblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGNhbGwoXG4gIG9uRG9uZTogKHJlc3VsdD86IHN0cmluZykgPT4gdm9pZCxcbiAgX2NvbnRleHQ6IHVua25vd24sXG4gIGFyZ3M/OiBzdHJpbmcsXG4pOiBQcm9taXNlPFJlYWN0LlJlYWN0Tm9kZSB8IG51bGw+IHtcbiAgY29uc3Qgc2V0dGluZ3MgPSBnZXRTZXR0aW5nc19ERVBSRUNBVEVEKClcbiAgY29uc3QgdGhlbWVOYW1lOiBUaGVtZU5hbWUgPSAoc2V0dGluZ3MudGhlbWUgYXMgVGhlbWVOYW1lKSB8fCAnbGlnaHQnXG5cbiAgY29uc3QgcGxhdGZvcm0gPSBnZXRQbGF0Zm9ybSgpXG5cbiAgaWYgKCFTYW5kYm94TWFuYWdlci5pc1N1cHBvcnRlZFBsYXRmb3JtKCkpIHtcbiAgICAvLyBXU0wxIHVzZXJzIHdpbGwgc2VlIHRoaXMgc2luY2UgaXNTdXBwb3J0ZWRQbGF0Zm9ybSByZXR1cm5zIGZhbHNlIGZvciBXU0wxXG4gICAgY29uc3QgZXJyb3JNZXNzYWdlID1cbiAgICAgIHBsYXRmb3JtID09PSAnd3NsJ1xuICAgICAgICA/ICdFcnJvcjogU2FuZGJveGluZyByZXF1aXJlcyBXU0wyLiBXU0wxIGlzIG5vdCBzdXBwb3J0ZWQuJ1xuICAgICAgICA6ICdFcnJvcjogU2FuZGJveGluZyBpcyBjdXJyZW50bHkgb25seSBzdXBwb3J0ZWQgb24gbWFjT1MsIExpbnV4LCBhbmQgV1NMMi4nXG4gICAgY29uc3QgbWVzc2FnZSA9IGNvbG9yKCdlcnJvcicsIHRoZW1lTmFtZSkoZXJyb3JNZXNzYWdlKVxuICAgIG9uRG9uZShtZXNzYWdlKVxuICAgIHJldHVybiBudWxsXG4gIH1cblxuICAvLyBDaGVjayBkZXBlbmRlbmNpZXMgLSBnZXQgc3RydWN0dXJlZCByZXN1bHQgd2l0aCBlcnJvcnMvd2FybmluZ3NcbiAgY29uc3QgZGVwQ2hlY2sgPSBTYW5kYm94TWFuYWdlci5jaGVja0RlcGVuZGVuY2llcygpXG5cbiAgLy8gQ2hlY2sgaWYgcGxhdGZvcm0gaXMgaW4gZW5hYmxlZFBsYXRmb3JtcyBsaXN0ICh1bmRvY3VtZW50ZWQgZW50ZXJwcmlzZSBzZXR0aW5nKVxuICBpZiAoIVNhbmRib3hNYW5hZ2VyLmlzUGxhdGZvcm1JbkVuYWJsZWRMaXN0KCkpIHtcbiAgICBjb25zdCBtZXNzYWdlID0gY29sb3IoXG4gICAgICAnZXJyb3InLFxuICAgICAgdGhlbWVOYW1lLFxuICAgICkoXG4gICAgICBgRXJyb3I6IFNhbmRib3hpbmcgaXMgZGlzYWJsZWQgZm9yIHRoaXMgcGxhdGZvcm0gKCR7cGxhdGZvcm19KSB2aWEgdGhlIGVuYWJsZWRQbGF0Zm9ybXMgc2V0dGluZy5gLFxuICAgIClcbiAgICBvbkRvbmUobWVzc2FnZSlcbiAgICByZXR1cm4gbnVsbFxuICB9XG5cbiAgLy8gQ2hlY2sgaWYgc2FuZGJveCBzZXR0aW5ncyBhcmUgbG9ja2VkIGJ5IGhpZ2hlci1wcmlvcml0eSBzZXR0aW5nc1xuICBpZiAoU2FuZGJveE1hbmFnZXIuYXJlU2FuZGJveFNldHRpbmdzTG9ja2VkQnlQb2xpY3koKSkge1xuICAgIGNvbnN0IG1lc3NhZ2UgPSBjb2xvcihcbiAgICAgICdlcnJvcicsXG4gICAgICB0aGVtZU5hbWUsXG4gICAgKShcbiAgICAgICdFcnJvcjogU2FuZGJveCBzZXR0aW5ncyBhcmUgb3ZlcnJpZGRlbiBieSBhIGhpZ2hlci1wcmlvcml0eSBjb25maWd1cmF0aW9uIGFuZCBjYW5ub3QgYmUgY2hhbmdlZCBsb2NhbGx5LicsXG4gICAgKVxuICAgIG9uRG9uZShtZXNzYWdlKVxuICAgIHJldHVybiBudWxsXG4gIH1cblxuICAvLyBQYXJzZSB0aGUgYXJndW1lbnRzXG4gIGNvbnN0IHRyaW1tZWRBcmdzID0gYXJncz8udHJpbSgpIHx8ICcnXG5cbiAgLy8gSWYgbm8gYXJncywgc2hvdyB0aGUgaW50ZXJhY3RpdmUgbWVudVxuICBpZiAoIXRyaW1tZWRBcmdzKSB7XG4gICAgcmV0dXJuIDxTYW5kYm94U2V0dGluZ3Mgb25Db21wbGV0ZT17b25Eb25lfSBkZXBDaGVjaz17ZGVwQ2hlY2t9IC8+XG4gIH1cblxuICAvLyBIYW5kbGUgc3ViY29tbWFuZHNcbiAgaWYgKHRyaW1tZWRBcmdzKSB7XG4gICAgY29uc3QgcGFydHMgPSB0cmltbWVkQXJncy5zcGxpdCgnICcpXG4gICAgY29uc3Qgc3ViY29tbWFuZCA9IHBhcnRzWzBdXG5cbiAgICBpZiAoc3ViY29tbWFuZCA9PT0gJ2V4Y2x1ZGUnKSB7XG4gICAgICAvLyBIYW5kbGUgZXhjbHVkZSBzdWJjb21tYW5kXG4gICAgICBjb25zdCBjb21tYW5kUGF0dGVybiA9IHRyaW1tZWRBcmdzLnNsaWNlKCdleGNsdWRlICcubGVuZ3RoKS50cmltKClcblxuICAgICAgaWYgKCFjb21tYW5kUGF0dGVybikge1xuICAgICAgICBjb25zdCBtZXNzYWdlID0gY29sb3IoXG4gICAgICAgICAgJ2Vycm9yJyxcbiAgICAgICAgICB0aGVtZU5hbWUsXG4gICAgICAgICkoXG4gICAgICAgICAgJ0Vycm9yOiBQbGVhc2UgcHJvdmlkZSBhIGNvbW1hbmQgcGF0dGVybiB0byBleGNsdWRlIChlLmcuLCAvc2FuZGJveCBleGNsdWRlIFwibnBtIHJ1biB0ZXN0OipcIiknLFxuICAgICAgICApXG4gICAgICAgIG9uRG9uZShtZXNzYWdlKVxuICAgICAgICByZXR1cm4gbnVsbFxuICAgICAgfVxuXG4gICAgICAvLyBSZW1vdmUgcXVvdGVzIGlmIHByZXNlbnRcbiAgICAgIGNvbnN0IGNsZWFuUGF0dGVybiA9IGNvbW1hbmRQYXR0ZXJuLnJlcGxhY2UoL15bXCInXXxbXCInXSQvZywgJycpXG5cbiAgICAgIC8vIEFkZCB0byBleGNsdWRlZENvbW1hbmRzXG4gICAgICBhZGRUb0V4Y2x1ZGVkQ29tbWFuZHMoY2xlYW5QYXR0ZXJuKVxuXG4gICAgICAvLyBHZXQgdGhlIGxvY2FsIHNldHRpbmdzIHBhdGggYW5kIG1ha2UgaXQgcmVsYXRpdmUgdG8gY3dkXG4gICAgICBjb25zdCBsb2NhbFNldHRpbmdzUGF0aCA9IGdldFNldHRpbmdzRmlsZVBhdGhGb3JTb3VyY2UoJ2xvY2FsU2V0dGluZ3MnKVxuICAgICAgY29uc3QgcmVsYXRpdmVQYXRoID0gbG9jYWxTZXR0aW5nc1BhdGhcbiAgICAgICAgPyByZWxhdGl2ZShnZXRDd2RTdGF0ZSgpLCBsb2NhbFNldHRpbmdzUGF0aClcbiAgICAgICAgOiAnLmNsYXVkZS9zZXR0aW5ncy5sb2NhbC5qc29uJ1xuXG4gICAgICBjb25zdCBtZXNzYWdlID0gY29sb3IoXG4gICAgICAgICdzdWNjZXNzJyxcbiAgICAgICAgdGhlbWVOYW1lLFxuICAgICAgKShgQWRkZWQgXCIke2NsZWFuUGF0dGVybn1cIiB0byBleGNsdWRlZCBjb21tYW5kcyBpbiAke3JlbGF0aXZlUGF0aH1gKVxuXG4gICAgICBvbkRvbmUobWVzc2FnZSlcbiAgICAgIHJldHVybiBudWxsXG4gICAgfSBlbHNlIHtcbiAgICAgIC8vIFVua25vd24gc3ViY29tbWFuZFxuICAgICAgY29uc3QgbWVzc2FnZSA9IGNvbG9yKFxuICAgICAgICAnZXJyb3InLFxuICAgICAgICB0aGVtZU5hbWUsXG4gICAgICApKFxuICAgICAgICBgRXJyb3I6IFVua25vd24gc3ViY29tbWFuZCBcIiR7c3ViY29tbWFuZH1cIi4gQXZhaWxhYmxlIHN1YmNvbW1hbmQ6IGV4Y2x1ZGVgLFxuICAgICAgKVxuICAgICAgb25Eb25lKG1lc3NhZ2UpXG4gICAgICByZXR1cm4gbnVsbFxuICAgIH1cbiAgfVxuXG4gIC8vIFNob3VsZCBuZXZlciByZWFjaCBoZXJlIHNpbmNlIHdlIGhhbmRsZSBhbGwgY2FzZXMgYWJvdmVcbiAgcmV0dXJuIG51bGxcbn1cbiJdLCJtYXBwaW5ncyI6IkFBQUEsU0FBU0EsUUFBUSxRQUFRLE1BQU07QUFDL0IsT0FBT0MsS0FBSyxNQUFNLE9BQU87QUFDekIsU0FBU0MsV0FBVyxRQUFRLDBCQUEwQjtBQUN0RCxTQUFTQyxlQUFlLFFBQVEsNkNBQTZDO0FBQzdFLFNBQVNDLEtBQUssUUFBUSxjQUFjO0FBQ3BDLFNBQVNDLFdBQVcsUUFBUSx5QkFBeUI7QUFDckQsU0FDRUMscUJBQXFCLEVBQ3JCQyxjQUFjLFFBQ1Qsd0NBQXdDO0FBQy9DLFNBQ0VDLHNCQUFzQixFQUN0QkMsNEJBQTRCLFFBQ3ZCLGtDQUFrQztBQUN6QyxjQUFjQyxTQUFTLFFBQVEsc0JBQXNCO0FBRXJELE9BQU8sZUFBZUMsSUFBSUEsQ0FDeEJDLE1BQU0sRUFBRSxDQUFDQyxNQUFlLENBQVIsRUFBRSxNQUFNLEVBQUUsR0FBRyxJQUFJLEVBQ2pDQyxRQUFRLEVBQUUsT0FBTyxFQUNqQkMsSUFBYSxDQUFSLEVBQUUsTUFBTSxDQUNkLEVBQUVDLE9BQU8sQ0FBQ2YsS0FBSyxDQUFDZ0IsU0FBUyxHQUFHLElBQUksQ0FBQyxDQUFDO0VBQ2pDLE1BQU1DLFFBQVEsR0FBR1Ysc0JBQXNCLENBQUMsQ0FBQztFQUN6QyxNQUFNVyxTQUFTLEVBQUVULFNBQVMsR0FBSVEsUUFBUSxDQUFDRSxLQUFLLElBQUlWLFNBQVMsSUFBSyxPQUFPO0VBRXJFLE1BQU1XLFFBQVEsR0FBR2hCLFdBQVcsQ0FBQyxDQUFDO0VBRTlCLElBQUksQ0FBQ0UsY0FBYyxDQUFDZSxtQkFBbUIsQ0FBQyxDQUFDLEVBQUU7SUFDekM7SUFDQSxNQUFNQyxZQUFZLEdBQ2hCRixRQUFRLEtBQUssS0FBSyxHQUNkLHlEQUF5RCxHQUN6RCwwRUFBMEU7SUFDaEYsTUFBTUcsT0FBTyxHQUFHcEIs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 \ No newline at end of file diff --git a/src/commands/security-review.ts b/src/commands/security-review.ts new file mode 100644 index 0000000..03f7057 --- /dev/null +++ b/src/commands/security-review.ts @@ -0,0 +1,243 @@ +import { parseFrontmatter } from '../utils/frontmatterParser.js' +import { parseSlashCommandToolsFromFrontmatter } from '../utils/markdownConfigLoader.js' +import { executeShellCommandsInPrompt } from '../utils/promptShellExecution.js' +import { createMovedToPluginCommand } from './createMovedToPluginCommand.js' + +const SECURITY_REVIEW_MARKDOWN = `--- +allowed-tools: Bash(git diff:*), Bash(git status:*), Bash(git log:*), Bash(git show:*), Bash(git remote show:*), Read, Glob, Grep, LS, Task +description: Complete a security review of the pending changes on the current branch +--- + +You are a senior security engineer conducting a focused security review of the changes on this branch. + +GIT STATUS: + +\`\`\` +!\`git status\` +\`\`\` + +FILES MODIFIED: + +\`\`\` +!\`git diff --name-only origin/HEAD...\` +\`\`\` + +COMMITS: + +\`\`\` +!\`git log --no-decorate origin/HEAD...\` +\`\`\` + +DIFF CONTENT: + +\`\`\` +!\`git diff origin/HEAD...\` +\`\`\` + +Review the complete diff above. This contains all code changes in the PR. + + +OBJECTIVE: +Perform a security-focused code review to identify HIGH-CONFIDENCE security vulnerabilities that could have real exploitation potential. This is not a general code review - focus ONLY on security implications newly added by this PR. Do not comment on existing security concerns. + +CRITICAL INSTRUCTIONS: +1. MINIMIZE FALSE POSITIVES: Only flag issues where you're >80% confident of actual exploitability +2. AVOID NOISE: Skip theoretical issues, style concerns, or low-impact findings +3. FOCUS ON IMPACT: Prioritize vulnerabilities that could lead to unauthorized access, data breaches, or system compromise +4. EXCLUSIONS: Do NOT report the following issue types: + - Denial of Service (DOS) vulnerabilities, even if they allow service disruption + - Secrets or sensitive data stored on disk (these are handled by other processes) + - Rate limiting or resource exhaustion issues + +SECURITY CATEGORIES TO EXAMINE: + +**Input Validation Vulnerabilities:** +- SQL injection via unsanitized user input +- Command injection in system calls or subprocesses +- XXE injection in XML parsing +- Template injection in templating engines +- NoSQL injection in database queries +- Path traversal in file operations + +**Authentication & Authorization Issues:** +- Authentication bypass logic +- Privilege escalation paths +- Session management flaws +- JWT token vulnerabilities +- Authorization logic bypasses + +**Crypto & Secrets Management:** +- Hardcoded API keys, passwords, or tokens +- Weak cryptographic algorithms or implementations +- Improper key storage or management +- Cryptographic randomness issues +- Certificate validation bypasses + +**Injection & Code Execution:** +- Remote code execution via deseralization +- Pickle injection in Python +- YAML deserialization vulnerabilities +- Eval injection in dynamic code execution +- XSS vulnerabilities in web applications (reflected, stored, DOM-based) + +**Data Exposure:** +- Sensitive data logging or storage +- PII handling violations +- API endpoint data leakage +- Debug information exposure + +Additional notes: +- Even if something is only exploitable from the local network, it can still be a HIGH severity issue + +ANALYSIS METHODOLOGY: + +Phase 1 - Repository Context Research (Use file search tools): +- Identify existing security frameworks and libraries in use +- Look for established secure coding patterns in the codebase +- Examine existing sanitization and validation patterns +- Understand the project's security model and threat model + +Phase 2 - Comparative Analysis: +- Compare new code changes against existing security patterns +- Identify deviations from established secure practices +- Look for inconsistent security implementations +- Flag code that introduces new attack surfaces + +Phase 3 - Vulnerability Assessment: +- Examine each modified file for security implications +- Trace data flow from user inputs to sensitive operations +- Look for privilege boundaries being crossed unsafely +- Identify injection points and unsafe deserialization + +REQUIRED OUTPUT FORMAT: + +You MUST output your findings in markdown. The markdown output should contain the file, line number, severity, category (e.g. \`sql_injection\` or \`xss\`), description, exploit scenario, and fix recommendation. + +For example: + +# Vuln 1: XSS: \`foo.py:42\` + +* Severity: High +* Description: User input from \`username\` parameter is directly interpolated into HTML without escaping, allowing reflected XSS attacks +* Exploit Scenario: Attacker crafts URL like /bar?q= to execute JavaScript in victim's browser, enabling session hijacking or data theft +* Recommendation: Use Flask's escape() function or Jinja2 templates with auto-escaping enabled for all user inputs rendered in HTML + +SEVERITY GUIDELINES: +- **HIGH**: Directly exploitable vulnerabilities leading to RCE, data breach, or authentication bypass +- **MEDIUM**: Vulnerabilities requiring specific conditions but with significant impact +- **LOW**: Defense-in-depth issues or lower-impact vulnerabilities + +CONFIDENCE SCORING: +- 0.9-1.0: Certain exploit path identified, tested if possible +- 0.8-0.9: Clear vulnerability pattern with known exploitation methods +- 0.7-0.8: Suspicious pattern requiring specific conditions to exploit +- Below 0.7: Don't report (too speculative) + +FINAL REMINDER: +Focus on HIGH and MEDIUM findings only. Better to miss some theoretical issues than flood the report with false positives. Each finding should be something a security engineer would confidently raise in a PR review. + +FALSE POSITIVE FILTERING: + +> You do not need to run commands to reproduce the vulnerability, just read the code to determine if it is a real vulnerability. Do not use the bash tool or write to any files. +> +> HARD EXCLUSIONS - Automatically exclude findings matching these patterns: +> 1. Denial of Service (DOS) vulnerabilities or resource exhaustion attacks. +> 2. Secrets or credentials stored on disk if they are otherwise secured. +> 3. Rate limiting concerns or service overload scenarios. +> 4. Memory consumption or CPU exhaustion issues. +> 5. Lack of input validation on non-security-critical fields without proven security impact. +> 6. Input sanitization concerns for GitHub Action workflows unless they are clearly triggerable via untrusted input. +> 7. A lack of hardening measures. Code is not expected to implement all security best practices, only flag concrete vulnerabilities. +> 8. Race conditions or timing attacks that are theoretical rather than practical issues. Only report a race condition if it is concretely problematic. +> 9. Vulnerabilities related to outdated third-party libraries. These are managed separately and should not be reported here. +> 10. Memory safety issues such as buffer overflows or use-after-free-vulnerabilities are impossible in rust. Do not report memory safety issues in rust or any other memory safe languages. +> 11. Files that are only unit tests or only used as part of running tests. +> 12. Log spoofing concerns. Outputting un-sanitized user input to logs is not a vulnerability. +> 13. SSRF vulnerabilities that only control the path. SSRF is only a concern if it can control the host or protocol. +> 14. Including user-controlled content in AI system prompts is not a vulnerability. +> 15. Regex injection. Injecting untrusted content into a regex is not a vulnerability. +> 16. Regex DOS concerns. +> 16. Insecure documentation. Do not report any findings in documentation files such as markdown files. +> 17. A lack of audit logs is not a vulnerability. +> +> PRECEDENTS - +> 1. Logging high value secrets in plaintext is a vulnerability. Logging URLs is assumed to be safe. +> 2. UUIDs can be assumed to be unguessable and do not need to be validated. +> 3. Environment variables and CLI flags are trusted values. Attackers are generally not able to modify them in a secure environment. Any attack that relies on controlling an environment variable is invalid. +> 4. Resource management issues such as memory or file descriptor leaks are not valid. +> 5. Subtle or low impact web vulnerabilities such as tabnabbing, XS-Leaks, prototype pollution, and open redirects should not be reported unless they are extremely high confidence. +> 6. React and Angular are generally secure against XSS. These frameworks do not need to sanitize or escape user input unless it is using dangerouslySetInnerHTML, bypassSecurityTrustHtml, or similar methods. Do not report XSS vulnerabilities in React or Angular components or tsx files unless they are using unsafe methods. +> 7. Most vulnerabilities in github action workflows are not exploitable in practice. Before validating a github action workflow vulnerability ensure it is concrete and has a very specific attack path. +> 8. A lack of permission checking or authentication in client-side JS/TS code is not a vulnerability. Client-side code is not trusted and does not need to implement these checks, they are handled on the server-side. The same applies to all flows that send untrusted data to the backend, the backend is responsible for validating and sanitizing all inputs. +> 9. Only include MEDIUM findings if they are obvious and concrete issues. +> 10. Most vulnerabilities in ipython notebooks (*.ipynb files) are not exploitable in practice. Before validating a notebook vulnerability ensure it is concrete and has a very specific attack path where untrusted input can trigger the vulnerability. +> 11. Logging non-PII data is not a vulnerability even if the data may be sensitive. Only report logging vulnerabilities if they expose sensitive information such as secrets, passwords, or personally identifiable information (PII). +> 12. Command injection vulnerabilities in shell scripts are generally not exploitable in practice since shell scripts generally do not run with untrusted user input. Only report command injection vulnerabilities in shell scripts if they are concrete and have a very specific attack path for untrusted input. +> +> SIGNAL QUALITY CRITERIA - For remaining findings, assess: +> 1. Is there a concrete, exploitable vulnerability with a clear attack path? +> 2. Does this represent a real security risk vs theoretical best practice? +> 3. Are there specific code locations and reproduction steps? +> 4. Would this finding be actionable for a security team? +> +> For each finding, assign a confidence score from 1-10: +> - 1-3: Low confidence, likely false positive or noise +> - 4-6: Medium confidence, needs investigation +> - 7-10: High confidence, likely true vulnerability + +START ANALYSIS: + +Begin your analysis now. Do this in 3 steps: + +1. Use a sub-task to identify vulnerabilities. Use the repository exploration tools to understand the codebase context, then analyze the PR changes for security implications. In the prompt for this sub-task, include all of the above. +2. Then for each vulnerability identified by the above sub-task, create a new sub-task to filter out false-positives. Launch these sub-tasks as parallel sub-tasks. In the prompt for these sub-tasks, include everything in the "FALSE POSITIVE FILTERING" instructions. +3. Filter out any vulnerabilities where the sub-task reported a confidence less than 8. + +Your final reply must contain the markdown report and nothing else.` + +export default createMovedToPluginCommand({ + name: 'security-review', + description: + 'Complete a security review of the pending changes on the current branch', + progressMessage: 'analyzing code changes for security risks', + pluginName: 'security-review', + pluginCommand: 'security-review', + async getPromptWhileMarketplaceIsPrivate(_args, context) { + // Parse frontmatter from the markdown + const parsed = parseFrontmatter(SECURITY_REVIEW_MARKDOWN) + + // Parse allowed tools from frontmatter + const allowedTools = parseSlashCommandToolsFromFrontmatter( + parsed.frontmatter['allowed-tools'], + ) + + // Execute bash commands in the prompt + const processedContent = await executeShellCommandsInPrompt( + parsed.content, + { + ...context, + getAppState() { + const appState = context.getAppState() + return { + ...appState, + toolPermissionContext: { + ...appState.toolPermissionContext, + alwaysAllowRules: { + ...appState.toolPermissionContext.alwaysAllowRules, + command: allowedTools, + }, + }, + } + }, + }, + 'security-review', + ) + + return [ + { + type: 'text', + text: processedContent, + }, + ] + }, +}) diff --git a/src/commands/session/index.ts b/src/commands/session/index.ts new file mode 100644 index 0000000..c661878 --- /dev/null +++ b/src/commands/session/index.ts @@ -0,0 +1,16 @@ +import { getIsRemoteMode } from '../../bootstrap/state.js' +import type { Command } from '../../commands.js' + +const session = { + type: 'local-jsx', + name: 'session', + aliases: ['remote'], + description: 'Show remote session URL and QR code', + isEnabled: () => getIsRemoteMode(), + get isHidden() { + return !getIsRemoteMode() + }, + load: () => import('./session.js'), +} satisfies Command + +export default session diff --git a/src/commands/session/session.tsx b/src/commands/session/session.tsx new file mode 100644 index 0000000..f4f6083 --- /dev/null +++ b/src/commands/session/session.tsx @@ -0,0 +1,140 @@ +import { c as _c } from "react/compiler-runtime"; +import { toString as qrToString } from 'qrcode'; +import * as React from 'react'; +import { useEffect, useState } from 'react'; +import { Pane } from '../../components/design-system/Pane.js'; +import { Box, Text } from '../../ink.js'; +import { useKeybinding } from '../../keybindings/useKeybinding.js'; +import { useAppState } from '../../state/AppState.js'; +import type { LocalJSXCommandCall } from '../../types/command.js'; +import { logForDebugging } from '../../utils/debug.js'; +type Props = { + onDone: () => void; +}; +function SessionInfo(t0) { + const $ = _c(19); + const { + onDone + } = t0; + const remoteSessionUrl = useAppState(_temp); + const [qrCode, setQrCode] = useState(""); + let t1; + let t2; + if ($[0] !== remoteSessionUrl) { + t1 = () => { + if (!remoteSessionUrl) { + return; + } + const url = remoteSessionUrl; + const generateQRCode = async function generateQRCode() { + const qr = await qrToString(url, { + type: "utf8", + errorCorrectionLevel: "L" + }); + setQrCode(qr); + }; + generateQRCode().catch(_temp2); + }; + t2 = [remoteSessionUrl]; + $[0] = remoteSessionUrl; + $[1] = t1; + $[2] = t2; + } else { + t1 = $[1]; + t2 = $[2]; + } + useEffect(t1, t2); + let t3; + if ($[3] === Symbol.for("react.memo_cache_sentinel")) { + t3 = { + context: "Confirmation" + }; + $[3] = t3; + } else { + t3 = $[3]; + } + useKeybinding("confirm:no", onDone, t3); + if (!remoteSessionUrl) { + let t4; + if ($[4] === Symbol.for("react.memo_cache_sentinel")) { + t4 = Not in remote mode. Start with `claude --remote` to use this command.(press esc to close); + $[4] = t4; + } else { + t4 = $[4]; + } + return t4; + } + let T0; + let t4; + let t5; + if ($[5] !== qrCode) { + const lines = qrCode.split("\n").filter(_temp3); + const isLoading = lines.length === 0; + T0 = Pane; + if ($[9] === Symbol.for("react.memo_cache_sentinel")) { + t4 = Remote session; + $[9] = t4; + } else { + t4 = $[9]; + } + t5 = isLoading ? Generating QR code… : lines.map(_temp4); + $[5] = qrCode; + $[6] = T0; + $[7] = t4; + $[8] = t5; + } else { + T0 = $[6]; + t4 = $[7]; + t5 = $[8]; + } + let t6; + if ($[10] === Symbol.for("react.memo_cache_sentinel")) { + t6 = Open in browser: ; + $[10] = t6; + } else { + t6 = $[10]; + } + let t7; + if ($[11] !== remoteSessionUrl) { + t7 = {t6}{remoteSessionUrl}; + $[11] = remoteSessionUrl; + $[12] = t7; + } else { + t7 = $[12]; + } + let t8; + if ($[13] === Symbol.for("react.memo_cache_sentinel")) { + t8 = (press esc to close); + $[13] = t8; + } else { + t8 = $[13]; + } + let t9; + if ($[14] !== T0 || $[15] !== t4 || $[16] !== t5 || $[17] !== t7) { + t9 = {t4}{t5}{t7}{t8}; + $[14] = T0; + $[15] = t4; + $[16] = t5; + $[17] = t7; + $[18] = t9; + } else { + t9 = $[18]; + } + return t9; +} +function _temp4(line_0, i) { + return {line_0}; +} +function _temp3(line) { + return line.length > 0; +} +function _temp2(e) { + logForDebugging("QR code generation failed", e); +} +function _temp(s) { + return s.remoteSessionUrl; +} +export const call: LocalJSXCommandCall = async onDone => { + return ; +}; +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJ0b1N0cmluZyIsInFyVG9TdHJpbmciLCJSZWFjdCIsInVzZUVmZmVjdCIsInVzZVN0YXRlIiwiUGFuZSIsIkJveCIsIlRleHQiLCJ1c2VLZXliaW5kaW5nIiwidXNlQXBwU3RhdGUiLCJMb2NhbEpTWENvbW1hbmRDYWxsIiwibG9nRm9yRGVidWdnaW5nIiwiUHJvcHMiLCJvbkRvbmUiLCJTZXNzaW9uSW5mbyIsInQwIiwiJCIsIl9jIiwicmVtb3RlU2Vzc2lvblVybCIsIl90ZW1wIiwicXJDb2RlIiwic2V0UXJDb2RlIiwidDEiLCJ0MiIsInVybCIsImdlbmVyYXRlUVJDb2RlIiwicXIiLCJ0eXBlIiwiZXJyb3JDb3JyZWN0aW9uTGV2ZWwiLCJjYXRjaCIsIl90ZW1wMiIsInQzIiwiU3ltYm9sIiwiZm9yIiwiY29udGV4dCIsInQ0IiwiVDAiLCJ0NSIsImxpbmVzIiwic3BsaXQiLCJmaWx0ZXIiLCJfdGVtcDMiLCJpc0xvYWRpbmciLCJsZW5ndGgiLCJtYXAiLCJfdGVtcDQiLCJ0NiIsInQ3IiwidDgiLCJ0OSIsImxpbmVfMCIsImkiLCJsaW5lIiwiZSIsInMiLCJjYWxsIl0sInNvdXJjZXMiOlsic2Vzc2lvbi50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgdG9TdHJpbmcgYXMgcXJUb1N0cmluZyB9IGZyb20gJ3FyY29kZSdcbmltcG9ydCAqIGFzIFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgdXNlRWZmZWN0LCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgUGFuZSB9IGZyb20gJy4uLy4uL2NvbXBvbmVudHMvZGVzaWduLXN5c3RlbS9QYW5lLmpzJ1xuaW1wb3J0IHsgQm94LCBUZXh0IH0gZnJvbSAnLi4vLi4vaW5rLmpzJ1xuaW1wb3J0IHsgdXNlS2V5YmluZGluZyB9IGZyb20gJy4uLy4uL2tleWJpbmRpbmdzL3VzZUtleWJpbmRpbmcuanMnXG5pbXBvcnQgeyB1c2VBcHBTdGF0ZSB9IGZyb20gJy4uLy4uL3N0YXRlL0FwcFN0YXRlLmpzJ1xuaW1wb3J0IHR5cGUgeyBMb2NhbEpTWENvbW1hbmRDYWxsIH0gZnJvbSAnLi4vLi4vdHlwZXMvY29tbWFuZC5qcydcbmltcG9ydCB7IGxvZ0ZvckRlYnVnZ2luZyB9IGZyb20gJy4uLy4uL3V0aWxzL2RlYnVnLmpzJ1xuXG50eXBlIFByb3BzID0ge1xuICBvbkRvbmU6ICgpID0+IHZvaWRcbn1cblxuZnVuY3Rpb24gU2Vzc2lvbkluZm8oeyBvbkRvbmUgfTogUHJvcHMpOiBSZWFjdC5SZWFjdE5vZGUge1xuICBjb25zdCByZW1vdGVTZXNzaW9uVXJsID0gdXNlQXBwU3RhdGUocyA9PiBzLnJlbW90ZVNlc3Npb25VcmwpXG4gIGNvbnN0IFtxckNvZGUsIHNldFFyQ29kZV0gPSB1c2VTdGF0ZTxzdHJpbmc+KCcnKVxuXG4gIC8vIEdlbmVyYXRlIFFSIGNvZGUgd2hlbiBVUkwgaXMgYXZhaWxhYmxlXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKCFyZW1vdGVTZXNzaW9uVXJsKSByZXR1cm5cblxuICAgIGNvbnN0IHVybCA9IHJlbW90ZVNlc3Npb25VcmxcbiAgICBhc3luYyBmdW5jdGlvbiBnZW5lcmF0ZVFSQ29kZSgpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICAgIGNvbnN0IHFyID0gYXdhaXQgcXJUb1N0cmluZyh1cmwsIHtcbiAgICAgICAgdHlwZTogJ3V0ZjgnLFxuICAgICAgICBlcnJvckNvcnJlY3Rpb25MZXZlbDogJ0wnLFxuICAgICAgfSlcbiAgICAgIHNldFFyQ29kZShxcilcbiAgICB9XG4gICAgLy8gSW50ZW50aW9uYWxseSBzaWxlbnQgZmFpbCAtIFVSTCBpcyBzdGlsbCBzaG93biBzbyBRUiBpcyBub24tY3JpdGljYWxcbiAgICBnZW5lcmF0ZVFSQ29kZSgpLmNhdGNoKGUgPT4ge1xuICAgICAgbG9nRm9yRGVidWdnaW5nKCdRUiBjb2RlIGdlbmVyYXRpb24gZmFpbGVkJywgZSlcbiAgICB9KVxuICB9LCBbcmVtb3RlU2Vzc2lvblVybF0pXG5cbiAgLy8gSGFuZGxlIEVTQyB0byBkaXNtaXNzXG4gIHVzZUtleWJpbmRpbmcoJ2NvbmZpcm06bm8nLCBvbkRvbmUsIHsgY29udGV4dDogJ0NvbmZpcm1hdGlvbicgfSlcblxuICAvLyBOb3QgaW4gcmVtb3RlIG1vZGVcbiAgaWYgKCFyZW1vdGVTZXNzaW9uVXJsKSB7XG4gICAgcmV0dXJuIChcbiAgICAgIDxQYW5lPlxuICAgICAgICA8VGV4dCBjb2xvcj1cIndhcm5pbmdcIj5cbiAgICAgICAgICBOb3QgaW4gcmVtb3RlIG1vZGUuIFN0YXJ0IHdpdGggYGNsYXVkZSAtLXJlbW90ZWAgdG8gdXNlIHRoaXMgY29tbWFuZC5cbiAgICAgICAgPC9UZXh0PlxuICAgICAgICA8VGV4dCBkaW1Db2xvcj4ocHJlc3MgZXNjIHRvIGNsb3NlKTwvVGV4dD5cbiAgICAgIDwvUGFuZT5cbiAgICApXG4gIH1cblxuICBjb25zdCBsaW5lcyA9IHFyQ29kZS5zcGxpdCgnXFxuJykuZmlsdGVyKGxpbmUgPT4gbGluZS5sZW5ndGggPiAwKVxuICBjb25zdCBpc0xvYWRpbmcgPSBsaW5lcy5sZW5ndGggPT09IDBcblxuICByZXR1cm4gKFxuICAgIDxQYW5lPlxuICAgICAgPEJveCBtYXJnaW5Cb3R0b209ezF9PlxuICAgICAgICA8VGV4dCBib2xkPlJlbW90ZSBzZXNzaW9uPC9UZXh0PlxuICAgICAgPC9Cb3g+XG5cbiAgICAgIHsvKiBRUiBDb2RlIC0gc2lsZW50bHkgZmFpbHMgaWYgZ2VuZXJhdGlvbiBlcnJvcnMsIFVSTCBpcyBzdGlsbCBzaG93biAqL31cbiAgICAgIHtpc0xvYWRpbmcgPyAoXG4gICAgICAgIDxUZXh0IGRpbUNvbG9yPkdlbmVyYXRpbmcgUVIgY29kZeKApjwvVGV4dD5cbiAgICAgICkgOiAoXG4gICAgICAgIGxpbmVzLm1hcCgobGluZSwgaSkgPT4gPFRleHQga2V5PXtpfT57bGluZX08L1RleHQ+KVxuICAgICAgKX1cblxuICAgICAgey8qIFVSTCAqL31cbiAgICAgIDxCb3ggbWFyZ2luVG9wPXsxfT5cbiAgICAgICAgPFRleHQgZGltQ29sb3I+T3BlbiBpbiBicm93c2VyOiA8L1RleHQ+XG4gICAgICAgIDxUZXh0IGNvbG9yPVwiaWRlXCI+e3JlbW90ZVNlc3Npb25Vcmx9PC9UZXh0PlxuICAgICAgPC9Cb3g+XG5cbiAgICAgIDxCb3ggbWFyZ2luVG9wPXsxfT5cbiAgICAgICAgPFRleHQgZGltQ29sb3I+KHByZXNzIGVzYyB0byBjbG9zZSk8L1RleHQ+XG4gICAgICA8L0JveD5cbiAgICA8L1BhbmU+XG4gIClcbn1cblxuZXhwb3J0IGNvbnN0IGNhbGw6IExvY2FsSlNYQ29tbWFuZENhbGwgPSBhc3luYyBvbkRvbmUgPT4ge1xuICByZXR1cm4gPFNlc3Npb25JbmZvIG9uRG9uZT17b25Eb25lfSAvPlxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsU0FBU0EsUUFBUSxJQUFJQyxVQUFVLFFBQVEsUUFBUTtBQUMvQyxPQUFPLEtBQUtDLEtBQUssTUFBTSxPQUFPO0FBQzlCLFNBQVNDLFNBQVMsRUFBRUMsUUFBUSxRQUFRLE9BQU87QUFDM0MsU0FBU0MsSUFBSSxRQUFRLHdDQUF3QztBQUM3RCxTQUFTQyxHQUFHLEVBQUVDLElBQUksUUFBUSxjQUFjO0FBQ3hDLFNBQVNDLGFBQWEsUUFBUSxvQ0FBb0M7QUFDbEUsU0FBU0MsV0FBVyxRQUFRLHlCQUF5QjtBQUNyRCxjQUFjQyxtQkFBbUIsUUFBUSx3QkFBd0I7QUFDakUsU0FBU0MsZUFBZSxRQUFRLHNCQUFzQjtBQUV0RCxLQUFLQyxLQUFLLEdBQUc7RUFDWEMsTUFBTSxFQUFFLEdBQUcsR0FBRyxJQUFJO0FBQ3BCLENBQUM7QUFFRCxTQUFBQyxZQUFBQyxFQUFBO0VBQUEsTUFBQUMsQ0FBQSxHQUFBQyxFQUFBO0VBQXFCO0lBQUFKO0VBQUEsSUFBQUUsRUFBaUI7RUFDcEMsTUFBQUcsZ0JBQUEsR0FBeUJULFdBQVcsQ0FBQ1UsS0FBdUIsQ0FBQztFQUM3RCxPQUFBQyxNQUFBLEVBQUFDLFNBQUEsSUFBNEJqQixRQUFRLENBQVMsRUFBRSxDQUFDO0VBQUEsSUFB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 \ No newline at end of file diff --git a/src/commands/share/index.js b/src/commands/share/index.js new file mode 100644 index 0000000..e1a619d --- /dev/null +++ b/src/commands/share/index.js @@ -0,0 +1 @@ +export default { isEnabled: () => false, isHidden: true, name: 'stub' }; diff --git a/src/commands/skills/index.ts b/src/commands/skills/index.ts new file mode 100644 index 0000000..90e1d7f --- /dev/null +++ b/src/commands/skills/index.ts @@ -0,0 +1,10 @@ +import type { Command } from '../../commands.js' + +const skills = { + type: 'local-jsx', + name: 'skills', + description: 'List available skills', + load: () => import('./skills.js'), +} satisfies Command + +export default skills diff --git a/src/commands/skills/skills.tsx b/src/commands/skills/skills.tsx new file mode 100644 index 0000000..c9bf0e6 --- /dev/null +++ b/src/commands/skills/skills.tsx @@ -0,0 +1,8 @@ +import * as React from 'react'; +import type { LocalJSXCommandContext } from '../../commands.js'; +import { SkillsMenu } from '../../components/skills/SkillsMenu.js'; +import type { LocalJSXCommandOnDone } from '../../types/command.js'; +export async function call(onDone: LocalJSXCommandOnDone, context: LocalJSXCommandContext): Promise { + return ; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkxvY2FsSlNYQ29tbWFuZENvbnRleHQiLCJTa2lsbHNNZW51IiwiTG9jYWxKU1hDb21tYW5kT25Eb25lIiwiY2FsbCIsIm9uRG9uZSIsImNvbnRleHQiLCJQcm9taXNlIiwiUmVhY3ROb2RlIiwib3B0aW9ucyIsImNvbW1hbmRzIl0sInNvdXJjZXMiOlsic2tpbGxzLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB0eXBlIHsgTG9jYWxKU1hDb21tYW5kQ29udGV4dCB9IGZyb20gJy4uLy4uL2NvbW1hbmRzLmpzJ1xuaW1wb3J0IHsgU2tpbGxzTWVudSB9IGZyb20gJy4uLy4uL2NvbXBvbmVudHMvc2tpbGxzL1NraWxsc01lbnUuanMnXG5pbXBvcnQgdHlwZSB7IExvY2FsSlNYQ29tbWFuZE9uRG9uZSB9IGZyb20gJy4uLy4uL3R5cGVzL2NvbW1hbmQuanMnXG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBjYWxsKFxuICBvbkRvbmU6IExvY2FsSlNYQ29tbWFuZE9uRG9uZSxcbiAgY29udGV4dDogTG9jYWxKU1hDb21tYW5kQ29udGV4dCxcbik6IFByb21pc2U8UmVhY3QuUmVhY3ROb2RlPiB7XG4gIHJldHVybiA8U2tpbGxzTWVudSBvbkV4aXQ9e29uRG9uZX0gY29tbWFuZHM9e2NvbnRleHQub3B0aW9ucy5jb21tYW5kc30gLz5cbn1cbiJdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLQSxLQUFLLE1BQU0sT0FBTztBQUM5QixjQUFjQyxzQkFBc0IsUUFBUSxtQkFBbUI7QUFDL0QsU0FBU0MsVUFBVSxRQUFRLHVDQUF1QztBQUNsRSxjQUFjQyxxQkFBcUIsUUFBUSx3QkFBd0I7QUFFbkUsT0FBTyxlQUFlQyxJQUFJQSxDQUN4QkMsTUFBTSxFQUFFRixxQkFBcUIsRUFDN0JHLE9BQU8sRUFBRUwsc0JBQXNCLENBQ2hDLEVBQUVNLE9BQU8sQ0FBQ1AsS0FBSyxDQUFDUSxTQUFTLENBQUMsQ0FBQztFQUMxQixPQUFPLENBQUMsVUFBVSxDQUFDLE1BQU0sQ0FBQyxDQUFDSCxNQUFNLENBQUMsQ0FBQyxRQUFRLENBQUMsQ0FBQ0MsT0FBTyxDQUFDRyxPQUFPLENBQUNDLFFBQVEsQ0FBQyxHQUFHO0FBQzNFIiwiaWdub3JlTGlzdCI6W119 \ No newline at end of file diff --git a/src/commands/stats/index.ts b/src/commands/stats/index.ts new file mode 100644 index 0000000..c9680d6 --- /dev/null +++ b/src/commands/stats/index.ts @@ -0,0 +1,10 @@ +import type { Command } from '../../commands.js' + +const stats = { + type: 'local-jsx', + name: 'stats', + description: 'Show your Claude Code usage statistics and activity', + load: () => import('./stats.js'), +} satisfies Command + +export default stats diff --git a/src/commands/stats/stats.tsx b/src/commands/stats/stats.tsx new file mode 100644 index 0000000..0e83433 --- /dev/null +++ b/src/commands/stats/stats.tsx @@ -0,0 +1,7 @@ +import * as React from 'react'; +import { Stats } from '../../components/Stats.js'; +import type { LocalJSXCommandCall } from '../../types/command.js'; +export const call: LocalJSXCommandCall = async onDone => { + return ; +}; +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIlN0YXRzIiwiTG9jYWxKU1hDb21tYW5kQ2FsbCIsImNhbGwiLCJvbkRvbmUiXSwic291cmNlcyI6WyJzdGF0cy50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBTdGF0cyB9IGZyb20gJy4uLy4uL2NvbXBvbmVudHMvU3RhdHMuanMnXG5pbXBvcnQgdHlwZSB7IExvY2FsSlNYQ29tbWFuZENhbGwgfSBmcm9tICcuLi8uLi90eXBlcy9jb21tYW5kLmpzJ1xuXG5leHBvcnQgY29uc3QgY2FsbDogTG9jYWxKU1hDb21tYW5kQ2FsbCA9IGFzeW5jIG9uRG9uZSA9PiB7XG4gIHJldHVybiA8U3RhdHMgb25DbG9zZT17b25Eb25lfSAvPlxufVxuIl0sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUtBLEtBQUssTUFBTSxPQUFPO0FBQzlCLFNBQVNDLEtBQUssUUFBUSwyQkFBMkI7QUFDakQsY0FBY0MsbUJBQW1CLFFBQVEsd0JBQXdCO0FBRWpFLE9BQU8sTUFBTUMsSUFBSSxFQUFFRCxtQkFBbUIsR0FBRyxNQUFNRSxNQUFNLElBQUk7RUFDdkQsT0FBTyxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQ0EsTUFBTSxDQUFDLEdBQUc7QUFDbkMsQ0FBQyIsImlnbm9yZUxpc3QiOltdfQ== \ No newline at end of file diff --git a/src/commands/status/index.ts b/src/commands/status/index.ts new file mode 100644 index 0000000..768b358 --- /dev/null +++ b/src/commands/status/index.ts @@ -0,0 +1,12 @@ +import type { Command } from '../../commands.js' + +const status = { + type: 'local-jsx', + name: 'status', + description: + 'Show Claude Code status including version, model, account, API connectivity, and tool statuses', + immediate: true, + load: () => import('./status.js'), +} satisfies Command + +export default status diff --git a/src/commands/status/status.tsx b/src/commands/status/status.tsx new file mode 100644 index 0000000..7d98ad1 --- /dev/null +++ b/src/commands/status/status.tsx @@ -0,0 +1,8 @@ +import * as React from 'react'; +import type { LocalJSXCommandContext } from '../../commands.js'; +import { Settings } from '../../components/Settings/Settings.js'; +import type { LocalJSXCommandOnDone } from '../../types/command.js'; +export async function call(onDone: LocalJSXCommandOnDone, context: LocalJSXCommandContext): Promise { + return ; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkxvY2FsSlNYQ29tbWFuZENvbnRleHQiLCJTZXR0aW5ncyIsIkxvY2FsSlNYQ29tbWFuZE9uRG9uZSIsImNhbGwiLCJvbkRvbmUiLCJjb250ZXh0IiwiUHJvbWlzZSIsIlJlYWN0Tm9kZSJdLCJzb3VyY2VzIjpbInN0YXR1cy50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgdHlwZSB7IExvY2FsSlNYQ29tbWFuZENvbnRleHQgfSBmcm9tICcuLi8uLi9jb21tYW5kcy5qcydcbmltcG9ydCB7IFNldHRpbmdzIH0gZnJvbSAnLi4vLi4vY29tcG9uZW50cy9TZXR0aW5ncy9TZXR0aW5ncy5qcydcbmltcG9ydCB0eXBlIHsgTG9jYWxKU1hDb21tYW5kT25Eb25lIH0gZnJvbSAnLi4vLi4vdHlwZXMvY29tbWFuZC5qcydcblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGNhbGwoXG4gIG9uRG9uZTogTG9jYWxKU1hDb21tYW5kT25Eb25lLFxuICBjb250ZXh0OiBMb2NhbEpTWENvbW1hbmRDb250ZXh0LFxuKTogUHJvbWlzZTxSZWFjdC5SZWFjdE5vZGU+IHtcbiAgcmV0dXJuIDxTZXR0aW5ncyBvbkNsb3NlPXtvbkRvbmV9IGNvbnRleHQ9e2NvbnRleHR9IGRlZmF1bHRUYWI9XCJTdGF0dXNcIiAvPlxufVxuIl0sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUtBLEtBQUssTUFBTSxPQUFPO0FBQzlCLGNBQWNDLHNCQUFzQixRQUFRLG1CQUFtQjtBQUMvRCxTQUFTQyxRQUFRLFFBQVEsdUNBQXVDO0FBQ2hFLGNBQWNDLHFCQUFxQixRQUFRLHdCQUF3QjtBQUVuRSxPQUFPLGVBQWVDLElBQUlBLENBQ3hCQyxNQUFNLEVBQUVGLHFCQUFxQixFQUM3QkcsT0FBTyxFQUFFTCxzQkFBc0IsQ0FDaEMsRUFBRU0sT0FBTyxDQUFDUCxLQUFLLENBQUNRLFNBQVMsQ0FBQyxDQUFDO0VBQzFCLE9BQU8sQ0FBQyxRQUFRLENBQUMsT0FBTyxDQUFDLENBQUNILE1BQU0sQ0FBQyxDQUFDLE9BQU8sQ0FBQyxDQUFDQyxPQUFPLENBQUMsQ0FBQyxVQUFVLENBQUMsUUFBUSxHQUFHO0FBQzVFIiwiaWdub3JlTGlzdCI6W119 \ No newline at end of file diff --git a/src/commands/statusline.tsx b/src/commands/statusline.tsx new file mode 100644 index 0000000..02c7f0b --- /dev/null +++ b/src/commands/statusline.tsx @@ -0,0 +1,24 @@ +import type { ContentBlockParam } from '@anthropic-ai/sdk/resources/index.mjs'; +import type { Command } from '../commands.js'; +import { AGENT_TOOL_NAME } from '../tools/AgentTool/constants.js'; +const statusline = { + type: 'prompt', + description: "Set up Claude Code's status line UI", + contentLength: 0, + // Dynamic content + aliases: [], + name: 'statusline', + progressMessage: 'setting up statusLine', + allowedTools: [AGENT_TOOL_NAME, 'Read(~/**)', 'Edit(~/.claude/settings.json)'], + source: 'builtin', + disableNonInteractive: true, + async getPromptForCommand(args): Promise { + const prompt = args.trim() || 'Configure my statusLine from my shell PS1 configuration'; + return [{ + type: 'text', + text: `Create an ${AGENT_TOOL_NAME} with subagent_type "statusline-setup" and the prompt "${prompt}"` + }]; + } +} satisfies Command; +export default statusline; +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJDb250ZW50QmxvY2tQYXJhbSIsIkNvbW1hbmQiLCJBR0VOVF9UT09MX05BTUUiLCJzdGF0dXNsaW5lIiwidHlwZSIsImRlc2NyaXB0aW9uIiwiY29udGVudExlbmd0aCIsImFsaWFzZXMiLCJuYW1lIiwicHJvZ3Jlc3NNZXNzYWdlIiwiYWxsb3dlZFRvb2xzIiwic291cmNlIiwiZGlzYWJsZU5vbkludGVyYWN0aXZlIiwiZ2V0UHJvbXB0Rm9yQ29tbWFuZCIsImFyZ3MiLCJQcm9taXNlIiwicHJvbXB0IiwidHJpbSIsInRleHQiXSwic291cmNlcyI6WyJzdGF0dXNsaW5lLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IENvbnRlbnRCbG9ja1BhcmFtIH0gZnJvbSAnQGFudGhyb3BpYy1haS9zZGsvcmVzb3VyY2VzL2luZGV4Lm1qcydcbmltcG9ydCB0eXBlIHsgQ29tbWFuZCB9IGZyb20gJy4uL2NvbW1hbmRzLmpzJ1xuaW1wb3J0IHsgQUdFTlRfVE9PTF9OQU1FIH0gZnJvbSAnLi4vdG9vbHMvQWdlbnRUb29sL2NvbnN0YW50cy5qcydcblxuY29uc3Qgc3RhdHVzbGluZSA9IHtcbiAgdHlwZTogJ3Byb21wdCcsXG4gIGRlc2NyaXB0aW9uOiBcIlNldCB1cCBDbGF1ZGUgQ29kZSdzIHN0YXR1cyBsaW5lIFVJXCIsXG4gIGNvbnRlbnRMZW5ndGg6IDAsIC8vIER5bmFtaWMgY29udGVudFxuICBhbGlhc2VzOiBbXSxcbiAgbmFtZTogJ3N0YXR1c2xpbmUnLFxuICBwcm9ncmVzc01lc3NhZ2U6ICdzZXR0aW5nIHVwIHN0YXR1c0xpbmUnLFxuICBhbGxvd2VkVG9vbHM6IFtcbiAgICBBR0VOVF9UT09MX05BTUUsXG4gICAgJ1JlYWQofi8qKiknLFxuICAgICdFZGl0KH4vLmNsYXVkZS9zZXR0aW5ncy5qc29uKScsXG4gIF0sXG4gIHNvdXJjZTogJ2J1aWx0aW4nLFxuICBkaXNhYmxlTm9uSW50ZXJhY3RpdmU6IHRydWUsXG4gIGFzeW5jIGdldFByb21wdEZvckNvbW1hbmQoYXJncyk6IFByb21pc2U8Q29udGVudEJsb2NrUGFyYW1bXT4ge1xuICAgIGNvbnN0IHByb21wdCA9XG4gICAgICBhcmdzLnRyaW0oKSB8fCAnQ29uZmlndXJlIG15IHN0YXR1c0xpbmUgZnJvbSBteSBzaGVsbCBQUzEgY29uZmlndXJhdGlvbidcbiAgICByZXR1cm4gW1xuICAgICAge1xuICAgICAgICB0eXBlOiAndGV4dCcsXG4gICAgICAgIHRleHQ6IGBDcmVhdGUgYW4gJHtBR0VOVF9UT09MX05BTUV9IHdpdGggc3ViYWdlbnRfdHlwZSBcInN0YXR1c2xpbmUtc2V0dXBcIiBhbmQgdGhlIHByb21wdCBcIiR7cHJvbXB0fVwiYCxcbiAgICAgIH0sXG4gICAgXVxuICB9LFxufSBzYXRpc2ZpZXMgQ29tbWFuZFxuXG5leHBvcnQgZGVmYXVsdCBzdGF0dXNsaW5lXG4iXSwibWFwcGluZ3MiOiJBQUFBLGNBQWNBLGlCQUFpQixRQUFRLHVDQUF1QztBQUM5RSxjQUFjQyxPQUFPLFFBQVEsZ0JBQWdCO0FBQzdDLFNBQVNDLGVBQWUsUUFBUSxpQ0FBaUM7QUFFakUsTUFBTUMsVUFBVSxHQUFHO0VBQ2pCQyxJQUFJLEVBQUUsUUFBUTtFQUNkQyxXQUFXLEVBQUUscUNBQXFDO0VBQ2xEQyxhQUFhLEVBQUUsQ0FBQztFQUFFO0VBQ2xCQyxPQUFPLEVBQUUsRUFBRTtFQUNYQyxJQUFJLEVBQUUsWUFBWTtFQUNsQkMsZUFBZSxFQUFFLHVCQUF1QjtFQUN4Q0MsWUFBWSxFQUFFLENBQ1pSLGVBQWUsRUFDZixZQUFZLEVBQ1osK0JBQStCLENBQ2hDO0VBQ0RTLE1BQU0sRUFBRSxTQUFTO0VBQ2pCQyxxQkFBcUIsRUFBRSxJQUFJO0VBQzNCLE1BQU1DLG1CQUFtQkEsQ0FBQ0MsSUFBSSxDQUFDLEVBQUVDLE9BQU8sQ0FBQ2YsaUJBQWlCLEVBQUUsQ0FBQyxDQUFDO0lBQzVELE1BQU1nQixNQUFNLEdBQ1ZGLElBQUksQ0FBQ0csSUFBSSxDQUFDLENBQUMsSUFBSSx5REFBeUQ7SUFDMUUsT0FBTyxDQUNMO01BQ0ViLElBQUksRUFBRSxNQUFNO01BQ1pjLElBQUksRUFBRSxhQUFhaEIsZUFBZSwwREFBMERjLE1BQU07SUFDcEcsQ0FBQyxDQUNGO0VBQ0g7QUFDRixDQUFDLFdBQVdmLE9BQU87QUFFbkIsZUFBZUUsVUFBVSIsImlnbm9yZUxpc3QiOltdfQ== \ No newline at end of file diff --git a/src/commands/stickers/index.ts b/src/commands/stickers/index.ts new file mode 100644 index 0000000..ebca453 --- /dev/null +++ b/src/commands/stickers/index.ts @@ -0,0 +1,11 @@ +import type { Command } from '../../commands.js' + +const stickers = { + type: 'local', + name: 'stickers', + description: 'Order Claude Code stickers', + supportsNonInteractive: false, + load: () => import('./stickers.js'), +} satisfies Command + +export default stickers diff --git a/src/commands/stickers/stickers.ts b/src/commands/stickers/stickers.ts new file mode 100644 index 0000000..ede5193 --- /dev/null +++ b/src/commands/stickers/stickers.ts @@ -0,0 +1,16 @@ +import type { LocalCommandResult } from '../../types/command.js' +import { openBrowser } from '../../utils/browser.js' + +export async function call(): Promise { + const url = 'https://www.stickermule.com/claudecode' + const success = await openBrowser(url) + + if (success) { + return { type: 'text', value: 'Opening sticker page in browser…' } + } else { + return { + type: 'text', + value: `Failed to open browser. Visit: ${url}`, + } + } +} diff --git a/src/commands/summary/index.js b/src/commands/summary/index.js new file mode 100644 index 0000000..e1a619d --- /dev/null +++ b/src/commands/summary/index.js @@ -0,0 +1 @@ +export default { isEnabled: () => false, isHidden: true, name: 'stub' }; diff --git a/src/commands/tag/index.ts b/src/commands/tag/index.ts new file mode 100644 index 0000000..8d0bd65 --- /dev/null +++ b/src/commands/tag/index.ts @@ -0,0 +1,12 @@ +import type { Command } from '../../commands.js' + +const tag = { + type: 'local-jsx', + name: 'tag', + description: 'Toggle a searchable tag on the current session', + isEnabled: () => process.env.USER_TYPE === 'ant', + argumentHint: '', + load: () => import('./tag.js'), +} satisfies Command + +export default tag diff --git a/src/commands/tag/tag.tsx b/src/commands/tag/tag.tsx new file mode 100644 index 0000000..3809a99 --- /dev/null +++ b/src/commands/tag/tag.tsx @@ -0,0 +1,215 @@ +import { c as _c } from "react/compiler-runtime"; +import chalk from 'chalk'; +import type { UUID } from 'crypto'; +import * as React from 'react'; +import { getSessionId } from '../../bootstrap/state.js'; +import type { CommandResultDisplay } from '../../commands.js'; +import { Select } from '../../components/CustomSelect/select.js'; +import { Dialog } from '../../components/design-system/Dialog.js'; +import { COMMON_HELP_ARGS, COMMON_INFO_ARGS } from '../../constants/xml.js'; +import { Box, Text } from '../../ink.js'; +import { logEvent } from '../../services/analytics/index.js'; +import type { LocalJSXCommandOnDone } from '../../types/command.js'; +import { recursivelySanitizeUnicode } from '../../utils/sanitization.js'; +import { getCurrentSessionTag, getTranscriptPath, saveTag } from '../../utils/sessionStorage.js'; +function ConfirmRemoveTag(t0) { + const $ = _c(11); + const { + tagName, + onConfirm, + onCancel + } = t0; + const t1 = `Current tag: #${tagName}`; + let t2; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t2 = This will remove the tag from the current session.; + $[0] = t2; + } else { + t2 = $[0]; + } + let t3; + if ($[1] !== onCancel || $[2] !== onConfirm) { + t3 = value => value === "yes" ? onConfirm() : onCancel(); + $[1] = onCancel; + $[2] = onConfirm; + $[3] = t3; + } else { + t3 = $[3]; + } + let t4; + if ($[4] === Symbol.for("react.memo_cache_sentinel")) { + t4 = [{ + label: "Yes, remove tag", + value: "yes" + }, { + label: "No, keep tag", + value: "no" + }]; + $[4] = t4; + } else { + t4 = $[4]; + } + let t5; + if ($[5] !== t3) { + t5 = {t2}; + $[10] = handleSelect; + $[11] = options; + $[12] = t5; + } else { + t5 = $[12]; + } + let t6; + if ($[13] !== t4 || $[14] !== t5) { + t6 = {t4}{t5}; + $[13] = t4; + $[14] = t5; + $[15] = t6; + } else { + t6 = $[15]; + } + let t7; + if ($[16] !== handleCancel || $[17] !== t6) { + t7 = {t6}; + $[16] = handleCancel; + $[17] = t6; + $[18] = t7; + } else { + t7 = $[18]; + } + return t7; +} +const EDIT_PROMPT = 'Use the Skill tool to invoke the "thinkback" skill with mode=edit to modify my existing Claude Code year in review animation. Ask me what I want to change. When the animation is ready, tell the user to run /think-back again to play it.'; +const FIX_PROMPT = 'Use the Skill tool to invoke the "thinkback" skill with mode=fix to fix validation or rendering errors in my existing Claude Code year in review animation. Run the validator, identify errors, and fix them. When the animation is ready, tell the user to run /think-back again to play it.'; +const REGENERATE_PROMPT = 'Use the Skill tool to invoke the "thinkback" skill with mode=regenerate to create a completely new Claude Code year in review animation from scratch. Delete the existing animation and start fresh. When the animation is ready, tell the user to run /think-back again to play it.'; +function ThinkbackFlow(t0) { + const $ = _c(27); + const { + onDone + } = t0; + const [installComplete, setInstallComplete] = useState(false); + const [installError, setInstallError] = useState(null); + const [skillDir, setSkillDir] = useState(null); + const [hasGenerated, setHasGenerated] = useState(null); + let t1; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t1 = function handleReady() { + setInstallComplete(true); + }; + $[0] = t1; + } else { + t1 = $[0]; + } + const handleReady = t1; + let t2; + if ($[1] !== onDone) { + t2 = message => { + setInstallError(message); + onDone(`Error with thinkback: ${message}. Try running /plugin to manually install the think-back plugin.`, { + display: "system" + }); + }; + $[1] = onDone; + $[2] = t2; + } else { + t2 = $[2]; + } + const handleError = t2; + let t3; + let t4; + if ($[3] !== handleError || $[4] !== installComplete || $[5] !== installError || $[6] !== skillDir) { + t3 = () => { + if (installComplete && !skillDir && !installError) { + getThinkbackSkillDir().then(dir => { + if (dir) { + logForDebugging(`Thinkback skill directory: ${dir}`); + setSkillDir(dir); + } else { + handleError("Could not find thinkback skill directory"); + } + }); + } + }; + t4 = [installComplete, skillDir, installError, handleError]; + $[3] = handleError; + $[4] = installComplete; + $[5] = installError; + $[6] = skillDir; + $[7] = t3; + $[8] = t4; + } else { + t3 = $[7]; + t4 = $[8]; + } + useEffect(t3, t4); + let t5; + let t6; + if ($[9] !== skillDir) { + t5 = () => { + if (!skillDir) { + return; + } + const dataPath = join(skillDir, "year_in_review.js"); + pathExists(dataPath).then(exists => { + logForDebugging(`Checking for ${dataPath}: ${exists ? "found" : "not found"}`); + setHasGenerated(exists); + }); + }; + t6 = [skillDir]; + $[9] = skillDir; + $[10] = t5; + $[11] = t6; + } else { + t5 = $[10]; + t6 = $[11]; + } + useEffect(t5, t6); + let t7; + if ($[12] !== onDone) { + t7 = function handleAction(action) { + const prompts = { + edit: EDIT_PROMPT, + fix: FIX_PROMPT, + regenerate: REGENERATE_PROMPT + }; + onDone(prompts[action], { + display: "user", + shouldQuery: true + }); + }; + $[12] = onDone; + $[13] = t7; + } else { + t7 = $[13]; + } + const handleAction = t7; + if (installError) { + let t8; + if ($[14] !== installError) { + t8 = Error: {installError}; + $[14] = installError; + $[15] = t8; + } else { + t8 = $[15]; + } + let t9; + if ($[16] === Symbol.for("react.memo_cache_sentinel")) { + t9 = Try running /plugin to manually install the think-back plugin.; + $[16] = t9; + } else { + t9 = $[16]; + } + let t10; + if ($[17] !== t8) { + t10 = {t8}{t9}; + $[17] = t8; + $[18] = t10; + } else { + t10 = $[18]; + } + return t10; + } + if (!installComplete) { + let t8; + if ($[19] !== handleError) { + t8 = ; + $[19] = handleError; + $[20] = t8; + } else { + t8 = $[20]; + } + return t8; + } + if (!skillDir || hasGenerated === null) { + let t8; + if ($[21] === Symbol.for("react.memo_cache_sentinel")) { + t8 = Loading thinkback skill…; + $[21] = t8; + } else { + t8 = $[21]; + } + return t8; + } + let t8; + if ($[22] !== handleAction || $[23] !== hasGenerated || $[24] !== onDone || $[25] !== skillDir) { + t8 = ; + $[22] = handleAction; + $[23] = hasGenerated; + $[24] = onDone; + $[25] = skillDir; + $[26] = t8; + } else { + t8 = $[26]; + } + return t8; +} +export async function call(onDone: (result?: string, options?: { + display?: CommandResultDisplay; + shouldQuery?: boolean; +}) => void): Promise { + return ; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJleGVjYSIsInJlYWRGaWxlIiwiam9pbiIsIlJlYWN0IiwidXNlQ2FsbGJhY2siLCJ1c2VFZmZlY3QiLCJ1c2VTdGF0ZSIsIkNvbW1hbmRSZXN1bHREaXNwbGF5IiwiU2VsZWN0IiwiRGlhbG9nIiwiU3Bpbm5lciIsImluc3RhbmNlcyIsIkJveCIsIlRleHQiLCJlbmFibGVQbHVnaW5PcCIsImxvZ0ZvckRlYnVnZ2luZyIsImlzRU5PRU5UIiwidG9FcnJvciIsImV4ZWNGaWxlTm9UaHJvdyIsInBhdGhFeGlzdHMiLCJsb2dFcnJvciIsImdldFBsYXRmb3JtIiwiY2xlYXJBbGxDYWNoZXMiLCJpc1BsdWdpbkluc3RhbGxlZCIsImFkZE1hcmtldHBsYWNlU291cmNlIiwiY2xlYXJNYXJrZXRwbGFjZXNDYWNoZSIsImxvYWRLbm93bk1hcmtldHBsYWNlc0NvbmZpZyIsInJlZnJlc2hNYXJrZXRwbGFjZSIsIk9GRklDSUFMX01BUktFVFBMQUNFX05BTUUiLCJsb2FkQWxsUGx1Z2lucyIsImluc3RhbGxTZWxlY3RlZFBsdWdpbnMiLCJJTlRFUk5BTF9NQVJLRVRQTEFDRV9OQU1FIiwiSU5URVJOQUxfTUFSS0VUUExBQ0VfUkVQTyIsIk9GRklDSUFMX01BUktFVFBMQUNFX1JFUE8iLCJnZXRNYXJrZXRwbGFjZU5hbWUiLCJnZXRNYXJrZXRwbGFjZVJlcG8iLCJnZXRQbHVnaW5JZCIsIlNLSUxMX05BTUUiLCJnZXRUaGlua2JhY2tTa2lsbERpciIsIlByb21pc2UiLCJlbmFibGVkIiwidGhpbmtiYWNrUGx1Z2luIiwiZmluZCIsInAiLCJuYW1lIiwic291cmNlIiwiaW5jbHVkZXMiLCJza2lsbERpciIsInBhdGgiLCJwbGF5QW5pbWF0aW9uIiwic3VjY2VzcyIsIm1lc3NhZ2UiLCJkYXRhUGF0aCIsInBsYXllclBhdGgiLCJlIiwiaW5rSW5zdGFuY2UiLCJnZXQiLCJwcm9jZXNzIiwic3Rkb3V0IiwiZW50ZXJBbHRlcm5hdGVTY3JlZW4iLCJzdGRpbyIsImN3ZCIsInJlamVjdCIsImV4aXRBbHRlcm5hdGVTY3JlZW4iLCJodG1sUGF0aCIsInBsYXRmb3JtIiwib3BlbkNtZCIsIkluc3RhbGxTdGF0ZSIsInBoYXNlIiwiVGhpbmtiYWNrSW5zdGFsbGVyIiwib25SZWFkeSIsIm9uRXJyb3IiLCJSZWFjdE5vZGUiLCJzdGF0ZSIsInNldFN0YXRlIiwicHJvZ3Jlc3NNZXNzYWdlIiwic2V0UHJvZ3Jlc3NNZXNzYWdlIiwiY2hlY2tBbmRJbnN0YWxsIiwia25vd25NYXJrZXRwbGFjZXMiLCJtYXJrZXRwbGFjZU5hbWUiLCJtYXJrZXRwbGFjZVJlcG8iLCJwbHVnaW5JZCIsIm1hcmtldHBsYWNlSW5zdGFsbGVkIiwicGx1Z2luQWxyZWFkeUluc3RhbGxlZCIsInJlcG8iLCJyZXN1bHQiLCJmYWlsZWQiLCJsZW5ndGgiLCJlcnJvck1zZyIsIm1hcCIsImYiLCJlcnJvciIsIkVycm9yIiwiZGlzYWJsZWQiLCJpc0Rpc2FibGVkIiwic29tZSIsImVuYWJsZVJlc3VsdCIsImVyciIsInN0YXR1c01lc3NhZ2UiLCJNZW51QWN0aW9uIiwiR2VuZXJhdGl2ZUFjdGlvbiIsIkV4Y2x1ZGUiLCJUaGlua2JhY2tNZW51IiwidDAiLCIkIiwiX2MiLCJvbkRvbmUiLCJvbkFjdGlvbiIsImhhc0dlbmVyYXRlZCIsImhhc1NlbGVjdGVkIiwic2V0SGFzU2VsZWN0ZWQiLCJ0MSIsImxhYmVsIiwidmFsdWUiLCJjb25zdCIsImRlc2NyaXB0aW9uIiwib3B0aW9ucyIsInQyIiwiaGFuZGxlU2VsZWN0IiwidGhlbiIsInVuZGVmaW5lZCIsImRpc3BsYXkiLCJ0MyIsImhhbmRsZUNhbmNlbCIsInQ0IiwidDUiLCJ0NiIsInQ3IiwiRURJVF9QUk9NUFQiLCJGSVhfUFJPTVBUIiwiUkVHRU5FUkFURV9QUk9NUFQiLCJUaGlua2JhY2tGbG93IiwiaW5zdGFsbENvbXBsZXRlIiwic2V0SW5zdGFsbENvbXBsZXRlIiwiaW5zdGFsbEVycm9yIiwic2V0SW5zdGFsbEVycm9yIiwic2V0U2tpbGxEaXIiLCJzZXRIYXNHZW5lcmF0ZWQiLCJTeW1ib2wiLCJmb3IiLCJoYW5kbGVSZWFkeSIsImhhbmRsZUVycm9yIiwiZGlyIiwiZXhpc3RzIiwiaGFuZGxlQWN0aW9uIiwiYWN0aW9uIiwicHJvbXB0cyIsImVkaXQiLCJmaXgiLCJyZWdlbmVyYXRlIiwic2hvdWxkUXVlcnkiLCJ0OCIsInQ5IiwidDEwIiwiY2FsbCJdLCJzb3VyY2VzIjpbInRoaW5rYmFjay50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgZXhlY2EgfSBmcm9tICdleGVjYSdcbmltcG9ydCB7IHJlYWRGaWxlIH0gZnJvbSAnZnMvcHJvbWlzZXMnXG5pbXBvcnQgeyBqb2luIH0gZnJvbSAncGF0aCdcbmltcG9ydCAqIGFzIFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgdXNlQ2FsbGJhY2ssIHVzZUVmZmVjdCwgdXNlU3RhdGUgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB0eXBlIHsgQ29tbWFuZFJlc3VsdERpc3BsYXkgfSBmcm9tICcuLi8uLi9jb21tYW5kcy5qcydcbmltcG9ydCB7IFNlbGVjdCB9IGZyb20gJy4uLy4uL2NvbXBvbmVudHMvQ3VzdG9tU2VsZWN0L3NlbGVjdC5qcydcbmltcG9ydCB7IERpYWxvZyB9IGZyb20gJy4uLy4uL2NvbXBvbmVudHMvZGVzaWduLXN5c3RlbS9EaWFsb2cuanMnXG5pbXBvcnQgeyBTcGlubmVyIH0gZnJvbSAnLi4vLi4vY29tcG9uZW50cy9TcGlubmVyLmpzJ1xuaW1wb3J0IGluc3RhbmNlcyBmcm9tICcuLi8uLi9pbmsvaW5zdGFuY2VzLmpzJ1xuaW1wb3J0IHsgQm94LCBUZXh0IH0gZnJvbSAnLi4vLi4vaW5rLmpzJ1xuaW1wb3J0IHsgZW5hYmxlUGx1Z2luT3AgfSBmcm9tICcuLi8uLi9zZXJ2aWNlcy9wbHVnaW5zL3BsdWdpbk9wZXJhdGlvbnMuanMnXG5pbXBvcnQgeyBsb2dGb3JEZWJ1Z2dpbmcgfSBmcm9tICcuLi8uLi91dGlscy9kZWJ1Zy5qcydcbmltcG9ydCB7IGlzRU5PRU5ULCB0b0Vycm9yIH0gZnJvbSAnLi4vLi4vdXRpbHMvZXJyb3JzLmpzJ1xuaW1wb3J0IHsgZXhlY0ZpbGVOb1Rocm93IH0gZnJvbSAnLi4vLi4vdXRpbHMvZXhlY0ZpbGVOb1Rocm93LmpzJ1xuaW1wb3J0IHsgcGF0aEV4aXN0cyB9IGZyb20gJy4uLy4uL3V0aWxzL2ZpbGUuanMnXG5pbXBvcnQgeyBsb2dFcnJvciB9IGZyb20gJy4uLy4uL3V0aWxzL2xvZy5qcydcbmltcG9ydCB7IGdldFBsYXRmb3JtIH0gZnJvbSAnLi4vLi4vdXRpbHMvcGxhdGZvcm0uanMnXG5pbXBvcnQgeyBjbGVhckFsbENhY2hlcyB9IGZyb20gJy4uLy4uL3V0aWxzL3BsdWdpbnMvY2FjaGVVdGlscy5qcydcbmltcG9ydCB7IGlzUGx1Z2luSW5zdGFsbGVkIH0gZnJvbSAnLi4vLi4vdXRpbHMvcGx1Z2lucy9pbnN0YWxsZWRQbHVnaW5zTWFuYWdlci5qcydcbmltcG9ydCB7XG4gIGFkZE1hcmtldHBsYWNlU291cmNlLFxuICBjbGVhck1hcmtldHBsYWNlc0NhY2hlLFxuICBsb2FkS25vd25NYXJrZXRwbGFjZXNDb25maWcsXG4gIHJlZnJlc2hNYXJrZXRwbGFjZSxcbn0gZnJvbSAnLi4vLi4vdXRpbHMvcGx1Z2lucy9tYXJrZXRwbGFjZU1hbmFnZXIuanMnXG5pbXBvcnQgeyBPRkZJQ0lBTF9NQVJLRVRQTEFDRV9OQU1FIH0gZnJvbSAnLi4vLi4vdXRpbHMvcGx1Z2lucy9vZmZpY2lhbE1hcmtldHBsYWNlLmpzJ1xuaW1wb3J0IHsgbG9hZEFsbFBsdWdpbnMgfSBmcm9tICcuLi8uLi91dGlscy9wbHVnaW5zL3BsdWdpbkxvYWRlci5qcydcbmltcG9ydCB7IGluc3RhbGxTZWxlY3RlZFBsdWdpbnMgfSBmcm9tICcuLi8uLi91dGlscy9wbHVnaW5zL3BsdWdpblN0YXJ0dXBDaGVjay5qcydcblxuLy8gTWFya2V0cGxhY2UgYW5kIHBsdWdpbiBpZGVudGlmaWVycyAtIHZhcmllcyBieSB1c2VyIHR5cGVcbmNvbnN0IElOVEVSTkFMX01BUktFVFBMQUNFX05BTUUgPSAnY2xhdWRlLWNvZGUtbWFya2V0cGxhY2UnXG5jb25zdCBJTlRFUk5BTF9NQVJLRVRQTEFDRV9SRVBPID0gJ2FudGhyb3BpY3MvY2xhdWRlLWNvZGUtbWFya2V0cGxhY2UnXG5jb25zdCBPRkZJQ0lBTF9NQVJLRVRQTEFDRV9SRVBPID0gJ2FudGhyb3BpY3MvY2xhdWRlLXBsdWdpbnMtb2ZmaWNpYWwnXG5cbmZ1bmN0aW9uIGdldE1hcmtldHBsYWNlTmFtZSgpOiBzdHJpbmcge1xuICByZXR1cm4gXCJleHRlcm5hbFwiID09PSAnYW50J1xuICAgID8gSU5URVJOQUxfTUFSS0VUUExBQ0VfTkFNRVxuICAgIDogT0ZGSUNJQUxfTUFSS0VUUExBQ0VfTkFNRVxufVxuXG5mdW5jdGlvbiBnZXRNYXJrZXRwbGFjZVJlcG8oKTogc3RyaW5nIHtcbiAgcmV0dXJuIFwiZXh0ZXJuYWxcIiA9PT0gJ2FudCdcbiAgICA/IElOVEVSTkFMX01BUktFVFBMQUNFX1JFUE9cbiAgICA6IE9GRklDSUFMX01BUktFVFBMQUNFX1JFUE9cbn1cblxuZnVuY3Rpb24gZ2V0UGx1Z2luSWQoKTogc3RyaW5nIHtcbiAgcmV0dXJuIGB0aGlua2JhY2tAJHtnZXRNYXJrZXRwbGFjZU5hbWUoKX1gXG59XG5cbmNvbnN0IFNLSUxMX05BTUUgPSAndGhpbmtiYWNrJ1xuXG4vKipcbiAqIEdldCB0aGUgdGhpbmtiYWNrIHNraWxsIGRpcmVjdG9yeSBmcm9tIHRoZSBpbnN0YWxsZWQgcGx1Z2luJ3MgY2FjaGUgcGF0aFxuICovXG5hc3luYyBmdW5jdGlvbiBnZXRUaGlua2JhY2tTa2lsbERpcigpOiBQcm9taXNlPHN0cmluZyB8IG51bGw+IHtcbiAgY29uc3QgeyBlbmFibGVkIH0gPSBhd2FpdCBsb2FkQWxsUGx1Z2lucygpXG4gIGNvbnN0IHRoaW5rYmFja1BsdWdpbiA9IGVuYWJsZWQuZmluZChcbiAgICBwID0+XG4gICAgICBwLm5hbWUgPT09ICd0aGlua2JhY2snIHx8IChwLnNvdXJjZSAmJiBwLnNvdXJjZS5pbmNsdWRlcyhnZXRQbHVnaW5JZCgpKSksXG4gIClcblxuICBpZiAoIXRoaW5rYmFja1BsdWdpbikge1xuICAgIHJldHVybiBudWxsXG4gIH1cblxuICBjb25zdCBza2lsbERpciA9IGpvaW4odGhpbmtiYWNrUGx1Z2luLnBhdGgsICdza2lsbHMnLCBTS0lMTF9OQU1FKVxuICBpZiAoYXdhaXQgcGF0aEV4aXN0cyhza2lsbERpcikpIHtcbiAgICByZXR1cm4gc2tpbGxEaXJcbiAgfVxuXG4gIHJldHVybiBudWxsXG59XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBwbGF5QW5pbWF0aW9uKHNraWxsRGlyOiBzdHJpbmcpOiBQcm9taXNlPHtcbiAgc3VjY2VzczogYm9vbGVhblxuICBtZXNzYWdlOiBzdHJpbmdcbn0+IHtcbiAgY29uc3QgZGF0YVBhdGggPSBqb2luKHNraWxsRGlyLCAneWVhcl9pbl9yZXZpZXcuanMnKVxuICBjb25zdCBwbGF5ZXJQYXRoID0gam9pbihza2lsbERpciwgJ3BsYXllci5qcycpXG5cbiAgLy8gQm90aCBmaWxlcyBhcmUgcHJlcmVxdWlzaXRlcyBmb3IgdGhlIG5vZGUgc3VicHJvY2Vzcy4gUmVhZCB0aGVtIGhlcmVcbiAgLy8gKG5vdCBhdCBjYWxsIHNpdGVzKSBzbyBhbGwgY2FsbGVycyBnZXQgY29uc2lzdGVudCBlcnJvciBtZXNzYWdpbmcuIFRoZVxuICAvLyBzdWJwcm9jZXNzIHJ1bnMgd2l0aCByZWplY3Q6IGZhbHNlLCBzbyBhIG1pc3NpbmcgZmlsZSB3b3VsZCBvdGhlcndpc2VcbiAgLy8gc2lsZW50bHkgcmV0dXJuIHN1Y2Nlc3MuIFVzaW5nIHJlYWRGaWxlIChub3QgYWNjZXNzKSBwZXIgQ0xBVURFLm1kLlxuICAvL1xuICAvLyBOb24tRU5PRU5UIGVycm9ycyAoRUFDQ0VTIGV0YykgYXJlIGxvZ2dlZCBhbmQgcmV0dXJuZWQgYXMgZmFpbHVyZXMgcmF0aGVyXG4gIC8vIHRoYW4gdGhyb3duIOKAlCB0aGUgb2xkIHBhdGhFeGlzdHMtYmFzZWQgY29kZSBuZXZlciB0aHJldywgYW5kIG9uZSBjYWxsZXJcbiAgLy8gKGhhbmRsZVNlbGVjdCkgdXNlcyBgdm9pZCBwbGF5QW5pbWF0aW9uKCkudGhlbiguLi4pYCB3aXRob3V0IGEgLmNhdGNoKCkuXG4gIHRyeSB7XG4gICAgYXdhaXQgcmVhZEZpbGUoZGF0YVBhdGgpXG4gIH0gY2F0Y2ggKGU6IHVua25vd24pIHtcbiAgICBpZiAoaXNFTk9FTlQoZSkpIHtcbiAgICAgIHJldHVybiB7XG4gICAgICAgIHN1Y2Nlc3M6IGZhbHNlLFxuICAgICAgICBtZXNzYWdlOiAnTm8gYW5pbWF0aW9uIGZvdW5kLiBSdW4gL3RoaW5rLWJhY2sgZmlyc3QgdG8gZ2VuZXJhdGUgb25lLicsXG4gICAgICB9XG4gICAgfVxuICAgIGxvZ0Vycm9yKGUpXG4gICAgcmV0dXJuIHtcbiAgICAgIHN1Y2Nlc3M6IGZhbHNlLFxuICAgICAgbWVzc2FnZTogYENvdWxkIG5vdCBhY2Nlc3MgYW5pbWF0aW9uIGRhdGE6ICR7dG9FcnJvcihlKS5tZXNzYWdlfWAsXG4gICAgfVxuICB9XG5cbiAgdHJ5IHtcbiAgICBhd2FpdCByZWFkRmlsZShwbGF5ZXJQYXRoKVxuICB9IGNhdGNoIChlOiB1bmtub3duKSB7XG4gICAgaWYgKGlzRU5PRU5UKGUpKSB7XG4gICAgICByZXR1cm4ge1xuICAgICAgICBzdWNjZXNzOiBmYWxzZSxcbiAgICAgICAgbWVzc2FnZTpcbiAgICAgICAgICAnUGxheWVyIHNjcmlwdCBub3QgZm91bmQuIFRoZSBwbGF5ZXIuanMgZmlsZSBpcyBtaXNzaW5nIGZyb20gdGhlIHRoaW5rYmFjayBza2lsbC4nLFxuICAgICAgfVxuICAgIH1cbiAgICBsb2dFcnJvcihlKVxuICAgIHJldHVybiB7XG4gICAgICBzdWNjZXNzOiBmYWxzZSxcbiAgICAgIG1lc3NhZ2U6IGBDb3VsZCBub3QgYWNjZXNzIHBsYXllciBzY3JpcHQ6ICR7dG9FcnJvcihlKS5tZXNzYWdlfWAsXG4gICAgfVxuICB9XG5cbiAgLy8gR2V0IGluayBpbnN0YW5jZSBmb3IgdGVybWluYWwgdGFrZW92ZXJcbiAgY29uc3QgaW5rSW5zdGFuY2UgPSBpbnN0YW5jZXMuZ2V0KHByb2Nlc3Muc3Rkb3V0KVxuICBpZiAoIWlua0luc3RhbmNlKSB7XG4gICAgcmV0dXJuIHsgc3VjY2VzczogZmFsc2UsIG1lc3NhZ2U6ICdGYWlsZWQgdG8gYWNjZXNzIHRlcm1pbmFsIGluc3RhbmNlJyB9XG4gIH1cblxuICBpbmtJbnN0YW5jZS5lbnRlckFsdGVybmF0ZVNjcmVlbigpXG4gIHRyeSB7XG4gICAgYXdhaXQgZXhlY2EoJ25vZGUnLCBbcGxheWVyUGF0aF0sIHtcbiAgICAgIHN0ZGlvOiAnaW5oZXJpdCcsXG4gICAgICBjd2Q6IHNraWxsRGlyLFxuICAgICAgcmVqZWN0OiBmYWxzZSxcbiAgICB9KVxuICB9IGNhdGNoIHtcbiAgICAvLyBBbmltYXRpb24gbWF5IGhhdmUgYmVlbiBpbnRlcnJ1cHRlZCAoZS5nLiwgQ3RybCtDKVxuICB9IGZpbmFsbHkge1xuICAgIGlua0luc3RhbmNlLmV4aXRBbHRlcm5hdGVTY3JlZW4oKVxuICB9XG5cbiAgLy8gT3BlbiB0aGUgSFRNTCBmaWxlIGluIGJyb3dzZXIgZm9yIHZpZGVvIGRvd25sb2FkXG4gIGNvbnN0IGh0bWxQYXRoID0gam9pbihza2lsbERpciwgJ3llYXJfaW5fcmV2aWV3Lmh0bWwnKVxuICBpZiAoYXdhaXQgcGF0aEV4aXN0cyhodG1sUGF0aCkpIHtcbiAgICBjb25zdCBwbGF0Zm9ybSA9IGdldFBsYXRmb3JtKClcbiAgICBjb25zdCBvcGVuQ21kID1cbiAgICAgIHBsYXRmb3JtID09PSAnbWFjb3MnXG4gICAgICAgID8gJ29wZW4nXG4gICAgICAgIDogcGxhdGZvcm0gPT09ICd3aW5kb3dzJ1xuICAgICAgICAgID8gJ3N0YXJ0J1xuICAgICAgICAgIDogJ3hkZy1vcGVuJ1xuICAgIHZvaWQgZXhlY0ZpbGVOb1Rocm93KG9wZW5DbWQsIFtodG1sUGF0aF0pXG4gIH1cblxuICByZXR1cm4geyBzdWNjZXNzOiB0cnVlLCBtZXNzYWdlOiAnWWVhciBpbiByZXZpZXcgYW5pbWF0aW9uIGNvbXBsZXRlIScgfVxufVxuXG50eXBlIEluc3RhbGxTdGF0ZSA9XG4gIHwgeyBwaGFzZTogJ2NoZWNraW5nJyB9XG4gIHwgeyBwaGFzZTogJ2luc3RhbGxpbmctbWFya2V0cGxhY2UnIH1cbiAgfCB7IHBoYXNlOiAnaW5zdGFsbGluZy1wbHVnaW4nIH1cbiAgfCB7IHBoYXNlOiAnZW5hYmxpbmctcGx1Z2luJyB9XG4gIHwgeyBwaGFzZTogJ3JlYWR5JyB9XG4gIHwgeyBwaGFzZTogJ2Vycm9yJzsgbWVzc2FnZTogc3RyaW5nIH1cblxuZnVuY3Rpb24gVGhpbmtiYWNrSW5zdGFsbGVyKHtcbiAgb25SZWFkeSxcbiAgb25FcnJvcixcbn06IHtcbiAgb25SZWFkeTogKCkgPT4gdm9pZFxuICBvbkVycm9yOiAobWVzc2FnZTogc3RyaW5nKSA9PiB2b2lkXG59KTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgY29uc3QgW3N0YXRlLCBzZXRTdGF0ZV0gPSB1c2VTdGF0ZTxJbnN0YWxsU3RhdGU+KHsgcGhhc2U6ICdjaGVja2luZycgfSlcbiAgY29uc3QgW3Byb2dyZXNzTWVzc2FnZSwgc2V0UHJvZ3Jlc3NNZXNzYWdlXSA9IHVzZVN0YXRlKCcnKVxuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgYXN5bmMgZnVuY3Rpb24gY2hlY2tBbmRJbnN0YWxsKCk6IFByb21pc2U8dm9pZD4ge1xuICAgICAgdHJ5IHtcbiAgICAgICAgLy8gQ2hlY2sgaWYgbWFya2V0cGxhY2UgaXMgaW5zdGFsbGVkXG4gICAgICAgIGNvbnN0IGtub3duTWFya2V0cGxhY2VzID0gYXdhaXQgbG9hZEtub3duTWFya2V0cGxhY2VzQ29uZmlnKClcbiAgICAgICAgY29uc3QgbWFya2V0cGxhY2VOYW1lID0gZ2V0TWFya2V0cGxhY2VOYW1lKClcbiAgICAgICAgY29uc3QgbWFya2V0cGxhY2VSZXBvID0gZ2V0TWFya2V0cGxhY2VSZXBvKClcbiAgICAgICAgY29uc3QgcGx1Z2luSWQgPSBnZXRQbHVnaW5JZCgpXG4gICAgICAgIGNvbnN0IG1hcmtldHBsYWNlSW5zdGFsbGVkID0gbWFya2V0cGxhY2VOYW1lIGluIGtub3duTWFya2V0cGxhY2VzXG5cbiAgICAgICAgLy8gQ2hlY2sgaWYgcGx1Z2luIGlzIGFscmVhZHkgaW5zdGFsbGVkIGZpcnN0XG4gICAgICAgIGNvbnN0IHBsdWdpbkFscmVhZHlJbnN0YWxsZWQgPSBpc1BsdWdpbkluc3RhbGxlZChwbHVnaW5JZClcblxuICAgICAgICBpZiAoIW1hcmtldHBsYWNlSW5zdGFsbGVkKSB7XG4gICAgICAgICAgLy8gSW5zdGFsbCB0aGUgbWFya2V0cGxhY2VcbiAgICAgICAgICBzZXRTdGF0ZSh7IHBoYXNlOiAnaW5zdGFsbGluZy1tYXJrZXRwbGFjZScgfSlcbiAgICAgICAgICBsb2dGb3JEZWJ1Z2dpbmcoYEluc3RhbGxpbmcgbWFya2V0cGxhY2UgJHttYXJrZXRwbGFjZVJlcG99YClcblxuICAgICAgICAgIGF3YWl0IGFkZE1hcmtldHBsYWNlU291cmNlKFxuICAgICAgICAgICAgeyBzb3VyY2U6ICdnaXRodWInLCByZXBvOiBtYXJrZXRwbGFjZVJlcG8gfSxcbiAgICAgICAgICAgIG1lc3NhZ2UgPT4ge1xuICAgICAgICAgICAgICBzZXRQcm9ncmVzc01lc3NhZ2UobWVzc2FnZSlcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgKVxuICAgICAgICAgIGNsZWFyQWxsQ2FjaGVzKClcbiAgICAgICAgICBsb2dGb3JEZWJ1Z2dpbmcoYE1hcmtldHBsYWNlICR7bWFya2V0cGxhY2VOYW1lfSBpbnN0YWxsZWRgKVxuICAgICAgICB9IGVsc2UgaWYgKCFwbHVnaW5BbHJlYWR5SW5zdGFsbGVkKSB7XG4gICAgICAgICAgLy8gTWFya2V0cGxhY2UgaW5zdGFsbGVkIGJ1dCBwbHVnaW4gbm90IGluc3RhbGxlZCAtIHJlZnJlc2ggdG8gZ2V0IGxhdGVzdCBwbHVnaW5zXG4gICAgICAgICAgLy8gT25seSByZWZyZXNoIHdoZW4gbmVlZGVkIHRvIGF2b2lkIHBvdGVudGlhbGx5IGRlc3RydWN0aXZlIGdpdCBvcGVyYXRpb25zXG4gICAgICAgICAgc2V0U3RhdGUoeyBwaGFzZTogJ2luc3RhbGxpbmctbWFya2V0cGxhY2UnIH0pXG4gICAgICAgICAgc2V0UHJvZ3Jlc3NNZXNzYWdlKCdVcGRhdGluZyBtYXJrZXRwbGFjZeKApicpXG4gICAgICAgICAgbG9nRm9yRGVidWdnaW5nKGBSZWZyZXNoaW5nIG1hcmtldHBsYWNlICR7bWFya2V0cGxhY2VOYW1lfWApXG5cbiAgICAgICAgICBhd2FpdCByZWZyZXNoTWFya2V0cGxhY2UobWFya2V0cGxhY2VOYW1lLCBtZXNzYWdlID0+IHtcbiAgICAgICAgICAgIHNldFByb2dyZXNzTWVzc2FnZShtZXNzYWdlKVxuICAgICAgICAgIH0pXG4gICAgICAgICAgY2xlYXJNYXJrZXRwbGFjZXNDYWNoZSgpXG4gICAgICAgICAgY2xlYXJBbGxDYWNoZXMoKVxuICAgICAgICAgIGxvZ0ZvckRlYnVnZ2luZyhgTWFya2V0cGxhY2UgJHttYXJrZXRwbGFjZU5hbWV9IHJlZnJlc2hlZGApXG4gICAgICAgIH1cblxuICAgICAgICBpZiAoIXBsdWdpbkFscmVhZHlJbnN0YWxsZWQpIHtcbiAgICAgICAgICAvLyBJbnN0YWxsIHRoZSBwbHVnaW5cbiAgICAgICAgICBzZXRTdGF0ZSh7IHBoYXNlOiAnaW5zdGFsbGluZy1wbHVnaW4nIH0pXG4gICAgICAgICAgbG9nRm9yRGVidWdnaW5nKGBJbnN0YWxsaW5nIHBsdWdpbiAke3BsdWdpbklkfWApXG5cbiAgICAgICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCBpbnN0YWxsU2VsZWN0ZWRQbHVnaW5zKFtwbHVnaW5JZF0pXG5cbiAgICAgICAgICBpZiAocmVzdWx0LmZhaWxlZC5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgICBjb25zdCBlcnJvck1zZyA9IHJlc3VsdC5mYWlsZWRcbiAgICAgICAgICAgICAgLm1hcChmID0+IGAke2YubmFtZX06ICR7Zi5lcnJvcn1gKVxuICAgICAgICAgICAgICAuam9pbignLCAnKVxuICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBGYWlsZWQgdG8gaW5zdGFsbCBwbHVnaW46ICR7ZXJyb3JNc2d9YClcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBjbGVhckFsbENhY2hlcygpXG4gICAgICAgICAgbG9nRm9yRGVidWdnaW5nKGBQbHVnaW4gJHtwbHVnaW5JZH0gaW5zdGFsbGVkYClcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAvLyBQbHVnaW4gaXMgaW5zdGFsbGVkLCBjaGVjayBpZiBpdCdzIGVuYWJsZWRcbiAgICAgICAgICBjb25zdCB7IGRpc2FibGVkIH0gPSBhd2FpdCBsb2FkQWxsUGx1Z2lucygpXG4gICAgICAgICAgY29uc3QgaXNEaXNhYmxlZCA9IGRpc2FibGVkLnNvbWUoXG4gICAgICAgICAgICBwID0+IHAubmFtZSA9PT0gJ3RoaW5rYmFjaycgfHwgcC5zb3VyY2U/LmluY2x1ZGVzKHBsdWdpbklkKSxcbiAgICAgICAgICApXG5cbiAgICAgICAgICBpZiAoaXNEaXNhYmxlZCkge1xuICAgICAgICAgICAgLy8gRW5hYmxlIHRoZSBwbHVnaW5cbiAgICAgICAgICAgIHNldFN0YXRlKHsgcGhhc2U6ICdlbmFibGluZy1wbHVnaW4nIH0pXG4gICAgICAgICAgICBsb2dGb3JEZWJ1Z2dpbmcoYEVuYWJsaW5nIHBsdWdpbiAke3BsdWdpbklkfWApXG5cbiAgICAgICAgICAgIGNvbnN0IGVuYWJsZVJlc3VsdCA9IGF3YWl0IGVuYWJsZVBsdWdpbk9wKHBsdWdpbklkKVxuICAgICAgICAgICAgaWYgKCFlbmFibGVSZXN1bHQuc3VjY2Vzcykge1xuICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgICAgICAgICAgYEZhaWxlZCB0byBlbmFibGUgcGx1Z2luOiAke2VuYWJsZVJlc3VsdC5tZXNzYWdlfWAsXG4gICAgICAgICAgICAgIClcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgY2xlYXJBbGxDYWNoZXMoKVxuICAgICAgICAgICAgbG9nRm9yRGVidWdnaW5nKGBQbHVnaW4gJHtwbHVnaW5JZH0gZW5hYmxlZGApXG4gICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgc2V0U3RhdGUoeyBwaGFzZTogJ3JlYWR5JyB9KVxuICAgICAgICBvblJlYWR5KClcbiAgICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICAgIGNvbnN0IGVyciA9IHRvRXJyb3IoZXJyb3IpXG4gICAgICAgIGxvZ0Vycm9yKGVycilcbiAgICAgICAgc2V0U3RhdGUoeyBwaGFzZTogJ2Vycm9yJywgbWVzc2FnZTogZXJyLm1lc3NhZ2UgfSlcbiAgICAgICAgb25FcnJvcihlcnIubWVzc2FnZSlcbiAgICAgIH1cbiAgICB9XG5cbiAgICB2b2lkIGNoZWNrQW5kSW5zdGFsbCgpXG4gIH0sIFtvblJlYWR5LCBvbkVycm9yXSlcblxuICBpZiAoc3RhdGUucGhhc2UgPT09ICdlcnJvcicpIHtcbiAgICByZXR1cm4gKFxuICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCI+XG4gICAgICAgIDxUZXh0IGNvbG9yPVwiZXJyb3JcIj5FcnJvcjoge3N0YXRlLm1lc3NhZ2V9PC9UZXh0PlxuICAgICAgPC9Cb3g+XG4gICAgKVxuICB9XG5cbiAgaWYgKHN0YXRlLnBoYXNlID09PSAncmVhZHknKSB7XG4gICAgcmV0dXJuIG51bGxcbiAgfVxuXG4gIGNvbnN0IHN0YXR1c01lc3NhZ2UgPVxuICAgIHN0YXRlLnBoYXNlID09PSAnY2hlY2tpbmcnXG4gICAgICA/ICdDaGVja2luZyB0aGlua2JhY2sgaW5zdGFsbGF0aW9u4oCmJ1xuICAgICAgOiBzdGF0ZS5waGFzZSA9PT0gJ2luc3RhbGxpbmctbWFya2V0cGxhY2UnXG4gICAgICAgID8gJ0luc3RhbGxpbmcgbWFya2V0cGxhY2XigKYnXG4gICAgICAgIDogc3RhdGUucGhhc2UgPT09ICdlbmFibGluZy1wbHVnaW4nXG4gICAgICAgICAgPyAnRW5hYmxpbmcgdGhpbmtiYWNrIHBsdWdpbuKApidcbiAgICAgICAgICA6ICdJbnN0YWxsaW5nIHRoaW5rYmFjayBwbHVnaW7igKYnXG5cbiAgcmV0dXJuIChcbiAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIj5cbiAgICAgIDxCb3g+XG4gICAgICAgIDxTcGlubmVyIC8+XG4gICAgICAgIDxUZXh0Pntwcm9ncmVzc01lc3NhZ2UgfHwgc3RhdHVzTWVzc2FnZX08L1RleHQ+XG4gICAgICA8L0JveD5cbiAgICA8L0JveD5cbiAgKVxufVxuXG50eXBlIE1lbnVBY3Rpb24gPSAncGxheScgfCAnZWRpdCcgfCAnZml4JyB8ICdyZWdlbmVyYXRlJ1xudHlwZSBHZW5lcmF0aXZlQWN0aW9uID0gRXhjbHVkZTxNZW51QWN0aW9uLCAncGxheSc+XG5cbmZ1bmN0aW9uIFRoaW5rYmFja01lbnUoe1xuICBvbkRvbmUsXG4gIG9uQWN0aW9uLFxuICBza2lsbERpcixcbiAgaGFzR2VuZXJhdGVkLFxufToge1xuICBvbkRvbmU6IChcbiAgICByZXN1bHQ/OiBzdHJpbmcsXG4gICAgb3B0aW9ucz86IHsgZGlzcGxheT86IENvbW1hbmRSZXN1bHREaXNwbGF5OyBzaG91bGRRdWVyeT86IGJvb2xlYW4gfSxcbiAgKSA9PiB2b2lkXG4gIG9uQWN0aW9uOiAoYWN0aW9uOiBHZW5lcmF0aXZlQWN0aW9uKSA9PiB2b2lkXG4gIHNraWxsRGlyOiBzdHJpbmdcbiAgaGFzR2VuZXJhdGVkOiBib29sZWFuXG59KTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgY29uc3QgW2hhc1NlbGVjdGVkLCBzZXRIYXNTZWxlY3RlZF0gPSB1c2VTdGF0ZShmYWxzZSlcblxuICBjb25zdCBvcHRpb25zID0gaGFzR2VuZXJhdGVkXG4gICAgPyBbXG4gICAgICAgIHtcbiAgICAgICAgICBsYWJlbDogJ1BsYXkgYW5pbWF0aW9uJyxcbiAgICAgICAgICB2YWx1ZTogJ3BsYXknIGFzIGNvbnN0LFxuICAgICAgICAgIGRlc2NyaXB0aW9uOiAnV2F0Y2ggeW91ciB5ZWFyIGluIHJldmlldycsXG4gICAgICAgIH0sXG4gICAgICAgIHtcbiAgICAgICAgICBsYWJlbDogJ0VkaXQgY29udGVudCcsXG4gICAgICAgICAgdmFsdWU6ICdlZGl0JyBhcyBjb25zdCxcbiAgICAgICAgICBkZXNjcmlwdGlvbjogJ01vZGlmeSB0aGUgYW5pbWF0aW9uJyxcbiAgICAgICAgfSxcbiAgICAgICAge1xuICAgICAgICAgIGxhYmVsOiAnRml4IGVycm9ycycsXG4gICAgICAgICAgdmFsdWU6ICdmaXgnIGFzIGNvbnN0LFxuICAgICAgICAgIGRlc2NyaXB0aW9uOiAnRml4IHZhbGlkYXRpb24gb3IgcmVuZGVyaW5nIGlzc3VlcycsXG4gICAgICAgIH0sXG4gICAgICAgIHtcbiAgICAgICAgICBsYWJlbDogJ1JlZ2VuZXJhdGUnLFxuICAgICAgICAgIHZhbHVlOiAncmVnZW5lcmF0ZScgYXMgY29uc3QsXG4gICAgICAgICAgZGVzY3JpcHRpb246ICdDcmVhdGUgYSBuZXcgYW5pbWF0aW9uIGZyb20gc2NyYXRjaCcsXG4gICAgICAgIH0sXG4gICAgICBdXG4gICAgOiBbXG4gICAgICAgIHtcbiAgICAgICAgICBsYWJlbDogXCJMZXQncyBnbyFcIixcbiAgICAgICAgICB2YWx1ZTogJ3JlZ2VuZXJhdGUnIGFzIGNvbnN0LFxuICAgICAgICAgIGRlc2NyaXB0aW9uOiAnR2VuZXJhdGUgeW91ciBwZXJzb25hbGl6ZWQgYW5pbWF0aW9uJyxcbiAgICAgICAgfSxcbiAgICAgIF1cblxuICBmdW5jdGlvbiBoYW5kbGVTZWxlY3QodmFsdWU6IE1lbnVBY3Rpb24pOiB2b2lkIHtcbiAgICBzZXRIYXNTZWxlY3RlZCh0cnVlKVxuICAgIGlmICh2YWx1ZSA9PT0gJ3BsYXknKSB7XG4gICAgICAvLyBQbGF5IHJ1bnMgdGhlIHRlcm1pbmFsLXRha2VvdmVyIGFuaW1hdGlvbiwgdGhlbiBzaWduYWwgZG9uZSB3aXRoIHNraXBcbiAgICAgIHZvaWQgcGxheUFuaW1hdGlvbihza2lsbERpcikudGhlbigoKSA9PiB7XG4gICAgICAgIG9uRG9uZSh1bmRlZmluZWQsIHsgZGlzcGxheTogJ3NraXAnIH0pXG4gICAgICB9KVxuICAgIH0gZWxzZSB7XG4gICAgICBvbkFjdGlvbih2YWx1ZSlcbiAgICB9XG4gIH1cblxuICBmdW5jdGlvbiBoYW5kbGVDYW5jZWwoKTogdm9pZCB7XG4gICAgb25Eb25lKHVuZGVmaW5lZCwgeyBkaXNwbGF5OiAnc2tpcCcgfSlcbiAgfVxuXG4gIGlmIChoYXNTZWxlY3RlZCkge1xuICAgIHJldHVybiBudWxsXG4gIH1cblxuICByZXR1cm4gKFxuICAgIDxEaWFsb2dcbiAgICAgIHRpdGxlPVwiVGhpbmsgQmFjayBvbiAyMDI1IHdpdGggQ2xhdWRlIENvZGVcIlxuICAgICAgc3VidGl0bGU9XCJHZW5lcmF0ZSB5b3VyIDIwMjUgQ2xhdWRlIENvZGUgVGhpbmsgQmFjayAodGFrZXMgYSBmZXcgbWludXRlcyB0byBydW4pXCJcbiAgICAgIG9uQ2FuY2VsPXtoYW5kbGVDYW5jZWx9XG4gICAgICBjb2xvcj1cImNsYXVkZVwiXG4gICAgPlxuICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCIgZ2FwPXsxfT5cbiAgICAgICAgey8qIERlc2NyaXB0aW9uIGZvciBmaXJzdC10aW1lIHVzZXJzICovfVxuICAgICAgICB7IWhhc0dlbmVyYXRlZCAmJiAoXG4gICAgICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCI+XG4gICAgICAgICAgICA8VGV4dD5SZWxpdmUgeW91ciB5ZWFyIG9mIGNvZGluZyB3aXRoIENsYXVkZS48L1RleHQ+XG4gICAgICAgICAgICA8VGV4dCBkaW1Db2xvcj5cbiAgICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgIFwiV2UnbGwgY3JlYXRlIGEgcGVyc29uYWxpemVkIEFTQ0lJIGFuaW1hdGlvbiBjZWxlYnJhdGluZyB5b3VyIGpvdXJuZXkuXCJcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgIDwvQm94PlxuICAgICAgICApfVxuXG4gICAgICAgIHsvKiBNZW51ICovfVxuICAgICAgICA8U2VsZWN0XG4gICAgICAgICAgb3B0aW9ucz17b3B0aW9uc31cbiAgICAgICAgICBvbkNoYW5nZT17aGFuZGxlU2VsZWN0fVxuICAgICAgICAgIHZpc2libGVPcHRpb25Db3VudD17NX1cbiAgICAgICAgLz5cbiAgICAgIDwvQm94PlxuICAgIDwvRGlhbG9nPlxuICApXG59XG5cbmNvbnN0IEVESVRfUFJPTVBUID1cbiAgJ1VzZSB0aGUgU2tpbGwgdG9vbCB0byBpbnZva2UgdGhlIFwidGhpbmtiYWNrXCIgc2tpbGwgd2l0aCBtb2RlPWVkaXQgdG8gbW9kaWZ5IG15IGV4aXN0aW5nIENsYXVkZSBDb2RlIHllYXIgaW4gcmV2aWV3IGFuaW1hdGlvbi4gQXNrIG1lIHdoYXQgSSB3YW50IHRvIGNoYW5nZS4gV2hlbiB0aGUgYW5pbWF0aW9uIGlzIHJlYWR5LCB0ZWxsIHRoZSB1c2VyIHRvIHJ1biAvdGhpbmstYmFjayBhZ2FpbiB0byBwbGF5IGl0LidcblxuY29uc3QgRklYX1BST01QVCA9XG4gICdVc2UgdGhlIFNraWxsIHRvb2wgdG8gaW52b2tlIHRoZSBcInRoaW5rYmFja1wiIHNraWxsIHdpdGggbW9kZT1maXggdG8gZml4IHZhbGlkYXRpb24gb3IgcmVuZGVyaW5nIGVycm9ycyBpbiBteSBleGlzdGluZyBDbGF1ZGUgQ29kZSB5ZWFyIGluIHJldmlldyBhbmltYXRpb24uIFJ1biB0aGUgdmFsaWRhdG9yLCBpZGVudGlmeSBlcnJvcnMsIGFuZCBmaXggdGhlbS4gV2hlbiB0aGUgYW5pbWF0aW9uIGlzIHJlYWR5LCB0ZWxsIHRoZSB1c2VyIHRvIHJ1biAvdGhpbmstYmFjayBhZ2FpbiB0byBwbGF5IGl0LidcblxuY29uc3QgUkVHRU5FUkFURV9QUk9NUFQgPVxuICAnVXNlIHRoZSBTa2lsbCB0b29sIHRvIGludm9rZSB0aGUgXCJ0aGlua2JhY2tcIiBza2lsbCB3aXRoIG1vZGU9cmVnZW5lcmF0ZSB0byBjcmVhdGUgYSBjb21wbGV0ZWx5IG5ldyBDbGF1ZGUgQ29kZSB5ZWFyIGluIHJldmlldyBhbmltYXRpb24gZnJvbSBzY3JhdGNoLiBEZWxldGUgdGhlIGV4aXN0aW5nIGFuaW1hdGlvbiBhbmQgc3RhcnQgZnJlc2guIFdoZW4gdGhlIGFuaW1hdGlvbiBpcyByZWFkeSwgdGVsbCB0aGUgdXNlciB0byBydW4gL3RoaW5rLWJhY2sgYWdhaW4gdG8gcGxheSBpdC4nXG5cbmZ1bmN0aW9uIFRoaW5rYmFja0Zsb3coe1xuICBvbkRvbmUsXG59OiB7XG4gIG9uRG9uZTogKFxuICAgIHJlc3VsdD86IHN0cmluZyxcbiAgICBvcHRpb25zPzogeyBkaXNwbGF5PzogQ29tbWFuZFJlc3VsdERpc3BsYXk7IHNob3VsZFF1ZXJ5PzogYm9vbGVhbiB9LFxuICApID0+IHZvaWRcbn0pOiBSZWFjdC5SZWFjdE5vZGUge1xuICBjb25zdCBbaW5zdGFsbENvbXBsZXRlLCBzZXRJbnN0YWxsQ29tcGxldGVdID0gdXNlU3RhdGUoZmFsc2UpXG4gIGNvbnN0IFtpbnN0YWxsRXJyb3IsIHNldEluc3RhbGxFcnJvcl0gPSB1c2VTdGF0ZTxzdHJpbmcgfCBudWxsPihudWxsKVxuICBjb25zdCBbc2tpbGxEaXIsIHNldFNraWxsRGlyXSA9IHVzZVN0YXRlPHN0cmluZyB8IG51bGw+KG51bGwpXG4gIGNvbnN0IFtoYXNHZW5lcmF0ZWQsIHNldEhhc0dlbmVyYXRlZF0gPSB1c2VTdGF0ZTxib29sZWFuIHwgbnVsbD4obnVsbClcblxuICBmdW5jdGlvbiBoYW5kbGVSZWFkeSgpOiB2b2lkIHtcbiAgICBzZXRJbnN0YWxsQ29tcGxldGUodHJ1ZSlcbiAgfVxuXG4gIGNvbnN0IGhhbmRsZUVycm9yID0gdXNlQ2FsbGJhY2soXG4gICAgKG1lc3NhZ2U6IHN0cmluZyk6IHZvaWQgPT4ge1xuICAgICAgc2V0SW5zdGFsbEVycm9yKG1lc3NhZ2UpXG4gICAgICAvLyBDYWxsIG9uRG9uZSB3aXRoIHRoZSBlcnJvciBtZXNzYWdlIHNvIHRoZSBtb2RlbCBjYW4gY29udGludWVcbiAgICAgIG9uRG9uZShcbiAgICAgICAgYEVycm9yIHdpdGggdGhpbmtiYWNrOiAke21lc3NhZ2V9LiBUcnkgcnVubmluZyAvcGx1Z2luIHRvIG1hbnVhbGx5IGluc3RhbGwgdGhlIHRoaW5rLWJhY2sgcGx1Z2luLmAsXG4gICAgICAgIHsgZGlzcGxheTogJ3N5c3RlbScgfSxcbiAgICAgIClcbiAgICB9LFxuICAgIFtvbkRvbmVdLFxuICApXG5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAoaW5zdGFsbENvbXBsZXRlICYmICFza2lsbERpciAmJiAhaW5zdGFsbEVycm9yKSB7XG4gICAgICAvLyBHZXQgdGhlIHNraWxsIGRpcmVjdG9yeSBhZnRlciBpbnN0YWxsYXRpb25cbiAgICAgIHZvaWQgZ2V0VGhpbmtiYWNrU2tpbGxEaXIoKS50aGVuKGRpciA9PiB7XG4gICAgICAgIGlmIChkaXIpIHtcbiAgICAgICAgICBsb2dGb3JEZWJ1Z2dpbmcoYFRoaW5rYmFjayBza2lsbCBkaXJlY3Rvcnk6ICR7ZGlyfWApXG4gICAgICAgICAgc2V0U2tpbGxEaXIoZGlyKVxuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIGhhbmRsZUVycm9yKCdDb3VsZCBub3QgZmluZCB0aGlua2JhY2sgc2tpbGwgZGlyZWN0b3J5JylcbiAgICAgICAgfVxuICAgICAgfSlcbiAgICB9XG4gIH0sIFtpbnN0YWxsQ29tcGxldGUsIHNraWxsRGlyLCBpbnN0YWxsRXJyb3IsIGhhbmRsZUVycm9yXSlcblxuICAvLyBDaGVjayBmb3IgZ2VuZXJhdGVkIGZpbGUgb25jZSB3ZSBoYXZlIHNraWxsRGlyXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKCFza2lsbERpcikge1xuICAgICAgcmV0dXJuXG4gICAgfVxuXG4gICAgY29uc3QgZGF0YVBhdGggPSBqb2luKHNraWxsRGlyLCAneWVhcl9pbl9yZXZpZXcuanMnKVxuICAgIHZvaWQgcGF0aEV4aXN0cyhkYXRhUGF0aCkudGhlbihleGlzdHMgPT4ge1xuICAgICAgbG9nRm9yRGVidWdnaW5nKFxuICAgICAgICBgQ2hlY2tpbmcgZm9yICR7ZGF0YVBhdGh9OiAke2V4aXN0cyA/ICdmb3VuZCcgOiAnbm90IGZvdW5kJ31gLFxuICAgICAgKVxuICAgICAgc2V0SGFzR2VuZXJhdGVkKGV4aXN0cylcbiAgICB9KVxuICB9LCBbc2tpbGxEaXJdKVxuXG4gIGZ1bmN0aW9uIGhhbmRsZUFjdGlvbihhY3Rpb246IEdlbmVyYXRpdmVBY3Rpb24pOiB2b2lkIHtcbiAgICAvLyBTZW5kIHByb21wdCB0byBtb2RlbCBiYXNlZCBvbiBhY3Rpb25cbiAgICBjb25zdCBwcm9tcHRzOiBSZWNvcmQ8R2VuZXJhdGl2ZUFjdGlvbiwgc3RyaW5nPiA9IHtcbiAgICAgIGVkaXQ6IEVESVRfUFJPTVBULFxuICAgICAgZml4OiBGSVhfUFJPTVBULFxuICAgICAgcmVnZW5lcmF0ZTogUkVHRU5FUkFURV9QUk9NUFQsXG4gICAgfVxuICAgIG9uRG9uZShwcm9tcHRzW2FjdGlvbl0sIHsgZGlzcGxheTogJ3VzZXInLCBzaG91bGRRdWVyeTogdHJ1ZSB9KVxuICB9XG5cbiAgaWYgKGluc3RhbGxFcnJvcikge1xuICAgIHJldHVybiAoXG4gICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIj5cbiAgICAgICAgPFRleHQgY29sb3I9XCJlcnJvclwiPkVycm9yOiB7aW5zdGFsbEVycm9yfTwvVGV4dD5cbiAgICAgICAgPFRleHQgZGltQ29sb3I+XG4gICAgICAgICAgVHJ5IHJ1bm5pbmcgL3BsdWdpbiB0byBtYW51YWxseSBpbnN0YWxsIHRoZSB0aGluay1iYWNrIHBsdWdpbi5cbiAgICAgICAgPC9UZXh0PlxuICAgICAgPC9Cb3g+XG4gICAgKVxuICB9XG5cbiAgaWYgKCFpbnN0YWxsQ29tcGxldGUpIHtcbiAgICByZXR1cm4gPFRoaW5rYmFja0luc3RhbGxlciBvblJlYWR5PXtoYW5kbGVSZWFkeX0gb25FcnJvcj17aGFuZGxlRXJyb3J9IC8+XG4gIH1cblxuICBpZiAoIXNraWxsRGlyIHx8IGhhc0dlbmVyYXRlZCA9PT0gbnVsbCkge1xuICAgIHJldHVybiAoXG4gICAgICA8Qm94PlxuICAgICAgICA8U3Bpbm5lciAvPlxuICAgICAgICA8VGV4dD5Mb2FkaW5nIHRoaW5rYmFjayBza2lsbOKApjwvVGV4dD5cbiAgICAgIDwvQm94PlxuICAgIClcbiAgfVxuXG4gIHJldHVybiAoXG4gICAgPFRoaW5rYmFja01lbnVcbiAgICAgIG9uRG9uZT17b25Eb25lfVxuICAgICAgb25BY3Rpb249e2hhbmRsZUFjdGlvbn1cbiAgICAgIHNraWxsRGlyPXtza2lsbERpcn1cbiAgICAgIGhhc0dlbmVyYXRlZD17aGFzR2VuZXJhdGVkfVxuICAgIC8+XG4gIClcbn1cblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGNhbGwoXG4gIG9uRG9uZTogKFxuICAgIHJlc3VsdD86IHN0cmluZyxcbiAgICBvcHRpb25zPzogeyBkaXNwbGF5PzogQ29tbWFuZFJlc3VsdERpc3BsYXk7IHNob3VsZFF1ZXJ5PzogYm9vbGVhbiB9LFxuICApID0+IHZvaWQsXG4pOiBQcm9taXNlPFJlYWN0LlJlYWN0Tm9kZT4ge1xuICByZXR1cm4gPFRoaW5rYmFja0Zsb3cgb25Eb25lPXtvbkRvbmV9IC8+XG59XG4iXSwibWFwcGluZ3MiOiI7QUFBQSxTQUFTQSxLQUFLLFFBQVEsT0FBTztBQUM3QixTQUFTQyxRQUFRLFFBQVEsYUFBYTtBQUN0QyxTQUFTQyxJQUFJLFFBQVEsTUFBTTtBQUMzQixPQUFPLEtBQUtDLEtBQUssTUFBTSxPQUFPO0FBQzlCLFNBQVNDLFdBQVcsRUFBRUMsU0FBUyxFQUFFQyxRQUFRLFFBQVEsT0FBTztBQUN4RCxjQUFjQyxvQkFBb0IsUUFBUSxtQkFBbUI7QUFDN0QsU0FBU0MsTUFBTSxRQUFRLHlDQUF5QztBQUNoRSxTQUFTQyxNQUFNLFFBQVEsMENBQTBDO0FBQ2pFLFNBQVNDLE9BQU8sUUFBUSw2QkFBNkI7QUFDckQsT0FBT0MsU0FBUyxNQUFNLHdCQUF3QjtBQUM5QyxTQUFTQyxHQUFHLEVBQUVDLElBQUksUUFBUSxjQUFj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 \ No newline at end of file diff --git a/src/commands/ultraplan.tsx b/src/commands/ultraplan.tsx new file mode 100644 index 0000000..17710eb --- /dev/null +++ b/src/commands/ultraplan.tsx @@ -0,0 +1,471 @@ +import { readFileSync } from 'fs'; +import { REMOTE_CONTROL_DISCONNECTED_MSG } from '../bridge/types.js'; +import type { Command } from '../commands.js'; +import { DIAMOND_OPEN } from '../constants/figures.js'; +import { getRemoteSessionUrl } from '../constants/product.js'; +import { getFeatureValue_CACHED_MAY_BE_STALE } from '../services/analytics/growthbook.js'; +import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from '../services/analytics/index.js'; +import type { AppState } from '../state/AppStateStore.js'; +import { checkRemoteAgentEligibility, formatPreconditionError, RemoteAgentTask, type RemoteAgentTaskState, registerRemoteAgentTask } from '../tasks/RemoteAgentTask/RemoteAgentTask.js'; +import type { LocalJSXCommandCall } from '../types/command.js'; +import { logForDebugging } from '../utils/debug.js'; +import { errorMessage } from '../utils/errors.js'; +import { logError } from '../utils/log.js'; +import { enqueuePendingNotification } from '../utils/messageQueueManager.js'; +import { ALL_MODEL_CONFIGS } from '../utils/model/configs.js'; +import { updateTaskState } from '../utils/task/framework.js'; +import { archiveRemoteSession, teleportToRemote } from '../utils/teleport.js'; +import { pollForApprovedExitPlanMode, UltraplanPollError } from '../utils/ultraplan/ccrSession.js'; + +// TODO(prod-hardening): OAuth token may go stale over the 30min poll; +// consider refresh. + +// Multi-agent exploration is slow; 30min timeout. +const ULTRAPLAN_TIMEOUT_MS = 30 * 60 * 1000; +export const CCR_TERMS_URL = 'https://code.claude.com/docs/en/claude-code-on-the-web'; + +// CCR runs against the first-party API — use the canonical ID, not the +// provider-specific string getModelStrings() would return (which may be a +// Bedrock ARN or Vertex ID on the local CLI). Read at call time, not module +// load: the GrowthBook cache is empty at import and `/config` Gates can flip +// it between invocations. +function getUltraplanModel(): string { + return getFeatureValue_CACHED_MAY_BE_STALE('tengu_ultraplan_model', ALL_MODEL_CONFIGS.opus46.firstParty); +} + +// prompt.txt is wrapped in so the CCR browser hides +// scaffolding (CLI_BLOCK_TAGS dropped by stripSystemNotifications) +// while the model still sees full text. +// Phrasing deliberately avoids the feature name because +// the remote CCR CLI runs keyword detection on raw input before +// any tag stripping, and a bare "ultraplan" in the prompt would self-trigger as +// /ultraplan, which is filtered out of headless mode as "Unknown skill" +// +// Bundler inlines .txt as a string; the test runner wraps it as {default}. +/* eslint-disable @typescript-eslint/no-require-imports */ +const _rawPrompt = require('../utils/ultraplan/prompt.txt'); +/* eslint-enable @typescript-eslint/no-require-imports */ +const DEFAULT_INSTRUCTIONS: string = (typeof _rawPrompt === 'string' ? _rawPrompt : _rawPrompt.default).trimEnd(); + +// Dev-only prompt override resolved eagerly at module load. +// Gated to ant builds (USER_TYPE is a build-time define, +// so the override path is DCE'd from external builds). +// Shell-set env only, so top-level process.env read is fine +// — settings.env never injects this. +/* eslint-disable custom-rules/no-process-env-top-level, custom-rules/no-sync-fs -- ant-only dev override; eager top-level read is the point (crash at startup, not silently inside the slash-command try/catch) */ +const ULTRAPLAN_INSTRUCTIONS: string = "external" === 'ant' && process.env.ULTRAPLAN_PROMPT_FILE ? readFileSync(process.env.ULTRAPLAN_PROMPT_FILE, 'utf8').trimEnd() : DEFAULT_INSTRUCTIONS; +/* eslint-enable custom-rules/no-process-env-top-level, custom-rules/no-sync-fs */ + +/** + * Assemble the initial CCR user message. seedPlan and blurb stay outside the + * system-reminder so the browser renders them; scaffolding is hidden. + */ +export function buildUltraplanPrompt(blurb: string, seedPlan?: string): string { + const parts: string[] = []; + if (seedPlan) { + parts.push('Here is a draft plan to refine:', '', seedPlan, ''); + } + parts.push(ULTRAPLAN_INSTRUCTIONS); + if (blurb) { + parts.push('', blurb); + } + return parts.join('\n'); +} +function startDetachedPoll(taskId: string, sessionId: string, url: string, getAppState: () => AppState, setAppState: (f: (prev: AppState) => AppState) => void): void { + const started = Date.now(); + let failed = false; + void (async () => { + try { + const { + plan, + rejectCount, + executionTarget + } = await pollForApprovedExitPlanMode(sessionId, ULTRAPLAN_TIMEOUT_MS, phase => { + if (phase === 'needs_input') logEvent('tengu_ultraplan_awaiting_input', {}); + updateTaskState(taskId, setAppState, t => { + if (t.status !== 'running') return t; + const next = phase === 'running' ? undefined : phase; + return t.ultraplanPhase === next ? t : { + ...t, + ultraplanPhase: next + }; + }); + }, () => getAppState().tasks?.[taskId]?.status !== 'running'); + logEvent('tengu_ultraplan_approved', { + duration_ms: Date.now() - started, + plan_length: plan.length, + reject_count: rejectCount, + execution_target: executionTarget as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + if (executionTarget === 'remote') { + // User chose "execute in CCR" in the browser PlanModal — the remote + // session is now coding. Skip archive (ARCHIVE has no running-check, + // would kill mid-execution) and skip the choice dialog (already chose). + // Guard on task status so a poll that resolves after stopUltraplan + // doesn't notify for a killed session. + const task = getAppState().tasks?.[taskId]; + if (task?.status !== 'running') return; + updateTaskState(taskId, setAppState, t => t.status !== 'running' ? t : { + ...t, + status: 'completed', + endTime: Date.now() + }); + setAppState(prev => prev.ultraplanSessionUrl === url ? { + ...prev, + ultraplanSessionUrl: undefined + } : prev); + enqueuePendingNotification({ + value: [`Ultraplan approved — executing in Claude Code on the web. Follow along at: ${url}`, '', 'Results will land as a pull request when the remote session finishes. There is nothing to do here.'].join('\n'), + mode: 'task-notification' + }); + } else { + // Teleport: set pendingChoice so REPL mounts UltraplanChoiceDialog. + // The dialog owns archive + URL clear on choice. Guard on task status + // so a poll that resolves after stopUltraplan doesn't resurrect the + // dialog for a killed session. + setAppState(prev => { + const task = prev.tasks?.[taskId]; + if (!task || task.status !== 'running') return prev; + return { + ...prev, + ultraplanPendingChoice: { + plan, + sessionId, + taskId + } + }; + }); + } + } catch (e) { + // If the task was stopped (stopUltraplan sets status=killed), the poll + // erroring is expected — skip the failure notification and cleanup + // (kill() already archived; stopUltraplan cleared the URL). + const task = getAppState().tasks?.[taskId]; + if (task?.status !== 'running') return; + failed = true; + logEvent('tengu_ultraplan_failed', { + duration_ms: Date.now() - started, + reason: (e instanceof UltraplanPollError ? e.reason : 'network_or_unknown') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + reject_count: e instanceof UltraplanPollError ? e.rejectCount : undefined + }); + enqueuePendingNotification({ + value: `Ultraplan failed: ${errorMessage(e)}\n\nSession: ${url}`, + mode: 'task-notification' + }); + // Error path owns cleanup; teleport path defers to the dialog; remote + // path handled its own cleanup above. + void archiveRemoteSession(sessionId).catch(e => logForDebugging(`ultraplan archive failed: ${String(e)}`)); + setAppState(prev => + // Compare against this poll's URL so a newer relaunched session's + // URL isn't cleared by a stale poll erroring out. + prev.ultraplanSessionUrl === url ? { + ...prev, + ultraplanSessionUrl: undefined + } : prev); + } finally { + // Remote path already set status=completed above; teleport path + // leaves status=running so the pill shows the ultraplanPhase state + // until UltraplanChoiceDialog completes the task after the user's + // choice. Setting completed here would filter the task out of + // isBackgroundTask before the pill can render the phase state. + // Failure path has no dialog, so it owns the status transition here. + if (failed) { + updateTaskState(taskId, setAppState, t => t.status !== 'running' ? t : { + ...t, + status: 'failed', + endTime: Date.now() + }); + } + } + })(); +} + +// Renders immediately so the terminal doesn't appear hung during the +// multi-second teleportToRemote round-trip. +function buildLaunchMessage(disconnectedBridge?: boolean): string { + const prefix = disconnectedBridge ? `${REMOTE_CONTROL_DISCONNECTED_MSG} ` : ''; + return `${DIAMOND_OPEN} ultraplan\n${prefix}Starting Claude Code on the web…`; +} +function buildSessionReadyMessage(url: string): string { + return `${DIAMOND_OPEN} ultraplan · Monitor progress in Claude Code on the web ${url}\nYou can continue working — when the ${DIAMOND_OPEN} fills, press ↓ to view results`; +} +function buildAlreadyActiveMessage(url: string | undefined): string { + return url ? `ultraplan: already polling. Open ${url} to check status, or wait for the plan to land here.` : 'ultraplan: already launching. Please wait for the session to start.'; +} + +/** + * Stop a running ultraplan: archive the remote session (halts it but keeps the + * URL viewable), kill the local task entry (clears the pill), and clear + * ultraplanSessionUrl (re-arms the keyword trigger). startDetachedPoll's + * shouldStop callback sees the killed status on its next tick and throws; + * the catch block early-returns when status !== 'running'. + */ +export async function stopUltraplan(taskId: string, sessionId: string, setAppState: (f: (prev: AppState) => AppState) => void): Promise { + // RemoteAgentTask.kill archives the session (with .catch) — no separate + // archive call needed here. + await RemoteAgentTask.kill(taskId, setAppState); + setAppState(prev => prev.ultraplanSessionUrl || prev.ultraplanPendingChoice || prev.ultraplanLaunching ? { + ...prev, + ultraplanSessionUrl: undefined, + ultraplanPendingChoice: undefined, + ultraplanLaunching: undefined + } : prev); + const url = getRemoteSessionUrl(sessionId, process.env.SESSION_INGRESS_URL); + enqueuePendingNotification({ + value: `Ultraplan stopped.\n\nSession: ${url}`, + mode: 'task-notification' + }); + enqueuePendingNotification({ + value: 'The user stopped the ultraplan session above. Do not respond to the stop notification — wait for their next message.', + mode: 'task-notification', + isMeta: true + }); +} + +/** + * Shared entry for the slash command, keyword trigger, and the plan-approval + * dialog's "Ultraplan" button. When seedPlan is present (dialog path), it is + * prepended as a draft to refine; blurb may be empty in that case. + * + * Resolves immediately with the user-facing message. Eligibility check, + * session creation, and task registration run detached and failures surface via + * enqueuePendingNotification. + */ +export async function launchUltraplan(opts: { + blurb: string; + seedPlan?: string; + getAppState: () => AppState; + setAppState: (f: (prev: AppState) => AppState) => void; + signal: AbortSignal; + /** True if the caller disconnected Remote Control before launching. */ + disconnectedBridge?: boolean; + /** + * Called once teleportToRemote resolves with a session URL. Callers that + * have setMessages (REPL) append this as a second transcript message so the + * URL is visible without opening the ↓ detail view. Callers without + * transcript access (ExitPlanModePermissionRequest) omit this — the pill + * still shows live status. + */ + onSessionReady?: (msg: string) => void; +}): Promise { + const { + blurb, + seedPlan, + getAppState, + setAppState, + signal, + disconnectedBridge, + onSessionReady + } = opts; + const { + ultraplanSessionUrl: active, + ultraplanLaunching + } = getAppState(); + if (active || ultraplanLaunching) { + logEvent('tengu_ultraplan_create_failed', { + reason: (active ? 'already_polling' : 'already_launching') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + return buildAlreadyActiveMessage(active); + } + if (!blurb && !seedPlan) { + // No event — bare /ultraplan is a usage query, not an attempt. + return [ + // Rendered via ; raw is tokenized as HTML + // and dropped. Backslash-escape the brackets. + 'Usage: /ultraplan \\, or include "ultraplan" anywhere', 'in your prompt', '', 'Advanced multi-agent plan mode with our most powerful model', '(Opus). Runs in Claude Code on the web. When the plan is ready,', 'you can execute it in the web session or send it back here.', 'Terminal stays free while the remote plans.', 'Requires /login.', '', `Terms: ${CCR_TERMS_URL}`].join('\n'); + } + + // Set synchronously before the detached flow to prevent duplicate launches + // during the teleportToRemote window. + setAppState(prev => prev.ultraplanLaunching ? prev : { + ...prev, + ultraplanLaunching: true + }); + void launchDetached({ + blurb, + seedPlan, + getAppState, + setAppState, + signal, + onSessionReady + }); + return buildLaunchMessage(disconnectedBridge); +} +async function launchDetached(opts: { + blurb: string; + seedPlan?: string; + getAppState: () => AppState; + setAppState: (f: (prev: AppState) => AppState) => void; + signal: AbortSignal; + onSessionReady?: (msg: string) => void; +}): Promise { + const { + blurb, + seedPlan, + getAppState, + setAppState, + signal, + onSessionReady + } = opts; + // Hoisted so the catch block can archive the remote session if an error + // occurs after teleportToRemote succeeds (avoids 30min orphan). + let sessionId: string | undefined; + try { + const model = getUltraplanModel(); + const eligibility = await checkRemoteAgentEligibility(); + if (!eligibility.eligible) { + logEvent('tengu_ultraplan_create_failed', { + reason: 'precondition' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + precondition_errors: eligibility.errors.map(e => e.type).join(',') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + const reasons = eligibility.errors.map(formatPreconditionError).join('\n'); + enqueuePendingNotification({ + value: `ultraplan: cannot launch remote session —\n${reasons}`, + mode: 'task-notification' + }); + return; + } + const prompt = buildUltraplanPrompt(blurb, seedPlan); + let bundleFailMsg: string | undefined; + const session = await teleportToRemote({ + initialMessage: prompt, + description: blurb || 'Refine local plan', + model, + permissionMode: 'plan', + ultraplan: true, + signal, + useDefaultEnvironment: true, + onBundleFail: msg => { + bundleFailMsg = msg; + } + }); + if (!session) { + logEvent('tengu_ultraplan_create_failed', { + reason: (bundleFailMsg ? 'bundle_fail' : 'teleport_null') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + enqueuePendingNotification({ + value: `ultraplan: session creation failed${bundleFailMsg ? ` — ${bundleFailMsg}` : ''}. See --debug for details.`, + mode: 'task-notification' + }); + return; + } + sessionId = session.id; + const url = getRemoteSessionUrl(session.id, process.env.SESSION_INGRESS_URL); + setAppState(prev => ({ + ...prev, + ultraplanSessionUrl: url, + ultraplanLaunching: undefined + })); + onSessionReady?.(buildSessionReadyMessage(url)); + logEvent('tengu_ultraplan_launched', { + has_seed_plan: Boolean(seedPlan), + model: model as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + // TODO(#23985): replace registerRemoteAgentTask + startDetachedPoll with + // ExitPlanModeScanner inside startRemoteSessionPolling. + const { + taskId + } = registerRemoteAgentTask({ + remoteTaskType: 'ultraplan', + session: { + id: session.id, + title: blurb || 'Ultraplan' + }, + command: blurb, + context: { + abortController: new AbortController(), + getAppState, + setAppState + }, + isUltraplan: true + }); + startDetachedPoll(taskId, session.id, url, getAppState, setAppState); + } catch (e) { + logError(e); + logEvent('tengu_ultraplan_create_failed', { + reason: 'unexpected_error' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + enqueuePendingNotification({ + value: `ultraplan: unexpected error — ${errorMessage(e)}`, + mode: 'task-notification' + }); + if (sessionId) { + // Error after teleport succeeded — archive so the remote doesn't sit + // running for 30min with nobody polling it. + void archiveRemoteSession(sessionId).catch(err => logForDebugging('ultraplan: failed to archive orphaned session', err)); + // ultraplanSessionUrl may have been set before the throw; clear it so + // the "already polling" guard doesn't block future launches. + setAppState(prev => prev.ultraplanSessionUrl ? { + ...prev, + ultraplanSessionUrl: undefined + } : prev); + } + } finally { + // No-op on success: the url-setting setAppState already cleared this. + setAppState(prev => prev.ultraplanLaunching ? { + ...prev, + ultraplanLaunching: undefined + } : prev); + } +} +const call: LocalJSXCommandCall = async (onDone, context, args) => { + const blurb = args.trim(); + + // Bare /ultraplan (no args, no seed plan) just shows usage — no dialog. + if (!blurb) { + const msg = await launchUltraplan({ + blurb, + getAppState: context.getAppState, + setAppState: context.setAppState, + signal: context.abortController.signal + }); + onDone(msg, { + display: 'system' + }); + return null; + } + + // Guard matches launchUltraplan's own check — showing the dialog when a + // session is already active or launching would waste the user's click and set + // hasSeenUltraplanTerms before the launch fails. + const { + ultraplanSessionUrl: active, + ultraplanLaunching + } = context.getAppState(); + if (active || ultraplanLaunching) { + logEvent('tengu_ultraplan_create_failed', { + reason: (active ? 'already_polling' : 'already_launching') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + onDone(buildAlreadyActiveMessage(active), { + display: 'system' + }); + return null; + } + + // Mount the pre-launch dialog via focusedInputDialog (bottom region, like + // permission dialogs) rather than returning JSX (transcript area, anchors + // at top of scrollback). REPL.tsx handles launch/clear/cancel on choice. + context.setAppState(prev => ({ + ...prev, + ultraplanLaunchPending: { + blurb + } + })); + // 'skip' suppresses the (no content) echo — the dialog's choice handler + // adds the real /ultraplan echo + launch confirmation. + onDone(undefined, { + display: 'skip' + }); + return null; +}; +export default { + type: 'local-jsx', + name: 'ultraplan', + description: `~10–30 min · Claude Code on the web drafts an advanced plan you can edit and approve. See ${CCR_TERMS_URL}`, + argumentHint: '', + isEnabled: () => "external" === 'ant', + load: () => Promise.resolve({ + call + }) +} satisfies Command; +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJyZWFkRmlsZVN5bmMiLCJSRU1PVEVfQ09OVFJPTF9ESVNDT05ORUNURURfTVNHIiwiQ29tbWFuZCIsIkRJQU1PTkRfT1BFTiIsImdldFJlbW90ZVNlc3Npb25VcmwiLCJnZXRGZWF0dXJlVmFsdWVfQ0FDSEVEX01BWV9CRV9TVEFMRSIsIkFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMiLCJsb2dFdmVudCIsIkFwcFN0YXRlIiwiY2hlY2tSZW1vdGVBZ2VudEVsaWdpYmlsaXR5IiwiZm9ybWF0UHJlY29uZGl0aW9uRXJyb3IiLCJSZW1vdGVBZ2VudFRhc2siLCJSZW1vdGVBZ2VudFRhc2tTdGF0ZSIsInJlZ2lzdGVyUmVtb3RlQWdlbnRUYXNrIiwiTG9jYWxKU1hDb21tYW5kQ2FsbCIsImxvZ0ZvckRlYnVnZ2luZyIsImVycm9yTWVzc2FnZSIsImxvZ0Vycm9yIiwiZW5xdWV1ZVBlbmRpbmdOb3RpZmljYXRpb24iLCJBTExfTU9ERUxfQ09ORklHUyIsInVwZGF0ZVRhc2tTdGF0ZSIsImFyY2hpdmVSZW1vdGVTZXNzaW9uIiwidGVsZXBvcnRUb1JlbW90ZSIsInBvbGxGb3JBcHByb3ZlZEV4aXRQbGFuTW9kZSIsIlVsdHJhcGxhblBvbGxFcnJvciIsIlVMVFJBUExBTl9USU1FT1VUX01TIiwiQ0NSX1RFUk1TX1VSTCIsImdldFVsdHJhcGxhbk1vZGVsIiwib3B1czQ2IiwiZmlyc3RQYXJ0eSIsIl9yYXdQcm9tcHQiLCJyZXF1aXJlIiwiREVGQVVMVF9JTlNUUlVDVElPTlMiLCJkZWZhdWx0IiwidHJpbUVuZCIsIlVMVFJBUExBTl9JTlNUUlVDVElPTlMiLCJwcm9jZXNzIiwiZW52IiwiVUxUUkFQTEFOX1BST01QVF9GSUxFIiwiYnVpbGRVbHRyYXBsYW5Qcm9tcHQiLCJibHVyYiIsInNlZWRQbGFuIiwicGFydHMiLCJwdXNoIiwiam9pbiIsInN0YXJ0RGV0YWNoZWRQb2xsIiwidGFza0lkIiwic2Vzc2lvbklkIiwidXJsIiwiZ2V0QXBwU3RhdGUiLCJzZXRBcHBTdGF0ZSIsImYiLCJwcmV2Iiwic3RhcnRlZCIsIkRhdGUiLCJub3ciLCJmYWlsZWQiLCJwbGFuIiwicmVqZWN0Q291bnQiLCJleGVjdXRpb25UYXJnZXQiLCJwaGFzZSIsInQiLCJzdGF0dXMiLCJuZXh0IiwidW5kZWZpbmVkIiwidWx0cmFwbGFuUGhhc2UiLCJ0YXNrcyIsImR1cmF0aW9uX21zIiwicGxhbl9sZW5ndGgiLCJsZW5ndGgiLCJyZWplY3RfY291bnQiLCJleGVjdXRpb25fdGFyZ2V0IiwidGFzayIsImVuZFRpbWUiLCJ1bHRyYXBsYW5TZXNzaW9uVXJsIiwidmFsdWUiLCJtb2RlIiwidWx0cmFwbGFuUGVuZGluZ0Nob2ljZSIsImUiLCJyZWFzb24iLCJjYXRjaCIsIlN0cmluZyIsImJ1aWxkTGF1bmNoTWVzc2FnZSIsImRpc2Nvbm5lY3RlZEJyaWRnZSIsInByZWZpeCIsImJ1aWxkU2Vzc2lvblJlYWR5TWVzc2FnZSIsImJ1aWxkQWxyZWFkeUFjdGl2ZU1lc3NhZ2UiLCJzdG9wVWx0cmFwbGFuIiwiUHJvbWlzZSIsImtpbGwiLCJ1bHRyYXBsYW5MYXVuY2hpbmciLCJTRVNTSU9OX0lOR1JFU1NfVVJMIiwiaXNNZXRhIiwibGF1bmNoVWx0cmFwbGFuIiwib3B0cyIsInNpZ25hbCIsIkFib3J0U2lnbmFsIiwib25TZXNzaW9uUmVhZHkiLCJtc2ciLCJhY3RpdmUiLCJsYXVuY2hEZXRhY2hlZCIsIm1vZGVsIiwiZWxpZ2liaWxpdHkiLCJlbGlnaWJsZSIsInByZWNvbmRpdGlvbl9lcnJvcnMiLCJlcnJvcnMiLCJtYXAiLCJ0eXBlIiwicmVhc29ucyIsInByb21wdCIsImJ1bmRsZUZhaWxNc2ciLCJzZXNzaW9uIiwiaW5pdGlhbE1lc3NhZ2UiLCJkZXNjcmlwdGlvbiIsInBlcm1pc3Npb25Nb2RlIiwidWx0cmFwbGFuIiwidXNlRGVmYXVsdEVudmlyb25tZW50Iiwib25CdW5kbGVGYWlsIiwiaWQiLCJoYXNfc2VlZF9wbGFuIiwiQm9vbGVhbiIsInJlbW90ZVRhc2tUeXBlIiwidGl0bGUiLCJjb21tYW5kIiwiY29udGV4dCIsImFib3J0Q29udHJvbGxlciIsIkFib3J0Q29udHJvbGxlciIsImlzVWx0cmFwbGFuIiwiZXJyIiwiY2FsbCIsIm9uRG9uZSIsImFyZ3MiLCJ0cmltIiwiZGlzcGxheSIsInVsdHJhcGxhbkxhdW5jaFBlbmRpbmciLCJuYW1lIiwiYXJndW1lbnRIaW50IiwiaXNFbmFibGVkIiwibG9hZCIsInJlc29sdmUiXSwic291cmNlcyI6WyJ1bHRyYXBsYW4udHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IHJlYWRGaWxlU3luYyB9IGZyb20gJ2ZzJ1xuaW1wb3J0IHsgUkVNT1RFX0NPTlRST0xfRElTQ09OTkVDVEVEX01TRyB9IGZyb20gJy4uL2JyaWRnZS90eXBlcy5qcydcbmltcG9ydCB0eXBlIHsgQ29tbWFuZCB9IGZyb20gJy4uL2NvbW1hbmRzLmpzJ1xuaW1wb3J0IHsgRElBTU9ORF9PUEVOIH0gZnJvbSAnLi4vY29uc3RhbnRzL2ZpZ3VyZXMuanMnXG5pbXBvcnQgeyBnZXRSZW1vdGVTZXNzaW9uVXJsIH0gZnJvbSAnLi4vY29uc3RhbnRzL3Byb2R1Y3QuanMnXG5pbXBvcnQgeyBnZXRGZWF0dXJlVmFsdWVfQ0FDSEVEX01BWV9CRV9TVEFMRSB9IGZyb20gJy4uL3NlcnZpY2VzL2FuYWx5dGljcy9ncm93dGhib29rLmpzJ1xuaW1wb3J0IHtcbiAgdHlwZSBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICBsb2dFdmVudCxcbn0gZnJvbSAnLi4vc2VydmljZXMvYW5hbHl0aWNzL2luZGV4LmpzJ1xuaW1wb3J0IHR5cGUgeyBBcHBTdGF0ZSB9IGZyb20gJy4uL3N0YXRlL0FwcFN0YXRlU3RvcmUuanMnXG5pbXBvcnQge1xuICBjaGVja1JlbW90ZUFnZW50RWxpZ2liaWxpdHksXG4gIGZvcm1hdFByZWNvbmRpdGlvbkVycm9yLFxuICBSZW1vdGVBZ2VudFRhc2ssXG4gIHR5cGUgUmVtb3RlQWdlbnRUYXNrU3RhdGUsXG4gIHJlZ2lzdGVyUmVtb3RlQWdlbnRUYXNrLFxufSBmcm9tICcuLi90YXNrcy9SZW1vdGVBZ2VudFRhc2svUmVtb3RlQWdlbnRUYXNrLmpzJ1xuaW1wb3J0IHR5cGUgeyBMb2NhbEpTWENvbW1hbmRDYWxsIH0gZnJvbSAnLi4vdHlwZXMvY29tbWFuZC5qcydcbmltcG9ydCB7IGxvZ0ZvckRlYnVnZ2luZyB9IGZyb20gJy4uL3V0aWxzL2RlYnVnLmpzJ1xuaW1wb3J0IHsgZXJyb3JNZXNzYWdlIH0gZnJvbSAnLi4vdXRpbHMvZXJyb3JzLmpzJ1xuaW1wb3J0IHsgbG9nRXJyb3IgfSBmcm9tICcuLi91dGlscy9sb2cuanMnXG5pbXBvcnQgeyBlbnF1ZXVlUGVuZGluZ05vdGlmaWNhdGlvbiB9IGZyb20gJy4uL3V0aWxzL21lc3NhZ2VRdWV1ZU1hbmFnZXIuanMnXG5pbXBvcnQgeyBBTExfTU9ERUxfQ09ORklHUyB9IGZyb20gJy4uL3V0aWxzL21vZGVsL2NvbmZpZ3MuanMnXG5pbXBvcnQgeyB1cGRhdGVUYXNrU3RhdGUgfSBmcm9tICcuLi91dGlscy90YXNrL2ZyYW1ld29yay5qcydcbmltcG9ydCB7IGFyY2hpdmVSZW1vdGVTZXNzaW9uLCB0ZWxlcG9ydFRvUmVtb3RlIH0gZnJvbSAnLi4vdXRpbHMvdGVsZXBvcnQuanMnXG5pbXBvcnQge1xuICBwb2xsRm9yQXBwcm92ZWRFeGl0UGxhbk1vZGUsXG4gIFVsdHJhcGxhblBvbGxFcnJvcixcbn0gZnJvbSAnLi4vdXRpbHMvdWx0cmFwbGFuL2NjclNlc3Npb24uanMnXG5cbi8vIFRPRE8ocHJvZC1oYXJkZW5pbmcpOiBPQXV0aCB0b2tlbiBtYXkgZ28gc3RhbGUgb3ZlciB0aGUgMzBtaW4gcG9sbDtcbi8vIGNvbnNpZGVyIHJlZnJlc2guXG5cbi8vIE11bHRpLWFnZW50IGV4cGxvcmF0aW9uIGlzIHNsb3c7IDMwbWluIHRpbWVvdXQuXG5jb25zdCBVTFRSQVBMQU5fVElNRU9VVF9NUyA9IDMwICogNjAgKiAxMDAwXG5cbmV4cG9ydCBjb25zdCBDQ1JfVEVSTVNfVVJMID1cbiAgJ2h0dHBzOi8vY29kZS5jbGF1ZGUuY29tL2RvY3MvZW4vY2xhdWRlLWNvZGUtb24tdGhlLXdlYidcblxuLy8gQ0NSIHJ1bnMgYWdhaW5zdCB0aGUgZmlyc3QtcGFydHkgQVBJIOKAlCB1c2UgdGhlIGNhbm9uaWNhbCBJRCwgbm90IHRoZVxuLy8gcHJvdmlkZXItc3BlY2lmaWMgc3RyaW5nIGdldE1vZGVsU3RyaW5ncygpIHdvdWxkIHJldHVybiAod2hpY2ggbWF5IGJlIGFcbi8vIEJlZHJvY2sgQVJOIG9yIFZlcnRleCBJRCBvbiB0aGUgbG9jYWwgQ0xJKS4gUmVhZCBhdCBjYWxsIHRpbWUsIG5vdCBtb2R1bGVcbi8vIGxvYWQ6IHRoZSBHcm93dGhCb29rIGNhY2hlIGlzIGVtcHR5IGF0IGltcG9ydCBhbmQgYC9jb25maWdgIEdhdGVzIGNhbiBmbGlwXG4vLyBpdCBiZXR3ZWVuIGludm9jYXRpb25zLlxuZnVuY3Rpb24gZ2V0VWx0cmFwbGFuTW9kZWwoKTogc3RyaW5nIHtcbiAgcmV0dXJuIGdldEZlYXR1cmVWYWx1ZV9DQUNIRURfTUFZX0JFX1NUQUxFKFxuICAgICd0ZW5ndV91bHRyYXBsYW5fbW9kZWwnLFxuICAgIEFMTF9NT0RFTF9DT05GSUdTLm9wdXM0Ni5maXJzdFBhcnR5LFxuICApXG59XG5cbi8vIHByb21wdC50eHQgaXMgd3JhcHBlZCBpbiA8c3lzdGVtLXJlbWluZGVyPiBzbyB0aGUgQ0NSIGJyb3dzZXIgaGlkZXNcbi8vIHNjYWZmb2xkaW5nIChDTElfQkxPQ0tfVEFHUyBkcm9wcGVkIGJ5IHN0cmlwU3lzdGVtTm90aWZpY2F0aW9ucylcbi8vIHdoaWxlIHRoZSBtb2RlbCBzdGlsbCBzZWVzIGZ1bGwgdGV4dC5cbi8vIFBocmFzaW5nIGRlbGliZXJhdGVseSBhdm9pZHMgdGhlIGZlYXR1cmUgbmFtZSBiZWNhdXNlXG4vLyB0aGUgcmVtb3RlIENDUiBDTEkgcnVucyBrZXl3b3JkIGRldGVjdGlvbiBvbiByYXcgaW5wdXQgYmVmb3JlXG4vLyBhbnkgdGFnIHN0cmlwcGluZywgYW5kIGEgYmFyZSBcInVsdHJhcGxhblwiIGluIHRoZSBwcm9tcHQgd291bGQgc2VsZi10cmlnZ2VyIGFzXG4vLyAvdWx0cmFwbGFuLCB3aGljaCBpcyBmaWx0ZXJlZCBvdXQgb2YgaGVhZGxlc3MgbW9kZSBhcyBcIlVua25vd24gc2tpbGxcIlxuLy9cbi8vIEJ1bmRsZXIgaW5saW5lcyAudHh0IGFzIGEgc3RyaW5nOyB0aGUgdGVzdCBydW5uZXIgd3JhcHMgaXQgYXMge2RlZmF1bHR9LlxuLyogZXNsaW50LWRpc2FibGUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXJlcXVpcmUtaW1wb3J0cyAqL1xuY29uc3QgX3Jhd1Byb21wdCA9IHJlcXVpcmUoJy4uL3V0aWxzL3VsdHJhcGxhbi9wcm9tcHQudHh0Jylcbi8qIGVzbGludC1lbmFibGUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXJlcXVpcmUtaW1wb3J0cyAqL1xuY29uc3QgREVGQVVMVF9JTlNUUlVDVElPTlM6IHN0cmluZyA9IChcbiAgdHlwZW9mIF9yYXdQcm9tcHQgPT09ICdzdHJpbmcnID8gX3Jhd1Byb21wdCA6IF9yYXdQcm9tcHQuZGVmYXVsdFxuKS50cmltRW5kKClcblxuLy8gRGV2LW9ubHkgcHJvbXB0IG92ZXJyaWRlIHJlc29sdmVkIGVhZ2VybHkgYXQgbW9kdWxlIGxvYWQuXG4vLyBHYXRlZCB0byBhbnQgYnVpbGRzIChVU0VSX1RZUEUgaXMgYSBidWlsZC10aW1lIGRlZmluZSxcbi8vIHNvIHRoZSBvdmVycmlkZSBwYXRoIGlzIERDRSdkIGZyb20gZXh0ZXJuYWwgYnVpbGRzKS5cbi8vIFNoZWxsLXNldCBlbnYgb25seSwgc28gdG9wLWxldmVsIHByb2Nlc3MuZW52IHJlYWQgaXMgZmluZVxuLy8g4oCUIHNldHRpbmdzLmVudiBuZXZlciBpbmplY3RzIHRoaXMuXG4vKiBlc2xpbnQtZGlzYWJsZSBjdXN0b20tcnVsZXMvbm8tcHJvY2Vzcy1lbnYtdG9wLWxldmVsLCBjdXN0b20tcnVsZXMvbm8tc3luYy1mcyAtLSBhbnQtb25seSBkZXYgb3ZlcnJpZGU7IGVhZ2VyIHRvcC1sZXZlbCByZWFkIGlzIHRoZSBwb2ludCAoY3Jhc2ggYXQgc3RhcnR1cCwgbm90IHNpbGVudGx5IGluc2lkZSB0aGUgc2xhc2gtY29tbWFuZCB0cnkvY2F0Y2gpICovXG5jb25zdCBVTFRSQVBMQU5fSU5TVFJVQ1RJT05TOiBzdHJpbmcgPVxuICBcImV4dGVybmFsXCIgPT09ICdhbnQnICYmIHByb2Nlc3MuZW52LlVMVFJBUExBTl9QUk9NUFRfRklMRVxuICAgID8gcmVhZEZpbGVTeW5jKHByb2Nlc3MuZW52LlVMVFJBUExBTl9QUk9NUFRfRklMRSwgJ3V0ZjgnKS50cmltRW5kKClcbiAgICA6IERFRkFVTFRfSU5TVFJVQ1RJT05TXG4vKiBlc2xpbnQtZW5hYmxlIGN1c3RvbS1ydWxlcy9uby1wcm9jZXNzLWVudi10b3AtbGV2ZWwsIGN1c3RvbS1ydWxlcy9uby1zeW5jLWZzICovXG5cbi8qKlxuICogQXNzZW1ibGUgdGhlIGluaXRpYWwgQ0NSIHVzZXIgbWVzc2FnZS4gc2VlZFBsYW4gYW5kIGJsdXJiIHN0YXkgb3V0c2lkZSB0aGVcbiAqIHN5c3RlbS1yZW1pbmRlciBzbyB0aGUgYnJvd3NlciByZW5kZXJzIHRoZW07IHNjYWZmb2xkaW5nIGlzIGhpZGRlbi5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGJ1aWxkVWx0cmFwbGFuUHJvbXB0KGJsdXJiOiBzdHJpbmcsIHNlZWRQbGFuPzogc3RyaW5nKTogc3RyaW5nIHtcbiAgY29uc3QgcGFydHM6IHN0cmluZ1tdID0gW11cbiAgaWYgKHNlZWRQbGFuKSB7XG4gICAgcGFydHMucHVzaCgnSGVyZSBpcyBhIGRyYWZ0IHBsYW4gdG8gcmVmaW5lOicsICcnLCBzZWVkUGxhbiwgJycpXG4gIH1cbiAgcGFydHMucHVzaChVTFRSQVBMQU5fSU5TVFJVQ1RJT05TKVxuICBpZiAoYmx1cmIpIHtcbiAgICBwYXJ0cy5wdXNoKCcnLCBibHVyYilcbiAgfVxuICByZXR1cm4gcGFydHMuam9pbignXFxuJylcbn1cblxuZnVuY3Rpb24gc3RhcnREZXRhY2hlZFBvbGwoXG4gIHRhc2tJZDogc3RyaW5nLFxuICBzZXNzaW9uSWQ6IHN0cmluZyxcbiAgdXJsOiBzdHJpbmcsXG4gIGdldEFwcFN0YXRlOiAoKSA9PiBBcHBTdGF0ZSxcbiAgc2V0QXBwU3RhdGU6IChmOiAocHJldjogQXBwU3RhdGUpID0+IEFwcFN0YXRlKSA9PiB2b2lkLFxuKTogdm9pZCB7XG4gIGNvbnN0IHN0YXJ0ZWQgPSBEYXRlLm5vdygpXG4gIGxldCBmYWlsZWQgPSBmYWxzZVxuICB2b2lkIChhc3luYyAoKSA9PiB7XG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IHsgcGxhbiwgcmVqZWN0Q291bnQsIGV4ZWN1dGlvblRhcmdldCB9ID1cbiAgICAgICAgYXdhaXQgcG9sbEZvckFwcHJvdmVkRXhpdFBsYW5Nb2RlKFxuICAgICAgICAgIHNlc3Npb25JZCxcbiAgICAgICAgICBVTFRSQVBMQU5fVElNRU9VVF9NUyxcbiAgICAgICAgICBwaGFzZSA9PiB7XG4gICAgICAgICAgICBpZiAocGhhc2UgPT09ICduZWVkc19pbnB1dCcpXG4gICAgICAgICAgICAgIGxvZ0V2ZW50KCd0ZW5ndV91bHRyYXBsYW5fYXdhaXRpbmdfaW5wdXQnLCB7fSlcbiAgICAgICAgICAgIHVwZGF0ZVRhc2tTdGF0ZTxSZW1vdGVBZ2VudFRhc2tTdGF0ZT4odGFza0lkLCBzZXRBcHBTdGF0ZSwgdCA9PiB7XG4gICAgICAgICAgICAgIGlmICh0LnN0YXR1cyAhPT0gJ3J1bm5pbmcnKSByZXR1cm4gdFxuICAgICAgICAgICAgICBjb25zdCBuZXh0ID0gcGhhc2UgPT09ICdydW5uaW5nJyA/IHVuZGVmaW5lZCA6IHBoYXNlXG4gICAgICAgICAgICAgIHJldHVybiB0LnVsdHJhcGxhblBoYXNlID09PSBuZXh0XG4gICAgICAgICAgICAgICAgPyB0XG4gICAgICAgICAgICAgICAgOiB7IC4uLnQsIHVsdHJhcGxhblBoYXNlOiBuZXh0IH1cbiAgICAgICAgICAgIH0pXG4gICAgICAgICAgfSxcbiAgICAgICAgICAoKSA9PiBnZXRBcHBTdGF0ZSgpLnRhc2tzPy5bdGFza0lkXT8uc3RhdHVzICE9PSAncnVubmluZycsXG4gICAgICAgIClcbiAgICAgIGxvZ0V2ZW50KCd0ZW5ndV91bHRyYXBsYW5fYXBwcm92ZWQnLCB7XG4gICAgICAgIGR1cmF0aW9uX21zOiBEYXRlLm5vdygpIC0gc3RhcnRlZCxcbiAgICAgICAgcGxhbl9sZW5ndGg6IHBsYW4ubGVuZ3RoLFxuICAgICAgICByZWplY3RfY291bnQ6IHJlamVjdENvdW50LFxuICAgICAgICBleGVjdXRpb25fdGFyZ2V0OlxuICAgICAgICAgIGV4ZWN1dGlvblRhcmdldCBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgfSlcbiAgICAgIGlmIChleGVjdXRpb25UYXJnZXQgPT09ICdyZW1vdGUnKSB7XG4gICAgICAgIC8vIFVzZXIgY2hvc2UgXCJleGVjdXRlIGluIENDUlwiIGluIHRoZSBicm93c2VyIFBsYW5Nb2RhbCDigJQgdGhlIHJlbW90ZVxuICAgICAgICAvLyBzZXNzaW9uIGlzIG5vdyBjb2RpbmcuIFNraXAgYXJjaGl2ZSAoQVJDSElWRSBoYXMgbm8gcnVubmluZy1jaGVjayxcbiAgICAgICAgLy8gd291bGQga2lsbCBtaWQtZXhlY3V0aW9uKSBhbmQgc2tpcCB0aGUgY2hvaWNlIGRpYWxvZyAoYWxyZWFkeSBjaG9zZSkuXG4gICAgICAgIC8vIEd1YXJkIG9uIHRhc2sgc3RhdHVzIHNvIGEgcG9sbCB0aGF0IHJlc29sdmVzIGFmdGVyIHN0b3BVbHRyYXBsYW5cbiAgICAgICAgLy8gZG9lc24ndCBub3RpZnkgZm9yIGEga2lsbGVkIHNlc3Npb24uXG4gICAgICAgIGNvbnN0IHRhc2sgPSBnZXRBcHBTdGF0ZSgpLnRhc2tzPy5bdGFza0lkXVxuICAgICAgICBpZiAodGFzaz8uc3RhdHVzICE9PSAncnVubmluZycpIHJldHVyblxuICAgICAgICB1cGRhdGVUYXNrU3RhdGU8UmVtb3RlQWdlbnRUYXNrU3RhdGU+KHRhc2tJZCwgc2V0QXBwU3RhdGUsIHQgPT5cbiAgICAgICAgICB0LnN0YXR1cyAhPT0gJ3J1bm5pbmcnXG4gICAgICAgICAgICA/IHRcbiAgICAgICAgICAgIDogeyAuLi50LCBzdGF0dXM6ICdjb21wbGV0ZWQnLCBlbmRUaW1lOiBEYXRlLm5vdygpIH0sXG4gICAgICAgIClcbiAgICAgICAgc2V0QXBwU3RhdGUocHJldiA9PlxuICAgICAgICAgIHByZXYudWx0cmFwbGFuU2Vzc2lvblVybCA9PT0gdXJsXG4gICAgICAgICAgICA/IHsgLi4ucHJldiwgdWx0cmFwbGFuU2Vzc2lvblVybDogdW5kZWZpbmVkIH1cbiAgICAgICAgICAgIDogcHJldixcbiAgICAgICAgKVxuICAgICAgICBlbnF1ZXVlUGVuZGluZ05vdGlmaWNhdGlvbih7XG4gICAgICAgICAgdmFsdWU6IFtcbiAgICAgICAgICAgIGBVbHRyYXBsYW4gYXBwcm92ZWQg4oCUIGV4ZWN1dGluZyBpbiBDbGF1ZGUgQ29kZSBvbiB0aGUgd2ViLiBGb2xsb3cgYWxvbmcgYXQ6ICR7dXJsfWAsXG4gICAgICAgICAgICAnJyxcbiAgICAgICAgICAgICdSZXN1bHRzIHdpbGwgbGFuZCBhcyBhIHB1bGwgcmVxdWVzdCB3aGVuIHRoZSByZW1vdGUgc2Vzc2lvbiBmaW5pc2hlcy4gVGhlcmUgaXMgbm90aGluZyB0byBkbyBoZXJlLicsXG4gICAgICAgICAgXS5qb2luKCdcXG4nKSxcbiAgICAgICAgICBtb2RlOiAndGFzay1ub3RpZmljYXRpb24nLFxuICAgICAgICB9KVxuICAgICAgfSBlbHNlIHtcbiAgICAgICAgLy8gVGVsZXBvcnQ6IHNldCBwZW5kaW5nQ2hvaWNlIHNvIFJFUEwgbW91bnRzIFVsdHJhcGxhbkNob2ljZURpYWxvZy5cbiAgICAgICAgLy8gVGhlIGRpYWxvZyBvd25zIGFyY2hpdmUgKyBVUkwgY2xlYXIgb24gY2hvaWNlLiBHdWFyZCBvbiB0YXNrIHN0YXR1c1xuICAgICAgICAvLyBzbyBhIHBvbGwgdGhhdCByZXNvbHZlcyBhZnRlciBzdG9wVWx0cmFwbGFuIGRvZXNuJ3QgcmVzdXJyZWN0IHRoZVxuICAgICAgICAvLyBkaWFsb2cgZm9yIGEga2lsbGVkIHNlc3Npb24uXG4gICAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4ge1xuICAgICAgICAgIGNvbnN0IHRhc2sgPSBwcmV2LnRhc2tzPy5bdGFza0lkXVxuICAgICAgICAgIGlmICghdGFzayB8fCB0YXNrLnN0YXR1cyAhPT0gJ3J1bm5pbmcnKSByZXR1cm4gcHJldlxuICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAuLi5wcmV2LFxuICAgICAgICAgICAgdWx0cmFwbGFuUGVuZGluZ0Nob2ljZTogeyBwbGFuLCBzZXNzaW9uSWQsIHRhc2tJZCB9LFxuICAgICAgICAgIH1cbiAgICAgICAgfSlcbiAgICAgIH1cbiAgICB9IGNhdGNoIChlKSB7XG4gICAgICAvLyBJZiB0aGUgdGFzayB3YXMgc3RvcHBlZCAoc3RvcFVsdHJhcGxhbiBzZXRzIHN0YXR1cz1raWxsZWQpLCB0aGUgcG9sbFxuICAgICAgLy8gZXJyb3JpbmcgaXMgZXhwZWN0ZWQg4oCUIHNraXAgdGhlIGZhaWx1cmUgbm90aWZpY2F0aW9uIGFuZCBjbGVhbnVwXG4gICAgICAvLyAoa2lsbCgpIGFscmVhZHkgYXJjaGl2ZWQ7IHN0b3BVbHRyYXBsYW4gY2xlYXJlZCB0aGUgVVJMKS5cbiAgICAgIGNvbnN0IHRhc2sgPSBnZXRBcHBTdGF0ZSgpLnRhc2tzPy5bdGFza0lkXVxuICAgICAgaWYgKHRhc2s/LnN0YXR1cyAhPT0gJ3J1bm5pbmcnKSByZXR1cm5cbiAgICAgIGZhaWxlZCA9IHRydWVcbiAgICAgIGxvZ0V2ZW50KCd0ZW5ndV91bHRyYXBsYW5fZmFpbGVkJywge1xuICAgICAgICBkdXJhdGlvbl9tczogRGF0ZS5ub3coKSAtIHN0YXJ0ZWQsXG4gICAgICAgIHJlYXNvbjogKGUgaW5zdGFuY2VvZiBVbHRyYXBsYW5Qb2xsRXJyb3JcbiAgICAgICAgICA/IGUucmVhc29uXG4gICAgICAgICAgOiAnbmV0d29ya19vcl91bmtub3duJykgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICAgICAgcmVqZWN0X2NvdW50OlxuICAgICAgICAgIGUgaW5zdGFuY2VvZiBVbHRyYXBsYW5Qb2xsRXJyb3IgPyBlLnJlamVjdENvdW50IDogdW5kZWZpbmVkLFxuICAgICAgfSlcbiAgICAgIGVucXVldWVQZW5kaW5nTm90aWZpY2F0aW9uKHtcbiAgICAgICAgdmFsdWU6IGBVbHRyYXBsYW4gZmFpbGVkOiAke2Vycm9yTWVzc2FnZShlKX1cXG5cXG5TZXNzaW9uOiAke3VybH1gLFxuICAgICAgICBtb2RlOiAndGFzay1ub3RpZmljYXRpb24nLFxuICAgICAgfSlcbiAgICAgIC8vIEVycm9yIHBhdGggb3ducyBjbGVhbnVwOyB0ZWxlcG9ydCBwYXRoIGRlZmVycyB0byB0aGUgZGlhbG9nOyByZW1vdGVcbiAgICAgIC8vIHBhdGggaGFuZGxlZCBpdHMgb3duIGNsZWFudXAgYWJvdmUuXG4gICAgICB2b2lkIGFyY2hpdmVSZW1vdGVTZXNzaW9uKHNlc3Npb25JZCkuY2F0Y2goZSA9PlxuICAgICAgICBsb2dGb3JEZWJ1Z2dpbmcoYHVsdHJhcGxhbiBhcmNoaXZlIGZhaWxlZDogJHtTdHJpbmcoZSl9YCksXG4gICAgICApXG4gICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+XG4gICAgICAgIC8vIENvbXBhcmUgYWdhaW5zdCB0aGlzIHBvbGwncyBVUkwgc28gYSBuZXdlciByZWxhdW5jaGVkIHNlc3Npb24nc1xuICAgICAgICAvLyBVUkwgaXNuJ3QgY2xlYXJlZCBieSBhIHN0YWxlIHBvbGwgZXJyb3Jpbmcgb3V0LlxuICAgICAgICBwcmV2LnVsdHJhcGxhblNlc3Npb25VcmwgPT09IHVybFxuICAgICAgICAgID8geyAuLi5wcmV2LCB1bHRyYXBsYW5TZXNzaW9uVXJsOiB1bmRlZmluZWQgfVxuICAgICAgICAgIDogcHJldixcbiAgICAgIClcbiAgICB9IGZpbmFsbHkge1xuICAgICAgLy8gUmVtb3RlIHBhdGggYWxyZWFkeSBzZXQgc3RhdHVzPWNvbXBsZXRlZCBhYm92ZTsgdGVsZXBvcnQgcGF0aFxuICAgICAgLy8gbGVhdmVzIHN0YXR1cz1ydW5uaW5nIHNvIHRoZSBwaWxsIHNob3dzIHRoZSB1bHRyYXBsYW5QaGFzZSBzdGF0ZVxuICAgICAgLy8gdW50aWwgVWx0cmFwbGFuQ2hvaWNlRGlhbG9nIGNvbXBsZXRlcyB0aGUgdGFzayBhZnRlciB0aGUgdXNlcidzXG4gICAgICAvLyBjaG9pY2UuIFNldHRpbmcgY29tcGxldGVkIGhlcmUgd291bGQgZmlsdGVyIHRoZSB0YXNrIG91dCBvZlxuICAgICAgLy8gaXNCYWNrZ3JvdW5kVGFzayBiZWZvcmUgdGhlIHBpbGwgY2FuIHJlbmRlciB0aGUgcGhhc2Ugc3RhdGUuXG4gICAgICAvLyBGYWlsdXJlIHBhdGggaGFzIG5vIGRpYWxvZywgc28gaXQgb3ducyB0aGUgc3RhdHVzIHRyYW5zaXRpb24gaGVyZS5cbiAgICAgIGlmIChmYWlsZWQpIHtcbiAgICAgICAgdXBkYXRlVGFza1N0YXRlPFJlbW90ZUFnZW50VGFza1N0YXRlPih0YXNrSWQsIHNldEFwcFN0YXRlLCB0ID0+XG4gICAgICAgICAgdC5zdGF0dXMgIT09ICdydW5uaW5nJ1xuICAgICAgICAgICAgPyB0XG4gICAgICAgICAgICA6IHsgLi4udCwgc3RhdHVzOiAnZmFpbGVkJywgZW5kVGltZTogRGF0ZS5ub3coKSB9LFxuICAgICAgICApXG4gICAgICB9XG4gICAgfVxuICB9KSgpXG59XG5cbi8vIFJlbmRlcnMgaW1tZWRpYXRlbHkgc28gdGhlIHRlcm1pbmFsIGRvZXNuJ3QgYXBwZWFyIGh1bmcgZHVyaW5nIHRoZVxuLy8gbXVsdGktc2Vjb25kIHRlbGVwb3J0VG9SZW1vdGUgcm91bmQtdHJpcC5cbmZ1bmN0aW9uIGJ1aWxkTGF1bmNoTWVzc2FnZShkaXNjb25uZWN0ZWRCcmlkZ2U/OiBib29sZWFuKTogc3RyaW5nIHtcbiAgY29uc3QgcHJlZml4ID0gZGlzY29ubmVjdGVkQnJpZGdlID8gYCR7UkVNT1RFX0NPTlRST0xfRElTQ09OTkVDVEVEX01TR30gYCA6ICcnXG4gIHJldHVybiBgJHtESUFNT05EX09QRU59IHVsdHJhcGxhblxcbiR7cHJlZml4fVN0YXJ0aW5nIENsYXVkZSBDb2RlIG9uIHRoZSB3ZWLigKZgXG59XG5cbmZ1bmN0aW9uIGJ1aWxkU2Vzc2lvblJlYWR5TWVzc2FnZSh1cmw6IHN0cmluZyk6IHN0cmluZyB7XG4gIHJldHVybiBgJHtESUFNT05EX09QRU59IHVsdHJhcGxhbiDCtyBNb25pdG9yIHByb2dyZXNzIGluIENsYXVkZSBDb2RlIG9uIHRoZSB3ZWIgJHt1cmx9XFxuWW91IGNhbiBjb250aW51ZSB3b3JraW5nIOKAlCB3aGVuIHRoZSAke0RJQU1PTkRfT1BFTn0gZmlsbHMsIHByZXNzIOKGkyB0byB2aWV3IHJlc3VsdHNgXG59XG5cbmZ1bmN0aW9uIGJ1aWxkQWxyZWFkeUFjdGl2ZU1lc3NhZ2UodXJsOiBzdHJpbmcgfCB1bmRlZmluZWQpOiBzdHJpbmcge1xuICByZXR1cm4gdXJsXG4gICAgPyBgdWx0cmFwbGFuOiBhbHJlYWR5IHBvbGxpbmcuIE9wZW4gJHt1cmx9IHRvIGNoZWNrIHN0YXR1cywgb3Igd2FpdCBmb3IgdGhlIHBsYW4gdG8gbGFuZCBoZXJlLmBcbiAgICA6ICd1bHRyYXBsYW46IGFscmVhZHkgbGF1bmNoaW5nLiBQbGVhc2Ugd2FpdCBmb3IgdGhlIHNlc3Npb24gdG8gc3RhcnQuJ1xufVxuXG4vKipcbiAqIFN0b3AgYSBydW5uaW5nIHVsdHJhcGxhbjogYXJjaGl2ZSB0aGUgcmVtb3RlIHNlc3Npb24gKGhhbHRzIGl0IGJ1dCBrZWVwcyB0aGVcbiAqIFVSTCB2aWV3YWJsZSksIGtpbGwgdGhlIGxvY2FsIHRhc2sgZW50cnkgKGNsZWFycyB0aGUgcGlsbCksIGFuZCBjbGVhclxuICogdWx0cmFwbGFuU2Vzc2lvblVybCAocmUtYXJtcyB0aGUga2V5d29yZCB0cmlnZ2VyKS4gc3RhcnREZXRhY2hlZFBvbGwnc1xuICogc2hvdWxkU3RvcCBjYWxsYmFjayBzZWVzIHRoZSBraWxsZWQgc3RhdHVzIG9uIGl0cyBuZXh0IHRpY2sgYW5kIHRocm93cztcbiAqIHRoZSBjYXRjaCBibG9jayBlYXJseS1yZXR1cm5zIHdoZW4gc3RhdHVzICE9PSAncnVubmluZycuXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBzdG9wVWx0cmFwbGFuKFxuICB0YXNrSWQ6IHN0cmluZyxcbiAgc2Vzc2lvbklkOiBzdHJpbmcsXG4gIHNldEFwcFN0YXRlOiAoZjogKHByZXY6IEFwcFN0YXRlKSA9PiBBcHBTdGF0ZSkgPT4gdm9pZCxcbik6IFByb21pc2U8dm9pZD4ge1xuICAvLyBSZW1vdGVBZ2VudFRhc2sua2lsbCBhcmNoaXZlcyB0aGUgc2Vzc2lvbiAod2l0aCAuY2F0Y2gpIOKAlCBubyBzZXBhcmF0ZVxuICAvLyBhcmNoaXZlIGNhbGwgbmVlZGVkIGhlcmUuXG4gIGF3YWl0IFJlbW90ZUFnZW50VGFzay5raWxsKHRhc2tJZCwgc2V0QXBwU3RhdGUpXG4gIHNldEFwcFN0YXRlKHByZXYgPT5cbiAgICBwcmV2LnVsdHJhcGxhblNlc3Npb25VcmwgfHxcbiAgICBwcmV2LnVsdHJhcGxhblBlbmRpbmdDaG9pY2UgfHxcbiAgICBwcmV2LnVsdHJhcGxhbkxhdW5jaGluZ1xuICAgICAgPyB7XG4gICAgICAgICAgLi4ucHJldixcbiAgICAgICAgICB1bHRyYXBsYW5TZXNzaW9uVXJsOiB1bmRlZmluZWQsXG4gICAgICAgICAgdWx0cmFwbGFuUGVuZGluZ0Nob2ljZTogdW5kZWZpbmVkLFxuICAgICAgICAgIHVsdHJhcGxhbkxhdW5jaGluZzogdW5kZWZpbmVkLFxuICAgICAgICB9XG4gICAgICA6IHByZXYsXG4gIClcbiAgY29uc3QgdXJsID0gZ2V0UmVtb3RlU2Vzc2lvblVybChzZXNzaW9uSWQsIHByb2Nlc3MuZW52LlNFU1NJT05fSU5HUkVTU19VUkwpXG4gIGVucXVldWVQZW5kaW5nTm90aWZpY2F0aW9uKHtcbiAgICB2YWx1ZTogYFVsdHJhcGxhbiBzdG9wcGVkLlxcblxcblNlc3Npb246ICR7dXJsfWAsXG4gICAgbW9kZTogJ3Rhc2stbm90aWZpY2F0aW9uJyxcbiAgfSlcbiAgZW5xdWV1ZVBlbmRpbmdOb3RpZmljYXRpb24oe1xuICAgIHZhbHVlOlxuICAgICAgJ1RoZSB1c2VyIHN0b3BwZWQgdGhlIHVsdHJhcGxhbiBzZXNzaW9uIGFib3ZlLiBEbyBub3QgcmVzcG9uZCB0byB0aGUgc3RvcCBub3RpZmljYXRpb24g4oCUIHdhaXQgZm9yIHRoZWlyIG5leHQgbWVzc2FnZS4nLFxuICAgIG1vZGU6ICd0YXNrLW5vdGlmaWNhdGlvbicsXG4gICAgaXNNZXRhOiB0cnVlLFxuICB9KVxufVxuXG4vKipcbiAqIFNoYXJlZCBlbnRyeSBmb3IgdGhlIHNsYXNoIGNvbW1hbmQsIGtleXdvcmQgdHJpZ2dlciwgYW5kIHRoZSBwbGFuLWFwcHJvdmFsXG4gKiBkaWFsb2cncyBcIlVsdHJhcGxhblwiIGJ1dHRvbi4gV2hlbiBzZWVkUGxhbiBpcyBwcmVzZW50IChkaWFsb2cgcGF0aCksIGl0IGlzXG4gKiBwcmVwZW5kZWQgYXMgYSBkcmFmdCB0byByZWZpbmU7IGJsdXJiIG1heSBiZSBlbXB0eSBpbiB0aGF0IGNhc2UuXG4gKlxuICogUmVzb2x2ZXMgaW1tZWRpYXRlbHkgd2l0aCB0aGUgdXNlci1mYWNpbmcgbWVzc2FnZS4gRWxpZ2liaWxpdHkgY2hlY2ssXG4gKiBzZXNzaW9uIGNyZWF0aW9uLCBhbmQgdGFzayByZWdpc3RyYXRpb24gcnVuIGRldGFjaGVkIGFuZCBmYWlsdXJlcyBzdXJmYWNlIHZpYVxuICogZW5xdWV1ZVBlbmRpbmdOb3RpZmljYXRpb24uXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBsYXVuY2hVbHRyYXBsYW4ob3B0czoge1xuICBibHVyYjogc3RyaW5nXG4gIHNlZWRQbGFuPzogc3RyaW5nXG4gIGdldEFwcFN0YXRlOiAoKSA9PiBBcHBTdGF0ZVxuICBzZXRBcHBTdGF0ZTogKGY6IChwcmV2OiBBcHBTdGF0ZSkgPT4gQXBwU3RhdGUpID0+IHZvaWRcbiAgc2lnbmFsOiBBYm9ydFNpZ25hbFxuICAvKiogVHJ1ZSBpZiB0aGUgY2FsbGVyIGRpc2Nvbm5lY3RlZCBSZW1vdGUgQ29udHJvbCBiZWZvcmUgbGF1bmNoaW5nLiAqL1xuICBkaXNjb25uZWN0ZWRCcmlkZ2U/OiBib29sZWFuXG4gIC8qKlxuICAgKiBDYWxsZWQgb25jZSB0ZWxlcG9ydFRvUmVtb3RlIHJlc29sdmVzIHdpdGggYSBzZXNzaW9uIFVSTC4gQ2FsbGVycyB0aGF0XG4gICAqIGhhdmUgc2V0TWVzc2FnZXMgKFJFUEwpIGFwcGVuZCB0aGlzIGFzIGEgc2Vjb25kIHRyYW5zY3JpcHQgbWVzc2FnZSBzbyB0aGVcbiAgICogVVJMIGlzIHZpc2libGUgd2l0aG91dCBvcGVuaW5nIHRoZSDihpMgZGV0YWlsIHZpZXcuIENhbGxlcnMgd2l0aG91dFxuICAgKiB0cmFuc2NyaXB0IGFjY2VzcyAoRXhpdFBsYW5Nb2RlUGVybWlzc2lvblJlcXVlc3QpIG9taXQgdGhpcyDigJQgdGhlIHBpbGxcbiAgICogc3RpbGwgc2hvd3MgbGl2ZSBzdGF0dXMuXG4gICAqL1xuICBvblNlc3Npb25SZWFkeT86IChtc2c6IHN0cmluZykgPT4gdm9pZFxufSk6IFByb21pc2U8c3RyaW5nPiB7XG4gIGNvbnN0IHtcbiAgICBibHVyYixcbiAgICBzZWVkUGxhbixcbiAgICBnZXRBcHBTdGF0ZSxcbiAgICBzZXRBcHBTdGF0ZSxcbiAgICBzaWduYWwsXG4gICAgZGlzY29ubmVjdGVkQnJpZGdlLFxuICAgIG9uU2Vzc2lvblJlYWR5LFxuICB9ID0gb3B0c1xuXG4gIGNvbnN0IHsgdWx0cmFwbGFuU2Vzc2lvblVybDogYWN0aXZlLCB1bHRyYXBsYW5MYXVuY2hpbmcgfSA9IGdldEFwcFN0YXRlKClcbiAgaWYgKGFjdGl2ZSB8fCB1bHRyYXBsYW5MYXVuY2hpbmcpIHtcbiAgICBsb2dFdmVudCgndGVuZ3VfdWx0cmFwbGFuX2NyZWF0ZV9mYWlsZWQnLCB7XG4gICAgICByZWFzb246IChhY3RpdmVcbiAgICAgICAgPyAnYWxyZWFkeV9wb2xsaW5nJ1xuICAgICAgICA6ICdhbHJlYWR5X2xhdW5jaGluZycpIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgfSlcbiAgICByZXR1cm4gYnVpbGRBbHJlYWR5QWN0aXZlTWVzc2FnZShhY3RpdmUpXG4gIH1cblxuICBpZiAoIWJsdXJiICYmICFzZWVkUGxhbikge1xuICAgIC8vIE5vIGV2ZW50IOKAlCBiYXJlIC91bHRyYXBsYW4gaXMgYSB1c2FnZSBxdWVyeSwgbm90IGFuIGF0dGVtcHQuXG4gICAgcmV0dXJuIFtcbiAgICAgIC8vIFJlbmRlcmVkIHZpYSA8TWFya2Rvd24+OyByYXcgPG1lc3NhZ2U+IGlzIHRva2VuaXplZCBhcyBIVE1MXG4gICAgICAvLyBhbmQgZHJvcHBlZC4gQmFja3NsYXNoLWVzY2FwZSB0aGUgYnJhY2tldHMuXG4gICAgICAnVXNhZ2U6IC91bHRyYXBsYW4gXFxcXDxwcm9tcHRcXFxcPiwgb3IgaW5jbHVkZSBcInVsdHJhcGxhblwiIGFueXdoZXJlJyxcbiAgICAgICdpbiB5b3VyIHByb21wdCcsXG4gICAgICAnJyxcbiAgICAgICdBZHZhbmNlZCBtdWx0aS1hZ2VudCBwbGFuIG1vZGUgd2l0aCBvdXIgbW9zdCBwb3dlcmZ1bCBtb2RlbCcsXG4gICAgICAnKE9wdXMpLiBSdW5zIGluIENsYXVkZSBDb2RlIG9uIHRoZSB3ZWIuIFdoZW4gdGhlIHBsYW4gaXMgcmVhZHksJyxcbiAgICAgICd5b3UgY2FuIGV4ZWN1dGUgaXQgaW4gdGhlIHdlYiBzZXNzaW9uIG9yIHNlbmQgaXQgYmFjayBoZXJlLicsXG4gICAgICAnVGVybWluYWwgc3RheXMgZnJlZSB3aGlsZSB0aGUgcmVtb3RlIHBsYW5zLicsXG4gICAgICAnUmVxdWlyZXMgL2xvZ2luLicsXG4gICAgICAnJyxcbiAgICAgIGBUZXJtczogJHtDQ1JfVEVSTVNfVVJMfWAsXG4gICAgXS5qb2luKCdcXG4nKVxuICB9XG5cbiAgLy8gU2V0IHN5bmNocm9ub3VzbHkgYmVmb3JlIHRoZSBkZXRhY2hlZCBmbG93IHRvIHByZXZlbnQgZHVwbGljYXRlIGxhdW5jaGVzXG4gIC8vIGR1cmluZyB0aGUgdGVsZXBvcnRUb1JlbW90ZSB3aW5kb3cuXG4gIHNldEFwcFN0YXRlKHByZXYgPT5cbiAgICBwcmV2LnVsdHJhcGxhbkxhdW5jaGluZyA/IHByZXYgOiB7IC4uLnByZXYsIHVsdHJhcGxhbkxhdW5jaGluZzogdHJ1ZSB9LFxuICApXG4gIHZvaWQgbGF1bmNoRGV0YWNoZWQoe1xuICAgIGJsdXJiLFxuICAgIHNlZWRQbGFuLFxuICAgIGdldEFwcFN0YXRlLFxuICAgIHNldEFwcFN0YXRlLFxuICAgIHNpZ25hbCxcbiAgICBvblNlc3Npb25SZWFkeSxcbiAgfSlcbiAgcmV0dXJuIGJ1aWxkTGF1bmNoTWVzc2FnZShkaXNjb25uZWN0ZWRCcmlkZ2UpXG59XG5cbmFzeW5jIGZ1bmN0aW9uIGxhdW5jaERldGFjaGVkKG9wdHM6IHtcbiAgYmx1cmI6IHN0cmluZ1xuICBzZWVkUGxhbj86IHN0cmluZ1xuICBnZXRBcHBTdGF0ZTogKCkgPT4gQXBwU3RhdGVcbiAgc2V0QXBwU3RhdGU6IChmOiAocHJldjogQXBwU3RhdGUpID0+IEFwcFN0YXRlKSA9PiB2b2lkXG4gIHNpZ25hbDogQWJvcnRTaWduYWxcbiAgb25TZXNzaW9uUmVhZHk/OiAobXNnOiBzdHJpbmcpID0+IHZvaWRcbn0pOiBQcm9taXNlPHZvaWQ+IHtcbiAgY29uc3QgeyBibHVyYiwgc2VlZFBsYW4sIGdldEFwcFN0YXRlLCBzZXRBcHBTdGF0ZSwgc2lnbmFsLCBvblNlc3Npb25SZWFkeSB9ID1cbiAgICBvcHRzXG4gIC8vIEhvaXN0ZWQgc28gdGhlIGNhdGNoIGJsb2NrIGNhbiBhcmNoaXZlIHRoZSByZW1vdGUgc2Vzc2lvbiBpZiBhbiBlcnJvclxuICAvLyBvY2N1cnMgYWZ0ZXIgdGVsZXBvcnRUb1JlbW90ZSBzdWNjZWVkcyAoYXZvaWRzIDMwbWluIG9ycGhhbikuXG4gIGxldCBzZXNzaW9uSWQ6IHN0cmluZyB8IHVuZGVmaW5lZFxuICB0cnkge1xuICAgIGNvbnN0IG1vZGVsID0gZ2V0VWx0cmFwbGFuTW9kZWwoKVxuXG4gICAgY29uc3QgZWxpZ2liaWxpdHkgPSBhd2FpdCBjaGVja1JlbW90ZUFnZW50RWxpZ2liaWxpdHkoKVxuICAgIGlmICghZWxpZ2liaWxpdHkuZWxpZ2libGUpIHtcbiAgICAgIGxvZ0V2ZW50KCd0ZW5ndV91bHRyYXBsYW5fY3JlYXRlX2ZhaWxlZCcsIHtcbiAgICAgICAgcmVhc29uOlxuICAgICAgICAgICdwcmVjb25kaXRpb24nIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgIHByZWNvbmRpdGlvbl9lcnJvcnM6IGVsaWdpYmlsaXR5LmVycm9yc1xuICAgICAgICAgIC5tYXAoZSA9PiBlLnR5cGUpXG4gICAgICAgICAgLmpvaW4oXG4gICAgICAgICAgICAnLCcsXG4gICAgICAgICAgKSBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgfSlcbiAgICAgIGNvbnN0IHJlYXNvbnMgPSBlbGlnaWJpbGl0eS5lcnJvcnMubWFwKGZvcm1hdFByZWNvbmRpdGlvbkVycm9yKS5qb2luKCdcXG4nKVxuICAgICAgZW5xdWV1ZVBlbmRpbmdOb3RpZmljYXRpb24oe1xuICAgICAgICB2YWx1ZTogYHVsdHJhcGxhbjogY2Fubm90IGxhdW5jaCByZW1vdGUgc2Vzc2lvbiDigJRcXG4ke3JlYXNvbnN9YCxcbiAgICAgICAgbW9kZTogJ3Rhc2stbm90aWZpY2F0aW9uJyxcbiAgICAgIH0pXG4gICAgICByZXR1cm5cbiAgICB9XG5cbiAgICBjb25zdCBwcm9tcHQgPSBidWlsZFVsdHJhcGxhblByb21wdChibHVyYiwgc2VlZFBsYW4pXG4gICAgbGV0IGJ1bmRsZUZhaWxNc2c6IHN0cmluZyB8IHVuZGVmaW5lZFxuICAgIGNvbnN0IHNlc3Npb24gPSBhd2FpdCB0ZWxlcG9ydFRvUmVtb3RlKHtcbiAgICAgIGluaXRpYWxNZXNzYWdlOiBwcm9tcHQsXG4gICAgICBkZXNjcmlwdGlvbjogYmx1cmIgfHwgJ1JlZmluZSBsb2NhbCBwbGFuJyxcbiAgICAgIG1vZGVsLFxuICAgICAgcGVybWlzc2lvbk1vZGU6ICdwbGFuJyxcbiAgICAgIHVsdHJhcGxhbjogdHJ1ZSxcbiAgICAgIHNpZ25hbCxcbiAgICAgIHVzZURlZmF1bHRFbnZpcm9ubWVudDogdHJ1ZSxcbiAgICAgIG9uQnVuZGxlRmFpbDogbXNnID0+IHtcbiAgICAgICAgYnVuZGxlRmFpbE1zZyA9IG1zZ1xuICAgICAgfSxcbiAgICB9KVxuICAgIGlmICghc2Vzc2lvbikge1xuICAgICAgbG9nRXZlbnQoJ3Rlbmd1X3VsdHJhcGxhbl9jcmVhdGVfZmFpbGVkJywge1xuICAgICAgICByZWFzb246IChidW5kbGVGYWlsTXNnXG4gICAgICAgICAgPyAnYnVuZGxlX2ZhaWwnXG4gICAgICAgICAgOiAndGVsZXBvcnRfbnVsbCcpIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICB9KVxuICAgICAgZW5xdWV1ZVBlbmRpbmdOb3RpZmljYXRpb24oe1xuICAgICAgICB2YWx1ZTogYHVsdHJhcGxhbjogc2Vzc2lvbiBjcmVhdGlvbiBmYWlsZWQke2J1bmRsZUZhaWxNc2cgPyBgIOKAlCAke2J1bmRsZUZhaWxNc2d9YCA6ICcnfS4gU2VlIC0tZGVidWcgZm9yIGRldGFpbHMuYCxcbiAgICAgICAgbW9kZTogJ3Rhc2stbm90aWZpY2F0aW9uJyxcbiAgICAgIH0pXG4gICAgICByZXR1cm5cbiAgICB9XG4gICAgc2Vzc2lvbklkID0gc2Vzc2lvbi5pZFxuXG4gICAgY29uc3QgdXJsID0gZ2V0UmVtb3RlU2Vzc2lvblVybChzZXNzaW9uLmlkLCBwcm9jZXNzLmVudi5TRVNTSU9OX0lOR1JFU1NfVVJMKVxuICAgIHNldEFwcFN0YXRlKHByZXYgPT4gKHtcbiAgICAgIC4uLnByZXYsXG4gICAgICB1bHRyYXBsYW5TZXNzaW9uVXJsOiB1cmwsXG4gICAgICB1bHRyYXBsYW5MYXVuY2hpbmc6IHVuZGVmaW5lZCxcbiAgICB9KSlcbiAgICBvblNlc3Npb25SZWFkeT8uKGJ1aWxkU2Vzc2lvblJlYWR5TWVzc2FnZSh1cmwpKVxuICAgIGxvZ0V2ZW50KCd0ZW5ndV91bHRyYXBsYW5fbGF1bmNoZWQnLCB7XG4gICAgICBoYXNfc2VlZF9wbGFuOiBCb29sZWFuKHNlZWRQbGFuKSxcbiAgICAgIG1vZGVsOlxuICAgICAgICBtb2RlbCBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgIH0pXG4gICAgLy8gVE9ETygjMjM5ODUpOiByZXBsYWNlIHJlZ2lzdGVyUmVtb3RlQWdlbnRUYXNrICsgc3RhcnREZXRhY2hlZFBvbGwgd2l0aFxuICAgIC8vIEV4aXRQbGFuTW9kZVNjYW5uZXIgaW5zaWRlIHN0YXJ0UmVtb3RlU2Vzc2lvblBvbGxpbmcuXG4gICAgY29uc3QgeyB0YXNrSWQgfSA9IHJlZ2lzdGVyUmVtb3RlQWdlbnRUYXNrKHtcbiAgICAgIHJlbW90ZVRhc2tUeXBlOiAndWx0cmFwbGFuJyxcbiAgICAgIHNlc3Npb246IHsgaWQ6IHNlc3Npb24uaWQsIHRpdGxlOiBibHVyYiB8fCAnVWx0cmFwbGFuJyB9LFxuICAgICAgY29tbWFuZDogYmx1cmIsXG4gICAgICBjb250ZXh0OiB7XG4gICAgICAgIGFib3J0Q29udHJvbGxlcjogbmV3IEFib3J0Q29udHJvbGxlcigpLFxuICAgICAgICBnZXRBcHBTdGF0ZSxcbiAgICAgICAgc2V0QXBwU3RhdGUsXG4gICAgICB9LFxuICAgICAgaXNVbHRyYXBsYW46IHRydWUsXG4gICAgfSlcbiAgICBzdGFydERldGFjaGVkUG9sbCh0YXNrSWQsIHNlc3Npb24uaWQsIHVybCwgZ2V0QXBwU3RhdGUsIHNldEFwcFN0YXRlKVxuICB9IGNhdGNoIChlKSB7XG4gICAgbG9nRXJyb3IoZSlcbiAgICBsb2dFdmVudCgndGVuZ3VfdWx0cmFwbGFuX2NyZWF0ZV9mYWlsZWQnLCB7XG4gICAgICByZWFzb246XG4gICAgICAgICd1bmV4cGVjdGVkX2Vycm9yJyBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgIH0pXG4gICAgZW5xdWV1ZVBlbmRpbmdOb3RpZmljYXRpb24oe1xuICAgICAgdmFsdWU6IGB1bHRyYXBsYW46IHVuZXhwZWN0ZWQgZXJyb3Ig4oCUICR7ZXJyb3JNZXNzYWdlKGUpfWAsXG4gICAgICBtb2RlOiAndGFzay1ub3RpZmljYXRpb24nLFxuICAgIH0pXG4gICAgaWYgKHNlc3Npb25JZCkge1xuICAgICAgLy8gRXJyb3IgYWZ0ZXIgdGVsZXBvcnQgc3VjY2VlZGVkIOKAlCBhcmNoaXZlIHNvIHRoZSByZW1vdGUgZG9lc24ndCBzaXRcbiAgICAgIC8vIHJ1bm5pbmcgZm9yIDMwbWluIHdpdGggbm9ib2R5IHBvbGxpbmcgaXQuXG4gICAgICB2b2lkIGFyY2hpdmVSZW1vdGVTZXNzaW9uKHNlc3Npb25JZCkuY2F0Y2goZXJyID0+XG4gICAgICAgIGxvZ0ZvckRlYnVnZ2luZygndWx0cmFwbGFuOiBmYWlsZWQgdG8gYXJjaGl2ZSBvcnBoYW5lZCBzZXNzaW9uJywgZXJyKSxcbiAgICAgIClcbiAgICAgIC8vIHVsdHJhcGxhblNlc3Npb25VcmwgbWF5IGhhdmUgYmVlbiBzZXQgYmVmb3JlIHRoZSB0aHJvdzsgY2xlYXIgaXQgc29cbiAgICAgIC8vIHRoZSBcImFscmVhZHkgcG9sbGluZ1wiIGd1YXJkIGRvZXNuJ3QgYmxvY2sgZnV0dXJlIGxhdW5jaGVzLlxuICAgICAgc2V0QXBwU3RhdGUocHJldiA9PlxuICAgICAgICBwcmV2LnVsdHJhcGxhblNlc3Npb25VcmxcbiAgICAgICAgICA/IHsgLi4ucHJldiwgdWx0cmFwbGFuU2Vzc2lvblVybDogdW5kZWZpbmVkIH1cbiAgICAgICAgICA6IHByZXYsXG4gICAgICApXG4gICAgfVxuICB9IGZpbmFsbHkge1xuICAgIC8vIE5vLW9wIG9uIHN1Y2Nlc3M6IHRoZSB1cmwtc2V0dGluZyBzZXRBcHBTdGF0ZSBhbHJlYWR5IGNsZWFyZWQgdGhpcy5cbiAgICBzZXRBcHBTdGF0ZShwcmV2ID0+XG4gICAgICBwcmV2LnVsdHJhcGxhbkxhdW5jaGluZ1xuICAgICAgICA/IHsgLi4ucHJldiwgdWx0cmFwbGFuTGF1bmNoaW5nOiB1bmRlZmluZWQgfVxuICAgICAgICA6IHByZXYsXG4gICAgKVxuICB9XG59XG5cbmNvbnN0IGNhbGw6IExvY2FsSlNYQ29tbWFuZENhbGwgPSBhc3luYyAob25Eb25lLCBjb250ZXh0LCBhcmdzKSA9PiB7XG4gIGNvbnN0IGJsdXJiID0gYXJncy50cmltKClcblxuICAvLyBCYXJlIC91bHRyYXBsYW4gKG5vIGFyZ3MsIG5vIHNlZWQgcGxhbikganVzdCBzaG93cyB1c2FnZSDigJQgbm8gZGlhbG9nLlxuICBpZiAoIWJsdXJiKSB7XG4gICAgY29uc3QgbXNnID0gYXdhaXQgbGF1bmNoVWx0cmFwbGFuKHtcbiAgICAgIGJsdXJiLFxuICAgICAgZ2V0QXBwU3RhdGU6IGNvbnRleHQuZ2V0QXBwU3RhdGUsXG4gICAgICBzZXRBcHBTdGF0ZTogY29udGV4dC5zZXRBcHBTdGF0ZSxcbiAgICAgIHNpZ25hbDogY29udGV4dC5hYm9ydENvbnRyb2xsZXIuc2lnbmFsLFxuICAgIH0pXG4gICAgb25Eb25lKG1zZywgeyBkaXNwbGF5OiAnc3lzdGVtJyB9KVxuICAgIHJldHVybiBudWxsXG4gIH1cblxuICAvLyBHdWFyZCBtYXRjaGVzIGxhdW5jaFVsdHJhcGxhbidzIG93biBjaGVjayDigJQgc2hvd2luZyB0aGUgZGlhbG9nIHdoZW4gYVxuICAvLyBzZXNzaW9uIGlzIGFscmVhZHkgYWN0aXZlIG9yIGxhdW5jaGluZyB3b3VsZCB3YXN0ZSB0aGUgdXNlcidzIGNsaWNrIGFuZCBzZXRcbiAgLy8gaGFzU2VlblVsdHJhcGxhblRlcm1zIGJlZm9yZSB0aGUgbGF1bmNoIGZhaWxzLlxuICBjb25zdCB7IHVsdHJhcGxhblNlc3Npb25Vcmw6IGFjdGl2ZSwgdWx0cmFwbGFuTGF1bmNoaW5nIH0gPVxuICAgIGNvbnRleHQuZ2V0QXBwU3RhdGUoKVxuICBpZiAoYWN0aXZlIHx8IHVsdHJhcGxhbkxhdW5jaGluZykge1xuICAgIGxvZ0V2ZW50KCd0ZW5ndV91bHRyYXBsYW5fY3JlYXRlX2ZhaWxlZCcsIHtcbiAgICAgIHJlYXNvbjogKGFjdGl2ZVxuICAgICAgICA/ICdhbHJlYWR5X3BvbGxpbmcnXG4gICAgICAgIDogJ2FscmVhZHlfbGF1bmNoaW5nJykgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICB9KVxuICAgIG9uRG9uZShidWlsZEFscmVhZHlBY3RpdmVNZXNzYWdlKGFjdGl2ZSksIHsgZGlzcGxheTogJ3N5c3RlbScgfSlcbiAgICByZXR1cm4gbnVsbFxuICB9XG5cbiAgLy8gTW91bnQgdGhlIHByZS1sYXVuY2ggZGlhbG9nIHZpYSBmb2N1c2VkSW5wdXREaWFsb2cgKGJvdHRvbSByZWdpb24sIGxpa2VcbiAgLy8gcGVybWlzc2lvbiBkaWFsb2dzKSByYXRoZXIgdGhhbiByZXR1cm5pbmcgSlNYICh0cmFuc2NyaXB0IGFyZWEsIGFuY2hvcnNcbiAgLy8gYXQgdG9wIG9mIHNjcm9sbGJhY2spLiBSRVBMLnRzeCBoYW5kbGVzIGxhdW5jaC9jbGVhci9jYW5jZWwgb24gY2hvaWNlLlxuICBjb250ZXh0LnNldEFwcFN0YXRlKHByZXYgPT4gKHsgLi4ucHJldiwgdWx0cmFwbGFuTGF1bmNoUGVuZGluZzogeyBibHVyYiB9IH0pKVxuICAvLyAnc2tpcCcgc3VwcHJlc3NlcyB0aGUgKG5vIGNvbnRlbnQpIGVjaG8g4oCUIHRoZSBkaWFsb2cncyBjaG9pY2UgaGFuZGxlclxuICAvLyBhZGRzIHRoZSByZWFsIC91bHRyYXBsYW4gZWNobyArIGxhdW5jaCBjb25maXJtYXRpb24uXG4gIG9uRG9uZSh1bmRlZmluZWQsIHsgZGlzcGxheTogJ3NraXAnIH0pXG4gIHJldHVybiBudWxsXG59XG5cbmV4cG9ydCBkZWZhdWx0IHtcbiAgdHlwZTogJ2xvY2FsLWpzeCcsXG4gIG5hbWU6ICd1bHRyYXBsYW4nLFxuICBkZXNjcmlwdGlvbjogYH4xMOKAkzMwIG1pbiDCtyBDbGF1ZGUgQ29kZSBvbiB0aGUgd2ViIGRyYWZ0cyBhbiBhZHZhbmNlZCBwbGFuIHlvdSBjYW4gZWRpdCBhbmQgYXBwcm92ZS4gU2VlICR7Q0NSX1RFUk1TX1VSTH1gLFxuICBhcmd1bWVudEhpbnQ6ICc8cHJvbXB0PicsXG4gIGlzRW5hYmxlZDogKCkgPT4gXCJleHRlcm5hbFwiID09PSAnYW50JyxcbiAgbG9hZDogKCkgPT4gUHJvbWlzZS5yZXNvbHZlKHsgY2FsbCB9KSxcbn0gc2F0aXNmaWVzIENvbW1hbmRcbiJdLCJtYXBwaW5ncyI6IkFBQUEsU0FBU0EsWUFBWSxRQUFRLElBQUk7QUFDakMsU0FBU0MsK0JBQStCLFFBQVEsb0JBQW9CO0FBQ3BFLGNBQWNDLE9BQU8sUUFBUSxnQkFBZ0I7QUFDN0MsU0FBU0MsWUFBWSxRQUFRLHlCQUF5QjtBQUN0RCxTQUFTQyxtQkFBbUIsUUFBUSx5QkFBeUI7QUFDN0QsU0FBU0MsbUNBQW1DLFFBQVEscUNBQXFDO0FBQ3pGLFNBQ0UsS0FBS0MsMERBQTBELEVBQy9EQyxRQUFRLFFBQ0gsZ0NBQWdDO0FBQ3ZDLGNBQWNDLFFBQVEsUUFBUSwyQkFBMkI7QUFDekQsU0FDRUMsMkJBQTJCLEVBQzNCQyx1QkFBdUIsRUFDdkJDLGVBQWUsRUFDZixLQUFLQyxvQkFBb0IsRUFDekJDLHVCQUF1QixRQUNsQiw2Q0FBNkM7QUFDcEQsY0FBY0MsbUJBQW1CLFFBQVEscUJBQXFCO0FBQzlELFNBQVNDLGVBQWUsUUFB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 \ No newline at end of file diff --git a/src/commands/upgrade/index.ts b/src/commands/upgrade/index.ts new file mode 100644 index 0000000..63dc5ff --- /dev/null +++ b/src/commands/upgrade/index.ts @@ -0,0 +1,16 @@ +import type { Command } from '../../commands.js' +import { getSubscriptionType } from '../../utils/auth.js' +import { isEnvTruthy } from '../../utils/envUtils.js' + +const upgrade = { + type: 'local-jsx', + name: 'upgrade', + description: 'Upgrade to Max for higher rate limits and more Opus', + availability: ['claude-ai'], + isEnabled: () => + !isEnvTruthy(process.env.DISABLE_UPGRADE_COMMAND) && + getSubscriptionType() !== 'enterprise', + load: () => import('./upgrade.js'), +} satisfies Command + +export default upgrade diff --git a/src/commands/upgrade/upgrade.tsx b/src/commands/upgrade/upgrade.tsx new file mode 100644 index 0000000..1daf73d --- /dev/null +++ b/src/commands/upgrade/upgrade.tsx @@ -0,0 +1,38 @@ +import * as React from 'react'; +import type { LocalJSXCommandContext } from '../../commands.js'; +import { getOauthProfileFromOauthToken } from '../../services/oauth/getOauthProfile.js'; +import type { LocalJSXCommandOnDone } from '../../types/command.js'; +import { getClaudeAIOAuthTokens, isClaudeAISubscriber } from '../../utils/auth.js'; +import { openBrowser } from '../../utils/browser.js'; +import { logError } from '../../utils/log.js'; +import { Login } from '../login/login.js'; +export async function call(onDone: LocalJSXCommandOnDone, context: LocalJSXCommandContext): Promise { + try { + // Check if user is already on the highest Max plan (20x) + if (isClaudeAISubscriber()) { + const tokens = getClaudeAIOAuthTokens(); + let isMax20x = false; + if (tokens?.subscriptionType && tokens?.rateLimitTier) { + isMax20x = tokens.subscriptionType === 'max' && tokens.rateLimitTier === 'default_claude_max_20x'; + } else if (tokens?.accessToken) { + const profile = await getOauthProfileFromOauthToken(tokens.accessToken); + isMax20x = profile?.organization?.organization_type === 'claude_max' && profile?.organization?.rate_limit_tier === 'default_claude_max_20x'; + } + if (isMax20x) { + setTimeout(onDone, 0, 'You are already on the highest Max subscription plan. For additional usage, run /login to switch to an API usage-billed account.'); + return null; + } + } + const url = 'https://claude.ai/upgrade/max'; + await openBrowser(url); + return { + context.onChangeAPIKey(); + onDone(success ? 'Login successful' : 'Login interrupted'); + }} />; + } catch (error) { + logError(error as Error); + setTimeout(onDone, 0, 'Failed to open browser. Please visit https://claude.ai/upgrade/max to upgrade.'); + } + return null; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkxvY2FsSlNYQ29tbWFuZENvbnRleHQiLCJnZXRPYXV0aFByb2ZpbGVGcm9tT2F1dGhUb2tlbiIsIkxvY2FsSlNYQ29tbWFuZE9uRG9uZSIsImdldENsYXVkZUFJT0F1dGhUb2tlbnMiLCJpc0NsYXVkZUFJU3Vic2NyaWJlciIsIm9wZW5Ccm93c2VyIiwibG9nRXJyb3IiLCJMb2dpbiIsImNhbGwiLCJvbkRvbmUiLCJjb250ZXh0IiwiUHJvbWlzZSIsIlJlYWN0Tm9kZSIsInRva2VucyIsImlzTWF4MjB4Iiwic3Vic2NyaXB0aW9uVHlwZSIsInJhdGVMaW1pdFRpZXIiLCJhY2Nlc3NUb2tlbiIsInByb2ZpbGUiLCJvcmdhbml6YXRpb24iLCJvcmdhbml6YXRpb25fdHlwZSIsInJhdGVfbGltaXRfdGllciIsInNldFRpbWVvdXQiLCJ1cmwiLCJzdWNjZXNzIiwib25DaGFuZ2VBUElLZXkiLCJlcnJvciIsIkVycm9yIl0sInNvdXJjZXMiOlsidXBncmFkZS50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgdHlwZSB7IExvY2FsSlNYQ29tbWFuZENvbnRleHQgfSBmcm9tICcuLi8uLi9jb21tYW5kcy5qcydcbmltcG9ydCB7IGdldE9hdXRoUHJvZmlsZUZyb21PYXV0aFRva2VuIH0gZnJvbSAnLi4vLi4vc2VydmljZXMvb2F1dGgvZ2V0T2F1dGhQcm9maWxlLmpzJ1xuaW1wb3J0IHR5cGUgeyBMb2NhbEpTWENvbW1hbmRPbkRvbmUgfSBmcm9tICcuLi8uLi90eXBlcy9jb21tYW5kLmpzJ1xuaW1wb3J0IHtcbiAgZ2V0Q2xhdWRlQUlPQXV0aFRva2VucyxcbiAgaXNDbGF1ZGVBSVN1YnNjcmliZXIsXG59IGZyb20gJy4uLy4uL3V0aWxzL2F1dGguanMnXG5pbXBvcnQgeyBvcGVuQnJvd3NlciB9IGZyb20gJy4uLy4uL3V0aWxzL2Jyb3dzZXIuanMnXG5pbXBvcnQgeyBsb2dFcnJvciB9IGZyb20gJy4uLy4uL3V0aWxzL2xvZy5qcydcbmltcG9ydCB7IExvZ2luIH0gZnJvbSAnLi4vbG9naW4vbG9naW4uanMnXG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBjYWxsKFxuICBvbkRvbmU6IExvY2FsSlNYQ29tbWFuZE9uRG9uZSxcbiAgY29udGV4dDogTG9jYWxKU1hDb21tYW5kQ29udGV4dCxcbik6IFByb21pc2U8UmVhY3QuUmVhY3ROb2RlIHwgbnVsbD4ge1xuICB0cnkge1xuICAgIC8vIENoZWNrIGlmIHVzZXIgaXMgYWxyZWFkeSBvbiB0aGUgaGlnaGVzdCBNYXggcGxhbiAoMjB4KVxuICAgIGlmIChpc0NsYXVkZUFJU3Vic2NyaWJlcigpKSB7XG4gICAgICBjb25zdCB0b2tlbnMgPSBnZXRDbGF1ZGVBSU9BdXRoVG9rZW5zKClcbiAgICAgIGxldCBpc01heDIweCA9IGZhbHNlXG5cbiAgICAgIGlmICh0b2tlbnM/LnN1YnNjcmlwdGlvblR5cGUgJiYgdG9rZW5zPy5yYXRlTGltaXRUaWVyKSB7XG4gICAgICAgIGlzTWF4MjB4ID1cbiAgICAgICAgICB0b2tlbnMuc3Vic2NyaXB0aW9uVHlwZSA9PT0gJ21heCcgJiZcbiAgICAgICAgICB0b2tlbnMucmF0ZUxpbWl0VGllciA9PT0gJ2RlZmF1bHRfY2xhdWRlX21heF8yMHgnXG4gICAgICB9IGVsc2UgaWYgKHRva2Vucz8uYWNjZXNzVG9rZW4pIHtcbiAgICAgICAgY29uc3QgcHJvZmlsZSA9IGF3YWl0IGdldE9hdXRoUHJvZmlsZUZyb21PYXV0aFRva2VuKHRva2Vucy5hY2Nlc3NUb2tlbilcbiAgICAgICAgaXNNYXgyMHggPVxuICAgICAgICAgIHByb2ZpbGU/Lm9yZ2FuaXphdGlvbj8ub3JnYW5pemF0aW9uX3R5cGUgPT09ICdjbGF1ZGVfbWF4JyAmJlxuICAgICAgICAgIHByb2ZpbGU/Lm9yZ2FuaXphdGlvbj8ucmF0ZV9saW1pdF90aWVyID09PSAnZGVmYXVsdF9jbGF1ZGVfbWF4XzIweCdcbiAgICAgIH1cblxuICAgICAgaWYgKGlzTWF4MjB4KSB7XG4gICAgICAgIHNldFRpbWVvdXQoXG4gICAgICAgICAgb25Eb25lLFxuICAgICAgICAgIDAsXG4gICAgICAgICAgJ1lvdSBhcmUgYWxyZWFkeSBvbiB0aGUgaGlnaGVzdCBNYXggc3Vic2NyaXB0aW9uIHBsYW4uIEZvciBhZGRpdGlvbmFsIHVzYWdlLCBydW4gL2xvZ2luIHRvIHN3aXRjaCB0byBhbiBBUEkgdXNhZ2UtYmlsbGVkIGFjY291bnQuJyxcbiAgICAgICAgKVxuICAgICAgICByZXR1cm4gbnVsbFxuICAgICAgfVxuICAgIH1cblxuICAgIGNvbnN0IHVybCA9ICdodHRwczovL2NsYXVkZS5haS91cGdyYWRlL21heCdcbiAgICBhd2FpdCBvcGVuQnJvd3Nlcih1cmwpXG5cbiAgICByZXR1cm4gKFxuICAgICAgPExvZ2luXG4gICAgICAgIHN0YXJ0aW5nTWVzc2FnZT17XG4gICAgICAgICAgJ1N0YXJ0aW5nIG5ldyBsb2dpbiBmb2xsb3dpbmcgL3VwZ3JhZGUuIEV4aXQgd2l0aCBDdHJsLUMgdG8gdXNlIGV4aXN0aW5nIGFjY291bnQuJ1xuICAgICAgICB9XG4gICAgICAgIG9uRG9uZT17c3VjY2VzcyA9PiB7XG4gICAgICAgICAgY29udGV4dC5vbkNoYW5nZUFQSUtleSgpXG4gICAgICAgICAgb25Eb25lKHN1Y2Nlc3MgPyAnTG9naW4gc3VjY2Vzc2Z1bCcgOiAnTG9naW4gaW50ZXJydXB0ZWQnKVxuICAgICAgICB9fVxuICAgICAgLz5cbiAgICApXG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgbG9nRXJyb3IoZXJyb3IgYXMgRXJyb3IpXG4gICAgc2V0VGltZW91dChcbiAgICAgIG9uRG9uZSxcbiAgICAgIDAsXG4gICAgICAnRmFpbGVkIHRvIG9wZW4gYnJvd3Nlci4gUGxlYXNlIHZpc2l0IGh0dHBzOi8vY2xhdWRlLmFpL3VwZ3JhZGUvbWF4IHRvIHVwZ3JhZGUuJyxcbiAgICApXG4gIH1cbiAgcmV0dXJuIG51bGxcbn1cbiJdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLQSxLQUFLLE1BQU0sT0FBTztBQUM5QixjQUFjQyxzQkFBc0IsUUFBUSxtQkFBbUI7QUFDL0QsU0FBU0MsNkJBQTZCLFFBQVEseUNBQXlDO0FBQ3ZGLGNBQWNDLHFCQUFxQixRQUFRLHdCQUF3QjtBQUNuRSxTQUNFQyxzQkFBc0IsRUFDdEJDLG9CQUFvQixRQUNmLHFCQUFxQjtBQUM1QixTQUFTQyxXQUFXLFFBQVEsd0JBQXdCO0FBQ3BELFNBQVNDLFFBQVEsUUFBUSxvQkFBb0I7QUFDN0MsU0FBU0MsS0FBSyxRQUFRLG1CQUFtQjtBQUV6QyxPQUFPLGVBQWVDLElBQUlBLENBQ3hCQyxNQUFNLEVBQUVQLHFCQUFxQixFQUM3QlEsT0FBTyxFQUFFVixzQkFBc0IsQ0FDaEMsRUFBRVcsT0FBTyxDQUFDWixLQUFLLENBQUNhLFNBQVMsR0FBRyxJQUFJLENBQUMsQ0FBQztFQUNqQyxJQUFJO0lBQ0Y7SUFDQSxJQUFJUixvQkFBb0IsQ0FBQyxDQUFDLEVBQUU7TUFDMUIsTUFBTVMsTUFBTSxHQUFHVixzQkFBc0IsQ0FBQyxDQUFDO01BQ3ZDLElBQUlXLFFBQVEsR0FBRyxLQUFLO01BRXBCLElBQUlELE1BQU0sRUFBRUUsZ0JBQWdCLElBQUlGLE1BQU0sRUFBRUcsYUFBYSxFQUFFO1FBQ3JERixRQUFRLEdBQ05ELE1BQU0sQ0FBQ0UsZ0JBQWdCLEtBQUssS0FBSyxJQUNqQ0YsTUFBTSxDQUFDRyxhQUFhLEtBQUssd0JBQXdCO01BQ3JELENBQUMsTUFBTSxJQUFJSCxNQUFNLEVBQUVJLFdBQVcsRUFBRTtRQUM5QixNQUFNQyxPQUFPLEdBQUcsTUFBTWpCLDZCQUE2QixDQUFDWSxNQUFNLENBQUNJLFdBQVcsQ0FBQztRQUN2RUgsUUFBUSxHQUNOSSxPQUFPLEVBQUVDLFlBQVksRUFBRUMsaUJBQWlCLEtBQUssWUFBWSxJQUN6REYsT0FBTyxFQUFFQyxZQUFZLEVBQUVFLGVBQWUsS0FBSyx3QkFBd0I7TUFDdkU7TUFFQSxJQUFJUCxRQUFRLEVBQUU7UUFDWlEsVUFBVSxDQUNSYixNQUFNLEVBQ04sQ0FBQyxFQUNELGtJQUNGLENBQUM7UUFDRCxPQUFPLElBQUk7TUFDYjtJQUNGO0lBRUEsTUFBTWMsR0FBRyxHQUFHLCtCQUErQjtJQUMzQyxNQUFNbEIsV0FBVyxDQUFDa0IsR0FBRyxDQUFDO0lBRXRCLE9BQ0UsQ0FBQyxLQUFLLENBQ0osZUFBZSxDQUFDLENBQ2Qsa0ZBQ0YsQ0FBQyxDQUNELE1BQU0sQ0FBQyxDQUFDQyxPQUFPLElBQUk7TUFDakJkLE9BQU8sQ0FBQ2UsY0FBYyxDQUFDLENBQUM7TUFDeEJoQixNQUFNLENBQUNlLE9BQU8sR0FBRyxrQkFBa0IsR0FBRyxtQkFBbUIsQ0FBQztJQUM1RCxDQUFDLENBQUMsR0FDRjtFQUVOLENBQUMsQ0FBQyxPQUFPRSxLQUFLLEVBQUU7SUFDZHBCLFFBQVEsQ0FBQ29CLEtBQUssSUFBSUMsS0FBSyxDQUFDO0lBQ3hCTCxVQUFVLENBQ1JiLE1BQU0sRUFDTixDQUFDLEVBQ0QsZ0ZBQ0YsQ0FBQztFQUNIO0VBQ0EsT0FBTyxJQUFJO0FBQ2IiLCJpZ25vcmVMaXN0IjpbXX0= \ No newline at end of file diff --git a/src/commands/usage/index.ts b/src/commands/usage/index.ts new file mode 100644 index 0000000..c387104 --- /dev/null +++ b/src/commands/usage/index.ts @@ -0,0 +1,9 @@ +import type { Command } from '../../commands.js' + +export default { + type: 'local-jsx', + name: 'usage', + description: 'Show plan usage limits', + availability: ['claude-ai'], + load: () => import('./usage.js'), +} satisfies Command diff --git a/src/commands/usage/usage.tsx b/src/commands/usage/usage.tsx new file mode 100644 index 0000000..b7deb40 --- /dev/null +++ b/src/commands/usage/usage.tsx @@ -0,0 +1,7 @@ +import * as React from 'react'; +import { Settings } from '../../components/Settings/Settings.js'; +import type { LocalJSXCommandCall } from '../../types/command.js'; +export const call: LocalJSXCommandCall = async (onDone, context) => { + return ; +}; +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIlNldHRpbmdzIiwiTG9jYWxKU1hDb21tYW5kQ2FsbCIsImNhbGwiLCJvbkRvbmUiLCJjb250ZXh0Il0sInNvdXJjZXMiOlsidXNhZ2UudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgU2V0dGluZ3MgfSBmcm9tICcuLi8uLi9jb21wb25lbnRzL1NldHRpbmdzL1NldHRpbmdzLmpzJ1xuaW1wb3J0IHR5cGUgeyBMb2NhbEpTWENvbW1hbmRDYWxsIH0gZnJvbSAnLi4vLi4vdHlwZXMvY29tbWFuZC5qcydcblxuZXhwb3J0IGNvbnN0IGNhbGw6IExvY2FsSlNYQ29tbWFuZENhbGwgPSBhc3luYyAob25Eb25lLCBjb250ZXh0KSA9PiB7XG4gIHJldHVybiA8U2V0dGluZ3Mgb25DbG9zZT17b25Eb25lfSBjb250ZXh0PXtjb250ZXh0fSBkZWZhdWx0VGFiPVwiVXNhZ2VcIiAvPlxufVxuIl0sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUtBLEtBQUssTUFBTSxPQUFPO0FBQzlCLFNBQVNDLFFBQVEsUUFBUSx1Q0FBdUM7QUFDaEUsY0FBY0MsbUJBQW1CLFFBQVEsd0JBQXdCO0FBRWpFLE9BQU8sTUFBTUMsSUFBSSxFQUFFRCxtQkFBbUIsR0FBRyxNQUFBQyxDQUFPQyxNQUFNLEVBQUVDLE9BQU8sS0FBSztFQUNsRSxPQUFPLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxDQUFDRCxNQUFNLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQ0MsT0FBTyxDQUFDLENBQUMsVUFBVSxDQUFDLE9BQU8sR0FBRztBQUMzRSxDQUFDIiwiaWdub3JlTGlzdCI6W119 \ No newline at end of file diff --git a/src/commands/version.ts b/src/commands/version.ts new file mode 100644 index 0000000..09f0a44 --- /dev/null +++ b/src/commands/version.ts @@ -0,0 +1,22 @@ +import type { Command, LocalCommandCall } from '../types/command.js' + +const call: LocalCommandCall = async () => { + return { + type: 'text', + value: MACRO.BUILD_TIME + ? `${MACRO.VERSION} (built ${MACRO.BUILD_TIME})` + : MACRO.VERSION, + } +} + +const version = { + type: 'local', + name: 'version', + description: + 'Print the version this session is running (not what autoupdate downloaded)', + isEnabled: () => process.env.USER_TYPE === 'ant', + supportsNonInteractive: true, + load: () => Promise.resolve({ call }), +} satisfies Command + +export default version diff --git a/src/commands/vim/index.ts b/src/commands/vim/index.ts new file mode 100644 index 0000000..f7f2592 --- /dev/null +++ b/src/commands/vim/index.ts @@ -0,0 +1,11 @@ +import type { Command } from '../../commands.js' + +const command = { + name: 'vim', + description: 'Toggle between Vim and Normal editing modes', + supportsNonInteractive: false, + type: 'local', + load: () => import('./vim.js'), +} satisfies Command + +export default command diff --git a/src/commands/vim/vim.ts b/src/commands/vim/vim.ts new file mode 100644 index 0000000..de5fd99 --- /dev/null +++ b/src/commands/vim/vim.ts @@ -0,0 +1,38 @@ +import { + type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + logEvent, +} from '../../services/analytics/index.js' +import type { LocalCommandCall } from '../../types/command.js' +import { getGlobalConfig, saveGlobalConfig } from '../../utils/config.js' + +export const call: LocalCommandCall = async () => { + const config = getGlobalConfig() + let currentMode = config.editorMode || 'normal' + + // Handle backward compatibility - treat 'emacs' as 'normal' + if (currentMode === 'emacs') { + currentMode = 'normal' + } + + const newMode = currentMode === 'normal' ? 'vim' : 'normal' + + saveGlobalConfig(current => ({ + ...current, + editorMode: newMode, + })) + + logEvent('tengu_editor_mode_changed', { + mode: newMode as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + source: + 'command' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + }) + + return { + type: 'text', + value: `Editor mode set to ${newMode}. ${ + newMode === 'vim' + ? 'Use Escape key to toggle between INSERT and NORMAL modes.' + : 'Using standard (readline) keyboard bindings.' + }`, + } +} diff --git a/src/commands/voice/index.ts b/src/commands/voice/index.ts new file mode 100644 index 0000000..61540d3 --- /dev/null +++ b/src/commands/voice/index.ts @@ -0,0 +1,20 @@ +import type { Command } from '../../commands.js' +import { + isVoiceGrowthBookEnabled, + isVoiceModeEnabled, +} from '../../voice/voiceModeEnabled.js' + +const voice = { + type: 'local', + name: 'voice', + description: 'Toggle voice mode', + availability: ['claude-ai'], + isEnabled: () => isVoiceGrowthBookEnabled(), + get isHidden() { + return !isVoiceModeEnabled() + }, + supportsNonInteractive: false, + load: () => import('./voice.js'), +} satisfies Command + +export default voice diff --git a/src/commands/voice/voice.ts b/src/commands/voice/voice.ts new file mode 100644 index 0000000..f369891 --- /dev/null +++ b/src/commands/voice/voice.ts @@ -0,0 +1,150 @@ +import { normalizeLanguageForSTT } from '../../hooks/useVoice.js' +import { getShortcutDisplay } from '../../keybindings/shortcutFormat.js' +import { logEvent } from '../../services/analytics/index.js' +import type { LocalCommandCall } from '../../types/command.js' +import { isAnthropicAuthEnabled } from '../../utils/auth.js' +import { getGlobalConfig, saveGlobalConfig } from '../../utils/config.js' +import { settingsChangeDetector } from '../../utils/settings/changeDetector.js' +import { + getInitialSettings, + updateSettingsForSource, +} from '../../utils/settings/settings.js' +import { isVoiceModeEnabled } from '../../voice/voiceModeEnabled.js' + +const LANG_HINT_MAX_SHOWS = 2 + +export const call: LocalCommandCall = async () => { + // Check auth and kill-switch before allowing voice mode + if (!isVoiceModeEnabled()) { + // Differentiate: OAuth-less users get an auth hint, everyone else + // gets nothing (command shouldn't be reachable when the kill-switch is on). + if (!isAnthropicAuthEnabled()) { + return { + type: 'text' as const, + value: + 'Voice mode requires a Claude.ai account. Please run /login to sign in.', + } + } + return { + type: 'text' as const, + value: 'Voice mode is not available.', + } + } + + const currentSettings = getInitialSettings() + const isCurrentlyEnabled = currentSettings.voiceEnabled === true + + // Toggle OFF — no checks needed + if (isCurrentlyEnabled) { + const result = updateSettingsForSource('userSettings', { + voiceEnabled: false, + }) + if (result.error) { + return { + type: 'text' as const, + value: + 'Failed to update settings. Check your settings file for syntax errors.', + } + } + settingsChangeDetector.notifyChange('userSettings') + logEvent('tengu_voice_toggled', { enabled: false }) + return { + type: 'text' as const, + value: 'Voice mode disabled.', + } + } + + // Toggle ON — run pre-flight checks first + const { isVoiceStreamAvailable } = await import( + '../../services/voiceStreamSTT.js' + ) + const { checkRecordingAvailability } = await import('../../services/voice.js') + + // Check recording availability (microphone access) + const recording = await checkRecordingAvailability() + if (!recording.available) { + return { + type: 'text' as const, + value: + recording.reason ?? 'Voice mode is not available in this environment.', + } + } + + // Check for API key + if (!isVoiceStreamAvailable()) { + return { + type: 'text' as const, + value: + 'Voice mode requires a Claude.ai account. Please run /login to sign in.', + } + } + + // Check for recording tools + const { checkVoiceDependencies, requestMicrophonePermission } = await import( + '../../services/voice.js' + ) + const deps = await checkVoiceDependencies() + if (!deps.available) { + const hint = deps.installCommand + ? `\nInstall audio recording tools? Run: ${deps.installCommand}` + : '\nInstall SoX manually for audio recording.' + return { + type: 'text' as const, + value: `No audio recording tool found.${hint}`, + } + } + + // Probe mic access so the OS permission dialog fires now rather than + // on the user's first hold-to-talk activation. + if (!(await requestMicrophonePermission())) { + let guidance: string + if (process.platform === 'win32') { + guidance = 'Settings \u2192 Privacy \u2192 Microphone' + } else if (process.platform === 'linux') { + guidance = "your system's audio settings" + } else { + guidance = 'System Settings \u2192 Privacy & Security \u2192 Microphone' + } + return { + type: 'text' as const, + value: `Microphone access is denied. To enable it, go to ${guidance}, then run /voice again.`, + } + } + + // All checks passed — enable voice + const result = updateSettingsForSource('userSettings', { voiceEnabled: true }) + if (result.error) { + return { + type: 'text' as const, + value: + 'Failed to update settings. Check your settings file for syntax errors.', + } + } + settingsChangeDetector.notifyChange('userSettings') + logEvent('tengu_voice_toggled', { enabled: true }) + const key = getShortcutDisplay('voice:pushToTalk', 'Chat', 'Space') + const stt = normalizeLanguageForSTT(currentSettings.language) + const cfg = getGlobalConfig() + // Reset the hint counter whenever the resolved STT language changes + // (including first-ever enable, where lastLanguage is undefined). + const langChanged = cfg.voiceLangHintLastLanguage !== stt.code + const priorCount = langChanged ? 0 : (cfg.voiceLangHintShownCount ?? 0) + const showHint = !stt.fellBackFrom && priorCount < LANG_HINT_MAX_SHOWS + let langNote = '' + if (stt.fellBackFrom) { + langNote = ` Note: "${stt.fellBackFrom}" is not a supported dictation language; using English. Change it via /config.` + } else if (showHint) { + langNote = ` Dictation language: ${stt.code} (/config to change).` + } + if (langChanged || showHint) { + saveGlobalConfig(prev => ({ + ...prev, + voiceLangHintShownCount: priorCount + (showHint ? 1 : 0), + voiceLangHintLastLanguage: stt.code, + })) + } + return { + type: 'text' as const, + value: `Voice mode enabled. Hold ${key} to record.${langNote}`, + } +} diff --git a/src/components/AgentProgressLine.tsx b/src/components/AgentProgressLine.tsx new file mode 100644 index 0000000..49fa502 --- /dev/null +++ b/src/components/AgentProgressLine.tsx @@ -0,0 +1,136 @@ +import { c as _c } from "react/compiler-runtime"; +import * as React from 'react'; +import { Box, Text } from '../ink.js'; +import { formatNumber } from '../utils/format.js'; +import type { Theme } from '../utils/theme.js'; +type Props = { + agentType: string; + description?: string; + name?: string; + descriptionColor?: keyof Theme; + taskDescription?: string; + toolUseCount: number; + tokens: number | null; + color?: keyof Theme; + isLast: boolean; + isResolved: boolean; + isError: boolean; + isAsync?: boolean; + shouldAnimate: boolean; + lastToolInfo?: string | null; + hideType?: boolean; +}; +export function AgentProgressLine(t0) { + const $ = _c(32); + const { + agentType, + description, + name, + descriptionColor, + taskDescription, + toolUseCount, + tokens, + color, + isLast, + isResolved, + isAsync: t1, + lastToolInfo, + hideType: t2 + } = t0; + const isAsync = t1 === undefined ? false : t1; + const hideType = t2 === undefined ? false : t2; + const treeChar = isLast ? "\u2514\u2500" : "\u251C\u2500"; + const isBackgrounded = isAsync && isResolved; + let t3; + if ($[0] !== isBackgrounded || $[1] !== isResolved || $[2] !== lastToolInfo || $[3] !== taskDescription) { + t3 = () => { + if (!isResolved) { + return lastToolInfo || "Initializing\u2026"; + } + if (isBackgrounded) { + return taskDescription ?? "Running in the background"; + } + return "Done"; + }; + $[0] = isBackgrounded; + $[1] = isResolved; + $[2] = lastToolInfo; + $[3] = taskDescription; + $[4] = t3; + } else { + t3 = $[4]; + } + const getStatusText = t3; + let t4; + if ($[5] !== treeChar) { + t4 = {treeChar} ; + $[5] = treeChar; + $[6] = t4; + } else { + t4 = $[6]; + } + const t5 = !isResolved; + let t6; + if ($[7] !== agentType || $[8] !== color || $[9] !== description || $[10] !== descriptionColor || $[11] !== hideType || $[12] !== name) { + t6 = hideType ? <>{name ?? description ?? agentType}{name && description && : {description}} : <>{agentType}{description && <>{" ("}{description}{")"}}; + $[7] = agentType; + $[8] = color; + $[9] = description; + $[10] = descriptionColor; + $[11] = hideType; + $[12] = name; + $[13] = t6; + } else { + t6 = $[13]; + } + let t7; + if ($[14] !== isBackgrounded || $[15] !== tokens || $[16] !== toolUseCount) { + t7 = !isBackgrounded && <>{" \xB7 "}{toolUseCount} tool {toolUseCount === 1 ? "use" : "uses"}{tokens !== null && <> · {formatNumber(tokens)} tokens}; + $[14] = isBackgrounded; + $[15] = tokens; + $[16] = toolUseCount; + $[17] = t7; + } else { + t7 = $[17]; + } + let t8; + if ($[18] !== t5 || $[19] !== t6 || $[20] !== t7) { + t8 = {t6}{t7}; + $[18] = t5; + $[19] = t6; + $[20] = t7; + $[21] = t8; + } else { + t8 = $[21]; + } + let t9; + if ($[22] !== t4 || $[23] !== t8) { + t9 = {t4}{t8}; + $[22] = t4; + $[23] = t8; + $[24] = t9; + } else { + t9 = $[24]; + } + let t10; + if ($[25] !== getStatusText || $[26] !== isBackgrounded || $[27] !== isLast) { + t10 = !isBackgrounded && {isLast ? " \u23BF " : "\u2502 \u23BF "}{getStatusText()}; + $[25] = getStatusText; + $[26] = isBackgrounded; + $[27] = isLast; + $[28] = t10; + } else { + t10 = $[28]; + } + let t11; + if ($[29] !== t10 || $[30] !== t9) { + t11 = {t9}{t10}; + $[29] = t10; + $[30] = t9; + $[31] = t11; + } else { + t11 = $[31]; + } + return t11; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkJveCIsIlRleHQiLCJmb3JtYXROdW1iZXIiLCJUaGVtZSIsIlByb3BzIiwiYWdlbnRUeXBlIiwiZGVzY3JpcHRpb24iLCJuYW1lIiwiZGVzY3JpcHRpb25Db2xvciIsInRhc2tEZXNjcmlwdGlvbiIsInRvb2xVc2VDb3VudCIsInRva2VucyIsImNvbG9yIiwiaXNMYXN0IiwiaXNSZXNvbHZlZCIsImlzRXJyb3IiLCJpc0FzeW5jIiwic2hvdWxkQW5pbWF0ZSIsImxhc3RUb29sSW5mbyIsImhpZGVUeXBlIiwiQWdlbnRQcm9ncmVzc0xpbmUiLCJ0MCIsIiQiLCJfYyIsInQxIiwidDIiLCJ1bmRlZmluZWQiLCJ0cmVlQ2hhciIsImlzQmFja2dyb3VuZGVkIiwidDMiLCJnZXRTdGF0dXNUZXh0IiwidDQiLCJ0NSIsInQ2IiwidDciLCJ0OCIsInQ5IiwidDEwIiwidDExIl0sInNvdXJjZXMiOlsiQWdlbnRQcm9ncmVzc0xpbmUudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgQm94LCBUZXh0IH0gZnJvbSAnLi4vaW5rLmpzJ1xuaW1wb3J0IHsgZm9ybWF0TnVtYmVyIH0gZnJvbSAnLi4vdXRpbHMvZm9ybWF0LmpzJ1xuaW1wb3J0IHR5cGUgeyBUaGVtZSB9IGZyb20gJy4uL3V0aWxzL3RoZW1lLmpzJ1xuXG50eXBlIFByb3BzID0ge1xuICBhZ2VudFR5cGU6IHN0cmluZ1xuICBkZXNjcmlwdGlvbj86IHN0cmluZ1xuICBuYW1lPzogc3RyaW5nXG4gIGRlc2NyaXB0aW9uQ29sb3I/OiBrZXlvZiBUaGVtZVxuICB0YXNrRGVzY3JpcHRpb24/OiBzdHJpbmdcbiAgdG9vbFVzZUNvdW50OiBudW1iZXJcbiAgdG9rZW5zOiBudW1iZXIgfCBudWxsXG4gIGNvbG9yPzoga2V5b2YgVGhlbWVcbiAgaXNMYXN0OiBib29sZWFuXG4gIGlzUmVzb2x2ZWQ6IGJvb2xlYW5cbiAgaXNFcnJvcjogYm9vbGVhblxuICBpc0FzeW5jPzogYm9vbGVhblxuICBzaG91bGRBbmltYXRlOiBib29sZWFuXG4gIGxhc3RUb29sSW5mbz86IHN0cmluZyB8IG51bGxcbiAgaGlkZVR5cGU/OiBib29sZWFuXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBBZ2VudFByb2dyZXNzTGluZSh7XG4gIGFnZW50VHlwZSxcbiAgZGVzY3JpcHRpb24sXG4gIG5hbWUsXG4gIGRlc2NyaXB0aW9uQ29sb3IsXG4gIHRhc2tEZXNjcmlwdGlvbixcbiAgdG9vbFVzZUNvdW50LFxuICB0b2tlbnMsXG4gIGNvbG9yLFxuICBpc0xhc3QsXG4gIGlzUmVzb2x2ZWQsXG4gIGlzRXJyb3I6IF9pc0Vycm9yLFxuICBpc0FzeW5jID0gZmFsc2UsXG4gIHNob3VsZEFuaW1hdGU6IF9zaG91bGRBbmltYXRlLFxuICBsYXN0VG9vbEluZm8sXG4gIGhpZGVUeXBlID0gZmFsc2UsXG59OiBQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IHRyZWVDaGFyID0gaXNMYXN0ID8gJ+KUlOKUgCcgOiAn4pSc4pSAJ1xuICBjb25zdCBpc0JhY2tncm91bmRlZCA9IGlzQXN5bmMgJiYgaXNSZXNvbHZlZFxuXG4gIC8vIERldGVybWluZSB0aGUgc3RhdHVzIHRleHRcbiAgY29uc3QgZ2V0U3RhdHVzVGV4dCA9ICgpOiBzdHJpbmcgPT4ge1xuICAgIGlmICghaXNSZXNvbHZlZCkge1xuICAgICAgcmV0dXJuIGxhc3RUb29sSW5mbyB8fCAnSW5pdGlhbGl6aW5n4oCmJ1xuICAgIH1cbiAgICBpZiAoaXNCYWNrZ3JvdW5kZWQpIHtcbiAgICAgIHJldHVybiB0YXNrRGVzY3JpcHRpb24gPz8gJ1J1bm5pbmcgaW4gdGhlIGJhY2tncm91bmQnXG4gICAgfVxuICAgIHJldHVybiAnRG9uZSdcbiAgfVxuXG4gIHJldHVybiAoXG4gICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCI+XG4gICAgICA8Qm94IHBhZGRpbmdMZWZ0PXszfT5cbiAgICAgICAgPFRleHQgZGltQ29sb3I+e3RyZWVDaGFyfSA8L1RleHQ+XG4gICAgICAgIDxUZXh0IGRpbUNvbG9yPXshaXNSZXNvbHZlZH0+XG4gICAgICAgICAge2hpZGVUeXBlID8gKFxuICAgICAgICAgICAgPD5cbiAgICAgICAgICAgICAgPFRleHQgYm9sZD57bmFtZSA/PyBkZXNjcmlwdGlvbiA/PyBhZ2VudFR5cGV9PC9UZXh0PlxuICAgICAgICAgICAgICB7bmFtZSAmJiBkZXNjcmlwdGlvbiAmJiA8VGV4dCBkaW1Db2xvcj46IHtkZXNjcmlwdGlvbn08L1RleHQ+fVxuICAgICAgICAgICAgPC8+XG4gICAgICAgICAgKSA6IChcbiAgICAgICAgICAgIDw+XG4gICAgICAgICAgICAgIDxUZXh0XG4gICAgICAgICAgICAgICAgYm9sZFxuICAgICAgICAgICAgICAgIGJhY2tncm91bmRDb2xvcj17Y29sb3J9XG4gICAgICAgICAgICAgICAgY29sb3I9e2NvbG9yID8gJ2ludmVyc2VUZXh0JyA6IHVuZGVmaW5lZH1cbiAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgIHthZ2VudFR5cGV9XG4gICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAge2Rlc2NyaXB0aW9uICYmIChcbiAgICAgICAgICAgICAgICA8PlxuICAgICAgICAgICAgICAgICAgeycgKCd9XG4gICAgICAgICAgICAgICAgICA8VGV4dFxuICAgICAgICAgICAgICAgICAgICBiYWNrZ3JvdW5kQ29sb3I9e2Rlc2NyaXB0aW9uQ29sb3J9XG4gICAgICAgICAgICAgICAgICAgIGNvbG9yPXtkZXNjcmlwdGlvbkNvbG9yID8gJ2ludmVyc2VUZXh0JyA6IHVuZGVmaW5lZH1cbiAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAge2Rlc2NyaXB0aW9ufVxuICAgICAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgICAgICAgeycpJ31cbiAgICAgICAgICAgICAgICA8Lz5cbiAgICAgICAgICAgICAgKX1cbiAgICAgICAgICAgIDwvPlxuICAgICAgICAgICl9XG4gICAgICAgICAgeyFpc0JhY2tncm91bmRlZCAmJiAoXG4gICAgICAgICAgICA8PlxuICAgICAgICAgICAgICB7JyDCtyAnfVxuICAgICAgICAgICAgICB7dG9vbFVzZUNvdW50fSB0b29sIHt0b29sVXNlQ291bnQgPT09IDEgPyAndXNlJyA6ICd1c2VzJ31cbiAgICAgICAgICAgICAge3Rva2VucyAhPT0gbnVsbCAmJiA8PiDCtyB7Zm9ybWF0TnVtYmVyKHRva2Vucyl9IHRva2VuczwvPn1cbiAgICAgICAgICAgIDwvPlxuICAgICAgICAgICl9XG4gICAgICAgIDwvVGV4dD5cbiAgICAgIDwvQm94PlxuICAgICAgeyFpc0JhY2tncm91bmRlZCAmJiAoXG4gICAgICAgIDxCb3ggcGFkZGluZ0xlZnQ9ezN9IGZsZXhEaXJlY3Rpb249XCJyb3dcIj5cbiAgICAgICAgICA8VGV4dCBkaW1Db2xvcj57aXNMYXN0ID8gJyAgIOKOvyAgJyA6ICfilIIgIOKOvyAgJ308L1RleHQ+XG4gICAgICAgICAgPFRleHQgZGltQ29sb3I+e2dldFN0YXR1c1RleHQoKX08L1RleHQ+XG4gICAgICAgIDwvQm94PlxuICAgICAgKX1cbiAgICA8L0JveD5cbiAgKVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsT0FBTyxLQUFLQSxLQUFLLE1BQU0sT0FBTztBQUM5QixTQUFTQyxHQUFHLEVBQUVDLElBQUksUUFBUSxXQUFXO0FBQ3JDLFNBQVNDLFlBQVksUUFBUSxvQkFBb0I7QUFDakQsY0FBY0MsS0FBSyxRQUFRLG1CQUFtQjtBQUU5QyxLQUFLQyxLQUFLLEdBQUc7RUFDWEMsU0FBUyxFQUFFLE1BQU07RUFDakJDLFdBQVcsQ0FBQyxFQUFFLE1BQU07RUFDcEJDLElBQUksQ0FBQyxFQUFFLE1BQU07RUFDYkMsZ0JBQWdCLENBQUMsRUFBRSxNQUFNTCxLQUFLO0VBQzlCTSxlQUFlLENBQUMsRUFBRSxNQUFNO0VBQ3hCQyxZQUFZLEVBQUUsTUFBTTtFQUNwQkMsTUFBTSxF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 \ No newline at end of file diff --git a/src/components/App.tsx b/src/components/App.tsx new file mode 100644 index 0000000..69ca968 --- /dev/null +++ b/src/components/App.tsx @@ -0,0 +1,56 @@ +import { c as _c } from "react/compiler-runtime"; +import React from 'react'; +import { FpsMetricsProvider } from '../context/fpsMetrics.js'; +import { StatsProvider, type StatsStore } from '../context/stats.js'; +import { type AppState, AppStateProvider } from '../state/AppState.js'; +import { onChangeAppState } from '../state/onChangeAppState.js'; +import type { FpsMetrics } from '../utils/fpsTracker.js'; +type Props = { + getFpsMetrics: () => FpsMetrics | undefined; + stats?: StatsStore; + initialState: AppState; + children: React.ReactNode; +}; + +/** + * Top-level wrapper for interactive sessions. + * Provides FPS metrics, stats context, and app state to the component tree. + */ +export function App(t0) { + const $ = _c(9); + const { + getFpsMetrics, + stats, + initialState, + children + } = t0; + let t1; + if ($[0] !== children || $[1] !== initialState) { + t1 = {children}; + $[0] = children; + $[1] = initialState; + $[2] = t1; + } else { + t1 = $[2]; + } + let t2; + if ($[3] !== stats || $[4] !== t1) { + t2 = {t1}; + $[3] = stats; + $[4] = t1; + $[5] = t2; + } else { + t2 = $[5]; + } + let t3; + if ($[6] !== getFpsMetrics || $[7] !== t2) { + t3 = {t2}; + $[6] = getFpsMetrics; + $[7] = t2; + $[8] = t3; + } else { + t3 = $[8]; + } + return t3; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkZwc01ldHJpY3NQcm92aWRlciIsIlN0YXRzUHJvdmlkZXIiLCJTdGF0c1N0b3JlIiwiQXBwU3RhdGUiLCJBcHBTdGF0ZVByb3ZpZGVyIiwib25DaGFuZ2VBcHBTdGF0ZSIsIkZwc01ldHJpY3MiLCJQcm9wcyIsImdldEZwc01ldHJpY3MiLCJzdGF0cyIsImluaXRpYWxTdGF0ZSIsImNoaWxkcmVuIiwiUmVhY3ROb2RlIiwiQXBwIiwidDAiLCIkIiwiX2MiLCJ0MSIsInQyIiwidDMiXSwic291cmNlcyI6WyJBcHAudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB7IEZwc01ldHJpY3NQcm92aWRlciB9IGZyb20gJy4uL2NvbnRleHQvZnBzTWV0cmljcy5qcydcbmltcG9ydCB7IFN0YXRzUHJvdmlkZXIsIHR5cGUgU3RhdHNTdG9yZSB9IGZyb20gJy4uL2NvbnRleHQvc3RhdHMuanMnXG5pbXBvcnQgeyB0eXBlIEFwcFN0YXRlLCBBcHBTdGF0ZVByb3ZpZGVyIH0gZnJvbSAnLi4vc3RhdGUvQXBwU3RhdGUuanMnXG5pbXBvcnQgeyBvbkNoYW5nZUFwcFN0YXRlIH0gZnJvbSAnLi4vc3RhdGUvb25DaGFuZ2VBcHBTdGF0ZS5qcydcbmltcG9ydCB0eXBlIHsgRnBzTWV0cmljcyB9IGZyb20gJy4uL3V0aWxzL2Zwc1RyYWNrZXIuanMnXG5cbnR5cGUgUHJvcHMgPSB7XG4gIGdldEZwc01ldHJpY3M6ICgpID0+IEZwc01ldHJpY3MgfCB1bmRlZmluZWRcbiAgc3RhdHM/OiBTdGF0c1N0b3JlXG4gIGluaXRpYWxTdGF0ZTogQXBwU3RhdGVcbiAgY2hpbGRyZW46IFJlYWN0LlJlYWN0Tm9kZVxufVxuXG4vKipcbiAqIFRvcC1sZXZlbCB3cmFwcGVyIGZvciBpbnRlcmFjdGl2ZSBzZXNzaW9ucy5cbiAqIFByb3ZpZGVzIEZQUyBtZXRyaWNzLCBzdGF0cyBjb250ZXh0LCBhbmQgYXBwIHN0YXRlIHRvIHRoZSBjb21wb25lbnQgdHJlZS5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIEFwcCh7XG4gIGdldEZwc01ldHJpY3MsXG4gIHN0YXRzLFxuICBpbml0aWFsU3RhdGUsXG4gIGNoaWxkcmVuLFxufTogUHJvcHMpOiBSZWFjdC5SZWFjdE5vZGUge1xuICByZXR1cm4gKFxuICAgIDxGcHNNZXRyaWNzUHJvdmlkZXIgZ2V0RnBzTWV0cmljcz17Z2V0RnBzTWV0cmljc30+XG4gICAgICA8U3RhdHNQcm92aWRlciBzdG9yZT17c3RhdHN9PlxuICAgICAgICA8QXBwU3RhdGVQcm92aWRlclxuICAgICAgICAgIGluaXRpYWxTdGF0ZT17aW5pdGlhbFN0YXRlfVxuICAgICAgICAgIG9uQ2hhbmdlQXBwU3RhdGU9e29uQ2hhbmdlQXBwU3RhdGV9XG4gICAgICAgID5cbiAgICAgICAgICB7Y2hpbGRyZW59XG4gICAgICAgIDwvQXBwU3RhdGVQcm92aWRlcj5cbiAgICAgIDwvU3RhdHNQcm92aWRlcj5cbiAgICA8L0Zwc01ldHJpY3NQcm92aWRlcj5cbiAgKVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsT0FBT0EsS0FBSyxNQUFNLE9BQU87QUFDekIsU0FBU0Msa0JBQWtCLFFBQVEsMEJBQTBCO0FBQzdELFNBQVNDLGFBQWEsRUFBRSxLQUFLQyxVQUFVLFFBQVEscUJBQXFCO0FBQ3BFLFNBQVMsS0FBS0MsUUFBUSxFQUFFQyxnQkFBZ0IsUUFBUSxzQkFBc0I7QUFDdEUsU0FBU0MsZ0JBQWdCLFFBQVEsOEJBQThCO0FBQy9ELGNBQWNDLFVBQVUsUUFBUSx3QkFBd0I7QUFFeEQsS0FBS0MsS0FBSyxHQUFHO0VBQ1hDLGFBQWEsRUFBRSxHQUFHLEdBQUdGLFVBQVUsR0FBRyxTQUFTO0VBQzNDRyxLQUFLLENBQUMsRUFBRVAsVUFBVTtFQUNsQlEsWUFBWSxFQUFFUCxRQUFRO0VBQ3RCUSxRQUFRLEVBQUVaLEtBQUssQ0FBQ2EsU0FBUztBQUMzQixDQUFDOztBQUVEO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsT0FBTyxTQUFBQyxJQUFBQyxFQUFBO0VBQUEsTUFBQUMsQ0FBQSxHQUFBQyxFQUFBO0VBQWE7SUFBQVIsYUFBQTtJQUFBQyxLQUFBO0lBQUFDLFlBQUE7SUFBQUM7RUFBQSxJQUFBRyxFQUtaO0VBQUEsSUFBQUcsRUFBQTtFQUFBLElBQUFGLENBQUEsUUFBQUosUUFBQSxJQUFBSSxDQUFBLFFBQUFMLFlBQUE7SUFJQU8sRUFBQSxJQUFDLGdCQUFnQixDQUNEUCxZQUFZLENBQVpBLGFBQVcsQ0FBQyxDQUNSTCxnQkFBZ0IsQ0FBaEJBLGlCQUFlLENBQUMsQ0FFakNNLFNBQU8sQ0FDVixFQUxDLGdCQUFnQixDQUtFO0lBQUFJLENBQUEsTUFBQUosUUFBQTtJQUFBSSxDQUFBLE1BQUFMLFlBQUE7SUFBQUssQ0FBQSxNQUFBRSxFQUFBO0VBQUE7SUFBQUEsRUFBQSxHQUFBRixDQUFBO0VBQUE7RUFBQSxJQUFBRyxFQUFBO0VBQUEsSUFBQUgsQ0FBQSxRQUFBTixLQUFBLElBQUFNLENBQUEsUUFBQUUsRUFBQTtJQU5yQkMsRUFBQSxJQUFDLGFBQWEsQ0FBUVQsS0FBSyxDQUFMQSxNQUFJLENBQUMsQ0FDekIsQ0FBQVEsRUFLa0IsQ0FDcEIsRUFQQyxhQUFhLENBT0U7SUFBQUYsQ0FBQSxNQUFBTixLQUFBO0lBQUFNLENBQUEsTUFBQUUsRUFBQTtJQUFBRixDQUFBLE1BQUFHLEVBQUE7RUFBQTtJQUFBQSxFQUFBLEdBQUFILENBQUE7RUFBQTtFQUFBLElBQUFJLEVBQUE7RUFBQSxJQUFBSixDQUFBLFFBQUFQLGFBQUEsSUFBQU8sQ0FBQSxRQUFBRyxFQUFBO0lBUmxCQyxFQUFBLElBQUMsa0JBQWtCLENBQWdCWCxhQUFhLENBQWJBLGNBQVksQ0FBQyxDQUM5QyxDQUFBVSxFQU9lLENBQ2pCLEVBVEMsa0JBQWtCLENBU0U7SUFBQUgsQ0FBQSxNQUFBUCxhQUFBO0lBQUFPLENBQUEsTUFBQUcsRUFBQTtJQUFBSCxDQUFBLE1BQUFJLEVBQUE7RUFBQTtJQUFBQSxFQUFBLEdBQUFKLENBQUE7RUFBQTtFQUFBLE9BVHJCSSxFQVNxQjtBQUFBIiwiaWdub3JlTGlzdCI6W119 \ No newline at end of file diff --git a/src/components/ApproveApiKey.tsx b/src/components/ApproveApiKey.tsx new file mode 100644 index 0000000..be54c0e --- /dev/null +++ b/src/components/ApproveApiKey.tsx @@ -0,0 +1,123 @@ +import { c as _c } from "react/compiler-runtime"; +import React from 'react'; +import { Text } from '../ink.js'; +import { saveGlobalConfig } from '../utils/config.js'; +import { Select } from './CustomSelect/index.js'; +import { Dialog } from './design-system/Dialog.js'; +type Props = { + customApiKeyTruncated: string; + onDone(approved: boolean): void; +}; +export function ApproveApiKey(t0) { + const $ = _c(17); + const { + customApiKeyTruncated, + onDone + } = t0; + let t1; + if ($[0] !== customApiKeyTruncated || $[1] !== onDone) { + t1 = function onChange(value) { + bb2: switch (value) { + case "yes": + { + saveGlobalConfig(current_0 => ({ + ...current_0, + customApiKeyResponses: { + ...current_0.customApiKeyResponses, + approved: [...(current_0.customApiKeyResponses?.approved ?? []), customApiKeyTruncated] + } + })); + onDone(true); + break bb2; + } + case "no": + { + saveGlobalConfig(current => ({ + ...current, + customApiKeyResponses: { + ...current.customApiKeyResponses, + rejected: [...(current.customApiKeyResponses?.rejected ?? []), customApiKeyTruncated] + } + })); + onDone(false); + } + } + }; + $[0] = customApiKeyTruncated; + $[1] = onDone; + $[2] = t1; + } else { + t1 = $[2]; + } + const onChange = t1; + let t2; + if ($[3] !== onChange) { + t2 = () => onChange("no"); + $[3] = onChange; + $[4] = t2; + } else { + t2 = $[4]; + } + let t3; + if ($[5] === Symbol.for("react.memo_cache_sentinel")) { + t3 = ANTHROPIC_API_KEY; + $[5] = t3; + } else { + t3 = $[5]; + } + let t4; + if ($[6] !== customApiKeyTruncated) { + t4 = {t3}: sk-ant-...{customApiKeyTruncated}; + $[6] = customApiKeyTruncated; + $[7] = t4; + } else { + t4 = $[7]; + } + let t5; + if ($[8] === Symbol.for("react.memo_cache_sentinel")) { + t5 = Do you want to use this API key?; + $[8] = t5; + } else { + t5 = $[8]; + } + let t6; + if ($[9] === Symbol.for("react.memo_cache_sentinel")) { + t6 = { + label: "Yes", + value: "yes" + }; + $[9] = t6; + } else { + t6 = $[9]; + } + let t7; + if ($[10] === Symbol.for("react.memo_cache_sentinel")) { + t7 = [t6, { + label: No (recommended), + value: "no" + }]; + $[10] = t7; + } else { + t7 = $[10]; + } + let t8; + if ($[11] !== onChange) { + t8 = ; + $[11] = onDecline; + $[12] = t7; + $[13] = t8; + $[14] = t9; + } else { + t9 = $[14]; + } + let t10; + if ($[15] !== onDecline || $[16] !== t9) { + t10 = {t3}{t9}; + $[15] = onDecline; + $[16] = t9; + $[17] = t10; + } else { + t10 = $[17]; + } + return t10; +} +function _temp() { + logEvent("tengu_auto_mode_opt_in_dialog_shown", {}); +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsImxvZ0V2ZW50IiwiQm94IiwiTGluayIsIlRleHQiLCJ1cGRhdGVTZXR0aW5nc0ZvclNvdXJjZSIsIlNlbGVjdCIsIkRpYWxvZyIsIkFVVE9fTU9ERV9ERVNDUklQVElPTiIsIlByb3BzIiwib25BY2NlcHQiLCJvbkRlY2xpbmUiLCJkZWNsaW5lRXhpdHMiLCJBdXRvTW9kZU9wdEluRGlhbG9nIiwidDAiLCIkIiwiX2MiLCJ0MSIsIlN5bWJvbCIsImZvciIsInVzZUVmZmVjdCIsIl90ZW1wIiwidDIiLCJvbkNoYW5nZSIsInZhbHVlIiwiYmIzIiwic2tpcEF1dG9QZXJtaXNzaW9uUHJvbXB0IiwicGVybWlzc2lvbnMiLCJkZWZhdWx0TW9kZSIsInQzIiwidDQiLCJsYWJlbCIsImNvbnN0IiwidDUiLCJ0NiIsInQ3IiwidDgiLCJ2YWx1ZV8wIiwidDkiLCJ0MTAiXSwic291cmNlcyI6WyJBdXRvTW9kZU9wdEluRGlhbG9nLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBsb2dFdmVudCB9IGZyb20gJ3NyYy9zZXJ2aWNlcy9hbmFseXRpY3MvaW5kZXguanMnXG5pbXBvcnQgeyBCb3gsIExpbmssIFRleHQgfSBmcm9tICcuLi9pbmsuanMnXG5pbXBvcnQgeyB1cGRhdGVTZXR0aW5nc0ZvclNvdXJjZSB9IGZyb20gJy4uL3V0aWxzL3NldHRpbmdzL3NldHRpbmdzLmpzJ1xuaW1wb3J0IHsgU2VsZWN0IH0gZnJvbSAnLi9DdXN0b21TZWxlY3QvaW5kZXguanMnXG5pbXBvcnQgeyBEaWFsb2cgfSBmcm9tICcuL2Rlc2lnbi1zeXN0ZW0vRGlhbG9nLmpzJ1xuXG4vLyBOT1RFOiBUaGlzIGNvcHkgaXMgbGVnYWxseSByZXZpZXdlZCDigJQgZG8gbm90IG1vZGlmeSB3aXRob3V0IExlZ2FsIHRlYW0gYXBwcm92YWwuXG5leHBvcnQgY29uc3QgQVVUT19NT0RFX0RFU0NSSVBUSU9OID1cbiAgXCJBdXRvIG1vZGUgbGV0cyBDbGF1ZGUgaGFuZGxlIHBlcm1pc3Npb24gcHJvbXB0cyBhdXRvbWF0aWNhbGx5IOKAlCBDbGF1ZGUgY2hlY2tzIGVhY2ggdG9vbCBjYWxsIGZvciByaXNreSBhY3Rpb25zIGFuZCBwcm9tcHQgaW5qZWN0aW9uIGJlZm9yZSBleGVjdXRpbmcuIEFjdGlvbnMgQ2xhdWRlIGlkZW50aWZpZXMgYXMgc2FmZSBhcmUgZXhlY3V0ZWQsIHdoaWxlIGFjdGlvbnMgQ2xhdWRlIGlkZW50aWZpZXMgYXMgcmlza3kgYXJlIGJsb2NrZWQgYW5kIENsYXVkZSBtYXkgdHJ5IGEgZGlmZmVyZW50IGFwcHJvYWNoLiBJZGVhbCBmb3IgbG9uZy1ydW5uaW5nIHRhc2tzLiBTZXNzaW9ucyBhcmUgc2xpZ2h0bHkgbW9yZSBleHBlbnNpdmUuIENsYXVkZSBjYW4gbWFrZSBtaXN0YWtlcyB0aGF0IGFsbG93IGhhcm1mdWwgY29tbWFuZHMgdG8gcnVuLCBpdCdzIHJlY29tbWVuZGVkIHRvIG9ubHkgdXNlIGluIGlzb2xhdGVkIGVudmlyb25tZW50cy4gU2hpZnQrVGFiIHRvIGNoYW5nZSBtb2RlLlwiXG5cbnR5cGUgUHJvcHMgPSB7XG4gIG9uQWNjZXB0KCk6IHZvaWRcbiAgb25EZWNsaW5lKCk6IHZvaWRcbiAgLy8gU3RhcnR1cCBnYXRlOiBkZWNsaW5lIGV4aXRzIHRoZSBwcm9jZXNzLCBzbyByZWxhYmVsIGFjY29yZGluZ2x5LlxuICBkZWNsaW5lRXhpdHM/OiBib29sZWFuXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBBdXRvTW9kZU9wdEluRGlhbG9nKHtcbiAgb25BY2NlcHQsXG4gIG9uRGVjbGluZSxcbiAgZGVjbGluZUV4aXRzLFxufTogUHJvcHMpOiBSZWFjdC5SZWFjdE5vZGUge1xuICBSZWFjdC51c2VFZmZlY3QoKCkgPT4ge1xuICAgIGxvZ0V2ZW50KCd0ZW5ndV9hdXRvX21vZGVfb3B0X2luX2RpYWxvZ19zaG93bicsIHt9KVxuICB9LCBbXSlcblxuICBmdW5jdGlvbiBvbkNoYW5nZSh2YWx1ZTogJ2FjY2VwdCcgfCAnYWNjZXB0LWRlZmF1bHQnIHwgJ2RlY2xpbmUnKSB7XG4gICAgc3dpdGNoICh2YWx1ZSkge1xuICAgICAgY2FzZSAnYWNjZXB0Jzoge1xuICAgICAgICBsb2dFdmVudCgndGVuZ3VfYXV0b19tb2RlX29wdF9pbl9kaWFsb2dfYWNjZXB0Jywge30pXG4gICAgICAgIHVwZGF0ZVNldHRpbmdzRm9yU291cmNlKCd1c2VyU2V0dGluZ3MnLCB7XG4gICAgICAgICAgc2tpcEF1dG9QZXJtaXNzaW9uUHJvbXB0OiB0cnVlLFxuICAgICAgICB9KVxuICAgICAgICBvbkFjY2VwdCgpXG4gICAgICAgIGJyZWFrXG4gICAgICB9XG4gICAgICBjYXNlICdhY2NlcHQtZGVmYXVsdCc6IHtcbiAgICAgICAgbG9nRXZlbnQoJ3Rlbmd1X2F1dG9fbW9kZV9vcHRfaW5fZGlhbG9nX2FjY2VwdF9kZWZhdWx0Jywge30pXG4gICAgICAgIHVwZGF0ZVNldHRpbmdzRm9yU291cmNlKCd1c2VyU2V0dGluZ3MnLCB7XG4gICAgICAgICAgc2tpcEF1dG9QZXJtaXNzaW9uUHJvbXB0OiB0cnVlLFxuICAgICAgICAgIHBlcm1pc3Npb25zOiB7IGRlZmF1bHRNb2RlOiAnYXV0bycgfSxcbiAgICAgICAgfSlcbiAgICAgICAgb25BY2NlcHQoKVxuICAgICAgICBicmVha1xuICAgICAgfVxuICAgICAgY2FzZSAnZGVjbGluZSc6IHtcbiAgICAgICAgbG9nRXZlbnQoJ3Rlbmd1X2F1dG9fbW9kZV9vcHRfaW5fZGlhbG9nX2RlY2xpbmUnLCB7fSlcbiAgICAgICAgb25EZWNsaW5lKClcbiAgICAgICAgYnJlYWtcbiAgICAgIH1cbiAgICB9XG4gIH1cblxuICByZXR1cm4gKFxuICAgIDxEaWFsb2cgdGl0bGU9XCJFbmFibGUgYXV0byBtb2RlP1wiIGNvbG9yPVwid2FybmluZ1wiIG9uQ2FuY2VsPXtvbkRlY2xpbmV9PlxuICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCIgZ2FwPXsxfT5cbiAgICAgICAgPFRleHQ+e0FVVE9fTU9ERV9ERVNDUklQVElPTn08L1RleHQ+XG5cbiAgICAgICAgPExpbmsgdXJsPVwiaHR0cHM6Ly9jb2RlLmNsYXVkZS5jb20vZG9jcy9lbi9zZWN1cml0eVwiIC8+XG4gICAgICA8L0JveD5cblxuICAgICAgPFNlbGVjdFxuICAgICAgICBvcHRpb25zPXtbXG4gICAgICAgICAgLi4uKFwiZXh0ZXJuYWxcIiAhPT0gJ2FudCdcbiAgICAgICAgICAgID8gW1xuICAgICAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgICAgIGxhYmVsOiAnWWVzLCBhbmQgbWFrZSBpdCBteSBkZWZhdWx0IG1vZGUnLFxuICAgICAgICAgICAgICAgICAgdmFsdWU6ICdhY2NlcHQtZGVmYXVsdCcgYXMgY29uc3QsXG4gICAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgICAgXVxuICAgICAgICAgICAgOiBbXSksXG4gICAgICAgICAgeyBsYWJlbDogJ1llcywgZW5hYmxlIGF1dG8gbW9kZScsIHZhbHVlOiAnYWNjZXB0JyBhcyBjb25zdCB9LFxuICAgICAgICAgIHtcbiAgICAgICAgICAgIGxhYmVsOiBkZWNsaW5lRXhpdHMgPyAnTm8sIGV4aXQnIDogJ05vLCBnbyBiYWNrJyxcbiAgICAgICAgICAgIHZhbHVlOiAnZGVjbGluZScgYXMgY29uc3QsXG4gICAgICAgICAgfSxcbiAgICAgICAgXX1cbiAgICAgICAgb25DaGFuZ2U9e3ZhbHVlID0+XG4gICAgICAgICAgb25DaGFuZ2UodmFsdWUgYXMgJ2FjY2VwdCcgfCAnYWNjZXB0LWRlZmF1bHQnIHwgJ2RlY2xpbmUnKVxuICAgICAgICB9XG4gICAgICAgIG9uQ2FuY2VsPXtvbkRlY2xpbmV9XG4gICAgICAvPlxuICAgIDwvRGlhbG9nPlxuICApXG59XG4iXSwibWFwcGluZ3MiOiI7QUFBQSxPQUFPQSxLQUFLLE1BQU0sT0FBTztBQUN6QixTQUFTQyxRQUFRLFFBQVEsaUNBQWlDO0FBQzFELFNBQVNDLEdBQUcsRUFBRUMsSUFBSSxFQUFFQyxJQUFJ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 \ No newline at end of file diff --git a/src/components/AutoUpdater.tsx b/src/components/AutoUpdater.tsx new file mode 100644 index 0000000..144e2c8 --- /dev/null +++ b/src/components/AutoUpdater.tsx @@ -0,0 +1,198 @@ +import * as React from 'react'; +import { useEffect, useRef, useState } from 'react'; +import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from 'src/services/analytics/index.js'; +import { useInterval } from 'usehooks-ts'; +import { useUpdateNotification } from '../hooks/useUpdateNotification.js'; +import { Box, Text } from '../ink.js'; +import { type AutoUpdaterResult, getLatestVersion, getMaxVersion, type InstallStatus, installGlobalPackage, shouldSkipVersion } from '../utils/autoUpdater.js'; +import { getGlobalConfig, isAutoUpdaterDisabled } from '../utils/config.js'; +import { logForDebugging } from '../utils/debug.js'; +import { getCurrentInstallationType } from '../utils/doctorDiagnostic.js'; +import { installOrUpdateClaudePackage, localInstallationExists } from '../utils/localInstaller.js'; +import { removeInstalledSymlink } from '../utils/nativeInstaller/index.js'; +import { gt, gte } from '../utils/semver.js'; +import { getInitialSettings } from '../utils/settings/settings.js'; +type Props = { + isUpdating: boolean; + onChangeIsUpdating: (isUpdating: boolean) => void; + onAutoUpdaterResult: (autoUpdaterResult: AutoUpdaterResult) => void; + autoUpdaterResult: AutoUpdaterResult | null; + showSuccessMessage: boolean; + verbose: boolean; +}; +export function AutoUpdater({ + isUpdating, + onChangeIsUpdating, + onAutoUpdaterResult, + autoUpdaterResult, + showSuccessMessage, + verbose +}: Props): React.ReactNode { + const [versions, setVersions] = useState<{ + global?: string | null; + latest?: string | null; + }>({}); + const [hasLocalInstall, setHasLocalInstall] = useState(false); + const updateSemver = useUpdateNotification(autoUpdaterResult?.version); + useEffect(() => { + void localInstallationExists().then(setHasLocalInstall); + }, []); + + // Track latest isUpdating value in a ref so the memoized checkForUpdates + // callback always sees the current value. Without this, the 30-minute + // interval fires with a stale closure where isUpdating is false, allowing + // a concurrent installGlobalPackage() to run while one is already in + // progress. + const isUpdatingRef = useRef(isUpdating); + isUpdatingRef.current = isUpdating; + const checkForUpdates = React.useCallback(async () => { + if (isUpdatingRef.current) { + return; + } + if ("production" === 'test' || "production" === 'development') { + logForDebugging('AutoUpdater: Skipping update check in test/dev environment'); + return; + } + const currentVersion = MACRO.VERSION; + const channel = getInitialSettings()?.autoUpdatesChannel ?? 'latest'; + let latestVersion = await getLatestVersion(channel); + const isDisabled = isAutoUpdaterDisabled(); + + // Check if max version is set (server-side kill switch for auto-updates) + const maxVersion = await getMaxVersion(); + if (maxVersion && latestVersion && gt(latestVersion, maxVersion)) { + logForDebugging(`AutoUpdater: maxVersion ${maxVersion} is set, capping update from ${latestVersion} to ${maxVersion}`); + if (gte(currentVersion, maxVersion)) { + logForDebugging(`AutoUpdater: current version ${currentVersion} is already at or above maxVersion ${maxVersion}, skipping update`); + setVersions({ + global: currentVersion, + latest: latestVersion + }); + return; + } + latestVersion = maxVersion; + } + setVersions({ + global: currentVersion, + latest: latestVersion + }); + + // Check if update needed and perform update + if (!isDisabled && currentVersion && latestVersion && !gte(currentVersion, latestVersion) && !shouldSkipVersion(latestVersion)) { + const startTime = Date.now(); + onChangeIsUpdating(true); + + // Remove native installer symlink since we're using JS-based updates + // But only if user hasn't migrated to native installation + const config = getGlobalConfig(); + if (config.installMethod !== 'native') { + await removeInstalledSymlink(); + } + + // Detect actual running installation type + const installationType = await getCurrentInstallationType(); + logForDebugging(`AutoUpdater: Detected installation type: ${installationType}`); + + // Skip update for development builds + if (installationType === 'development') { + logForDebugging('AutoUpdater: Cannot auto-update development build'); + onChangeIsUpdating(false); + return; + } + + // Choose the appropriate update method based on what's actually running + let installStatus: InstallStatus; + let updateMethod: 'local' | 'global'; + if (installationType === 'npm-local') { + // Use local update for local installations + logForDebugging('AutoUpdater: Using local update method'); + updateMethod = 'local'; + installStatus = await installOrUpdateClaudePackage(channel); + } else if (installationType === 'npm-global') { + // Use global update for global installations + logForDebugging('AutoUpdater: Using global update method'); + updateMethod = 'global'; + installStatus = await installGlobalPackage(); + } else if (installationType === 'native') { + // This shouldn't happen - native should use NativeAutoUpdater + logForDebugging('AutoUpdater: Unexpected native installation in non-native updater'); + onChangeIsUpdating(false); + return; + } else { + // Fallback to config-based detection for unknown types + logForDebugging(`AutoUpdater: Unknown installation type, falling back to config`); + const isMigrated = config.installMethod === 'local'; + updateMethod = isMigrated ? 'local' : 'global'; + if (isMigrated) { + installStatus = await installOrUpdateClaudePackage(channel); + } else { + installStatus = await installGlobalPackage(); + } + } + onChangeIsUpdating(false); + if (installStatus === 'success') { + logEvent('tengu_auto_updater_success', { + fromVersion: currentVersion as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + toVersion: latestVersion as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + durationMs: Date.now() - startTime, + wasMigrated: updateMethod === 'local', + installationType: installationType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + } else { + logEvent('tengu_auto_updater_fail', { + fromVersion: currentVersion as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + attemptedVersion: latestVersion as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + status: installStatus as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + durationMs: Date.now() - startTime, + wasMigrated: updateMethod === 'local', + installationType: installationType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + } + onAutoUpdaterResult({ + version: latestVersion, + status: installStatus + }); + } + // isUpdating intentionally omitted from deps; we read isUpdatingRef + // instead so the guard is always current without changing callback + // identity (which would re-trigger the initial-check useEffect below). + // eslint-disable-next-line react-hooks/exhaustive-deps + // biome-ignore lint/correctness/useExhaustiveDependencies: isUpdating read via ref + }, [onAutoUpdaterResult]); + + // Initial check + useEffect(() => { + void checkForUpdates(); + }, [checkForUpdates]); + + // Check every 30 minutes + useInterval(checkForUpdates, 30 * 60 * 1000); + if (!autoUpdaterResult?.version && (!versions.global || !versions.latest)) { + return null; + } + if (!autoUpdaterResult?.version && !isUpdating) { + return null; + } + return + {verbose && + globalVersion: {versions.global} · latestVersion:{' '} + {versions.latest} + } + {isUpdating ? <> + + + Auto-updating… + + + : autoUpdaterResult?.status === 'success' && showSuccessMessage && updateSemver && + ✓ Update installed · Restart to apply + } + {(autoUpdaterResult?.status === 'install_failed' || autoUpdaterResult?.status === 'no_permissions') && + ✗ Auto-update failed · Try claude doctor or{' '} + + {hasLocalInstall ? `cd ~/.claude/local && npm update ${MACRO.PACKAGE_URL}` : `npm i -g ${MACRO.PACKAGE_URL}`} + + } + ; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsInVzZUVmZmVjdCIsInVzZVJlZiIsInVzZVN0YXRlIiwiQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyIsImxvZ0V2ZW50IiwidXNlSW50ZXJ2YWwiLCJ1c2VVcGRhdGVOb3RpZmljYXRpb24iLCJCb3giLCJUZXh0IiwiQXV0b1VwZGF0ZXJSZXN1bHQiLCJnZXRMYXRlc3RWZXJzaW9uIiwiZ2V0TWF4VmVyc2lvbiIsIkluc3RhbGxTdGF0dXMiLCJpbnN0YWxsR2xvYmFsUGFja2FnZSIsInNob3VsZFNraXBWZXJzaW9uIiwiZ2V0R2xvYmFsQ29uZmlnIiwiaXNBdXRvVXBkYXRlckRpc2FibGVkIiwibG9nRm9yRGVidWdnaW5nIiwiZ2V0Q3VycmVudEluc3RhbGxhdGlvblR5cGUiLCJpbnN0YWxsT3JVcGRhdGVDbGF1ZGVQYWNrYWdlIiwibG9jYWxJbnN0YWxsYXRpb25FeGlzdHMiLCJyZW1vdmVJbnN0YWxsZWRTeW1saW5rIiwiZ3QiLCJndGUiLCJnZXRJbml0aWFsU2V0dGluZ3MiLCJQcm9wcyIsImlzVXBkYXRpbmciLCJvbkNoYW5nZUlzVXBkYXRpbmciLCJvbkF1dG9VcGRhdGVyUmVzdWx0IiwiYXV0b1VwZGF0ZXJSZXN1bHQiLCJzaG93U3VjY2Vzc01lc3NhZ2UiLCJ2ZXJib3NlIiwiQXV0b1VwZGF0ZXIiLCJSZWFjdE5vZGUiLCJ2ZXJzaW9ucyIsInNldFZlcnNpb25zIiwiZ2xvYmFsIiwibGF0ZXN0IiwiaGFzTG9jYWxJbnN0YWxsIiwic2V0SGFzTG9jYWxJbnN0YWxsIiwidXBkYXRlU2VtdmVyIiwidmVyc2lvbiIsInRoZW4iLCJpc1VwZGF0aW5nUmVmIiwiY3VycmVudCIsImNoZWNrRm9yVXBkYXRlcyIsInVzZUNhbGxiYWNrIiwiY3VycmVudFZlcnNpb24iLCJNQUNSTyIsIlZFUlNJT04iLCJjaGFubmVsIiwiYXV0b1VwZGF0ZXNDaGFubmVsIiwibGF0ZXN0VmVyc2lvbiIsImlzRGlzYWJsZWQiLCJtYXhWZXJzaW9uIiwic3RhcnRUaW1lIiwiRGF0ZSIsIm5vdyIsImNvbmZpZyIsImluc3RhbGxNZXRob2QiLCJpbnN0YWxsYXRpb25UeXBlIiwiaW5zdGFsbFN0YXR1cyIsInVwZGF0ZU1ldGhvZCIsImlzTWlncmF0ZWQiLCJmcm9tVmVyc2lvbiIsInRvVmVyc2lvbiIsImR1cmF0aW9uTXMiLCJ3YXNNaWdyYXRlZCIsImF0dGVtcHRlZFZlcnNpb24iLCJzdGF0dXMiLCJQQUNLQUdFX1VSTCJdLCJzb3VyY2VzIjpbIkF1dG9VcGRhdGVyLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZUVmZmVjdCwgdXNlUmVmLCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHtcbiAgdHlwZSBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICBsb2dFdmVudCxcbn0gZnJvbSAnc3JjL3NlcnZpY2VzL2FuYWx5dGljcy9pbmRleC5qcydcbmltcG9ydCB7IHVzZUludGVydmFsIH0gZnJvbSAndXNlaG9va3MtdHMnXG5pbXBvcnQgeyB1c2VVcGRhdGVOb3RpZmljYXRpb24gfSBmcm9tICcuLi9ob29rcy91c2VVcGRhdGVOb3RpZmljYXRpb24uanMnXG5pbXBvcnQgeyBCb3gsIFRleHQgfSBmcm9tICcuLi9pbmsuanMnXG5pbXBvcnQge1xuICB0eXBlIEF1dG9VcGRhdGVyUmVzdWx0LFxuICBnZXRMYXRlc3RWZXJzaW9uLFxuICBnZXRNYXhWZXJzaW9uLFxuICB0eXBlIEluc3RhbGxTdGF0dXMsXG4gIGluc3RhbGxHbG9iYWxQYWNrYWdlLFxuICBzaG91bGRTa2lwVmVyc2lvbixcbn0gZnJvbSAnLi4vdXRpbHMvYXV0b1VwZGF0ZXIuanMnXG5pbXBvcnQgeyBnZXRHbG9iYWxDb25maWcsIGlzQXV0b1VwZGF0ZXJEaXNhYmxlZCB9IGZyb20gJy4uL3V0aWxzL2NvbmZpZy5qcydcbmltcG9ydCB7IGxvZ0ZvckRlYnVnZ2luZyB9IGZyb20gJy4uL3V0aWxzL2RlYnVnLmpzJ1xuaW1wb3J0IHsgZ2V0Q3VycmVudEluc3RhbGxhdGlvblR5cGUgfSBmcm9tICcuLi91dGlscy9kb2N0b3JEaWFnbm9zdGljLmpzJ1xuaW1wb3J0IHtcbiAgaW5zdGFsbE9yVXBkYXRlQ2xhdWRlUGFja2FnZSxcbiAgbG9jYWxJbnN0YWxsYXRpb25FeGlzdHMsXG59IGZyb20gJy4uL3V0aWxzL2xvY2FsSW5zdGFsbGVyLmpzJ1xuaW1wb3J0IHsgcmVtb3ZlSW5zdGFsbGVkU3ltbGluayB9IGZyb20gJy4uL3V0aWxzL25hdGl2ZUluc3RhbGxlci9pbmRleC5qcydcbmltcG9ydCB7IGd0LCBndGUgfSBmcm9tICcuLi91dGlscy9zZW12ZXIuanMnXG5pbXBvcnQgeyBnZXRJbml0aWFsU2V0dGluZ3MgfSBmcm9tICcuLi91dGlscy9zZXR0aW5ncy9zZXR0aW5ncy5qcydcblxudHlwZSBQcm9wcyA9IHtcbiAgaXNVcGRhdGluZzogYm9vbGVhblxuICBvbkNoYW5nZUlzVXBkYXRpbmc6IChpc1VwZGF0aW5nOiBib29sZWFuKSA9PiB2b2lkXG4gIG9uQXV0b1VwZGF0ZXJSZXN1bHQ6IChhdXRvVXBkYXRlclJlc3VsdDogQXV0b1VwZGF0ZXJSZXN1bHQpID0+IHZvaWRcbiAgYXV0b1VwZGF0ZXJSZXN1bHQ6IEF1dG9VcGRhdGVyUmVzdWx0IHwgbnVsbFxuICBzaG93U3VjY2Vzc01lc3NhZ2U6IGJvb2xlYW5cbiAgdmVyYm9zZTogYm9vbGVhblxufVxuXG5leHBvcnQgZnVuY3Rpb24gQXV0b1VwZGF0ZXIoe1xuICBpc1VwZGF0aW5nLFxuICBvbkNoYW5nZUlzVXBkYXRpbmcsXG4gIG9uQXV0b1VwZGF0ZXJSZXN1bHQsXG4gIGF1dG9VcGRhdGVyUmVzdWx0LFxuICBzaG93U3VjY2Vzc01lc3NhZ2UsXG4gIHZlcmJvc2UsXG59OiBQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IFt2ZXJzaW9ucywgc2V0VmVyc2lvbnNdID0gdXNlU3RhdGU8e1xuICAgIGdsb2JhbD86IHN0cmluZyB8IG51bGxcbiAgICBsYXRlc3Q/OiBzdHJpbmcgfCBudWxsXG4gIH0+KHt9KVxuICBjb25zdCBbaGFzTG9jYWxJbnN0YWxsLCBzZXRIYXNMb2NhbEluc3RhbGxdID0gdXNlU3RhdGUoZmFsc2UpXG4gIGNvbnN0IHVwZGF0ZVNlbXZlciA9IHVzZVVwZGF0ZU5vdGlmaWNhdGlvbihhdXRvVXBkYXRlclJlc3VsdD8udmVyc2lvbilcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIHZvaWQgbG9jYWxJbnN0YWxsYXRpb25FeGlzdHMoKS50aGVuKHNldEhhc0xvY2FsSW5zdGFsbClcbiAgfSwgW10pXG5cbiAgLy8gVHJhY2sgbGF0ZXN0IGlzVXBkYXRpbmcgdmFsdWUgaW4gYSByZWYgc28gdGhlIG1lbW9pemVkIGNoZWNrRm9yVXBkYXRlc1xuICAvLyBjYWxsYmFjayBhbHdheXMgc2VlcyB0aGUgY3VycmVudCB2YWx1ZS4gV2l0aG91dCB0aGlzLCB0aGUgMzAtbWludXRlXG4gIC8vIGludGVydmFsIGZpcmVzIHdpdGggYSBzdGFsZSBjbG9zdXJlIHdoZXJlIGlzVXBkYXRpbmcgaXMgZmFsc2UsIGFsbG93aW5nXG4gIC8vIGEgY29uY3VycmVudCBpbnN0YWxsR2xvYmFsUGFja2FnZSgpIHRvIHJ1biB3aGlsZSBvbmUgaXMgYWxyZWFkeSBpblxuICAvLyBwcm9ncmVzcy5cbiAgY29uc3QgaXNVcGRhdGluZ1JlZiA9IHVzZVJlZihpc1VwZGF0aW5nKVxuICBpc1VwZGF0aW5nUmVmLmN1cnJlbnQgPSBpc1VwZGF0aW5nXG5cbiAgY29uc3QgY2hlY2tGb3JVcGRhdGVzID0gUmVhY3QudXNlQ2FsbGJhY2soYXN5bmMgKCkgPT4ge1xuICAgIGlmIChpc1VwZGF0aW5nUmVmLmN1cnJlbnQpIHtcbiAgICAgIHJldHVyblxuICAgIH1cblxuICAgIGlmIChcbiAgICAgIFwicHJvZHVjdGlvblwiID09PSAndGVzdCcgfHxcbiAgICAgIFwicHJvZHVjdGlvblwiID09PSAnZGV2ZWxvcG1lbnQnXG4gICAgKSB7XG4gICAgICBsb2dGb3JEZWJ1Z2dpbmcoXG4gICAgICAgICdBdXRvVXBkYXRlcjogU2tpcHBpbmcgdXBkYXRlIGNoZWNrIGluIHRlc3QvZGV2IGVudmlyb25tZW50JyxcbiAgICAgIClcbiAgICAgIHJldHVyblxuICAgIH1cblxuICAgIGNvbnN0IGN1cnJlbnRWZXJzaW9uID0gTUFDUk8uVkVSU0lPTlxuICAgIGNvbnN0IGNoYW5uZWwgPSBnZXRJbml0aWFsU2V0dGluZ3MoKT8uYXV0b1VwZGF0ZXNDaGFubmVsID8/ICdsYXRlc3QnXG4gICAgbGV0IGxhdGVzdFZlcnNpb24gPSBhd2FpdCBnZXRMYXRlc3RWZXJzaW9uKGNoYW5uZWwpXG4gICAgY29uc3QgaXNEaXNhYmxlZCA9IGlzQXV0b1VwZGF0ZXJEaXNhYmxlZCgpXG5cbiAgICAvLyBDaGVjayBpZiBtYXggdmVyc2lvbiBpcyBzZXQgKHNlcnZlci1zaWRlIGtpbGwgc3dpdGNoIGZvciBhdXRvLXVwZGF0ZXMpXG4gICAgY29uc3QgbWF4VmVyc2lvbiA9IGF3YWl0IGdldE1heFZlcnNpb24oKVxuICAgIGlmIChtYXhWZXJzaW9uICYmIGxhdGVzdFZlcnNpb24gJiYgZ3QobGF0ZXN0VmVyc2lvbiwgbWF4VmVyc2lvbikpIHtcbiAgICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgICAgYEF1dG9VcGRhdGVyOiBtYXhWZXJzaW9uICR7bWF4VmVyc2lvbn0gaXMgc2V0LCBjYXBwaW5nIHVwZGF0ZSBmcm9tICR7bGF0ZXN0VmVyc2lvbn0gdG8gJHttYXhWZXJzaW9ufWAsXG4gICAgICApXG4gICAgICBpZiAoZ3RlKGN1cnJlbnRWZXJzaW9uLCBtYXhWZXJzaW9uKSkge1xuICAgICAgICBsb2dGb3JEZWJ1Z2dpbmcoXG4gICAgICAgICAgYEF1dG9VcGRhdGVyOiBjdXJyZW50IHZlcnNpb24gJHtjdXJyZW50VmVyc2lvbn0gaXMgYWxyZWFkeSBhdCBvciBhYm92ZSBtYXhWZXJzaW9uICR7bWF4VmVyc2lvbn0sIHNraXBwaW5nIHVwZGF0ZWAsXG4gICAgICAgIClcbiAgICAgICAgc2V0VmVyc2lvbnMoeyBnbG9iYWw6IGN1cnJlbnRWZXJzaW9uLCBsYXRlc3Q6IGxhdGVzdFZlcnNpb24gfSlcbiAgICAgICAgcmV0dXJuXG4gICAgICB9XG4gICAgICBsYXRlc3RWZXJzaW9uID0gbWF4VmVyc2lvblxuICAgIH1cblxuICAgIHNldFZlcnNpb25zKHsgZ2xvYmFsOiBjdXJyZW50VmVyc2lvbiwgbGF0ZXN0OiBsYXRlc3RWZXJzaW9uIH0pXG5cbiAgICAvLyBDaGVjayBpZiB1cGRhdGUgbmVlZGVkIGFuZCBwZXJmb3JtIHVwZGF0ZVxuICAgIGlmIChcbiAgICAgICFpc0Rpc2FibGVkICYmXG4gICAgICBjdXJyZW50VmVyc2lvbiAmJlxuICAgICAgbGF0ZXN0VmVyc2lvbiAmJlxuICAgICAgIWd0ZShjdXJyZW50VmVyc2lvbiwgbGF0ZXN0VmVyc2lvbikgJiZcbiAgICAgICFzaG91bGRTa2lwVmVyc2lvbihsYXRlc3RWZXJzaW9uKVxuICAgICkge1xuICAgICAgY29uc3Qgc3RhcnRUaW1lID0gRGF0ZS5ub3coKVxuICAgICAgb25DaGFuZ2VJc1VwZGF0aW5nKHRydWUpXG5cbiAgICAgIC8vIFJlbW92ZSBuYXRpdmUgaW5zdGFsbGVyIHN5bWxpbmsgc2luY2Ugd2UncmUgdXNpbmcgSlMtYmFzZWQgdXBkYXRlc1xuICAgICAgLy8gQnV0IG9ubHkgaWYgdXNlciBoYXNuJ3QgbWlncmF0ZWQgdG8gbmF0aXZlIGluc3RhbGxhdGlvblxuICAgICAgY29uc3QgY29uZmlnID0gZ2V0R2xvYmFsQ29uZmlnKClcbiAgICAgIGlmIChjb25maWcuaW5zdGFsbE1ldGhvZCAhPT0gJ25hdGl2ZScpIHtcbiAgICAgICAgYXdhaXQgcmVtb3ZlSW5zdGFsbGVkU3ltbGluaygpXG4gICAgICB9XG5cbiAgICAgIC8vIERldGVjdCBhY3R1YWwgcnVubmluZyBpbnN0YWxsYXRpb24gdHlwZVxuICAgICAgY29uc3QgaW5zdGFsbGF0aW9uVHlwZSA9IGF3YWl0IGdldEN1cnJlbnRJbnN0YWxsYXRpb25UeXBlKClcbiAgICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgICAgYEF1dG9VcGRhdGVyOiBEZXRlY3RlZCBpbnN0YWxsYXRpb24gdHlwZTogJHtpbnN0YWxsYXRpb25UeXBlfWAsXG4gICAgICApXG5cbiAgICAgIC8vIFNraXAgdXBkYXRlIGZvciBkZXZlbG9wbWVudCBidWlsZHNcbiAgICAgIGlmIChpbnN0YWxsYXRpb25UeXBlID09PSAnZGV2ZWxvcG1lbnQnKSB7XG4gICAgICAgIGxvZ0ZvckRlYnVnZ2luZygnQXV0b1VwZGF0ZXI6IENhbm5vdCBhdXRvLXVwZGF0ZSBkZXZlbG9wbWVudCBidWlsZCcpXG4gICAgICAgIG9uQ2hhbmdlSXNVcGRhdGluZyhmYWxzZSlcbiAgICAgICAgcmV0dXJuXG4gICAgICB9XG5cbiAgICAgIC8vIENob29zZSB0aGUgYXBwcm9wcmlhdGUgdXBkYXRlIG1ldGhvZCBiYXNlZCBvbiB3aGF0J3MgYWN0dWFsbHkgcnVubmluZ1xuICAgICAgbGV0IGluc3RhbGxTdGF0dXM6IEluc3RhbGxTdGF0dXNcbiAgICAgIGxldCB1cGRhdGVNZXRob2Q6ICdsb2NhbCcgfCAnZ2xvYmFsJ1xuXG4gICAgICBpZiAoaW5zdGFsbGF0aW9uVHlwZSA9PT0gJ25wbS1sb2NhbCcpIHtcbiAgICAgICAgLy8gVXNlIGxvY2FsIHVwZGF0ZSBmb3IgbG9jYWwgaW5zdGFsbGF0aW9uc1xuICAgICAgICBsb2dGb3JEZWJ1Z2dpbmcoJ0F1dG9VcGRhdGVyOiBVc2luZyBsb2NhbCB1cGRhdGUgbWV0aG9kJylcbiAgICAgICAgdXBkYXRlTWV0aG9kID0gJ2xvY2FsJ1xuICAgICAgICBpbnN0YWxsU3RhdHVzID0gYXdhaXQgaW5zdGFsbE9yVXBkYXRlQ2xhdWRlUGFja2FnZShjaGFubmVsKVxuICAgICAgfSBlbHNlIGlmIChpbnN0YWxsYXRpb25UeXBlID09PSAnbnBtLWdsb2JhbCcpIHtcbiAgICAgICAgLy8gVXNlIGdsb2JhbCB1cGRhdGUgZm9yIGdsb2JhbCBpbnN0YWxsYXRpb25zXG4gICAgICAgIGxvZ0ZvckRlYnVnZ2luZygnQXV0b1VwZGF0ZXI6IFVzaW5nIGdsb2JhbCB1cGRhdGUgbWV0aG9kJylcbiAgICAgICAgdXBkYXRlTWV0aG9kID0gJ2dsb2JhbCdcbiAgICAgICAgaW5zdGFsbFN0YXR1cyA9IGF3YWl0IGluc3RhbGxHbG9iYWxQYWNrYWdlKClcbiAgICAgIH0gZWxzZSBpZiAoaW5zdGFsbGF0aW9uVHlwZSA9PT0gJ25hdGl2ZScpIHtcbiAgICAgICAgLy8gVGhpcyBzaG91bGRuJ3QgaGFwcGVuIC0gbmF0aXZlIHNob3VsZCB1c2UgTmF0aXZlQXV0b1VwZGF0ZXJcbiAgICAgICAgbG9nRm9yRGVidWdnaW5nKFxuICAgICAgICAgICdBdXRvVXBkYXRlcjogVW5leHBlY3RlZCBuYXRpdmUgaW5zdGFsbGF0aW9uIGluIG5vbi1uYXRpdmUgdXBkYXRlcicsXG4gICAgICAgIClcbiAgICAgICAgb25DaGFuZ2VJc1VwZGF0aW5nKGZhbHNlKVxuICAgICAgICByZXR1cm5cbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIC8vIEZhbGxiYWNrIHRvIGNvbmZpZy1iYXNlZCBkZXRlY3Rpb24gZm9yIHVua25vd24gdHlwZXNcbiAgICAgICAgbG9nRm9yRGVidWdnaW5nKFxuICAgICAgICAgIGBBdXRvVXBkYXRlcjogVW5rbm93biBpbnN0YWxsYXRpb24gdHlwZSwgZmFsbGluZyBiYWNrIHRvIGNvbmZpZ2AsXG4gICAgICAgIClcbiAgICAgICAgY29uc3QgaXNNaWdyYXRlZCA9IGNvbmZpZy5pbnN0YWxsTWV0aG9kID09PSAnbG9jYWwnXG4gICAgICAgIHVwZGF0ZU1ldGhvZCA9IGlzTWlncmF0ZWQgPyAnbG9jYWwnIDogJ2dsb2JhbCdcblxuICAgICAgICBpZiAoaXNNaWdyYXRlZCkge1xuICAgICAgICAgIGluc3RhbGxTdGF0dXMgPSBhd2FpdCBpbnN0YWxsT3JVcGRhdGVDbGF1ZGVQYWNrYWdlKGNoYW5uZWwpXG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgaW5zdGFsbFN0YXR1cyA9IGF3YWl0IGluc3RhbGxHbG9iYWxQYWNrYWdlKClcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBvbkNoYW5nZUlzVXBkYXRpbmcoZmFsc2UpXG5cbiAgICAgIGlmIChpbnN0YWxsU3RhdHVzID09PSAnc3VjY2VzcycpIHtcbiAgICAgICAgbG9nRXZlbnQoJ3Rlbmd1X2F1dG9fdXBkYXRlcl9zdWNjZXNzJywge1xuICAgICAgICAgIGZyb21WZXJzaW9uOlxuICAgICAgICAgICAgY3VycmVudFZlcnNpb24gYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICAgICAgICB0b1ZlcnNpb246XG4gICAgICAgICAgICBsYXRlc3RWZXJzaW9uIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgICAgZHVyYXRpb25NczogRGF0ZS5ub3coKSAtIHN0YXJ0VGltZSxcbiAgICAgICAgICB3YXNNaWdyYXRlZDogdXBkYXRlTWV0aG9kID09PSAnbG9jYWwnLFxuICAgICAgICAgIGluc3RhbGxhdGlvblR5cGU6XG4gICAgICAgICAgICBpbnN0YWxsYXRpb25UeXBlIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgIH0pXG4gICAgICB9IGVsc2Uge1xuICAgICAgICBsb2dFdmVudCgndGVuZ3VfYXV0b191cGRhdGVyX2ZhaWwnLCB7XG4gICAgICAgICAgZnJvbVZlcnNpb246XG4gICAgICAgICAgICBjdXJyZW50VmVyc2lvbiBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICAgIGF0dGVtcHRlZFZlcnNpb246XG4gICAgICAgICAgICBsYXRlc3RWZXJzaW9uIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgICAgc3RhdHVzOlxuICAgICAgICAgICAgaW5zdGFsbFN0YXR1cyBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICAgIGR1cmF0aW9uTXM6IERhdGUubm93KCkgLSBzdGFydFRpbWUsXG4gICAgICAgICAgd2FzTWlncmF0ZWQ6IHVwZGF0ZU1ldGhvZCA9PT0gJ2xvY2FsJyxcbiAgICAgICAgICBpbnN0YWxsYXRpb25UeXBlOlxuICAgICAgICAgICAgaW5zdGFsbGF0aW9uVHlwZSBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICB9KVxuICAgICAgfVxuXG4gICAgICBvbkF1dG9VcGRhdGVyUmVzdWx0KHtcbiAgICAgICAgdmVyc2lvbjogbGF0ZXN0VmVyc2lvbixcbiAgICAgICAgc3RhdHVzOiBpbnN0YWxsU3RhdHVzLFxuICAgICAgfSlcbiAgICB9XG4gICAgLy8gaXNVcGRhdGluZyBpbnRlbnRpb25hbGx5IG9taXR0ZWQgZnJvbSBkZXBzOyB3ZSByZWFkIGlzVXBkYXRpbmdSZWZcbiAgICAvLyBpbnN0ZWFkIHNvIHRoZSBndWFyZCBpcyBhbHdheXMgY3VycmVudCB3aXRob3V0IGNoYW5naW5nIGNhbGxiYWNrXG4gICAgLy8gaWRlbnRpdHkgKHdoaWNoIHdvdWxkIHJlLXRyaWdnZXIgdGhlIGluaXRpYWwtY2hlY2sgdXNlRWZmZWN0IGJlbG93KS5cbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgcmVhY3QtaG9va3MvZXhoYXVzdGl2ZS1kZXBzXG4gICAgLy8gYmlvbWUtaWdub3JlIGxpbnQvY29ycmVjdG5lc3MvdXNlRXhoYXVzdGl2ZURlcGVuZGVuY2llczogaXNVcGRhdGluZyByZWFkIHZpYSByZWZcbiAgfSwgW29uQXV0b1VwZGF0ZXJSZXN1bHRdKVxuXG4gIC8vIEluaXRpYWwgY2hlY2tcbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICB2b2lkIGNoZWNrRm9yVXBkYXRlcygpXG4gIH0sIFtjaGVja0ZvclVwZGF0ZXNdKVxuXG4gIC8vIENoZWNrIGV2ZXJ5IDMwIG1pbnV0ZXNcbiAgdXNlSW50ZXJ2YWwoY2hlY2tGb3JVcGRhdGVzLCAzMCAqIDYwICogMTAwMClcblxuICBpZiAoIWF1dG9VcGRhdGVyUmVzdWx0Py52ZXJzaW9uICYmICghdmVyc2lvbnMuZ2xvYmFsIHx8ICF2ZXJzaW9ucy5sYXRlc3QpKSB7XG4gICAgcmV0dXJuIG51bGxcbiAgfVxuXG4gIGlmICghYXV0b1VwZGF0ZXJSZXN1bHQ/LnZlcnNpb24gJiYgIWlzVXBkYXRpbmcpIHtcbiAgICByZXR1cm4gbnVsbFxuICB9XG5cbiAgcmV0dXJuIChcbiAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJyb3dcIiBnYXA9ezF9PlxuICAgICAge3ZlcmJvc2UgJiYgKFxuICAgICAgICA8VGV4dCBkaW1Db2xvciB3cmFwPVwidHJ1bmNhdGVcIj5cbiAgICAgICAgICBnbG9iYWxWZXJzaW9uOiB7dmVyc2lvbnMuZ2xvYmFsfSAmbWlkZG90OyBsYXRlc3RWZXJzaW9uOnsnICd9XG4gICAgICAgICAge3ZlcnNpb25zLmxhdGVzdH1cbiAgICAgICAgPC9UZXh0PlxuICAgICAgKX1cbiAgICAgIHtpc1VwZGF0aW5nID8gKFxuICAgICAgICA8PlxuICAgICAgICAgIDxCb3g+XG4gICAgICAgICAgICA8VGV4dCBjb2xvcj1cInRleHRcIiBkaW1Db2xvciB3cmFwPVwidHJ1bmNhdGVcIj5cbiAgICAgICAgICAgICAgQXV0by11cGRhdGluZ+KAplxuICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgIDwvQm94PlxuICAgICAgICA8Lz5cbiAgICAgICkgOiAoXG4gICAgICAgIGF1dG9VcGRhdGVyUmVzdWx0Py5zdGF0dXMgPT09ICdzdWNjZXNzJyAmJlxuICAgICAgICBzaG93U3VjY2Vzc01lc3NhZ2UgJiZcbiAgICAgICAgdXBkYXRlU2VtdmVyICYmIChcbiAgICAgICAgICA8VGV4dCBjb2xvcj1cInN1Y2Nlc3NcIiB3cmFwPVwidHJ1bmNhdGVcIj5cbiAgICAgICAgICAgIOKckyBVcGRhdGUgaW5zdGFsbGVkIMK3IFJlc3RhcnQgdG8gYXBwbHlcbiAgICAgICAgICA8L1RleHQ+XG4gICAgICAgIClcbiAgICAgICl9XG4gICAgICB7KGF1dG9VcGRhdGVyUmVzdWx0Py5zdGF0dXMgPT09ICdpbnN0YWxsX2ZhaWxlZCcgfHxcbiAgICAgICAgYXV0b1VwZGF0ZXJSZXN1bHQ/LnN0YXR1cyA9PT0gJ25vX3Blcm1pc3Npb25zJykgJiYgKFxuICAgICAgICA8VGV4dCBjb2xvcj1cImVycm9yXCIgd3JhcD1cInRydW5jYXRlXCI+XG4gICAgICAgICAg4pyXIEF1dG8tdXBkYXRlIGZhaWxlZCAmbWlkZG90OyBUcnkgPFRleHQgYm9sZD5jbGF1ZGUgZG9jdG9yPC9UZXh0PiBvcnsnICd9XG4gICAgICAgICAgPFRleHQgYm9sZD5cbiAgICAgICAgICAgIHtoYXNMb2NhbEluc3RhbGxcbiAgICAgICAgICAgICAgPyBgY2Qgfi8uY2xhdWRlL2xvY2FsICYmIG5wbSB1cGRhdGUgJHtNQUNSTy5QQUNLQUdFX1VSTH1gXG4gICAgICAgICAgICAgIDogYG5wbSBpIC1nICR7TUFDUk8uUEFDS0FHRV9VUkx9YH1cbiAgICAgICAgICA8L1RleHQ+XG4gICAgICAgIDwvVGV4dD5cbiAgICAgICl9XG4gICAgPC9Cb3g+XG4gIClcbn1cbiJdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLQSxLQUFLLE1BQU0sT0FBTztBQUM5QixTQUFTQyxTQUFTLEVBQUVDLE1BQU0sRUFBRUMsUUFBUSxRQUFRLE9BQU87QUFDbkQsU0FDRSxLQUFLQywwREFBMEQsRUFDL0RDLFFBQVEsUUFDSCxpQ0FBaUM7QUFDeEMsU0FBU0MsV0FBVyxRQUFRLGFBQWE7QUFDekMsU0FBU0MscUJB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QUFDLEVBQUU7SUFDekUsT0FBTyxJQUFJO0VBQ2I7RUFFQSxJQUFJLENBQUNSLGlCQUFpQixFQUFFWSxPQUFPLElBQUksQ0FBQ2YsVUFBVSxFQUFFO0lBQzlDLE9BQU8sSUFBSTtFQUNiO0VBRUEsT0FDRSxDQUFDLEdBQUcsQ0FBQyxhQUFhLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQztBQUNwQyxNQUFNLENBQUNLLE9BQU8sSUFDTixDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLFVBQVU7QUFDdEMseUJBQXlCLENBQUNHLFFBQVEsQ0FBQ0UsTUFBTSxDQUFDLHdCQUF3QixDQUFDLEdBQUc7QUFDdEUsVUFBVSxDQUFDRixRQUFRLENBQUNHLE1BQU07QUFDMUIsUUFBUSxFQUFFLElBQUksQ0FDUDtBQUNQLE1BQU0sQ0FBQ1gsVUFBVSxHQUNUO0FBQ1IsVUFBVSxDQUFDLEdBQUc7QUFDZCxZQUFZLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxNQUFNLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQyxVQUFVO0FBQ3ZEO0FBQ0EsWUFBWSxFQUFFLElBQUk7QUFDbEIsVUFBVSxFQUFFLEdBQUc7QUFDZixRQUFRLEdBQUcsR0FFSEcsaUJBQWlCLEVBQUV3QyxNQUFNLEtBQUssU0FBUyxJQUN2Q3ZDLGtCQUFrQixJQUNsQlUsWUFBWSxJQUNWLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLFVBQVU7QUFDL0M7QUFDQSxVQUFVLEVBQUUsSUFBSSxDQUVUO0FBQ1AsTUFBTSxDQUFDLENBQUNYLGlCQUFpQixFQUFFd0MsTUFBTSxLQUFLLGdCQUFnQixJQUM5Q3hDLGlCQUFpQixFQUFFd0MsTUFBTSxLQUFLLGdCQUFnQixLQUM5QyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxVQUFVO0FBQzNDLDRDQUE0QyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsYUFBYSxFQUFFLElBQUksQ0FBQyxHQUFHLENBQUMsR0FBRztBQUNsRixVQUFVLENBQUMsSUFBSSxDQUFDLElBQUk7QUFDcEIsWUFBWSxDQUFDL0IsZUFBZSxHQUNaLG9DQUFvQ1UsS0FBSyxDQUFDc0IsV0FBVyxFQUFFLEdBQ3ZELFlBQVl0QixLQUFLLENBQUNzQixXQUFXLEVBQUU7QUFDL0MsVUFBVSxFQUFFLElBQUk7QUFDaEIsUUFBUSxFQUFFLElBQUksQ0FDUDtBQUNQLElBQUksRUFBRSxHQUFHLENBQUM7QUFFViIsImlnbm9yZUxpc3QiOltdfQ== \ No newline at end of file diff --git a/src/components/AutoUpdaterWrapper.tsx b/src/components/AutoUpdaterWrapper.tsx new file mode 100644 index 0000000..d812ec3 --- /dev/null +++ b/src/components/AutoUpdaterWrapper.tsx @@ -0,0 +1,91 @@ +import { c as _c } from "react/compiler-runtime"; +import { feature } from 'bun:bundle'; +import * as React from 'react'; +import type { AutoUpdaterResult } from '../utils/autoUpdater.js'; +import { isAutoUpdaterDisabled } from '../utils/config.js'; +import { logForDebugging } from '../utils/debug.js'; +import { getCurrentInstallationType } from '../utils/doctorDiagnostic.js'; +import { AutoUpdater } from './AutoUpdater.js'; +import { NativeAutoUpdater } from './NativeAutoUpdater.js'; +import { PackageManagerAutoUpdater } from './PackageManagerAutoUpdater.js'; +type Props = { + isUpdating: boolean; + onChangeIsUpdating: (isUpdating: boolean) => void; + onAutoUpdaterResult: (autoUpdaterResult: AutoUpdaterResult) => void; + autoUpdaterResult: AutoUpdaterResult | null; + showSuccessMessage: boolean; + verbose: boolean; +}; +export function AutoUpdaterWrapper(t0) { + const $ = _c(17); + const { + isUpdating, + onChangeIsUpdating, + onAutoUpdaterResult, + autoUpdaterResult, + showSuccessMessage, + verbose + } = t0; + const [useNativeInstaller, setUseNativeInstaller] = React.useState(null); + const [isPackageManager, setIsPackageManager] = React.useState(null); + let t1; + let t2; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t1 = () => { + const checkInstallation = async function checkInstallation() { + if (feature("SKIP_DETECTION_WHEN_AUTOUPDATES_DISABLED") && isAutoUpdaterDisabled()) { + logForDebugging("AutoUpdaterWrapper: Skipping detection, auto-updates disabled"); + return; + } + const installationType = await getCurrentInstallationType(); + logForDebugging(`AutoUpdaterWrapper: Installation type: ${installationType}`); + setUseNativeInstaller(installationType === "native"); + setIsPackageManager(installationType === "package-manager"); + }; + checkInstallation(); + }; + t2 = []; + $[0] = t1; + $[1] = t2; + } else { + t1 = $[0]; + t2 = $[1]; + } + React.useEffect(t1, t2); + if (useNativeInstaller === null || isPackageManager === null) { + return null; + } + if (isPackageManager) { + let t3; + if ($[2] !== autoUpdaterResult || $[3] !== isUpdating || $[4] !== onAutoUpdaterResult || $[5] !== onChangeIsUpdating || $[6] !== showSuccessMessage || $[7] !== verbose) { + t3 = ; + $[2] = autoUpdaterResult; + $[3] = isUpdating; + $[4] = onAutoUpdaterResult; + $[5] = onChangeIsUpdating; + $[6] = showSuccessMessage; + $[7] = verbose; + $[8] = t3; + } else { + t3 = $[8]; + } + return t3; + } + const Updater = useNativeInstaller ? NativeAutoUpdater : AutoUpdater; + let t3; + if ($[9] !== Updater || $[10] !== autoUpdaterResult || $[11] !== isUpdating || $[12] !== onAutoUpdaterResult || $[13] !== onChangeIsUpdating || $[14] !== showSuccessMessage || $[15] !== verbose) { + t3 = ; + $[9] = Updater; + $[10] = autoUpdaterResult; + $[11] = isUpdating; + $[12] = onAutoUpdaterResult; + $[13] = onChangeIsUpdating; + $[14] = showSuccessMessage; + $[15] = verbose; + $[16] = t3; + } else { + t3 = $[16]; + } + return t3; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJmZWF0dXJlIiwiUmVhY3QiLCJBdXRvVXBkYXRlclJlc3VsdCIsImlzQXV0b1VwZGF0ZXJEaXNhYmxlZCIsImxvZ0ZvckRlYnVnZ2luZyIsImdldEN1cnJlbnRJbnN0YWxsYXRpb25UeXBlIiwiQXV0b1VwZGF0ZXIiLCJOYXRpdmVBdXRvVXBkYXRlciIsIlBhY2thZ2VNYW5hZ2VyQXV0b1VwZGF0ZXIiLCJQcm9wcyIsImlzVXBkYXRpbmciLCJvbkNoYW5nZUlzVXBkYXRpbmciLCJvbkF1dG9VcGRhdGVyUmVzdWx0IiwiYXV0b1VwZGF0ZXJSZXN1bHQiLCJzaG93U3VjY2Vzc01lc3NhZ2UiLCJ2ZXJib3NlIiwiQXV0b1VwZGF0ZXJXcmFwcGVyIiwidDAiLCIkIiwiX2MiLCJ1c2VOYXRpdmVJbnN0YWxsZXIiLCJzZXRVc2VOYXRpdmVJbnN0YWxsZXIiLCJ1c2VTdGF0ZSIsImlzUGFja2FnZU1hbmFnZXIiLCJzZXRJc1BhY2thZ2VNYW5hZ2VyIiwidDEiLCJ0MiIsIlN5bWJvbCIsImZvciIsImNoZWNrSW5zdGFsbGF0aW9uIiwiaW5zdGFsbGF0aW9uVHlwZSIsInVzZUVmZmVjdCIsInQzIiwiVXBkYXRlciJdLCJzb3VyY2VzIjpbIkF1dG9VcGRhdGVyV3JhcHBlci50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgZmVhdHVyZSB9IGZyb20gJ2J1bjpidW5kbGUnXG5pbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB0eXBlIHsgQXV0b1VwZGF0ZXJSZXN1bHQgfSBmcm9tICcuLi91dGlscy9hdXRvVXBkYXRlci5qcydcbmltcG9ydCB7IGlzQXV0b1VwZGF0ZXJEaXNhYmxlZCB9IGZyb20gJy4uL3V0aWxzL2NvbmZpZy5qcydcbmltcG9ydCB7IGxvZ0ZvckRlYnVnZ2luZyB9IGZyb20gJy4uL3V0aWxzL2RlYnVnLmpzJ1xuaW1wb3J0IHsgZ2V0Q3VycmVudEluc3RhbGxhdGlvblR5cGUgfSBmcm9tICcuLi91dGlscy9kb2N0b3JEaWFnbm9zdGljLmpzJ1xuaW1wb3J0IHsgQXV0b1VwZGF0ZXIgfSBmcm9tICcuL0F1dG9VcGRhdGVyLmpzJ1xuaW1wb3J0IHsgTmF0aXZlQXV0b1VwZGF0ZXIgfSBmcm9tICcuL05hdGl2ZUF1dG9VcGRhdGVyLmpzJ1xuaW1wb3J0IHsgUGFja2FnZU1hbmFnZXJBdXRvVXBkYXRlciB9IGZyb20gJy4vUGFja2FnZU1hbmFnZXJBdXRvVXBkYXRlci5qcydcblxudHlwZSBQcm9wcyA9IHtcbiAgaXNVcGRhdGluZzogYm9vbGVhblxuICBvbkNoYW5nZUlzVXBkYXRpbmc6IChpc1VwZGF0aW5nOiBib29sZWFuKSA9PiB2b2lkXG4gIG9uQXV0b1VwZGF0ZXJSZXN1bHQ6IChhdXRvVXBkYXRlclJlc3VsdDogQXV0b1VwZGF0ZXJSZXN1bHQpID0+IHZvaWRcbiAgYXV0b1VwZGF0ZXJSZXN1bHQ6IEF1dG9VcGRhdGVyUmVzdWx0IHwgbnVsbFxuICBzaG93U3VjY2Vzc01lc3NhZ2U6IGJvb2xlYW5cbiAgdmVyYm9zZTogYm9vbGVhblxufVxuXG5leHBvcnQgZnVuY3Rpb24gQXV0b1VwZGF0ZXJXcmFwcGVyKHtcbiAgaXNVcGRhdGluZyxcbiAgb25DaGFuZ2VJc1VwZGF0aW5nLFxuICBvbkF1dG9VcGRhdGVyUmVzdWx0LFxuICBhdXRvVXBkYXRlclJlc3VsdCxcbiAgc2hvd1N1Y2Nlc3NNZXNzYWdlLFxuICB2ZXJib3NlLFxufTogUHJvcHMpOiBSZWFjdC5SZWFjdE5vZGUge1xuICBjb25zdCBbdXNlTmF0aXZlSW5zdGFsbGVyLCBzZXRVc2VOYXRpdmVJbnN0YWxsZXJdID0gUmVhY3QudXNlU3RhdGU8XG4gICAgYm9vbGVhbiB8IG51bGxcbiAgPihudWxsKVxuICBjb25zdCBbaXNQYWNrYWdlTWFuYWdlciwgc2V0SXNQYWNrYWdlTWFuYWdlcl0gPSBSZWFjdC51c2VTdGF0ZTxcbiAgICBib29sZWFuIHwgbnVsbFxuICA+KG51bGwpXG5cbiAgUmVhY3QudXNlRWZmZWN0KCgpID0+IHtcbiAgICBhc3luYyBmdW5jdGlvbiBjaGVja0luc3RhbGxhdGlvbigpIHtcbiAgICAgIC8vIFNraXAgaW5zdGFsbGF0aW9uIHR5cGUgZGV0ZWN0aW9uIGlmIGF1dG8tdXBkYXRlcyBhcmUgZGlzYWJsZWQgKGFudC1vbmx5KVxuICAgICAgLy8gVGhpcyBhdm9pZHMgcG90ZW50aWFsbHkgc2xvdyBwYWNrYWdlIG1hbmFnZXIgZGV0ZWN0aW9uIChzcGF3blN5bmMgY2FsbHMpXG4gICAgICBpZiAoXG4gICAgICAgIGZlYXR1cmUoJ1NLSVBfREVURUNUSU9OX1dIRU5fQVVUT1VQREFURVNfRElTQUJMRUQnKSAmJlxuICAgICAgICBpc0F1dG9VcGRhdGVyRGlzYWJsZWQoKVxuICAgICAgKSB7XG4gICAgICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgICAgICAnQXV0b1VwZGF0ZXJXcmFwcGVyOiBTa2lwcGluZyBkZXRlY3Rpb24sIGF1dG8tdXBkYXRlcyBkaXNhYmxlZCcsXG4gICAgICAgIClcbiAgICAgICAgcmV0dXJuXG4gICAgICB9XG5cbiAgICAgIGNvbnN0IGluc3RhbGxhdGlvblR5cGUgPSBhd2FpdCBnZXRDdXJyZW50SW5zdGFsbGF0aW9uVHlwZSgpXG4gICAgICBsb2dGb3JEZWJ1Z2dpbmcoXG4gICAgICAgIGBBdXRvVXBkYXRlcldyYXBwZXI6IEluc3RhbGxhdGlvbiB0eXBlOiAke2luc3RhbGxhdGlvblR5cGV9YCxcbiAgICAgIClcbiAgICAgIHNldFVzZU5hdGl2ZUluc3RhbGxlcihpbnN0YWxsYXRpb25UeXBlID09PSAnbmF0aXZlJylcbiAgICAgIHNldElzUGFja2FnZU1hbmFnZXIoaW5zdGFsbGF0aW9uVHlwZSA9PT0gJ3BhY2thZ2UtbWFuYWdlcicpXG4gICAgfVxuXG4gICAgdm9pZCBjaGVja0luc3RhbGxhdGlvbigpXG4gIH0sIFtdKVxuXG4gIC8vIERvbid0IHJlbmRlciB1bnRpbCB3ZSBrbm93IHRoZSBpbnN0YWxsYXRpb24gdHlwZVxuICBpZiAodXNlTmF0aXZlSW5zdGFsbGVyID09PSBudWxsIHx8IGlzUGFja2FnZU1hbmFnZXIgPT09IG51bGwpIHtcbiAgICByZXR1cm4gbnVsbFxuICB9XG5cbiAgaWYgKGlzUGFja2FnZU1hbmFnZXIpIHtcbiAgICByZXR1cm4gKFxuICAgICAgPFBhY2thZ2VNYW5hZ2VyQXV0b1VwZGF0ZXJcbiAgICAgICAgdmVyYm9zZT17dmVyYm9zZX1cbiAgICAgICAgb25BdXRvVXBkYXRlclJlc3VsdD17b25BdXRvVXBkYXRlclJlc3VsdH1cbiAgICAgICAgYXV0b1VwZGF0ZXJSZXN1bHQ9e2F1dG9VcGRhdGVyUmVzdWx0fVxuICAgICAgICBpc1VwZGF0aW5nPXtpc1VwZGF0aW5nfVxuICAgICAgICBvbkNoYW5nZUlzVXBkYXRpbmc9e29uQ2hhbmdlSXNVcGRhdGluZ31cbiAgICAgICAgc2hvd1N1Y2Nlc3NNZXNzYWdlPXtzaG93U3VjY2Vzc01lc3NhZ2V9XG4gICAgICAvPlxuICAgIClcbiAgfVxuXG4gIGNvbnN0IFVwZGF0ZXIgPSB1c2VOYXRpdmVJbnN0YWxsZXIgPyBOYXRpdmVBdXRvVXBkYXRlciA6IEF1dG9VcGRhdGVyXG5cbiAgcmV0dXJuIChcbiAgICA8VXBkYXRlclxuICAgICAgdmVyYm9zZT17dmVyYm9zZX1cbiAgICAgIG9uQXV0b1VwZGF0ZXJSZXN1bHQ9e29uQXV0b1VwZGF0ZXJSZXN1bHR9XG4gICAgICBhdXRvVXBkYXRlclJlc3VsdD17YXV0b1VwZGF0ZXJSZXN1bHR9XG4gICAgICBpc1VwZGF0aW5nPXtpc1VwZGF0aW5nfVxuICAgICAgb25DaGFuZ2VJc1VwZGF0aW5nPXtvbkNoYW5nZUlzVXBkYXRpbmd9XG4gICAgICBzaG93U3VjY2Vzc01lc3NhZ2U9e3Nob3dTdWNjZXNzTWVzc2FnZX1cbiAgICAvPlxuICApXG59XG4iXSwibWFwcGluZ3MiOiI7QUFBQSxTQUFTQSxPQUFPLFFBQVEsWUFBWTtBQUNwQyxPQUFPLEtBQUtDLEtBQUssTUFBTSxPQUFPO0FBQzlCLGNBQWNDLGlCQUFpQixRQUFRLHlCQUF5QjtBQUNoRSxTQUFTQyxxQkFBcUIsUUFBUSxvQkFBb0I7QUFDMUQs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 \ No newline at end of file diff --git a/src/components/AwsAuthStatusBox.tsx b/src/components/AwsAuthStatusBox.tsx new file mode 100644 index 0000000..9dd5849 --- /dev/null +++ b/src/components/AwsAuthStatusBox.tsx @@ -0,0 +1,82 @@ +import { c as _c } from "react/compiler-runtime"; +import React, { useEffect, useState } from 'react'; +import { Box, Link, Text } from '../ink.js'; +import { type AwsAuthStatus, AwsAuthStatusManager } from '../utils/awsAuthStatusManager.js'; +const URL_RE = /https?:\/\/\S+/; +export function AwsAuthStatusBox() { + const $ = _c(11); + let t0; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t0 = AwsAuthStatusManager.getInstance().getStatus(); + $[0] = t0; + } else { + t0 = $[0]; + } + const [status, setStatus] = useState(t0); + let t1; + let t2; + if ($[1] === Symbol.for("react.memo_cache_sentinel")) { + t1 = () => { + const unsubscribe = AwsAuthStatusManager.getInstance().subscribe(setStatus); + return unsubscribe; + }; + t2 = []; + $[1] = t1; + $[2] = t2; + } else { + t1 = $[1]; + t2 = $[2]; + } + useEffect(t1, t2); + if (!status.isAuthenticating && !status.error && status.output.length === 0) { + return null; + } + if (!status.isAuthenticating && !status.error) { + return null; + } + let t3; + if ($[3] === Symbol.for("react.memo_cache_sentinel")) { + t3 = Cloud Authentication; + $[3] = t3; + } else { + t3 = $[3]; + } + let t4; + if ($[4] !== status.output) { + t4 = status.output.length > 0 && {status.output.slice(-5).map(_temp)}; + $[4] = status.output; + $[5] = t4; + } else { + t4 = $[5]; + } + let t5; + if ($[6] !== status.error) { + t5 = status.error && {status.error}; + $[6] = status.error; + $[7] = t5; + } else { + t5 = $[7]; + } + let t6; + if ($[8] !== t4 || $[9] !== t5) { + t6 = {t3}{t4}{t5}; + $[8] = t4; + $[9] = t5; + $[10] = t6; + } else { + t6 = $[10]; + } + return t6; +} +function _temp(line, index) { + const m = line.match(URL_RE); + if (!m) { + return {line}; + } + const url = m[0]; + const start = m.index ?? 0; + const before = line.slice(0, start); + const after = line.slice(start + url.length); + return {before}{url}{after}; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsInVzZUVmZmVjdCIsInVzZVN0YXRlIiwiQm94IiwiTGluayIsIlRleHQiLCJBd3NBdXRoU3RhdHVzIiwiQXdzQXV0aFN0YXR1c01hbmFnZXIiLCJVUkxfUkUiLCJBd3NBdXRoU3RhdHVzQm94IiwiJCIsIl9jIiwidDAiLCJTeW1ib2wiLCJmb3IiLCJnZXRJbnN0YW5jZSIsImdldFN0YXR1cyIsInN0YXR1cyIsInNldFN0YXR1cyIsInQxIiwidDIiLCJ1bnN1YnNjcmliZSIsInN1YnNjcmliZSIsImlzQXV0aGVudGljYXRpbmciLCJlcnJvciIsIm91dHB1dCIsImxlbmd0aCIsInQzIiwidDQiLCJzbGljZSIsIm1hcCIsIl90ZW1wIiwidDUiLCJ0NiIsImxpbmUiLCJpbmRleCIsIm0iLCJtYXRjaCIsInVybCIsInN0YXJ0IiwiYmVmb3JlIiwiYWZ0ZXIiXSwic291cmNlcyI6WyJBd3NBdXRoU3RhdHVzQm94LnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgUmVhY3QsIHsgdXNlRWZmZWN0LCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgQm94LCBMaW5rLCBUZXh0IH0gZnJvbSAnLi4vaW5rLmpzJ1xuaW1wb3J0IHtcbiAgdHlwZSBBd3NBdXRoU3RhdHVzLFxuICBBd3NBdXRoU3RhdHVzTWFuYWdlcixcbn0gZnJvbSAnLi4vdXRpbHMvYXdzQXV0aFN0YXR1c01hbmFnZXIuanMnXG5cbmNvbnN0IFVSTF9SRSA9IC9odHRwcz86XFwvXFwvXFxTKy9cblxuZXhwb3J0IGZ1bmN0aW9uIEF3c0F1dGhTdGF0dXNCb3goKTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgY29uc3QgW3N0YXR1cywgc2V0U3RhdHVzXSA9IHVzZVN0YXRlPEF3c0F1dGhTdGF0dXM+KFxuICAgIEF3c0F1dGhTdGF0dXNNYW5hZ2VyLmdldEluc3RhbmNlKCkuZ2V0U3RhdHVzKCksXG4gIClcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIC8vIFN1YnNjcmliZSB0byBzdGF0dXMgdXBkYXRlc1xuICAgIGNvbnN0IHVuc3Vic2NyaWJlID0gQXdzQXV0aFN0YXR1c01hbmFnZXIuZ2V0SW5zdGFuY2UoKS5zdWJzY3JpYmUoc2V0U3RhdHVzKVxuICAgIHJldHVybiB1bnN1YnNjcmliZVxuICB9LCBbXSlcblxuICAvLyBEb24ndCBzaG93IGFueXRoaW5nIGlmIG5vdCBhdXRoZW50aWNhdGluZyBhbmQgbm8gZXJyb3JcbiAgaWYgKCFzdGF0dXMuaXNBdXRoZW50aWNhdGluZyAmJiAhc3RhdHVzLmVycm9yICYmIHN0YXR1cy5vdXRwdXQubGVuZ3RoID09PSAwKSB7XG4gICAgcmV0dXJuIG51bGxcbiAgfVxuXG4gIC8vIERvbid0IHNob3cgaWYgYXV0aGVudGljYXRpb24gc3VjY2VlZGVkIChubyBlcnJvciBhbmQgbm90IGF1dGhlbnRpY2F0aW5nKVxuICBpZiAoIXN0YXR1cy5pc0F1dGhlbnRpY2F0aW5nICYmICFzdGF0dXMuZXJyb3IpIHtcbiAgICByZXR1cm4gbnVsbFxuICB9XG5cbiAgcmV0dXJuIChcbiAgICA8Qm94XG4gICAgICBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCJcbiAgICAgIGJvcmRlclN0eWxlPVwicm91bmRcIlxuICAgICAgYm9yZGVyQ29sb3I9XCJwZXJtaXNzaW9uXCJcbiAgICAgIHBhZGRpbmdYPXsxfVxuICAgICAgbWFyZ2luWT17MX1cbiAgICA+XG4gICAgICA8VGV4dCBib2xkIGNvbG9yPVwicGVybWlzc2lvblwiPlxuICAgICAgICBDbG91ZCBBdXRoZW50aWNhdGlvblxuICAgICAgPC9UZXh0PlxuXG4gICAgICB7c3RhdHVzLm91dHB1dC5sZW5ndGggPiAwICYmIChcbiAgICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCIgbWFyZ2luVG9wPXsxfT5cbiAgICAgICAgICB7c3RhdHVzLm91dHB1dC5zbGljZSgtNSkubWFwKChsaW5lLCBpbmRleCkgPT4ge1xuICAgICAgICAgICAgY29uc3QgbSA9IGxpbmUubWF0Y2goVVJMX1JFKVxuICAgICAgICAgICAgaWYgKCFtKSB7XG4gICAgICAgICAgICAgIHJldHVybiAoXG4gICAgICAgICAgICAgICAgPFRleHQga2V5PXtpbmRleH0gZGltQ29sb3I+XG4gICAgICAgICAgICAgICAgICB7bGluZX1cbiAgICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICAgIClcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGNvbnN0IHVybCA9IG1bMF1cbiAgICAgICAgICAgIGNvbnN0IHN0YXJ0ID0gbS5pbmRleCA/PyAwXG4gICAgICAgICAgICBjb25zdCBiZWZvcmUgPSBsaW5lLnNsaWNlKDAsIHN0YXJ0KVxuICAgICAgICAgICAgY29uc3QgYWZ0ZXIgPSBsaW5lLnNsaWNlKHN0YXJ0ICsgdXJsLmxlbmd0aClcbiAgICAgICAgICAgIHJldHVybiAoXG4gICAgICAgICAgICAgIDxUZXh0IGtleT17aW5kZXh9IGRpbUNvbG9yPlxuICAgICAgICAgICAgICAgIHtiZWZvcmV9XG4gICAgICAgICAgICAgICAgPExpbmsgdXJsPXt1cmx9Pnt1cmx9PC9MaW5rPlxuICAgICAgICAgICAgICAgIHthZnRlcn1cbiAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgKVxuICAgICAgICAgIH0pfVxuICAgICAgICA8L0JveD5cbiAgICAgICl9XG5cbiAgICAgIHtzdGF0dXMuZXJyb3IgJiYgKFxuICAgICAgICA8Qm94IG1hcmdpblRvcD17MX0+XG4gICAgICAgICAgPFRleHQgY29sb3I9XCJlcnJvclwiPntzdGF0dXMuZXJyb3J9PC9UZXh0PlxuICAgICAgICA8L0JveD5cbiAgICAgICl9XG4gICAgPC9Cb3g+XG4gIClcbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU9BLEtBQUssSUFBSUMsU0FBUyxFQUFFQyxRQUFRLFFBQVEsT0FBTztBQUNsRCxTQUFTQyxHQUFHLEVBQUVDLElBQUksRUFBRUMsSUFBSSxRQUFRLFdBQVc7QUFDM0MsU0FDRSxLQUFLQyxhQUFhLEVBQ2xCQyxvQkFBb0Is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 \ No newline at end of file diff --git a/src/components/BaseTextInput.tsx b/src/components/BaseTextInput.tsx new file mode 100644 index 0000000..8edc605 --- /dev/null +++ b/src/components/BaseTextInput.tsx @@ -0,0 +1,136 @@ +import { c as _c } from "react/compiler-runtime"; +import React from 'react'; +import { renderPlaceholder } from '../hooks/renderPlaceholder.js'; +import { usePasteHandler } from '../hooks/usePasteHandler.js'; +import { useDeclaredCursor } from '../ink/hooks/use-declared-cursor.js'; +import { Ansi, Box, Text, useInput } from '../ink.js'; +import type { BaseInputState, BaseTextInputProps } from '../types/textInputTypes.js'; +import type { TextHighlight } from '../utils/textHighlighting.js'; +import { HighlightedInput } from './PromptInput/ShimmeredInput.js'; +type BaseTextInputComponentProps = BaseTextInputProps & { + inputState: BaseInputState; + children?: React.ReactNode; + terminalFocus: boolean; + highlights?: TextHighlight[]; + invert?: (text: string) => string; + hidePlaceholderText?: boolean; +}; + +/** + * A base component for text inputs that handles rendering and basic input + */ +export function BaseTextInput(t0) { + const $ = _c(14); + const { + inputState, + children, + terminalFocus, + invert, + hidePlaceholderText, + ...props + } = t0; + const { + onInput, + renderedValue, + cursorLine, + cursorColumn + } = inputState; + const t1 = Boolean(props.focus && props.showCursor && terminalFocus); + let t2; + if ($[0] !== cursorColumn || $[1] !== cursorLine || $[2] !== t1) { + t2 = { + line: cursorLine, + column: cursorColumn, + active: t1 + }; + $[0] = cursorColumn; + $[1] = cursorLine; + $[2] = t1; + $[3] = t2; + } else { + t2 = $[3]; + } + const cursorRef = useDeclaredCursor(t2); + const { + wrappedOnInput, + isPasting: t3 + } = usePasteHandler({ + onPaste: props.onPaste, + onInput: (input, key) => { + if (isPasting && key.return) { + return; + } + onInput(input, key); + }, + onImagePaste: props.onImagePaste + }); + const isPasting = t3; + const { + onIsPastingChange + } = props; + React.useEffect(() => { + if (onIsPastingChange) { + onIsPastingChange(isPasting); + } + }, [isPasting, onIsPastingChange]); + const { + showPlaceholder, + renderedPlaceholder + } = renderPlaceholder({ + placeholder: props.placeholder, + value: props.value, + showCursor: props.showCursor, + focus: props.focus, + terminalFocus, + invert, + hidePlaceholderText + }); + useInput(wrappedOnInput, { + isActive: props.focus + }); + const commandWithoutArgs = props.value && props.value.trim().indexOf(" ") === -1 || props.value && props.value.endsWith(" "); + const showArgumentHint = Boolean(props.argumentHint && props.value && commandWithoutArgs && props.value.startsWith("/")); + const cursorFiltered = props.showCursor && props.highlights ? props.highlights.filter(h => h.dimColor || props.cursorOffset < h.start || props.cursorOffset >= h.end) : props.highlights; + const { + viewportCharOffset, + viewportCharEnd + } = inputState; + const filteredHighlights = cursorFiltered && viewportCharOffset > 0 ? cursorFiltered.filter(h_0 => h_0.end > viewportCharOffset && h_0.start < viewportCharEnd).map(h_1 => ({ + ...h_1, + start: Math.max(0, h_1.start - viewportCharOffset), + end: h_1.end - viewportCharOffset + })) : cursorFiltered; + const hasHighlights = filteredHighlights && filteredHighlights.length > 0; + if (hasHighlights) { + return {showArgumentHint && {props.value?.endsWith(" ") ? "" : " "}{props.argumentHint}}{children}; + } + const T0 = Box; + const T1 = Text; + const t4 = "truncate-end"; + const t5 = showPlaceholder && props.placeholderElement ? props.placeholderElement : showPlaceholder && renderedPlaceholder ? {renderedPlaceholder} : {renderedValue}; + const t6 = showArgumentHint && {props.value?.endsWith(" ") ? "" : " "}{props.argumentHint}; + let t7; + if ($[4] !== T1 || $[5] !== children || $[6] !== props || $[7] !== t5 || $[8] !== t6) { + t7 = {t5}{t6}{children}; + $[4] = T1; + $[5] = children; + $[6] = props; + $[7] = t5; + $[8] = t6; + $[9] = t7; + } else { + t7 = $[9]; + } + let t8; + if ($[10] !== T0 || $[11] !== cursorRef || $[12] !== t7) { + t8 = {t7}; + $[10] = T0; + $[11] = cursorRef; + $[12] = t7; + $[13] = t8; + } else { + t8 = $[13]; + } + return t8; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsInJlbmRlclBsYWNlaG9sZGVyIiwidXNlUGFzdGVIYW5kbGVyIiwidXNlRGVjbGFyZWRDdXJzb3IiLCJBbnNpIiwiQm94IiwiVGV4dCIsInVzZUlucHV0IiwiQmFzZUlucHV0U3RhdGUiLCJCYXNlVGV4dElucHV0UHJvcHMiLCJUZXh0SGlnaGxpZ2h0IiwiSGlnaGxpZ2h0ZWRJbnB1dCIsIkJhc2VUZXh0SW5wdXRDb21wb25lbnRQcm9wcyIsImlucHV0U3RhdGUiLCJjaGlsZHJlbiIsIlJlYWN0Tm9kZSIsInRlcm1pbmFsRm9jdXMiLCJoaWdobGlnaHRzIiwiaW52ZXJ0IiwidGV4dCIsImhpZGVQbGFjZWhvbGRlclRleHQiLCJCYXNlVGV4dElucHV0IiwidDAiLCIkIiwiX2MiLCJwcm9wcyIsIm9uSW5wdXQiLCJyZW5kZXJlZFZhbHVlIiwiY3Vyc29yTGluZSIsImN1cnNvckNvbHVtbiIsInQxIiwiQm9vbGVhbiIsImZvY3VzIiwic2hvd0N1cnNvciIsInQyIiwibGluZSIsImNvbHVtbiIsImFjdGl2ZSIsImN1cnNvclJlZiIsIndyYXBwZWRPbklucHV0IiwiaXNQYXN0aW5nIiwidDMiLCJvblBhc3RlIiwiaW5wdXQiLCJrZXkiLCJyZXR1cm4iLCJvbkltYWdlUGFzdGUiLCJvbklzUGFzdGluZ0NoYW5nZSIsInVzZUVmZmVjdCIsInNob3dQbGFjZWhvbGRlciIsInJlbmRlcmVkUGxhY2Vob2xkZXIiLCJwbGFjZWhvbGRlciIsInZhbHVlIiwiaXNBY3RpdmUiLCJjb21tYW5kV2l0aG91dEFyZ3MiLCJ0cmltIiwiaW5kZXhPZiIsImVuZHNXaXRoIiwic2hvd0FyZ3VtZW50SGludCIsImFyZ3VtZW50SGludCIsInN0YXJ0c1dpdGgiLCJjdXJzb3JGaWx0ZXJlZCIsImZpbHRlciIsImgiLCJkaW1Db2xvciIsImN1cnNvck9mZnNldCIsInN0YXJ0IiwiZW5kIiwidmlld3BvcnRDaGFyT2Zmc2V0Iiwidmlld3BvcnRDaGFyRW5kIiwiZmlsdGVyZWRIaWdobGlnaHRzIiwiaF8wIiwibWFwIiwiaF8xIiwiTWF0aCIsIm1heCIsImhhc0hpZ2hsaWdodHMiLCJsZW5ndGgiLCJUMCIsIlQxIiwidDQiLCJ0NSIsInBsYWNlaG9sZGVyRWxlbWVudCIsInQ2IiwidDciLCJ0OCJdLCJzb3VyY2VzIjpbIkJhc2VUZXh0SW5wdXQudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHJlbmRlclBsYWNlaG9sZGVyIH0gZnJvbSAnLi4vaG9va3MvcmVuZGVyUGxhY2Vob2xkZXIuanMnXG5pbXBvcnQgeyB1c2VQYXN0ZUhhbmRsZXIgfSBmcm9tICcuLi9ob29rcy91c2VQYXN0ZUhhbmRsZXIuanMnXG5pbXBvcnQgeyB1c2VEZWNsYXJlZEN1cnNvciB9IGZyb20gJy4uL2luay9ob29rcy91c2UtZGVjbGFyZWQtY3Vyc29yLmpzJ1xuaW1wb3J0IHsgQW5zaSwgQm94LCBUZXh0LCB1c2VJbnB1dCB9IGZyb20gJy4uL2luay5qcydcbmltcG9ydCB0eXBlIHtcbiAgQmFzZUlucHV0U3RhdGUsXG4gIEJhc2VUZXh0SW5wdXRQcm9wcyxcbn0gZnJvbSAnLi4vdHlwZXMvdGV4dElucHV0VHlwZXMuanMnXG5pbXBvcnQgdHlwZSB7IFRleHRIaWdobGlnaHQgfSBmcm9tICcuLi91dGlscy90ZXh0SGlnaGxpZ2h0aW5nLmpzJ1xuaW1wb3J0IHsgSGlnaGxpZ2h0ZWRJbnB1dCB9IGZyb20gJy4vUHJvbXB0SW5wdXQvU2hpbW1lcmVkSW5wdXQuanMnXG5cbnR5cGUgQmFzZVRleHRJbnB1dENvbXBvbmVudFByb3BzID0gQmFzZVRleHRJbnB1dFByb3BzICYge1xuICBpbnB1dFN0YXRlOiBCYXNlSW5wdXRTdGF0ZVxuICBjaGlsZHJlbj86IFJlYWN0LlJlYWN0Tm9kZVxuICB0ZXJtaW5hbEZvY3VzOiBib29sZWFuXG4gIGhpZ2hsaWdodHM/OiBUZXh0SGlnaGxpZ2h0W11cbiAgaW52ZXJ0PzogKHRleHQ6IHN0cmluZykgPT4gc3RyaW5nXG4gIGhpZGVQbGFjZWhvbGRlclRleHQ/OiBib29sZWFuXG59XG5cbi8qKlxuICogQSBiYXNlIGNvbXBvbmVudCBmb3IgdGV4dCBpbnB1dHMgdGhhdCBoYW5kbGVzIHJlbmRlcmluZyBhbmQgYmFzaWMgaW5wdXRcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIEJhc2VUZXh0SW5wdXQoe1xuICBpbnB1dFN0YXRlLFxuICBjaGlsZHJlbixcbiAgdGVybWluYWxGb2N1cyxcbiAgaW52ZXJ0LFxuICBoaWRlUGxhY2Vob2xkZXJUZXh0LFxuICAuLi5wcm9wc1xufTogQmFzZVRleHRJbnB1dENvbXBvbmVudFByb3BzKTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgY29uc3QgeyBvbklucHV0LCByZW5kZXJlZFZhbHVlLCBjdXJzb3JMaW5lLCBjdXJzb3JDb2x1bW4gfSA9IGlucHV0U3RhdGVcblxuICAvLyBQYXJrIHRoZSBuYXRpdmUgdGVybWluYWwgY3Vyc29yIGF0IHRoZSBpbnB1dCBjYXJldC4gVGVybWluYWwgZW11bGF0b3JzXG4gIC8vIHBvc2l0aW9uIElNRSBwcmVlZGl0IHRleHQgYXQgdGhlIHBoeXNpY2FsIGN1cnNvciwgYW5kIHNjcmVlbiByZWFkZXJzIC9cbiAgLy8gc2NyZWVuIG1hZ25pZmllcnMgdHJhY2sgaXQg4oCUIHNvIHBhcmtpbmcgaGVyZSBtYWtlcyBDSksgaW5wdXQgYXBwZWFyXG4gIC8vIGlubGluZSBhbmQgbGV0cyBhY2Nlc3NpYmlsaXR5IHRvb2xzIGZvbGxvdyB0aGUgaW5wdXQuIFRoZSBCb3ggcmVmIGJlbG93XG4gIC8vIGlzIHRoZSB5b2dhIGxheW91dCBvcmlnaW47IChjdXJzb3JMaW5lLCBjdXJzb3JDb2x1bW4pIGlzIHJlbGF0aXZlIHRvIGl0LlxuICAvLyBPbmx5IGFjdGl2ZSB3aGVuIHRoZSBpbnB1dCBpcyBmb2N1c2VkLCBzaG93aW5nIGl0cyBjdXJzb3IsIGFuZCB0aGVcbiAgLy8gdGVybWluYWwgaXRzZWxmIGhhcyBmb2N1cy5cbiAgY29uc3QgY3Vyc29yUmVmID0gdXNlRGVjbGFyZWRDdXJzb3Ioe1xuICAgIGxpbmU6IGN1cnNvckxpbmUsXG4gICAgY29sdW1uOiBjdXJzb3JDb2x1bW4sXG4gICAgYWN0aXZlOiBCb29sZWFuKHByb3BzLmZvY3VzICYmIHByb3BzLnNob3dDdXJzb3IgJiYgdGVybWluYWxGb2N1cyksXG4gIH0pXG5cbiAgY29uc3QgeyB3cmFwcGVkT25JbnB1dCwgaXNQYXN0aW5nIH0gPSB1c2VQYXN0ZUhhbmRsZXIoe1xuICAgIG9uUGFzdGU6IHByb3BzLm9uUGFzdGUsXG4gICAgb25JbnB1dDogKGlucHV0LCBrZXkpID0+IHtcbiAgICAgIC8vIFByZXZlbnQgRW50ZXIga2V5IGZyb20gdHJpZ2dlcmluZyBzdWJtaXNzaW9uIGR1cmluZyBwYXN0ZVxuICAgICAgaWYgKGlzUGFzdGluZyAmJiBrZXkucmV0dXJuKSB7XG4gICAgICAgIHJldHVyblxuICAgICAgfVxuICAgICAgb25JbnB1dChpbnB1dCwga2V5KVxuICAgIH0sXG4gICAgb25JbWFnZVBhc3RlOiBwcm9wcy5vbkltYWdlUGFzdGUsXG4gIH0pXG5cbiAgLy8gTm90aWZ5IHBhcmVudCB3aGVuIHBhc3RlIHN0YXRlIGNoYW5nZXNcbiAgY29uc3QgeyBvbklzUGFzdGluZ0NoYW5nZSB9ID0gcHJvcHNcbiAgUmVhY3QudXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAob25Jc1Bhc3RpbmdDaGFuZ2UpIHtcbiAgICAgIG9uSXNQYXN0aW5nQ2hhbmdlKGlzUGFzdGluZylcbiAgICB9XG4gIH0sIFtpc1Bhc3RpbmcsIG9uSXNQYXN0aW5nQ2hhbmdlXSlcblxuICBjb25zdCB7IHNob3dQbGFjZWhvbGRlciwgcmVuZGVyZWRQbGFjZWhvbGRlciB9ID0gcmVuZGVyUGxhY2Vob2xkZXIoe1xuICAgIHBsYWNlaG9sZGVyOiBwcm9wcy5wbGFjZWhvbGRlcixcbiAgICB2YWx1ZTogcHJvcHMudmFsdWUsXG4gICAgc2hvd0N1cnNvcjogcHJvcHMuc2hvd0N1cnNvcixcbiAgICBmb2N1czogcHJvcHMuZm9jdXMsXG4gICAgdGVybWluYWxGb2N1cyxcbiAgICBpbnZlcnQsXG4gICAgaGlkZVBsYWNlaG9sZGVyVGV4dCxcbiAgfSlcblxuICB1c2VJbnB1dCh3cmFwcGVkT25JbnB1dCwgeyBpc0FjdGl2ZTogcHJvcHMuZm9jdXMgfSlcblxuICAvLyBTaG93IGFyZ3VtZW50IGhpbnQgb25seSB3aGVuIHdlIGhhdmUgYSB2YWx1ZSBhbmQgdGhlIGhpbnQgaXMgcHJvdmlkZWRcbiAgLy8gT25seSBzaG93IHRoZSBhcmd1bWVudCBoaW50IHdoZW46XG4gIC8vIDEuIFdlIGhhdmUgYSBoaW50IHRvIHNob3dcbiAgLy8gMi4gV2UgaGF2ZSBhIGNvbW1hbmQgdHlwZWQgKHZhbHVlIGlzIG5vdCBlbXB0eSlcbiAgLy8gMy4gVGhlIGNvbW1hbmQgZG9lc24ndCBoYXZlIGFyZ3VtZW50cyB5ZXQgKG5vIHRleHQgYWZ0ZXIgdGhlIHNwYWNlKVxuICAvLyA0LiBXZSdyZSBhY3R1YWxseSB0eXBpbmcgYSBjb21tYW5kICh0aGUgdmFsdWUgc3RhcnRzIHdpdGggLylcbiAgY29uc3QgY29tbWFuZFdpdGhvdXRBcmdzID1cbiAgICAocHJvcHMudmFsdWUgJiYgcHJvcHMudmFsdWUudHJpbSgpLmluZGV4T2YoJyAnKSA9PT0gLTEpIHx8XG4gICAgKHByb3BzLnZhbHVlICYmIHByb3BzLnZhbHVlLmVuZHNXaXRoKCcgJykpXG5cbiAgY29uc3Qgc2hvd0FyZ3VtZW50SGludCA9IEJvb2xlYW4oXG4gICAgcHJvcHMuYXJndW1lbnRIaW50ICYmXG4gICAgICBwcm9wcy52YWx1ZSAmJlxuICAgICAgY29tbWFuZFdpdGhvdXRBcmdzICYmXG4gICAgICBwcm9wcy52YWx1ZS5zdGFydHNXaXRoKCcvJyksXG4gIClcblxuICAvLyBGaWx0ZXIgb3V0IGhpZ2hsaWdodHMgdGhhdCBjb250YWluIHRoZSBjdXJzb3IgcG9zaXRpb25cbiAgY29uc3QgY3Vyc29yRmlsdGVyZWQgPVxuICAgIHByb3BzLnNob3dDdXJzb3IgJiYgcHJvcHMuaGlnaGxpZ2h0c1xuICAgICAgPyBwcm9wcy5oaWdobGlnaHRzLmZpbHRlcihcbiAgICAgICAgICBoID0+XG4gICAgICAgICAgICBoLmRpbUNvbG9yIHx8XG4gICAgICAgICAgICBwcm9wcy5jdXJzb3JPZmZzZXQgPCBoLnN0YXJ0IHx8XG4gICAgICAgICAgICBwcm9wcy5jdXJzb3JPZmZzZXQgPj0gaC5lbmQsXG4gICAgICAgIClcbiAgICAgIDogcHJvcHMuaGlnaGxpZ2h0c1xuXG4gIC8vIEFkanVzdCBoaWdobGlnaHRzIGZvciB2aWV3cG9ydCB3aW5kb3dpbmc6IGhpZ2hsaWdodCBwb3NpdGlvbnMgcmVmZXJlbmNlIHRoZVxuICAvLyBmdWxsIGlucHV0IHRleHQsIGJ1dCByZW5kZXJlZFZhbHVlIG9ubHkgY29udGFpbnMgdGhlIHdpbmRvd2VkIHN1YnNldC5cbiAgY29uc3QgeyB2aWV3cG9ydENoYXJPZmZzZXQsIHZpZXdwb3J0Q2hhckVuZCB9ID0gaW5wdXRTdGF0ZVxuICBjb25zdCBmaWx0ZXJlZEhpZ2hsaWdodHMgPVxuICAgIGN1cnNvckZpbHRlcmVkICYmIHZpZXdwb3J0Q2hhck9mZnNldCA+IDBcbiAgICAgID8gY3Vyc29yRmlsdGVyZWRcbiAgICAgICAgICAuZmlsdGVyKGggPT4gaC5lbmQgPiB2aWV3cG9ydENoYXJPZmZzZXQgJiYgaC5zdGFydCA8IHZpZXdwb3J0Q2hhckVuZClcbiAgICAgICAgICAubWFwKGggPT4gKHtcbiAgICAgICAgICAgIC4uLmgsXG4gICAgICAgICAgICBzdGFydDogTWF0aC5tYXgoMCwgaC5zdGFydCAtIHZpZXdwb3J0Q2hhck9mZnNldCksXG4gICAgICAgICAgICBlbmQ6IGguZW5kIC0gdmlld3BvcnRDaGFyT2Zmc2V0LFxuICAgICAgICAgIH0pKVxuICAgICAgOiBjdXJzb3JGaWx0ZXJlZFxuXG4gIGNvbnN0IGhhc0hpZ2hsaWdodHMgPSBmaWx0ZXJlZEhpZ2hsaWdodHMgJiYgZmlsdGVyZWRIaWdobGlnaHRzLmxlbmd0aCA+IDBcblxuICBpZiAoaGFzSGlnaGxpZ2h0cykge1xuICAgIHJldHVybiAoXG4gICAgICA8Qm94IHJlZj17Y3Vyc29yUmVmfT5cbiAgICAgICAgPEhpZ2hsaWdodGVkSW5wdXRcbiAgICAgICAgICB0ZXh0PXtyZW5kZXJlZFZhbHVlfVxuICAgICAgICAgIGhpZ2hsaWdodHM9e2ZpbHRlcmVkSGlnaGxpZ2h0c31cbiAgICAgICAgLz5cbiAgICAgICAge3Nob3dBcmd1bWVudEhpbnQgJiYgKFxuICAgICAgICAgIDxUZXh0IGRpbUNvbG9yPlxuICAgICAgICAgICAge3Byb3BzLnZhbHVlPy5lbmRzV2l0aCgnICcpID8gJycgOiAnICd9XG4gICAgICAgICAgICB7cHJvcHMuYXJndW1lbnRIaW50fVxuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgKX1cbiAgICAgICAge2NoaWxkcmVufVxuICAgICAgPC9Cb3g+XG4gICAgKVxuICB9XG5cbiAgcmV0dXJuIChcbiAgICA8Qm94IHJlZj17Y3Vyc29yUmVmfT5cbiAgICAgIDxUZXh0IHdyYXA9XCJ0cnVuY2F0ZS1lbmRcIiBkaW1Db2xvcj17cHJvcHMuZGltQ29sb3J9PlxuICAgICAgICB7c2hvd1BsYWNlaG9sZGVyICYmIHByb3BzLnBsYWNlaG9sZGVyRWxlbWVudCA/IChcbiAgICAgICAgICBwcm9wcy5wbGFjZWhvbGRlckVsZW1lbnRcbiAgICAgICAgKSA6IHNob3dQbGFjZWhvbGRlciAmJiByZW5kZXJlZFBsYWNlaG9sZGVyID8gKFxuICAgICAgICAgIDxBbnNpPntyZW5kZXJlZFBsYWNlaG9sZGVyfTwvQW5zaT5cbiAgICAgICAgKSA6IChcbiAgICAgICAgICA8QW5zaT57cmVuZGVyZWRWYWx1ZX08L0Fuc2k+XG4gICAgICAgICl9XG4gICAgICAgIHtzaG93QXJndW1lbnRIaW50ICYmIChcbiAgICAgICAgICA8VGV4dCBkaW1Db2xvcj5cbiAgICAgICAgICAgIHtwcm9wcy52YWx1ZT8uZW5kc1dpdGgoJyAnKSA/ICcnIDogJyAnfVxuICAgICAgICAgICAge3Byb3BzLmFyZ3VtZW50SGludH1cbiAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICl9XG4gICAgICAgIHtjaGlsZHJlbn1cbiAgICAgIDwvVGV4dD5cbiAgICA8L0JveD5cbiAgKVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsT0FBT0EsS0FBSyxNQUFNLE9BQU87QUFDekIsU0FBU0MsaUJBQWlCLFFBQVEsK0JBQStCO0FBQ2pFLFNBQVNDLGVBQWUsUUFBUSw2QkFBNkI7QUFDN0QsU0FBU0MsaUJBQWlCLFFBQVEscUNBQXFDO0FBQ3ZFLFNBQVNDLElBQUksRUFBRUMsR0FBRyxFQUFFQyxJQUFJLEVBQUVDLFFBQVEsUUFBUSxXQUFXO0FBQ3JELGNBQ0VDLGNBQWMsRUFDZEMsa0JBQWtCLFFBQ2IsNEJBQTRCO0FBQ25DLGNBQWNDLGFBQWEsUUFBUSw4QkFBOEI7QUFDakUsU0FBU0MsZ0JBQWdCLFFBQVEsaUNBQWlDO0FBRWxFLEtBQUtDLDJCQUEyQixHQUFHSCxrQkFBa0IsR0FBRztFQUN0REksVUFBVSxFQUFFTCxjQUFjO0VBQzFCTSxRQUFRLENBQUMsRUFBRWQsS0FBSyxDQUFDZSxTQUFTO0VBQzFCQyxhQUFhLEVBQUUsT0FBTztFQUN0QkMsVUFBVSxDQUFDLEVBQUVQLGFBQWEsRUFBRTtFQUM1QlEsTUFBTSxDQUFDLEVBQUUsQ0FBQ0MsSUFBSSxFQUFFLE1BQU0sRUFBRSxHQUFHLE1BQU07RUFDakNDLG1CQUFtQixDQUFDLEVBQUUsT0FBTztBQUMvQixDQUFDOztBQUVEO0FBQ0E7QUFDQTtBQUNBLE9BQU8sU0FBQUMsY0FBQUMsRUFBQTtFQUFBLE1BQUFDLENBQUEsR0FBQUMsRUFBQTtFQUF1QjtJQUFBWCxVQUFBO0lBQUFDLFFBQUE7SUFBQUUsYUFBQTtJQUFBRSxNQUFBO0lBQUFFLG1CQUFBO0lBQUEsR0FBQUs7RUFBQSxJQUFBSCxFQU9BO0VBQzVCO0lBQUFJLE9BQUE7SUFBQUMsYUFBQTtJQUFBQyxVQUFBO0lBQUFDO0VBQUEsSUFBNkRoQixVQUFVO0VBWTdELE1BQUFpQixFQUFBLEdBQUFDLE9BQU8sQ0FBQ04sS0FBSyxDQUFBTyxLQUEwQixJQUFoQlAs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 \ No newline at end of file diff --git a/src/components/BashModeProgress.tsx b/src/components/BashModeProgress.tsx new file mode 100644 index 0000000..dbbce04 --- /dev/null +++ b/src/components/BashModeProgress.tsx @@ -0,0 +1,56 @@ +import { c as _c } from "react/compiler-runtime"; +import React from 'react'; +import { Box } from '../ink.js'; +import { BashTool } from '../tools/BashTool/BashTool.js'; +import type { ShellProgress } from '../types/tools.js'; +import { UserBashInputMessage } from './messages/UserBashInputMessage.js'; +import { ShellProgressMessage } from './shell/ShellProgressMessage.js'; +type Props = { + input: string; + progress: ShellProgress | null; + verbose: boolean; +}; +export function BashModeProgress(t0) { + const $ = _c(8); + const { + input, + progress, + verbose + } = t0; + const t1 = `${input}`; + let t2; + if ($[0] !== t1) { + t2 = ; + $[0] = t1; + $[1] = t2; + } else { + t2 = $[1]; + } + let t3; + if ($[2] !== progress || $[3] !== verbose) { + t3 = progress ? : BashTool.renderToolUseProgressMessage?.([], { + verbose, + tools: [], + terminalSize: undefined + }); + $[2] = progress; + $[3] = verbose; + $[4] = t3; + } else { + t3 = $[4]; + } + let t4; + if ($[5] !== t2 || $[6] !== t3) { + t4 = {t2}{t3}; + $[5] = t2; + $[6] = t3; + $[7] = t4; + } else { + t4 = $[7]; + } + return t4; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkJveCIsIkJhc2hUb29sIiwiU2hlbGxQcm9ncmVzcyIsIlVzZXJCYXNoSW5wdXRNZXNzYWdlIiwiU2hlbGxQcm9ncmVzc01lc3NhZ2UiLCJQcm9wcyIsImlucHV0IiwicHJvZ3Jlc3MiLCJ2ZXJib3NlIiwiQmFzaE1vZGVQcm9ncmVzcyIsInQwIiwiJCIsIl9jIiwidDEiLCJ0MiIsInRleHQiLCJ0eXBlIiwidDMiLCJmdWxsT3V0cHV0Iiwib3V0cHV0IiwiZWxhcHNlZFRpbWVTZWNvbmRzIiwidG90YWxMaW5lcyIsInJlbmRlclRvb2xVc2VQcm9ncmVzc01lc3NhZ2UiLCJ0b29scyIsInRlcm1pbmFsU2l6ZSIsInVuZGVmaW5lZCIsInQ0Il0sInNvdXJjZXMiOlsiQmFzaE1vZGVQcm9ncmVzcy50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgQm94IH0gZnJvbSAnLi4vaW5rLmpzJ1xuaW1wb3J0IHsgQmFzaFRvb2wgfSBmcm9tICcuLi90b29scy9CYXNoVG9vbC9CYXNoVG9vbC5qcydcbmltcG9ydCB0eXBlIHsgU2hlbGxQcm9ncmVzcyB9IGZyb20gJy4uL3R5cGVzL3Rvb2xzLmpzJ1xuaW1wb3J0IHsgVXNlckJhc2hJbnB1dE1lc3NhZ2UgfSBmcm9tICcuL21lc3NhZ2VzL1VzZXJCYXNoSW5wdXRNZXNzYWdlLmpzJ1xuaW1wb3J0IHsgU2hlbGxQcm9ncmVzc01lc3NhZ2UgfSBmcm9tICcuL3NoZWxsL1NoZWxsUHJvZ3Jlc3NNZXNzYWdlLmpzJ1xuXG50eXBlIFByb3BzID0ge1xuICBpbnB1dDogc3RyaW5nXG4gIHByb2dyZXNzOiBTaGVsbFByb2dyZXNzIHwgbnVsbFxuICB2ZXJib3NlOiBib29sZWFuXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBCYXNoTW9kZVByb2dyZXNzKHtcbiAgaW5wdXQsXG4gIHByb2dyZXNzLFxuICB2ZXJib3NlLFxufTogUHJvcHMpOiBSZWFjdC5SZWFjdE5vZGUge1xuICByZXR1cm4gKFxuICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiIG1hcmdpblRvcD17MX0+XG4gICAgICA8VXNlckJhc2hJbnB1dE1lc3NhZ2VcbiAgICAgICAgYWRkTWFyZ2luPXtmYWxzZX1cbiAgICAgICAgcGFyYW09e3sgdGV4dDogYDxiYXNoLWlucHV0PiR7aW5wdXR9PC9iYXNoLWlucHV0PmAsIHR5cGU6ICd0ZXh0JyB9fVxuICAgICAgLz5cbiAgICAgIHtwcm9ncmVzcyA/IChcbiAgICAgICAgPFNoZWxsUHJvZ3Jlc3NNZXNzYWdlXG4gICAgICAgICAgZnVsbE91dHB1dD17cHJvZ3Jlc3MuZnVsbE91dHB1dH1cbiAgICAgICAgICBvdXRwdXQ9e3Byb2dyZXNzLm91dHB1dH1cbiAgICAgICAgICBlbGFwc2VkVGltZVNlY29uZHM9e3Byb2dyZXNzLmVsYXBzZWRUaW1lU2Vjb25kc31cbiAgICAgICAgICB0b3RhbExpbmVzPXtwcm9ncmVzcy50b3RhbExpbmVzfVxuICAgICAgICAgIHZlcmJvc2U9e3ZlcmJvc2V9XG4gICAgICAgIC8+XG4gICAgICApIDogKFxuICAgICAgICBCYXNoVG9vbC5yZW5kZXJUb29sVXNlUHJvZ3Jlc3NNZXNzYWdlPy4oW10sIHtcbiAgICAgICAgICB2ZXJib3NlLFxuICAgICAgICAgIHRvb2xzOiBbXSxcbiAgICAgICAgICB0ZXJtaW5hbFNpemU6IHVuZGVmaW5lZCxcbiAgICAgICAgfSlcbiAgICAgICl9XG4gICAgPC9Cb3g+XG4gIClcbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU9BLEtBQUssTUFBTSxPQUFPO0FBQ3pCLFNBQVNDLEdBQUcsUUFBUSxXQUFXO0FBQy9CLFNBQVNDLFFBQVEsUUFBUSwrQkFBK0I7QUFDeEQsY0FBY0MsYUFBYSxRQUFRLG1CQUFtQjtBQUN0RCxTQUFTQyxvQkFBb0IsUUFBUSxvQ0FBb0M7QUFDekUsU0FBU0Msb0JBQW9CLFFBQVEsaUNBQWlDO0FBRXRFLEtBQUtDLEtBQUssR0FBRztFQUNYQyxLQUFLLEVBQUUsTUFBTTtFQUNiQyxRQUFRLEVBQUVMLGFBQWEsR0FBRyxJQUFJO0VBQzlCTSxPQUFPLEVBQUUsT0FBTztBQUNsQixDQUFDO0FBRUQsT0FBTyxTQUFBQyxpQkFBQUMsRUFBQTtFQUFBLE1BQUFDLENBQUEsR0FBQUMsRUFBQTtFQUEwQjtJQUFBTixLQUFBO0lBQUFDLFFBQUE7SUFBQUM7RUFBQSxJQUFBRSxFQUl6QjtFQUtlLE1BQUFHLEVBQUEsa0JBQWVQLEtBQUssZUFBZTtFQUFBLElBQUFRLEVBQUE7RUFBQSxJQUFBSCxDQUFBLFFBQUFFLEVBQUE7SUFGcERDLEVBQUEsSUFBQyxvQkFBb0IsQ0FDUixTQUFLLENBQUwsTUFBSSxDQUFDLENBQ1QsS0FBMkQsQ0FBM0Q7TUFBQUMsSUFBQSxFQUFRRixFQUFtQztNQUFBRyxJQUFBLEVBQVE7SUFBTyxFQUFDLEdBQ2xFO0lBQUFMLENBQUEsTUFBQUUsRUFBQTtJQUFBRixDQUFBLE1BQUFHLEVBQUE7RUFBQTtJQUFBQSxFQUFBLEdBQUFILENBQUE7RUFBQTtFQUFBLElBQUFNLEVBQUE7RUFBQSxJQUFBTixDQUFBLFFBQUFKLFFBQUEsSUFBQUksQ0FBQSxRQUFBSCxPQUFBO0lBQ0RTLEVBQUEsR0FBQVYsUUFBUSxHQUNQLENBQUMsb0JBQW9CLENBQ1AsVUFBbUIsQ0FBbkIsQ0FBQUEsUUFBUSxDQUFBVyxVQUFVLENBQUMsQ0FDdkIsTUFBZSxDQUFmLENBQUFYLFFBQVEsQ0FBQVksTUFBTSxDQUFDLENBQ0gsa0JBQTJCLENBQTNCLENBQUFaLFFBQVEsQ0FBQWEsa0JBQWtCLENBQUMsQ0FDbkMsVUFBbUIsQ0FBbkIsQ0FBQWIsUUFBUSxDQUFBYyxVQUFVLENBQUMsQ0FDdEJiLE9BQU8sQ0FBUEEsUUFBTSxDQUFDLEdBUW5CLEdBTENQLFFBQVEsQ0FBQXFCLDRCQUlOLEdBSnNDLEVBQUUsRUFBRTtNQUFBZCxPQUFBO01BQUFlLEtBQUEsRUFFbkMsRUFBRTtNQUFBQyxZQUFBLEVBQ0tDO0lBQ2hCLENBQ0YsQ0FBQztJQUFBZCxDQUFBLE1BQUFKLFFBQUE7SUFBQUksQ0FBQSxNQUFBSCxPQUFBO0lBQUFHLENBQUEsTUFBQU0sRUFBQTtFQUFBO0lBQUFBLEVBQUEsR0FBQU4sQ0FBQTtFQUFBO0VBQUEsSUFBQWUsRUFBQTtFQUFBLElBQUFmLENBQUEsUUFBQUcsRUFBQSxJQUFBSCxDQUFBLFFBQUFNLEVBQUE7SUFuQkhTLEVBQUEsSUFBQyxHQUFHLENBQWUsYUFBUSxDQUFSLFFBQVEsQ0FBWSxTQUFDLENBQUQsR0FBQyxDQUN0QyxDQUFBWixFQUdDLENBQ0EsQ0FBQUcsRUFjRCxDQUNGLEVBcEJDLEdBQUcsQ0FvQkU7SUFBQU4sQ0FBQSxNQUFBRyxFQUFBO0lBQUFILENBQUEsTUFBQU0sRUFBQTtJQUFBTixDQUFBLE1BQUFlLEVBQUE7RUFBQTtJQUFBQSxFQUFBLEdBQUFmLENBQUE7RUFBQTtFQUFBLE9BcEJOZSxFQW9CTTtBQUFBIiwiaWdub3JlTGlzdCI6W119 \ No newline at end of file diff --git a/src/components/BridgeDialog.tsx b/src/components/BridgeDialog.tsx new file mode 100644 index 0000000..4f25836 --- /dev/null +++ b/src/components/BridgeDialog.tsx @@ -0,0 +1,401 @@ +import { c as _c } from "react/compiler-runtime"; +import { basename } from 'path'; +import { toString as qrToString } from 'qrcode'; +import * as React from 'react'; +import { useEffect, useState } from 'react'; +import { getOriginalCwd } from '../bootstrap/state.js'; +import { buildActiveFooterText, buildIdleFooterText, FAILED_FOOTER_TEXT, getBridgeStatus } from '../bridge/bridgeStatusUtil.js'; +import { BRIDGE_FAILED_INDICATOR, BRIDGE_READY_INDICATOR } from '../constants/figures.js'; +import { useRegisterOverlay } from '../context/overlayContext.js'; +// eslint-disable-next-line custom-rules/prefer-use-keybindings -- raw 'd' key for disconnect, not a configurable keybinding action +import { Box, Text, useInput } from '../ink.js'; +import { useKeybindings } from '../keybindings/useKeybinding.js'; +import { useAppState, useSetAppState } from '../state/AppState.js'; +import { saveGlobalConfig } from '../utils/config.js'; +import { getBranch } from '../utils/git.js'; +import { Dialog } from './design-system/Dialog.js'; +type Props = { + onDone: () => void; +}; +export function BridgeDialog(t0) { + const $ = _c(87); + const { + onDone + } = t0; + useRegisterOverlay("bridge-dialog"); + const connected = useAppState(_temp); + const sessionActive = useAppState(_temp2); + const reconnecting = useAppState(_temp3); + const connectUrl = useAppState(_temp4); + const sessionUrl = useAppState(_temp5); + const error = useAppState(_temp6); + const explicit = useAppState(_temp7); + const environmentId = useAppState(_temp8); + const sessionId = useAppState(_temp9); + const verbose = useAppState(_temp0); + const setAppState = useSetAppState(); + const [showQR, setShowQR] = useState(false); + const [qrText, setQrText] = useState(""); + const [branchName, setBranchName] = useState(""); + let t1; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t1 = basename(getOriginalCwd()); + $[0] = t1; + } else { + t1 = $[0]; + } + const repoName = t1; + let t2; + let t3; + if ($[1] === Symbol.for("react.memo_cache_sentinel")) { + t2 = () => { + getBranch().then(setBranchName).catch(_temp1); + }; + t3 = []; + $[1] = t2; + $[2] = t3; + } else { + t2 = $[1]; + t3 = $[2]; + } + useEffect(t2, t3); + const displayUrl = sessionActive ? sessionUrl : connectUrl; + let t4; + let t5; + if ($[3] !== displayUrl || $[4] !== showQR) { + t4 = () => { + if (!showQR || !displayUrl) { + setQrText(""); + return; + } + qrToString(displayUrl, { + type: "utf8", + errorCorrectionLevel: "L", + small: true + }).then(setQrText).catch(() => setQrText("")); + }; + t5 = [showQR, displayUrl]; + $[3] = displayUrl; + $[4] = showQR; + $[5] = t4; + $[6] = t5; + } else { + t4 = $[5]; + t5 = $[6]; + } + useEffect(t4, t5); + let t6; + if ($[7] === Symbol.for("react.memo_cache_sentinel")) { + t6 = () => { + setShowQR(_temp10); + }; + $[7] = t6; + } else { + t6 = $[7]; + } + let t7; + if ($[8] !== onDone) { + t7 = { + "confirm:yes": onDone, + "confirm:toggle": t6 + }; + $[8] = onDone; + $[9] = t7; + } else { + t7 = $[9]; + } + let t8; + if ($[10] === Symbol.for("react.memo_cache_sentinel")) { + t8 = { + context: "Confirmation" + }; + $[10] = t8; + } else { + t8 = $[10]; + } + useKeybindings(t7, t8); + let t9; + if ($[11] !== explicit || $[12] !== onDone || $[13] !== setAppState) { + t9 = input => { + if (input === "d") { + if (explicit) { + saveGlobalConfig(_temp11); + } + setAppState(_temp12); + onDone(); + } + }; + $[11] = explicit; + $[12] = onDone; + $[13] = setAppState; + $[14] = t9; + } else { + t9 = $[14]; + } + useInput(t9); + let t10; + if ($[15] !== connected || $[16] !== error || $[17] !== reconnecting || $[18] !== sessionActive) { + t10 = getBridgeStatus({ + error, + connected, + sessionActive, + reconnecting + }); + $[15] = connected; + $[16] = error; + $[17] = reconnecting; + $[18] = sessionActive; + $[19] = t10; + } else { + t10 = $[19]; + } + const { + label: statusLabel, + color: statusColor + } = t10; + const indicator = error ? BRIDGE_FAILED_INDICATOR : BRIDGE_READY_INDICATOR; + let T0; + let T1; + let footerText; + let t11; + let t12; + let t13; + let t14; + let t15; + let t16; + let t17; + if ($[20] !== branchName || $[21] !== displayUrl || $[22] !== environmentId || $[23] !== error || $[24] !== indicator || $[25] !== onDone || $[26] !== qrText || $[27] !== sessionActive || $[28] !== sessionId || $[29] !== showQR || $[30] !== statusColor || $[31] !== statusLabel || $[32] !== verbose) { + const qrLines = qrText ? qrText.split("\n").filter(_temp13) : []; + let contextParts; + if ($[43] !== branchName) { + contextParts = []; + if (repoName) { + contextParts.push(repoName); + } + if (branchName) { + contextParts.push(branchName); + } + $[43] = branchName; + $[44] = contextParts; + } else { + contextParts = $[44]; + } + const contextSuffix = contextParts.length > 0 ? " \xB7 " + contextParts.join(" \xB7 ") : ""; + let t18; + if ($[45] !== displayUrl || $[46] !== error || $[47] !== sessionActive) { + t18 = error ? FAILED_FOOTER_TEXT : displayUrl ? sessionActive ? buildActiveFooterText(displayUrl) : buildIdleFooterText(displayUrl) : undefined; + $[45] = displayUrl; + $[46] = error; + $[47] = sessionActive; + $[48] = t18; + } else { + t18 = $[48]; + } + footerText = t18; + T1 = Dialog; + t15 = "Remote Control"; + t16 = onDone; + t17 = true; + T0 = Box; + t11 = "column"; + t12 = 1; + let t19; + if ($[49] !== indicator || $[50] !== statusColor || $[51] !== statusLabel) { + t19 = {indicator} {statusLabel}; + $[49] = indicator; + $[50] = statusColor; + $[51] = statusLabel; + $[52] = t19; + } else { + t19 = $[52]; + } + let t20; + if ($[53] !== contextSuffix) { + t20 = {contextSuffix}; + $[53] = contextSuffix; + $[54] = t20; + } else { + t20 = $[54]; + } + let t21; + if ($[55] !== t19 || $[56] !== t20) { + t21 = {t19}{t20}; + $[55] = t19; + $[56] = t20; + $[57] = t21; + } else { + t21 = $[57]; + } + let t22; + if ($[58] !== error) { + t22 = error && {error}; + $[58] = error; + $[59] = t22; + } else { + t22 = $[59]; + } + let t23; + if ($[60] !== environmentId || $[61] !== verbose) { + t23 = verbose && environmentId && Environment: {environmentId}; + $[60] = environmentId; + $[61] = verbose; + $[62] = t23; + } else { + t23 = $[62]; + } + let t24; + if ($[63] !== sessionId || $[64] !== verbose) { + t24 = verbose && sessionId && Session: {sessionId}; + $[63] = sessionId; + $[64] = verbose; + $[65] = t24; + } else { + t24 = $[65]; + } + if ($[66] !== t21 || $[67] !== t22 || $[68] !== t23 || $[69] !== t24) { + t13 = {t21}{t22}{t23}{t24}; + $[66] = t21; + $[67] = t22; + $[68] = t23; + $[69] = t24; + $[70] = t13; + } else { + t13 = $[70]; + } + t14 = showQR && qrLines.length > 0 && {qrLines.map(_temp14)}; + $[20] = branchName; + $[21] = displayUrl; + $[22] = environmentId; + $[23] = error; + $[24] = indicator; + $[25] = onDone; + $[26] = qrText; + $[27] = sessionActive; + $[28] = sessionId; + $[29] = showQR; + $[30] = statusColor; + $[31] = statusLabel; + $[32] = verbose; + $[33] = T0; + $[34] = T1; + $[35] = footerText; + $[36] = t11; + $[37] = t12; + $[38] = t13; + $[39] = t14; + $[40] = t15; + $[41] = t16; + $[42] = t17; + } else { + T0 = $[33]; + T1 = $[34]; + footerText = $[35]; + t11 = $[36]; + t12 = $[37]; + t13 = $[38]; + t14 = $[39]; + t15 = $[40]; + t16 = $[41]; + t17 = $[42]; + } + let t18; + if ($[71] !== footerText) { + t18 = footerText && {footerText}; + $[71] = footerText; + $[72] = t18; + } else { + t18 = $[72]; + } + let t19; + if ($[73] === Symbol.for("react.memo_cache_sentinel")) { + t19 = d to disconnect · space for QR code · Enter/Esc to close; + $[73] = t19; + } else { + t19 = $[73]; + } + let t20; + if ($[74] !== T0 || $[75] !== t11 || $[76] !== t12 || $[77] !== t13 || $[78] !== t14 || $[79] !== t18) { + t20 = {t13}{t14}{t18}{t19}; + $[74] = T0; + $[75] = t11; + $[76] = t12; + $[77] = t13; + $[78] = t14; + $[79] = t18; + $[80] = t20; + } else { + t20 = $[80]; + } + let t21; + if ($[81] !== T1 || $[82] !== t15 || $[83] !== t16 || $[84] !== t17 || $[85] !== t20) { + t21 = {t20}; + $[81] = T1; + $[82] = t15; + $[83] = t16; + $[84] = t17; + $[85] = t20; + $[86] = t21; + } else { + t21 = $[86]; + } + return t21; +} +function _temp14(line, i) { + return {line}; +} +function _temp13(l) { + return l.length > 0; +} +function _temp12(prev_0) { + if (!prev_0.replBridgeEnabled) { + return prev_0; + } + return { + ...prev_0, + replBridgeEnabled: false + }; +} +function _temp11(current) { + if (current.remoteControlAtStartup === false) { + return current; + } + return { + ...current, + remoteControlAtStartup: false + }; +} +function _temp10(prev) { + return !prev; +} +function _temp1() {} +function _temp0(s_8) { + return s_8.verbose; +} +function _temp9(s_7) { + return s_7.replBridgeSessionId; +} +function _temp8(s_6) { + return s_6.replBridgeEnvironmentId; +} +function _temp7(s_5) { + return s_5.replBridgeExplicit; +} +function _temp6(s_4) { + return s_4.replBridgeError; +} +function _temp5(s_3) { + return s_3.replBridgeSessionUrl; +} +function _temp4(s_2) { + return s_2.replBridgeConnectUrl; +} +function _temp3(s_1) { + return s_1.replBridgeReconnecting; +} +function _temp2(s_0) { + return s_0.replBridgeSessionActive; +} +function _temp(s) { + return s.replBridgeConnected; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJiYXNlbmFtZSIsInRvU3RyaW5nIiwicXJUb1N0cmluZyIsIlJlYWN0IiwidXNlRWZmZWN0IiwidXNlU3RhdGUiLCJnZXRPcmlnaW5hbEN3ZCIsImJ1aWxkQWN0aXZlRm9vdGVyVGV4dCIsImJ1aWxkSWRsZUZvb3RlclRleHQiLCJGQUlMRURfRk9PVEVSX1RFWFQiLCJnZXRCcmlkZ2VTdGF0dXMiLCJCUklER0VfRkFJTEVEX0lORElDQVRPUiIsIkJSSURHRV9SRUFEWV9JTkRJQ0FUT1IiLCJ1c2VSZWdpc3Rlck92ZXJsYXkiLCJCb3giLCJUZXh0IiwidXNlSW5wdXQiLCJ1c2VLZXliaW5kaW5ncyIsInVzZUFwcFN0YXRlIiwidXNlU2V0QXBwU3RhdGUiLCJzYXZlR2xvYmFsQ29uZmlnIiwiZ2V0QnJhbmNoIiwiRGlhbG9nIiwiUHJvcHMiLCJvbkRvbmUiLCJCcmlkZ2VEaWFsb2ciLCJ0MCIsIiQiLCJfYyIsImNvbm5lY3RlZCIsIl90ZW1wIiwic2Vzc2lvbkFjdGl2ZSIsIl90ZW1wMiIsInJlY29ubmVjdGluZyIsIl90ZW1wMyIsImNvbm5lY3RVcmwiLCJfdGVtcDQiLCJzZXNzaW9uVXJsIiwiX3RlbXA1IiwiZXJyb3IiLCJfdGVtcDYiLCJleHBsaWNpdCIsIl90ZW1wNyIsImVudmlyb25tZW50SWQiLCJfdGVtcDgiLCJzZXNzaW9uSWQiLCJfdGVtcDkiLCJ2ZXJib3NlIiwiX3RlbXAwIiwic2V0QXBwU3RhdGUiLCJzaG93UVIiLCJzZXRTaG93UVIiLCJxclRleHQiLCJzZXRRclRleHQiLCJicmFuY2hOYW1lIiwic2V0QnJhbmNoTmFtZSIsInQxIiwiU3ltYm9sIiwiZm9yIiwicmVwb05hbWUiLCJ0MiIsInQzIiwidGhlbiIsImNhdGNoIiwiX3RlbXAxIiwiZGlzcGxheVVybCIsInQ0IiwidDUiLCJ0eXBlIiwiZXJyb3JDb3JyZWN0aW9uTGV2ZWwiLCJzbWFsbCIsInQ2IiwiX3RlbXAxMCIsInQ3IiwidDgiLCJjb250ZXh0IiwidDkiLCJpbnB1dCIsIl90ZW1wMTEiLCJfdGVtcDEyIiwidDEwIiwibGFiZWwiLCJzdGF0dXNMYWJlbCIsImNvbG9yIiwic3RhdHVzQ29sb3IiLCJpbmRpY2F0b3IiLCJUMCIsIlQxIiwiZm9vdGVyVGV4dCIsInQxMSIsInQxMiIsInQxMyIsInQxNCIsInQxNSIsInQxNiIsInQxNyIsInFyTGluZXMiLCJzcGxpdCIsImZpbHRlciIsIl90ZW1wMTMiLCJjb250ZXh0UGFydHMiLCJwdXNoIiwiY29udGV4dFN1ZmZpeCIsImxlbmd0aCIsImpvaW4iLCJ0MTgiLCJ1bmRlZmluZWQiLCJ0MTkiLCJ0MjAiLCJ0MjEiLCJ0MjIiLCJ0MjMiLCJ0MjQiLCJtYXAiLCJfdGVtcDE0IiwibGluZSIsImkiLCJsIiwicHJldl8wIiwicHJldiIsInJlcGxCcmlkZ2VFbmFibGVkIiwiY3VycmVudCIsInJlbW90ZUNvbnRyb2xBdFN0YXJ0dXAiLCJzXzgiLCJzIiwic183IiwicmVwbEJyaWRnZVNlc3Npb25JZCIsInNfNiIsInJlcGxCcmlkZ2VFbnZpcm9ubWVudElkIiwic181IiwicmVwbEJyaWRnZUV4cGxpY2l0Iiwic180IiwicmVwbEJyaWRnZUVycm9yIiwic18zIiwicmVwbEJyaWRnZVNlc3Npb25VcmwiLCJzXzIiLCJyZXBsQnJpZGdlQ29ubmVjdFVybCIsInNfMSIsInJlcGxCcmlkZ2VSZWNvbm5lY3RpbmciLCJzXzAiLCJyZXBsQnJpZGdlU2Vzc2lvbkFjdGl2ZSIsInJlcGxCcmlkZ2VDb25uZWN0ZWQiXSwic291cmNlcyI6WyJCcmlkZ2VEaWFsb2cudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGJhc2VuYW1lIH0gZnJvbSAncGF0aCdcbmltcG9ydCB7IHRvU3RyaW5nIGFzIHFyVG9TdHJpbmcgfSBmcm9tICdxcmNvZGUnXG5pbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZUVmZmVjdCwgdXNlU3RhdGUgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IGdldE9yaWdpbmFsQ3dkIH0gZnJvbSAnLi4vYm9vdHN0cmFwL3N0YXRlLmpzJ1xuaW1wb3J0IHtcbiAgYnVpbGRBY3RpdmVGb290ZXJUZXh0LFxuICBidWlsZElkbGVGb290ZXJUZXh0LFxuICBGQUlMRURfRk9PVEVSX1RFWFQsXG4gIGdldEJyaWRnZVN0YXR1cyxcbn0gZnJvbSAnLi4vYnJpZGdlL2JyaWRnZVN0YXR1c1V0aWwuanMnXG5pbXBvcnQge1xuICBCUklER0VfRkFJTEVEX0lORElDQVRPUixcbiAgQlJJREdFX1JFQURZX0lORElDQVRPUixcbn0gZnJvbSAnLi4vY29uc3RhbnRzL2ZpZ3VyZXMuanMnXG5pbXBvcnQgeyB1c2VSZWdpc3Rlck92ZXJsYXkgfSBmcm9tICcuLi9jb250ZXh0L292ZXJsYXlDb250ZXh0LmpzJ1xuLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIGN1c3RvbS1ydWxlcy9wcmVmZXItdXNlLWtleWJpbmRpbmdzIC0tIHJhdyAnZCcga2V5IGZvciBkaXNjb25uZWN0LCBub3QgYSBjb25maWd1cmFibGUga2V5YmluZGluZyBhY3Rpb25cbmltcG9ydCB7IEJveCwgVGV4dCwgdXNlSW5wdXQgfSBmcm9tICcuLi9pbmsuanMnXG5pbXBvcnQgeyB1c2VLZXliaW5kaW5ncyB9IGZyb20gJy4uL2tleWJpbmRpbmdzL3VzZUtleWJpbmRpbmcuanMnXG5pbXBvcnQgeyB1c2VBcHBTdGF0ZSwgdXNlU2V0QXBwU3RhdGUgfSBmcm9tICcuLi9zdGF0ZS9BcHBTdGF0ZS5qcydcbmltcG9ydCB7IHNhdmVHbG9iYWxDb25maWcgfSBmcm9tICcuLi91dGlscy9jb25maWcuanMnXG5pbXBvcnQgeyBnZXRCcmFuY2ggfSBmcm9tICcuLi91dGlscy9naXQuanMnXG5pbXBvcnQgeyBEaWFsb2cgfSBmcm9tICcuL2Rlc2lnbi1zeXN0ZW0vRGlhbG9nLmpzJ1xuXG50eXBlIFByb3BzID0ge1xuICBvbkRvbmU6ICgpID0+IHZvaWRcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIEJyaWRnZURpYWxvZyh7IG9uRG9uZSB9OiBQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIHVzZVJlZ2lzdGVyT3ZlcmxheSgnYnJpZGdlLWRpYWxvZycpXG5cbiAgY29uc3QgY29ubmVjdGVkID0gdXNlQXBwU3RhdGUocyA9PiBzLnJlcGxCcmlkZ2VDb25uZWN0ZWQpXG4gIGNvbnN0IHNlc3Npb25BY3RpdmUgPSB1c2VBcHBTdGF0ZShzID0+IHMucmVwbEJyaWRnZVNlc3Npb25BY3RpdmUpXG4gIGNvbnN0IHJlY29ubmVjdGluZyA9IHVzZUFwcFN0YXRlKHMgPT4gcy5yZXBsQnJpZGdlUmVjb25uZWN0aW5nKVxuICBjb25zdCBjb25uZWN0VXJsID0gdXNlQXBwU3RhdGUocyA9PiBzLnJlcGxCcmlkZ2VDb25uZWN0VXJsKVxuICBjb25zdCBzZXNzaW9uVXJsID0gdXNlQXBwU3RhdGUocyA9PiBzLnJlcGxCcmlkZ2VTZXNzaW9uVXJsKVxuICBjb25zdCBlcnJvciA9IHVzZUFwcFN0YXRlKHMgPT4gcy5yZXBsQnJpZGdlRXJyb3IpXG4gIGNvbnN0IGV4cGxpY2l0ID0gdXNlQXBwU3RhdGUocyA9PiBzLnJlcGxCcmlkZ2VFeHBsaWNpdClcbiAgY29uc3QgZW52aXJvbm1lbnRJZCA9IHVzZUFwcFN0YXRlKHMgPT4gcy5yZXBsQnJpZGdlRW52aXJvbm1lbnRJZClcbiAgY29uc3Qgc2Vzc2lvbklkID0gdXNlQXBwU3RhdGUocyA9PiBzLnJlcGxCcmlkZ2VTZXNzaW9uSWQpXG4gIGNvbnN0IHZlcmJvc2UgPSB1c2VBcHBTdGF0ZShzID0+IHMudmVyYm9zZSlcbiAgY29uc3Qgc2V0QXBwU3RhdGUgPSB1c2VTZXRBcHBTdGF0ZSgpXG5cbiAgY29uc3QgW3Nob3dRUiwgc2V0U2hvd1FSXSA9IHVzZVN0YXRlKGZhbHNlKVxuICBjb25zdCBbcXJUZXh0LCBzZXRRclRleHRdID0gdXNlU3RhdGUoJycpXG4gIGNvbnN0IFticmFuY2hOYW1lLCBzZXRCcmFuY2hOYW1lXSA9IHVzZVN0YXRlKCcnKVxuXG4gIGNvbnN0IHJlcG9OYW1lID0gYmFzZW5hbWUoZ2V0T3JpZ2luYWxDd2QoKSlcblxuICAvLyBGZXRjaCBicmFuY2ggbmFtZSBvbiBtb3VudFxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGdldEJyYW5jaCgpXG4gICAgICAudGhlbihzZXRCcmFuY2hOYW1lKVxuICAgICAgLmNhdGNoKCgpID0+IHt9KVxuICB9LCBbXSlcblxuICAvLyBUaGUgVVJMIHRvIGRpc3BsYXkvUVI6IHNlc3Npb24gVVJMIHdoZW4gY29ubmVjdGVkLCBjb25uZWN0IFVSTCB3aGVuIHJlYWR5XG4gIGNvbnN0IGRpc3BsYXlVcmwgPSBzZXNzaW9uQWN0aXZlID8gc2Vzc2lvblVybCA6IGNvbm5lY3RVcmxcblxuICAvLyBHZW5lcmF0ZSBRUiBjb2RlIHdoZW4gVVJMIGNoYW5nZXMgb3IgUVIgaXMgdG9nZ2xlZCBvblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmICghc2hvd1FSIHx8ICFkaXNwbGF5VXJsKSB7XG4gICAgICBzZXRRclRleHQoJycpXG4gICAgICByZXR1cm5cbiAgICB9XG4gICAgcXJUb1N0cmluZyhkaXNwbGF5VXJsLCB7XG4gICAgICB0eXBlOiAndXRmOCcsXG4gICAgICBlcnJvckNvcnJlY3Rpb25MZXZlbDogJ0wnLFxuICAgICAgc21hbGw6IHRydWUsXG4gICAgfSlcbiAgICAgIC50aGVuKHNldFFyVGV4dClcbiAgICAgIC5jYXRjaCgoKSA9PiBzZXRRclRleHQoJycpKVxuICB9LCBbc2hvd1FSLCBkaXNwbGF5VXJsXSlcblxuICB1c2VLZXliaW5kaW5ncyhcbiAgICB7XG4gICAgICAnY29uZmlybTp5ZXMnOiBvbkRvbmUsXG4gICAgICAnY29uZmlybTp0b2dnbGUnOiAoKSA9PiB7XG4gICAgICAgIHNldFNob3dRUihwcmV2ID0+ICFwcmV2KVxuICAgICAgfSxcbiAgICB9LFxuICAgIHsgY29udGV4dDogJ0NvbmZpcm1hdGlvbicgfSxcbiAgKVxuXG4gIHVzZUlucHV0KGlucHV0ID0+IHtcbiAgICBpZiAoaW5wdXQgPT09ICdkJykge1xuICAgICAgLy8gUGVyc2lzdCBvcHQtb3V0IG9ubHkgZm9yIENMSS1mbGFnL2NvbW1hbmQtYWN0aXZhdGVkIGJyaWRnZS5cbiAgICAgIC8vIENvbmZpZy1kcml2ZW4gYW5kIEdCLWF1dG8tY29ubmVjdCB1c2VycyBnZXQgc2Vzc2lvbi1vbmx5IGRpc2Nvbm5lY3RcbiAgICAgIC8vIOKAlCB3cml0aW5nIGZhbHNlIHdvdWxkIHNpbGVudGx5IHVuZG8gYSBTZXR0aW5ncyBjaG9pY2Ugb3Igb3B0IGFcbiAgICAgIC8vIEdCLXJvbGxvdXQgdXNlciBvdXQgcGVybWFuZW50bHkuXG4gICAgICBpZiAoZXhwbGljaXQpIHtcbiAgICAgICAgc2F2ZUdsb2JhbENvbmZpZyhjdXJyZW50ID0+IHtcbiAgICAgICAgICBpZiAoY3VycmVudC5yZW1vdGVDb250cm9sQXRTdGFydHVwID09PSBmYWxzZSkgcmV0dXJuIGN1cnJlbnRcbiAgICAgICAgICByZXR1cm4geyAuLi5jdXJyZW50LCByZW1vdGVDb250cm9sQXRTdGFydHVwOiBmYWxzZSB9XG4gICAgICAgIH0pXG4gICAgICB9XG4gICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+IHtcbiAgICAgICAgaWYgKCFwcmV2LnJlcGxCcmlkZ2VFbmFibGVkKSByZXR1cm4gcHJldlxuICAgICAgICByZXR1cm4geyAuLi5wcmV2LCByZXBsQnJpZGdlRW5hYmxlZDogZmFsc2UgfVxuICAgICAgfSlcbiAgICAgIG9uRG9uZSgpXG4gICAgfVxuICB9KVxuXG4gIGNvbnN0IHsgbGFiZWw6IHN0YXR1c0xhYmVsLCBjb2xvcjogc3RhdHVzQ29sb3IgfSA9IGdldEJyaWRnZVN0YXR1cyh7XG4gICAgZXJyb3IsXG4gICAgY29ubmVjdGVkLFxuICAgIHNlc3Npb25BY3RpdmUsXG4gICAgcmVjb25uZWN0aW5nLFxuICB9KVxuICBjb25zdCBpbmRpY2F0b3IgPSBlcnJvciA/IEJSSURHRV9GQUlMRURfSU5ESUNBVE9SIDogQlJJREdFX1JFQURZX0lORElDQVRPUlxuICBjb25zdCBxckxpbmVzID0gcXJUZXh0ID8gcXJUZXh0LnNwbGl0KCdcXG4nKS5maWx0ZXIobCA9PiBsLmxlbmd0aCA+IDApIDogW11cblxuICAvLyBCdWlsZCBzdWZmaXggd2l0aCByZXBvIGFuZCBicmFuY2ggKG1hdGNoZXMgc3RhbmRhbG9uZSBicmlkZ2UgZm9ybWF0KVxuICBjb25zdCBjb250ZXh0UGFydHM6IHN0cmluZ1tdID0gW11cbiAgaWYgKHJlcG9OYW1lKSBjb250ZXh0UGFydHMucHVzaChyZXBvTmFtZSlcbiAgaWYgKGJyYW5jaE5hbWUpIGNvbnRleHRQYXJ0cy5wdXNoKGJyYW5jaE5hbWUpXG4gIGNvbnN0IGNvbnRleHRTdWZmaXggPVxuICAgIGNvbnRleHRQYXJ0cy5sZW5ndGggPiAwID8gJyBcXHUwMGI3ICcgKyBjb250ZXh0UGFydHMuam9pbignIFxcdTAwYjcgJykgOiAnJ1xuXG4gIC8vIEZvb3RlciB0ZXh0IG1hdGNoZXMgc3RhbmRhbG9uZSBicmlkZ2VcbiAgY29uc3QgZm9vdGVyVGV4dCA9IGVycm9yXG4gICAgPyBGQUlMRURfRk9PVEVSX1RFWFRcbiAgICA6IGRpc3BsYXlVcmxcbiAgICAgID8gc2Vzc2lvbkFjdGl2ZVxuICAgICAgICA/IGJ1aWxkQWN0aXZlRm9vdGVyVGV4dChkaXNwbGF5VXJsKVxuICAgICAgICA6IGJ1aWxkSWRsZUZvb3RlclRleHQoZGlzcGxheVVybClcbiAgICAgIDogdW5kZWZpbmVkXG5cbiAgcmV0dXJuIChcbiAgICA8RGlhbG9nIHRpdGxlPVwiUmVtb3RlIENvbnRyb2xcIiBvbkNhbmNlbD17b25Eb25lfSBoaWRlSW5wdXRHdWlkZT5cbiAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiIGdhcD17MX0+XG4gICAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiPlxuICAgICAgICAgIDxUZXh0PlxuICAgICAgICAgICAgPFRleHQgY29sb3I9e3N0YXR1c0NvbG9yfT5cbiAgICAgICAgICAgICAge2luZGljYXRvcn0ge3N0YXR1c0xhYmVsfVxuICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgPFRleHQgZGltQ29sb3I+e2NvbnRleHRTdWZmaXh9PC9UZXh0PlxuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICB7ZXJyb3IgJiYgPFRleHQgY29sb3I9XCJlcnJvclwiPntlcnJvcn08L1RleHQ+fVxuICAgICAgICAgIHt2ZXJib3NlICYmIGVudmlyb25tZW50SWQgJiYgKFxuICAgICAgICAgICAgPFRleHQgZGltQ29sb3I+RW52aXJvbm1lbnQ6IHtlbnZpcm9ubWVudElkfTwvVGV4dD5cbiAgICAgICAgICApfVxuICAgICAgICAgIHt2ZXJib3NlICYmIHNlc3Npb25JZCAmJiA8VGV4dCBkaW1Db2xvcj5TZXNzaW9uOiB7c2Vzc2lvbklkfTwvVGV4dD59XG4gICAgICAgIDwvQm94PlxuICAgICAgICB7c2hvd1FSICYmIHFyTGluZXMubGVuZ3RoID4gMCAmJiAoXG4gICAgICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCI+XG4gICAgICAgICAgICB7cXJMaW5lcy5tYXAoKGxpbmUsIGkpID0+IChcbiAgICAgICAgICAgICAgPFRleHQga2V5PXtpfT57bGluZX08L1RleHQ+XG4gICAgICAgICAgICApKX1cbiAgICAgICAgICA8L0JveD5cbiAgICAgICAgKX1cbiAgICAgICAge2Zvb3RlclRleHQgJiYgPFRleHQgZGltQ29sb3I+e2Zvb3RlclRleHR9PC9UZXh0Pn1cbiAgICAgICAgPFRleHQgZGltQ29sb3I+XG4gICAgICAgICAgZCB0byBkaXNjb25uZWN0IMK3IHNwYWNlIGZvciBRUiBjb2RlIMK3IEVudGVyL0VzYyB0byBjbG9zZVxuICAgICAgICA8L1RleHQ+XG4gICAgICA8L0JveD5cbiAgICA8L0RpYWxvZz5cbiAgKVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsU0FBU0EsUUFBUSxRQUFRLE1BQU07QUFDL0IsU0FBU0MsUUFBUSxJQUFJQyxVQUFVLFFBQVEsUUFBUTtBQUMvQyxPQUFPLEtBQUtDLEtBQUssTUFBTSxPQUFPO0FBQzlCLFNBQVNDLFNBQVMsRUFBRUMsUUFBUSxRQUFRLE9B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 \ No newline at end of file diff --git a/src/components/BypassPermissionsModeDialog.tsx b/src/components/BypassPermissionsModeDialog.tsx new file mode 100644 index 0000000..ed09416 --- /dev/null +++ b/src/components/BypassPermissionsModeDialog.tsx @@ -0,0 +1,87 @@ +import { c as _c } from "react/compiler-runtime"; +import React, { useCallback } from 'react'; +import { logEvent } from 'src/services/analytics/index.js'; +import { Box, Link, Newline, Text } from '../ink.js'; +import { gracefulShutdownSync } from '../utils/gracefulShutdown.js'; +import { updateSettingsForSource } from '../utils/settings/settings.js'; +import { Select } from './CustomSelect/index.js'; +import { Dialog } from './design-system/Dialog.js'; +type Props = { + onAccept(): void; +}; +export function BypassPermissionsModeDialog(t0) { + const $ = _c(7); + const { + onAccept + } = t0; + let t1; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t1 = []; + $[0] = t1; + } else { + t1 = $[0]; + } + React.useEffect(_temp, t1); + let t2; + if ($[1] !== onAccept) { + t2 = function onChange(value) { + bb3: switch (value) { + case "accept": + { + logEvent("tengu_bypass_permissions_mode_dialog_accept", {}); + updateSettingsForSource("userSettings", { + skipDangerousModePermissionPrompt: true + }); + onAccept(); + break bb3; + } + case "decline": + { + gracefulShutdownSync(1); + } + } + }; + $[1] = onAccept; + $[2] = t2; + } else { + t2 = $[2]; + } + const onChange = t2; + const handleEscape = _temp2; + let t3; + if ($[3] === Symbol.for("react.memo_cache_sentinel")) { + t3 = In Bypass Permissions mode, Claude Code will not ask for your approval before running potentially dangerous commands.This mode should only be used in a sandboxed container/VM that has restricted internet access and can easily be restored if damaged.By proceeding, you accept all responsibility for actions taken while running in Bypass Permissions mode.; + $[3] = t3; + } else { + t3 = $[3]; + } + let t4; + if ($[4] === Symbol.for("react.memo_cache_sentinel")) { + t4 = [{ + label: "No, exit", + value: "decline" + }, { + label: "Yes, I accept", + value: "accept" + }]; + $[4] = t4; + } else { + t4 = $[4]; + } + let t5; + if ($[5] !== onChange) { + t5 = {t3}; + $[10] = handleSelect; + $[11] = t7; + $[12] = t8; + } else { + t8 = $[12]; + } + let t9; + if ($[13] !== handleCancel || $[14] !== t3 || $[15] !== t8) { + t9 = {t3}{t4}{t8}; + $[13] = handleCancel; + $[14] = t3; + $[15] = t8; + $[16] = t9; + } else { + t9 = $[16]; + } + return t9; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIlRleHQiLCJTZWxlY3QiLCJEaWFsb2ciLCJDaGFubmVsRG93bmdyYWRlQ2hvaWNlIiwiUHJvcHMiLCJjdXJyZW50VmVyc2lvbiIsIm9uQ2hvaWNlIiwiY2hvaWNlIiwiQ2hhbm5lbERvd25ncmFkZURpYWxvZyIsInQwIiwiJCIsIl9jIiwidDEiLCJoYW5kbGVTZWxlY3QiLCJ2YWx1ZSIsInQyIiwiaGFuZGxlQ2FuY2VsIiwidDMiLCJ0NCIsIlN5bWJvbCIsImZvciIsInQ1IiwibGFiZWwiLCJ0NiIsInQ3IiwidDgiLCJ0OSJdLCJzb3VyY2VzIjpbIkNoYW5uZWxEb3duZ3JhZGVEaWFsb2cudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB7IFRleHQgfSBmcm9tICcuLi9pbmsuanMnXG5pbXBvcnQgeyBTZWxlY3QgfSBmcm9tICcuL0N1c3RvbVNlbGVjdC9pbmRleC5qcydcbmltcG9ydCB7IERpYWxvZyB9IGZyb20gJy4vZGVzaWduLXN5c3RlbS9EaWFsb2cuanMnXG5cbmV4cG9ydCB0eXBlIENoYW5uZWxEb3duZ3JhZGVDaG9pY2UgPSAnZG93bmdyYWRlJyB8ICdzdGF5JyB8ICdjYW5jZWwnXG5cbnR5cGUgUHJvcHMgPSB7XG4gIGN1cnJlbnRWZXJzaW9uOiBzdHJpbmdcbiAgb25DaG9pY2U6IChjaG9pY2U6IENoYW5uZWxEb3duZ3JhZGVDaG9pY2UpID0+IHZvaWRcbn1cblxuLyoqXG4gKiBEaWFsb2cgc2hvd24gd2hlbiBzd2l0Y2hpbmcgZnJvbSBsYXRlc3QgdG8gc3RhYmxlIGNoYW5uZWwuXG4gKiBBbGxvd3MgdXNlciB0byBjaG9vc2Ugd2hldGhlciB0byBkb3duZ3JhZGUgb3Igc3RheSBvbiBjdXJyZW50IHZlcnNpb24uXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBDaGFubmVsRG93bmdyYWRlRGlhbG9nKHtcbiAgY3VycmVudFZlcnNpb24sXG4gIG9uQ2hvaWNlLFxufTogUHJvcHMpOiBSZWFjdC5SZWFjdE5vZGUge1xuICBmdW5jdGlvbiBoYW5kbGVTZWxlY3QodmFsdWU6IENoYW5uZWxEb3duZ3JhZGVDaG9pY2UpOiB2b2lkIHtcbiAgICBvbkNob2ljZSh2YWx1ZSlcbiAgfVxuXG4gIGZ1bmN0aW9uIGhhbmRsZUNhbmNlbCgpOiB2b2lkIHtcbiAgICBvbkNob2ljZSgnY2FuY2VsJylcbiAgfVxuXG4gIHJldHVybiAoXG4gICAgPERpYWxvZ1xuICAgICAgdGl0bGU9XCJTd2l0Y2ggdG8gU3RhYmxlIENoYW5uZWxcIlxuICAgICAgb25DYW5jZWw9e2hhbmRsZUNhbmNlbH1cbiAgICAgIGNvbG9yPVwicGVybWlzc2lvblwiXG4gICAgICBoaWRlQm9yZGVyXG4gICAgICBoaWRlSW5wdXRHdWlkZVxuICAgID5cbiAgICAgIDxUZXh0PlxuICAgICAgICBUaGUgc3RhYmxlIGNoYW5uZWwgbWF5IGhhdmUgYW4gb2xkZXIgdmVyc2lvbiB0aGFuIHdoYXQgeW91JmFwb3M7cmVcbiAgICAgICAgY3VycmVudGx5IHJ1bm5pbmcgKHtjdXJyZW50VmVyc2lvbn0pLlxuICAgICAgPC9UZXh0PlxuICAgICAgPFRleHQgZGltQ29sb3I+SG93IHdvdWxkIHlvdSBsaWtlIHRvIGhhbmRsZSB0aGlzPzwvVGV4dD5cbiAgICAgIDxTZWxlY3RcbiAgICAgICAgb3B0aW9ucz17W1xuICAgICAgICAgIHtcbiAgICAgICAgICAgIGxhYmVsOiAnQWxsb3cgcG9zc2libGUgZG93bmdyYWRlIHRvIHN0YWJsZSB2ZXJzaW9uJyxcbiAgICAgICAgICAgIHZhbHVlOiAnZG93bmdyYWRlJyBhcyBDaGFubmVsRG93bmdyYWRlQ2hvaWNlLFxuICAgICAgICAgIH0sXG4gICAgICAgICAge1xuICAgICAgICAgICAgbGFiZWw6IGBTdGF5IG9uIGN1cnJlbnQgdmVyc2lvbiAoJHtjdXJyZW50VmVyc2lvbn0pIHVudGlsIHN0YWJsZSBjYXRjaGVzIHVwYCxcbiAgICAgICAgICAgIHZhbHVlOiAnc3RheScgYXMgQ2hhbm5lbERvd25ncmFkZUNob2ljZSxcbiAgICAgICAgICB9LFxuICAgICAgICBdfVxuICAgICAgICBvbkNoYW5nZT17aGFuZGxlU2VsZWN0fVxuICAgICAgLz5cbiAgICA8L0RpYWxvZz5cbiAgKVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsT0FBT0EsS0FBSyxNQUFNLE9BQU87QUFDekIsU0FBU0MsSUFBSSxRQUFRLFdBQVc7QUFDaEMsU0FBU0MsTUFBTSxRQUFRLHlCQUF5QjtBQUNoRCxTQUFTQyxNQUFNLFFBQVEsMkJBQTJCO0FBRWxELE9BQU8sS0FBS0Msc0JBQXNCLEdBQUcsV0FBVyxHQUFHLE1BQU0sR0FBRyxRQUFRO0FBRXBFLEtBQUtDLEtBQUssR0FBRztFQUNYQyxjQUFjLEVBQUUsTUFBTTtFQUN0QkMsUUFBUSxFQUFFLENBQUNDLE1BQU0sRUFBRUosc0JBQXNCLEVBQUUsR0FBRyxJQUFJO0FBQ3BELENBQUM7O0FBRUQ7QUFDQTtBQUNBO0FBQ0E7QUFDQSxPQUFPLFNBQUFLLHVCQUFBQyxFQUFBO0VBQUEsTUFBQUMsQ0FBQSxHQUFBQyxFQUFBO0VBQWdDO0lBQUFOLGNBQUE7SUFBQUM7RUFBQSxJQUFBRyxFQUcvQjtFQUFBLElBQUFHLEVBQUE7RUFBQSxJQUFBRixDQUFBLFFBQUFKLFFBQUE7SUFDTk0sRUFBQSxZQUFBQyxhQUFBQyxLQUFBO01BQ0VSLFFBQVEsQ0FBQ1EsS0FBSyxDQUFDO0lBQUEsQ0FDaEI7SUFBQUosQ0FBQSxNQUFBSixRQUFBO0lBQUFJLENBQUEsTUFBQUUsRUFBQTtFQUFBO0lBQUFBLEVBQUEsR0FBQUYsQ0FBQTtFQUFBO0VBRkQsTUFBQUcsWUFBQSxHQUFBRCxFQUVDO0VBQUEsSUFBQUcsRUFBQTtFQUFBLElBQUFMLENBQUEsUUFBQUosUUFBQTtJQUVEUyxFQUFBLFlBQUFDLGFBQUE7TUFDRVYsUUFBUSxDQUFDLFFBQVEsQ0FBQztJQUFBLENBQ25CO0lBQUFJLENBQUEsTUFBQUosUUFBQTtJQUFBSSxDQUFBLE1BQUFLLEVBQUE7RUFBQTtJQUFBQSxFQUFBLEdBQUFMLENBQUE7RUFB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 \ No newline at end of file diff --git a/src/components/ClaudeCodeHint/PluginHintMenu.tsx b/src/components/ClaudeCodeHint/PluginHintMenu.tsx new file mode 100644 index 0000000..8ebd16e --- /dev/null +++ b/src/components/ClaudeCodeHint/PluginHintMenu.tsx @@ -0,0 +1,78 @@ +import * as React from 'react'; +import { Box, Text } from '../../ink.js'; +import { Select } from '../CustomSelect/select.js'; +import { PermissionDialog } from '../permissions/PermissionDialog.js'; +type Props = { + pluginName: string; + pluginDescription?: string; + marketplaceName: string; + sourceCommand: string; + onResponse: (response: 'yes' | 'no' | 'disable') => void; +}; +const AUTO_DISMISS_MS = 30_000; +export function PluginHintMenu({ + pluginName, + pluginDescription, + marketplaceName, + sourceCommand, + onResponse +}: Props): React.ReactNode { + const onResponseRef = React.useRef(onResponse); + onResponseRef.current = onResponse; + React.useEffect(() => { + const timeoutId = setTimeout(ref => ref.current('no'), AUTO_DISMISS_MS, onResponseRef); + return () => clearTimeout(timeoutId); + }, []); + function onSelect(value: string): void { + switch (value) { + case 'yes': + onResponse('yes'); + break; + case 'disable': + onResponse('disable'); + break; + default: + onResponse('no'); + } + } + const options = [{ + label: + Yes, install {pluginName} + , + value: 'yes' + }, { + label: 'No', + value: 'no' + }, { + label: "No, and don't show plugin installation hints again", + value: 'disable' + }]; + return + + + + The {sourceCommand} command suggests installing a + plugin. + + + + Plugin: + {pluginName} + + + Marketplace: + {marketplaceName} + + {pluginDescription && + {pluginDescription} + } + + Would you like to install it? + + + handleSelection(value_0 as 'yes' | 'no')} />; + $[10] = handleSelection; + $[11] = t10; + } else { + t10 = $[11]; + } + let t11; + if ($[12] !== handleEscape || $[13] !== t10 || $[14] !== t4 || $[15] !== t5 || $[16] !== t7) { + t11 = {t6}{t7}{t8}{t10}; + $[12] = handleEscape; + $[13] = t10; + $[14] = t4; + $[15] = t5; + $[16] = t7; + $[17] = t11; + } else { + t11 = $[17]; + } + return t11; +} +function _temp4(include, i) { + return {" "}{include.path}; +} +function _temp3(current_0) { + return { + ...current_0, + hasClaudeMdExternalIncludesApproved: true, + hasClaudeMdExternalIncludesWarningShown: true + }; +} +function _temp2(current) { + return { + ...current, + hasClaudeMdExternalIncludesApproved: false, + hasClaudeMdExternalIncludesWarningShown: true + }; +} +function _temp() { + logEvent("tengu_claude_md_includes_dialog_shown", {}); +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsInVzZUNhbGxiYWNrIiwibG9nRXZlbnQiLCJCb3giLCJMaW5rIiwiVGV4dCIsIkV4dGVybmFsQ2xhdWRlTWRJbmNsdWRlIiwic2F2ZUN1cnJlbnRQcm9qZWN0Q29uZmlnIiwiU2VsZWN0IiwiRGlhbG9nIiwiUHJvcHMiLCJvbkRvbmUiLCJpc1N0YW5kYWxvbmVEaWFsb2ciLCJleHRlcm5hbEluY2x1ZGVzIiwiQ2xhdWRlTWRFeHRlcm5hbEluY2x1ZGVzRGlhbG9nIiwidDAiLCIkIiwiX2MiLCJ0MSIsIlN5bWJvbCIsImZvciIsInVzZUVmZmVjdCIsIl90ZW1wIiwidDIiLCJ2YWx1ZSIsIl90ZW1wMiIsIl90ZW1wMyIsImhhbmRsZVNlbGVjdGlvbiIsInQzIiwiaGFuZGxlRXNjYXBlIiwidDQiLCJ0NSIsInQ2IiwidDciLCJsZW5ndGgiLCJtYXAiLCJfdGVtcDQiLCJ0OCIsInQ5IiwibGFiZWwiLCJ0MTAiLCJ2YWx1ZV8wIiwidDExIiwiaW5jbHVkZSIsImkiLCJwYXRoIiwiY3VycmVudF8wIiwiY3VycmVudCIsImhhc0NsYXVkZU1kRXh0ZXJuYWxJbmNsdWRlc0FwcHJvdmVkIiwiaGFzQ2xhdWRlTWRFeHRlcm5hbEluY2x1ZGVzV2FybmluZ1Nob3duIl0sInNvdXJjZXMiOlsiQ2xhdWRlTWRFeHRlcm5hbEluY2x1ZGVzRGlhbG9nLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgUmVhY3QsIHsgdXNlQ2FsbGJhY2sgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IGxvZ0V2ZW50IH0gZnJvbSAnc3JjL3NlcnZpY2VzL2FuYWx5dGljcy9pbmRleC5qcydcbmltcG9ydCB7IEJveCwgTGluaywgVGV4dCB9IGZyb20gJy4uL2luay5qcydcbmltcG9ydCB0eXBlIHsgRXh0ZXJuYWxDbGF1ZGVNZEluY2x1ZGUgfSBmcm9tICcuLi91dGlscy9jbGF1ZGVtZC5qcydcbmltcG9ydCB7IHNhdmVDdXJyZW50UHJvamVjdENvbmZpZyB9IGZyb20gJy4uL3V0aWxzL2NvbmZpZy5qcydcbmltcG9ydCB7IFNlbGVjdCB9IGZyb20gJy4vQ3VzdG9tU2VsZWN0L2luZGV4LmpzJ1xuaW1wb3J0IHsgRGlhbG9nIH0gZnJvbSAnLi9kZXNpZ24tc3lzdGVtL0RpYWxvZy5qcydcblxudHlwZSBQcm9wcyA9IHtcbiAgb25Eb25lKCk6IHZvaWRcbiAgaXNTdGFuZGFsb25lRGlhbG9nPzogYm9vbGVhblxuICBleHRlcm5hbEluY2x1ZGVzPzogRXh0ZXJuYWxDbGF1ZGVNZEluY2x1ZGVbXVxufVxuXG5leHBvcnQgZnVuY3Rpb24gQ2xhdWRlTWRFeHRlcm5hbEluY2x1ZGVzRGlhbG9nKHtcbiAgb25Eb25lLFxuICBpc1N0YW5kYWxvbmVEaWFsb2csXG4gIGV4dGVybmFsSW5jbHVkZXMsXG59OiBQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIFJlYWN0LnVzZUVmZmVjdCgoKSA9PiB7XG4gICAgLy8gTG9nIHdoZW4gZGlhbG9nIGlzIHNob3duXG4gICAgbG9nRXZlbnQoJ3Rlbmd1X2NsYXVkZV9tZF9pbmNsdWRlc19kaWFsb2dfc2hvd24nLCB7fSlcbiAgfSwgW10pXG5cbiAgY29uc3QgaGFuZGxlU2VsZWN0aW9uID0gdXNlQ2FsbGJhY2soXG4gICAgKHZhbHVlOiAneWVzJyB8ICdubycpID0+IHtcbiAgICAgIGlmICh2YWx1ZSA9PT0gJ25vJykge1xuICAgICAgICBsb2dFdmVudCgndGVuZ3VfY2xhdWRlX21kX2V4dGVybmFsX2luY2x1ZGVzX2RpYWxvZ19kZWNsaW5lZCcsIHt9KVxuICAgICAgICAvLyBNYXJrIHRoYXQgd2UndmUgc2hvd24gdGhlIGRpYWxvZyBidXQgaXQgd2FzIGRlY2xpbmVkXG4gICAgICAgIHNhdmVDdXJyZW50UHJvamVjdENvbmZpZyhjdXJyZW50ID0+ICh7XG4gICAgICAgICAgLi4uY3VycmVudCxcbiAgICAgICAgICBoYXNDbGF1ZGVNZEV4dGVybmFsSW5jbHVkZXNBcHByb3ZlZDogZmFsc2UsXG4gICAgICAgICAgaGFzQ2xhdWRlTWRFeHRlcm5hbEluY2x1ZGVzV2FybmluZ1Nob3duOiB0cnVlLFxuICAgICAgICB9KSlcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGxvZ0V2ZW50KCd0ZW5ndV9jbGF1ZGVfbWRfZXh0ZXJuYWxfaW5jbHVkZXNfZGlhbG9nX2FjY2VwdGVkJywge30pXG4gICAgICAgIHNhdmVDdXJyZW50UHJvamVjdENvbmZpZyhjdXJyZW50ID0+ICh7XG4gICAgICAgICAgLi4uY3VycmVudCxcbiAgICAgICAgICBoYXNDbGF1ZGVNZEV4dGVybmFsSW5jbHVkZXNBcHByb3ZlZDogdHJ1ZSxcbiAgICAgICAgICBoYXNDbGF1ZGVNZEV4dGVybmFsSW5jbHVkZXNXYXJuaW5nU2hvd246IHRydWUsXG4gICAgICAgIH0pKVxuICAgICAgfVxuXG4gICAgICBvbkRvbmUoKVxuICAgIH0sXG4gICAgW29uRG9uZV0sXG4gIClcblxuICBjb25zdCBoYW5kbGVFc2NhcGUgPSB1c2VDYWxsYmFjaygoKSA9PiB7XG4gICAgaGFuZGxlU2VsZWN0aW9uKCdubycpXG4gIH0sIFtoYW5kbGVTZWxlY3Rpb25dKVxuXG4gIHJldHVybiAoXG4gICAgPERpYWxvZ1xuICAgICAgdGl0bGU9XCJBbGxvdyBleHRlcm5hbCBDTEFVREUubWQgZmlsZSBpbXBvcnRzP1wiXG4gICAgICBjb2xvcj1cIndhcm5pbmdcIlxuICAgICAgb25DYW5jZWw9e2hhbmRsZUVzY2FwZX1cbiAgICAgIGhpZGVCb3JkZXI9eyFpc1N0YW5kYWxvbmVEaWFsb2d9XG4gICAgICBoaWRlSW5wdXRHdWlkZT17IWlzU3RhbmRhbG9uZURpYWxvZ31cbiAgICA+XG4gICAgICA8VGV4dD5cbiAgICAgICAgVGhpcyBwcm9qZWN0JmFwb3M7cyBDTEFVREUubWQgaW1wb3J0cyBmaWxlcyBvdXRzaWRlIHRoZSBjdXJyZW50IHdvcmtpbmdcbiAgICAgICAgZGlyZWN0b3J5LiBOZXZlciBhbGxvdyB0aGlzIGZvciB0aGlyZC1wYXJ0eSByZXBvc2l0b3JpZXMuXG4gICAgICA8L1RleHQ+XG5cbiAgICAgIHtleHRlcm5hbEluY2x1ZGVzICYmIGV4dGVybmFsSW5jbHVkZXMubGVuZ3RoID4gMCAmJiAoXG4gICAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiPlxuICAgICAgICAgIDxUZXh0IGRpbUNvbG9yPkV4dGVybmFsIGltcG9ydHM6PC9UZXh0PlxuICAgICAgICAgIHtleHRlcm5hbEluY2x1ZGVzLm1hcCgoaW5jbHVkZSwgaSkgPT4gKFxuICAgICAgICAgICAgPFRleHQga2V5PXtpfSBkaW1Db2xvcj5cbiAgICAgICAgICAgICAgeycgICd9XG4gICAgICAgICAgICAgIHtpbmNsdWRlLnBhdGh9XG4gICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgKSl9XG4gICAgICAgIDwvQm94PlxuICAgICAgKX1cblxuICAgICAgPFRleHQgZGltQ29sb3I+XG4gICAgICAgIEltcG9ydGFudDogT25seSB1c2UgQ2xhdWRlIENvZGUgd2l0aCBmaWxlcyB5b3UgdHJ1c3QuIEFjY2Vzc2luZ1xuICAgICAgICB1bnRydXN0ZWQgZmlsZXMgbWF5IHBvc2Ugc2VjdXJpdHkgcmlza3N7JyAnfVxuICAgICAgICA8TGluayB1cmw9XCJodHRwczovL2NvZGUuY2xhdWRlLmNvbS9kb2NzL2VuL3NlY3VyaXR5XCIgLz57JyAnfVxuICAgICAgPC9UZXh0PlxuXG4gICAgICA8U2VsZWN0XG4gICAgICAgIG9wdGlvbnM9e1tcbiAgICAgICAgICB7IGxhYmVsOiAnWWVzLCBhbGxvdyBleHRlcm5hbCBpbXBvcnRzJywgdmFsdWU6ICd5ZXMnIH0sXG4gICAgICAgICAgeyBsYWJlbDogJ05vLCBkaXNhYmxlIGV4dGVybmFsIGltcG9ydHMnLCB2YWx1ZTogJ25vJyB9LFxuICAgICAgICBdfVxuICAgICAgICBvbkNoYW5nZT17dmFsdWUgPT4gaGFuZGxlU2VsZWN0aW9uKHZhbHVlIGFzICd5ZXMnIHwgJ25vJyl9XG4gICAgICAvPlxuICAgIDwvRGlhbG9nPlxuICApXG59XG4iXSwibWFwcGluZ3MiOiI7QUFBQSxPQUFPQSxLQUFL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 \ No newline at end of file diff --git a/src/components/ClickableImageRef.tsx b/src/components/ClickableImageRef.tsx new file mode 100644 index 0000000..ff48a72 --- /dev/null +++ b/src/components/ClickableImageRef.tsx @@ -0,0 +1,73 @@ +import { c as _c } from "react/compiler-runtime"; +import * as React from 'react'; +import { pathToFileURL } from 'url'; +import Link from '../ink/components/Link.js'; +import { supportsHyperlinks } from '../ink/supports-hyperlinks.js'; +import { Text } from '../ink.js'; +import { getStoredImagePath } from '../utils/imageStore.js'; +import type { Theme } from '../utils/theme.js'; +type Props = { + imageId: number; + backgroundColor?: keyof Theme; + isSelected?: boolean; +}; + +/** + * Renders an image reference like [Image #1] as a clickable link. + * When clicked, opens the stored image file in the default viewer. + * + * Falls back to styled text if: + * - Terminal doesn't support hyperlinks + * - Image file is not found in the store + */ +export function ClickableImageRef(t0) { + const $ = _c(13); + const { + imageId, + backgroundColor, + isSelected: t1 + } = t0; + const isSelected = t1 === undefined ? false : t1; + const imagePath = getStoredImagePath(imageId); + const displayText = `[Image #${imageId}]`; + if (imagePath && supportsHyperlinks()) { + const fileUrl = pathToFileURL(imagePath).href; + let t2; + let t3; + if ($[0] !== backgroundColor || $[1] !== displayText || $[2] !== isSelected) { + t2 = {displayText}; + t3 = {displayText}; + $[0] = backgroundColor; + $[1] = displayText; + $[2] = isSelected; + $[3] = t2; + $[4] = t3; + } else { + t2 = $[3]; + t3 = $[4]; + } + let t4; + if ($[5] !== fileUrl || $[6] !== t2 || $[7] !== t3) { + t4 = {t3}; + $[5] = fileUrl; + $[6] = t2; + $[7] = t3; + $[8] = t4; + } else { + t4 = $[8]; + } + return t4; + } + let t2; + if ($[9] !== backgroundColor || $[10] !== displayText || $[11] !== isSelected) { + t2 = {displayText}; + $[9] = backgroundColor; + $[10] = displayText; + $[11] = isSelected; + $[12] = t2; + } else { + t2 = $[12]; + } + return t2; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsInBhdGhUb0ZpbGVVUkwiLCJMaW5rIiwic3VwcG9ydHNIeXBlcmxpbmtzIiwiVGV4dCIsImdldFN0b3JlZEltYWdlUGF0aCIsIlRoZW1lIiwiUHJvcHMiLCJpbWFnZUlkIiwiYmFja2dyb3VuZENvbG9yIiwiaXNTZWxlY3RlZCIsIkNsaWNrYWJsZUltYWdlUmVmIiwidDAiLCIkIiwiX2MiLCJ0MSIsInVuZGVmaW5lZCIsImltYWdlUGF0aCIsImRpc3BsYXlUZXh0IiwiZmlsZVVybCIsImhyZWYiLCJ0MiIsInQzIiwidDQiXSwic291cmNlcyI6WyJDbGlja2FibGVJbWFnZVJlZi50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBwYXRoVG9GaWxlVVJMIH0gZnJvbSAndXJsJ1xuaW1wb3J0IExpbmsgZnJvbSAnLi4vaW5rL2NvbXBvbmVudHMvTGluay5qcydcbmltcG9ydCB7IHN1cHBvcnRzSHlwZXJsaW5rcyB9IGZyb20gJy4uL2luay9zdXBwb3J0cy1oeXBlcmxpbmtzLmpzJ1xuaW1wb3J0IHsgVGV4dCB9IGZyb20gJy4uL2luay5qcydcbmltcG9ydCB7IGdldFN0b3JlZEltYWdlUGF0aCB9IGZyb20gJy4uL3V0aWxzL2ltYWdlU3RvcmUuanMnXG5pbXBvcnQgdHlwZSB7IFRoZW1lIH0gZnJvbSAnLi4vdXRpbHMvdGhlbWUuanMnXG5cbnR5cGUgUHJvcHMgPSB7XG4gIGltYWdlSWQ6IG51bWJlclxuICBiYWNrZ3JvdW5kQ29sb3I/OiBrZXlvZiBUaGVtZVxuICBpc1NlbGVjdGVkPzogYm9vbGVhblxufVxuXG4vKipcbiAqIFJlbmRlcnMgYW4gaW1hZ2UgcmVmZXJlbmNlIGxpa2UgW0ltYWdlICMxXSBhcyBhIGNsaWNrYWJsZSBsaW5rLlxuICogV2hlbiBjbGlja2VkLCBvcGVucyB0aGUgc3RvcmVkIGltYWdlIGZpbGUgaW4gdGhlIGRlZmF1bHQgdmlld2VyLlxuICpcbiAqIEZhbGxzIGJhY2sgdG8gc3R5bGVkIHRleHQgaWY6XG4gKiAtIFRlcm1pbmFsIGRvZXNuJ3Qgc3VwcG9ydCBoeXBlcmxpbmtzXG4gKiAtIEltYWdlIGZpbGUgaXMgbm90IGZvdW5kIGluIHRoZSBzdG9yZVxuICovXG5leHBvcnQgZnVuY3Rpb24gQ2xpY2thYmxlSW1hZ2VSZWYoe1xuICBpbWFnZUlkLFxuICBiYWNrZ3JvdW5kQ29sb3IsXG4gIGlzU2VsZWN0ZWQgPSBmYWxzZSxcbn06IFByb3BzKTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgY29uc3QgaW1hZ2VQYXRoID0gZ2V0U3RvcmVkSW1hZ2VQYXRoKGltYWdlSWQpXG4gIGNvbnN0IGRpc3BsYXlUZXh0ID0gYFtJbWFnZSAjJHtpbWFnZUlkfV1gXG5cbiAgLy8gSWYgd2UgaGF2ZSBhIHN0b3JlZCBpbWFnZSBhbmQgdGVybWluYWwgc3VwcG9ydHMgaHlwZXJsaW5rcywgbWFrZSBpdCBjbGlja2FibGVcbiAgaWYgKGltYWdlUGF0aCAmJiBzdXBwb3J0c0h5cGVybGlua3MoKSkge1xuICAgIGNvbnN0IGZpbGVVcmwgPSBwYXRoVG9GaWxlVVJMKGltYWdlUGF0aCkuaHJlZlxuXG4gICAgcmV0dXJuIChcbiAgICAgIDxMaW5rXG4gICAgICAgIHVybD17ZmlsZVVybH1cbiAgICAgICAgZmFsbGJhY2s9e1xuICAgICAgICAgIDxUZXh0IGJhY2tncm91bmRDb2xvcj17YmFja2dyb3VuZENvbG9yfSBpbnZlcnNlPXtpc1NlbGVjdGVkfT5cbiAgICAgICAgICAgIHtkaXNwbGF5VGV4dH1cbiAgICAgICAgICA8L1RleHQ+XG4gICAgICAgIH1cbiAgICAgID5cbiAgICAgICAgPFRleHRcbiAgICAgICAgICBiYWNrZ3JvdW5kQ29sb3I9e2JhY2tncm91bmRDb2xvcn1cbiAgICAgICAgICBpbnZlcnNlPXtpc1NlbGVjdGVkfVxuICAgICAgICAgIGJvbGQ9e2lzU2VsZWN0ZWR9XG4gICAgICAgID5cbiAgICAgICAgICB7ZGlzcGxheVRleHR9XG4gICAgICAgIDwvVGV4dD5cbiAgICAgIDwvTGluaz5cbiAgICApXG4gIH1cblxuICAvLyBGYWxsYmFjazogc3R5bGVkIGJ1dCBub3QgY2xpY2thYmxlXG4gIHJldHVybiAoXG4gICAgPFRleHQgYmFja2dyb3VuZENvbG9yPXtiYWNrZ3JvdW5kQ29sb3J9IGludmVyc2U9e2lzU2VsZWN0ZWR9PlxuICAgICAge2Rpc3BsYXlUZXh0fVxuICAgIDwvVGV4dD5cbiAgKVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsT0FBTyxLQUFLQSxLQUFLLE1BQU0sT0FBTztBQUM5QixTQUFTQyxhQUFhLFFBQVEsS0FBSztBQUNuQyxPQUFPQyxJQUFJLE1BQU0sMkJBQTJCO0FBQzVDLFNBQVNDLGtCQUFrQixRQUFRLCtCQUErQjtBQUNsRSxTQUFTQyxJQUFJLFFBQVEsV0FBVztBQUNoQyxTQUFTQyxrQkFBa0IsUUFBUSx3QkFBd0I7QUFDM0QsY0FBY0MsS0FBSyxRQUFRLG1CQUFtQjtBQUU5QyxLQUFLQyxLQUFLLEdBQUc7RUFDWEMsT0FBTyxFQUFFLE1BQU07RUFDZkMsZUFBZSxDQUFDLEVBQUUsTUFBTUgsS0FBSztFQUM3QkksVUFBVSxDQUFDLEVBQUUsT0FBTztBQUN0QixDQUFDOztBQUVEO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxPQUFPLFNBQUFDLGtCQUFBQyxFQUFBO0VBQUEsTUFBQUMsQ0FBQSxHQUFBQyxFQUFBO0VBQTJCO0lBQUFOLE9BQUE7SUFBQUMsZUFBQTtJQUFBQyxVQUFBLEVBQUFLO0VBQUEsSUFBQUgsRUFJMUI7RUFETixNQUFBRixVQUFBLEdBQUFLLEVBQWtCLEtBQWxCQyxTQUFrQixHQUFsQixLQUFrQixHQUFsQkQsRUFBa0I7RUFFbEIsTUFBQUUsU0FBQSxHQUFrQlosa0JBQWtCLENBQUNHLE9BQU8sQ0FBQztFQUM3QyxNQUFBVSxXQUFBLEdBQW9CLFdBQVdWLE9BQU8sR0FBRztFQUd6QyxJQUFJUyxTQUFpQyxJQUFwQmQsa0JBQWtCLENBQUMsQ0FBQztJQUNuQyxNQUFBZ0IsT0FBQSxHQUFnQmxCLGFBQWEsQ0FBQ2dCLFNBQVMsQ0FBQyxDQUFBRyxJQUFLO0lBQUEsSUFBQUMsRUFBQTtJQUFBLElBQUFDLEVBQUE7SUFBQSxJQUFBVCxDQUFBLFFBQUFKLGVBQUEsSUFBQUksQ0FBQSxRQUFBSyxXQUFBLElBQUFMLENBQUEsUUFBQUgsVUFBQTtNQU12Q1csRUFBQSxJQUFDLElBQUksQ0FBa0JaLGVBQWUsQ0FBZkEsZ0JBQWMsQ0FBQyxDQUFXQyxPQUFVLENBQVZBLFdBQVMsQ0FBQyxDQUN4RFEsWUFBVSxDQUNiLEVBRkMsSUFBSSxDQUVFO01BR1RJLEVBQUEsSUFBQyxJQUFJLENBQ2NiLGVBQWUsQ0FBZkEsZ0JBQWMsQ0FBQyxDQUN2QkMsT0FBVSxDQUFWQSxXQUFTLENBQUMsQ0FDYkEsSUFBVSxDQUFWQSxXQUFTLENBQUMsQ0FFZlEsWUFBVSxDQUNiLEVBTkMsSUFBSSxDQU1FO01BQUFMLENBQUEsTUFBQUosZUFBQTtNQUFBSSxDQUFBLE1BQUFLLFdBQUE7TUFBQUwsQ0FBQSxNQUFBSCxVQUFBO01BQUFHLENBQUEsTUFBQVEsRUFBQTtNQUFBUixDQUFBLE1BQUFTLEVBQUE7SUFBQTtNQUFBRCxFQUFBLEdBQUFSLENBQUE7TUFBQVMsRUFBQSxHQUFBVCxDQUFBO0lBQUE7SUFBQSxJQUFBVSxFQUFBO0lBQUEsSUFBQVYsQ0FBQSxRQUFBTSxPQUFBLElBQUFOLENBQUEsUUFBQVEsRUFBQSxJQUFBUixDQUFBLFFBQUFTLEVBQUE7TUFkVEMsRUFBQSxJQUFDLElBQUksQ0FDRUosR0FBTyxDQUFQQSxRQUFNLENBQUMsQ0FFVixRQUVPLENBRlAsQ0FBQUUsRUFFTSxDQUFDLENBR1QsQ0FBQUMsRUFNTSxDQUNSLEVBZkMsSUFBSSxDQWVFO01BQUFULENBQUEsTUFBQU0sT0FBQTtNQUFBTixDQUFBLE1BQUFRLEVBQUE7TUFBQVIsQ0FBQSxNQUFBUyxFQUFBO01BQUFULENBQUEsTUFBQVUsRUFBQTtJQUFBO01BQUFBLEVBQUEsR0FBQVYsQ0FBQTtJQUFBO0lBQUEsT0FmUFUsRUFlTztFQUFBO0VBRVYsSUFBQUYsRUFBQTtFQUFBLElBQUFSLENBQUEsUUFBQUosZUFBQSxJQUFBSSxDQUFBLFNBQUFLLFdBQUEsSUFBQUwsQ0FBQSxTQUFBSCxVQUFBO0lBSUNXLEVBQUEsSUFBQyxJQUFJLENBQWtCWixlQUFlLENBQWZBLGdCQUFjLENBQUMsQ0FBV0MsT0FBVSxDQUFWQSxXQUFTLENBQUMsQ0FDeERRLFlBQVUsQ0FDYixFQUZDLElBQUksQ0FFRTtJQUFBTCxDQUFBLE1BQUFKLGVBQUE7SUFBQUksQ0FBQSxPQUFBSyxXQUFBO0lBQUFMLENBQUEsT0FBQUgsVUFBQTtJQUFBRyxDQUFBLE9BQUFRLEVBQUE7RUFBQTtJQUFBQSxFQUFBLEdBQUFSLENBQUE7RUFBQTtFQUFBLE9BRlBRLEVBRU87QUFBQSIsImlnbm9yZUxpc3QiOltdfQ== \ No newline at end of file diff --git a/src/components/CompactSummary.tsx b/src/components/CompactSummary.tsx new file mode 100644 index 0000000..72e1b18 --- /dev/null +++ b/src/components/CompactSummary.tsx @@ -0,0 +1,118 @@ +import { c as _c } from "react/compiler-runtime"; +import * as React from 'react'; +import { BLACK_CIRCLE } from '../constants/figures.js'; +import { Box, Text } from '../ink.js'; +import type { Screen } from '../screens/REPL.js'; +import type { NormalizedUserMessage } from '../types/message.js'; +import { getUserMessageText } from '../utils/messages.js'; +import { ConfigurableShortcutHint } from './ConfigurableShortcutHint.js'; +import { MessageResponse } from './MessageResponse.js'; +type Props = { + message: NormalizedUserMessage; + screen: Screen; +}; +export function CompactSummary(t0) { + const $ = _c(24); + const { + message, + screen + } = t0; + const isTranscriptMode = screen === "transcript"; + let t1; + if ($[0] !== message) { + t1 = getUserMessageText(message) || ""; + $[0] = message; + $[1] = t1; + } else { + t1 = $[1]; + } + const textContent = t1; + const metadata = message.summarizeMetadata; + if (metadata) { + let t2; + if ($[2] === Symbol.for("react.memo_cache_sentinel")) { + t2 = {BLACK_CIRCLE}; + $[2] = t2; + } else { + t2 = $[2]; + } + let t3; + if ($[3] === Symbol.for("react.memo_cache_sentinel")) { + t3 = Summarized conversation; + $[3] = t3; + } else { + t3 = $[3]; + } + let t4; + if ($[4] !== isTranscriptMode || $[5] !== metadata) { + t4 = !isTranscriptMode && Summarized {metadata.messagesSummarized} messages{" "}{metadata.direction === "up_to" ? "up to this point" : "from this point"}{metadata.userContext && Context: {"\u201C"}{metadata.userContext}{"\u201D"}}; + $[4] = isTranscriptMode; + $[5] = metadata; + $[6] = t4; + } else { + t4 = $[6]; + } + let t5; + if ($[7] !== isTranscriptMode || $[8] !== textContent) { + t5 = isTranscriptMode && {textContent}; + $[7] = isTranscriptMode; + $[8] = textContent; + $[9] = t5; + } else { + t5 = $[9]; + } + let t6; + if ($[10] !== t4 || $[11] !== t5) { + t6 = {t2}{t3}{t4}{t5}; + $[10] = t4; + $[11] = t5; + $[12] = t6; + } else { + t6 = $[12]; + } + return t6; + } + let t2; + if ($[13] === Symbol.for("react.memo_cache_sentinel")) { + t2 = {BLACK_CIRCLE}; + $[13] = t2; + } else { + t2 = $[13]; + } + let t3; + if ($[14] !== isTranscriptMode) { + t3 = !isTranscriptMode && {" "}; + $[14] = isTranscriptMode; + $[15] = t3; + } else { + t3 = $[15]; + } + let t4; + if ($[16] !== t3) { + t4 = {t2}Compact summary{t3}; + $[16] = t3; + $[17] = t4; + } else { + t4 = $[17]; + } + let t5; + if ($[18] !== isTranscriptMode || $[19] !== textContent) { + t5 = isTranscriptMode && {textContent}; + $[18] = isTranscriptMode; + $[19] = textContent; + $[20] = t5; + } else { + t5 = $[20]; + } + let t6; + if ($[21] !== t4 || $[22] !== t5) { + t6 = {t4}{t5}; + $[21] = t4; + $[22] = t5; + $[23] = t6; + } else { + t6 = $[23]; + } + return t6; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkJMQUNLX0NJUkNMRSIsIkJveCIsIlRleHQiLCJTY3JlZW4iLCJOb3JtYWxpemVkVXNlck1lc3NhZ2UiLCJnZXRVc2VyTWVzc2FnZVRleHQiLCJDb25maWd1cmFibGVTaG9ydGN1dEhpbnQiLCJNZXNzYWdlUmVzcG9uc2UiLCJQcm9wcyIsIm1lc3NhZ2UiLCJzY3JlZW4iLCJDb21wYWN0U3VtbWFyeSIsInQwIiwiJCIsIl9jIiwiaXNUcmFuc2NyaXB0TW9kZSIsInQxIiwidGV4dENvbnRlbnQiLCJtZXRhZGF0YSIsInN1bW1hcml6ZU1ldGFkYXRhIiwidDIiLCJTeW1ib2wiLCJmb3IiLCJ0MyIsInQ0IiwibWVzc2FnZXNTdW1tYXJpemVkIiwiZGlyZWN0aW9uIiwidXNlckNvbnRleHQiLCJ0NSIsInQ2Il0sInNvdXJjZXMiOlsiQ29tcGFjdFN1bW1hcnkudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgQkxBQ0tfQ0lSQ0xFIH0gZnJvbSAnLi4vY29uc3RhbnRzL2ZpZ3VyZXMuanMnXG5pbXBvcnQgeyBCb3gsIFRleHQgfSBmcm9tICcuLi9pbmsuanMnXG5pbXBvcnQgdHlwZSB7IFNjcmVlbiB9IGZyb20gJy4uL3NjcmVlbnMvUkVQTC5qcydcbmltcG9ydCB0eXBlIHsgTm9ybWFsaXplZFVzZXJNZXNzYWdlIH0gZnJvbSAnLi4vdHlwZXMvbWVzc2FnZS5qcydcbmltcG9ydCB7IGdldFVzZXJNZXNzYWdlVGV4dCB9IGZyb20gJy4uL3V0aWxzL21lc3NhZ2VzLmpzJ1xuaW1wb3J0IHsgQ29uZmlndXJhYmxlU2hvcnRjdXRIaW50IH0gZnJvbSAnLi9Db25maWd1cmFibGVTaG9ydGN1dEhpbnQuanMnXG5pbXBvcnQgeyBNZXNzYWdlUmVzcG9uc2UgfSBmcm9tICcuL01lc3NhZ2VSZXNwb25zZS5qcydcblxudHlwZSBQcm9wcyA9IHtcbiAgbWVzc2FnZTogTm9ybWFsaXplZFVzZXJNZXNzYWdlXG4gIHNjcmVlbjogU2NyZWVuXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBDb21wYWN0U3VtbWFyeSh7IG1lc3NhZ2UsIHNjcmVlbiB9OiBQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IGlzVHJhbnNjcmlwdE1vZGUgPSBzY3JlZW4gPT09ICd0cmFuc2NyaXB0J1xuICBjb25zdCB0ZXh0Q29udGVudCA9IGdldFVzZXJNZXNzYWdlVGV4dChtZXNzYWdlKSB8fCAnJ1xuICBjb25zdCBtZXRhZGF0YSA9IG1lc3NhZ2Uuc3VtbWFyaXplTWV0YWRhdGFcblxuICAvLyBcIlN1bW1hcml6ZSBmcm9tIGhlcmVcIiB3aXRoIG1ldGFkYXRhXG4gIGlmIChtZXRhZGF0YSkge1xuICAgIHJldHVybiAoXG4gICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIiBtYXJnaW5Ub3A9ezF9PlxuICAgICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJyb3dcIj5cbiAgICAgICAgICA8Qm94IG1pbldpZHRoPXsyfT5cbiAgICAgICAgICAgIDxUZXh0IGNvbG9yPVwidGV4dFwiPntCTEFDS19DSVJDTEV9PC9UZXh0PlxuICAgICAgICAgIDwvQm94PlxuICAgICAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiPlxuICAgICAgICAgICAgPFRleHQgYm9sZD5TdW1tYXJpemVkIGNvbnZlcnNhdGlvbjwvVGV4dD5cbiAgICAgICAgICAgIHshaXNUcmFuc2NyaXB0TW9kZSAmJiAoXG4gICAgICAgICAgICAgIDxNZXNzYWdlUmVzcG9uc2U+XG4gICAgICAgICAgICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCI+XG4gICAgICAgICAgICAgICAgICA8VGV4dCBkaW1Db2xvcj5cbiAgICAgICAgICAgICAgICAgICAgU3VtbWFyaXplZCB7bWV0YWRhdGEubWVzc2FnZXNTdW1tYXJpemVkfSBtZXNzYWdlc3snICd9XG4gICAgICAgICAgICAgICAgICAgIHttZXRhZGF0YS5kaXJlY3Rpb24gPT09ICd1cF90bydcbiAgICAgICAgICAgICAgICAgICAgICA/ICd1cCB0byB0aGlzIHBvaW50J1xuICAgICAgICAgICAgICAgICAgICAgIDogJ2Zyb20gdGhpcyBwb2ludCd9XG4gICAgICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICAgICAgICB7bWV0YWRhdGEudXNlckNvbnRleHQgJiYgKFxuICAgICAgICAgICAgICAgICAgICA8VGV4dCBkaW1Db2xvcj5cbiAgICAgICAgICAgICAgICAgICAgICBDb250ZXh0OiB7J1xcdTIwMWMnfVxuICAgICAgICAgICAgICAgICAgICAgIHttZXRhZGF0YS51c2VyQ29udGV4dH1cbiAgICAgICAgICAgICAgICAgICAgICB7J1xcdTIwMWQnfVxuICAgICAgICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgICAgICAgPFRleHQgZGltQ29sb3I+XG4gICAgICAgICAgICAgICAgICAgIDxDb25maWd1cmFibGVTaG9ydGN1dEhpbnRcbiAgICAgICAgICAgICAgICAgICAgICBhY3Rpb249XCJhcHA6dG9nZ2xlVHJhbnNjcmlwdFwiXG4gICAgICAgICAgICAgICAgICAgICAgY29udGV4dD1cIkdsb2JhbFwiXG4gICAgICAgICAgICAgICAgICAgICAgZmFsbGJhY2s9XCJjdHJsK29cIlxuICAgICAgICAgICAgICAgICAgICAgIGRlc2NyaXB0aW9uPVwiZXhwYW5kIGhpc3RvcnlcIlxuICAgICAgICAgICAgICAgICAgICAgIHBhcmVuc1xuICAgICAgICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgICAgIDwvQm94PlxuICAgICAgICAgICAgICA8L01lc3NhZ2VSZXNwb25zZT5cbiAgICAgICAgICAgICl9XG4gICAgICAgICAgICB7aXNUcmFuc2NyaXB0TW9kZSAmJiAoXG4gICAgICAgICAgICAgIDxNZXNzYWdlUmVzcG9uc2U+XG4gICAgICAgICAgICAgICAgPFRleHQ+e3RleHRDb250ZW50fTwvVGV4dD5cbiAgICAgICAgICAgICAgPC9NZXNzYWdlUmVzcG9uc2U+XG4gICAgICAgICAgICApfVxuICAgICAgICAgIDwvQm94PlxuICAgICAgICA8L0JveD5cbiAgICAgIDwvQm94PlxuICAgIClcbiAgfVxuXG4gIC8vIERlZmF1bHQgY29tcGFjdCBzdW1tYXJ5IChhdXRvLWNvbXBhY3QpXG4gIHJldHVybiAoXG4gICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCIgbWFyZ2luVG9wPXsxfT5cbiAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cInJvd1wiPlxuICAgICAgICA8Qm94IG1pbldpZHRoPXsyfT5cbiAgICAgICAgICA8VGV4dCBjb2xvcj1cInRleHRcIj57QkxBQ0tfQ0lSQ0xFfTwvVGV4dD5cbiAgICAgICAgPC9Cb3g+XG4gICAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiPlxuICAgICAgICAgIDxUZXh0IGJvbGQ+XG4gICAgICAgICAgICBDb21wYWN0IHN1bW1hcnlcbiAgICAgICAgICAgIHshaXNUcmFuc2NyaXB0TW9kZSAmJiAoXG4gICAgICAgICAgICAgIDxUZXh0IGRpbUNvbG9yPlxuICAgICAgICAgICAgICAgIHsnICd9XG4gICAgICAgICAgICAgICAgPENvbmZpZ3VyYWJsZVNob3J0Y3V0SGludFxuICAgICAgICAgICAgICAgICAgYWN0aW9uPVwiYXBwOnRvZ2dsZVRyYW5zY3JpcHRcIlxuICAgICAgICAgICAgICAgICAgY29udGV4dD1cIkdsb2JhbFwiXG4gICAgICAgICAgICAgICAgICBmYWxsYmFjaz1cImN0cmwrb1wiXG4gICAgICAgICAgICAgICAgICBkZXNjcmlwdGlvbj1cImV4cGFuZFwiXG4gICAgICAgICAgICAgICAgICBwYXJlbnNcbiAgICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICApfVxuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgPC9Cb3g+XG4gICAgICA8L0JveD5cbiAgICAgIHtpc1RyYW5zY3JpcHRNb2RlICYmIChcbiAgICAgICAgPE1lc3NhZ2VSZXNwb25zZT5cbiAgICAgICAgICA8VGV4dD57dGV4dENvbnRlbnR9PC9UZXh0PlxuICAgICAgICA8L01lc3NhZ2VSZXNwb25zZT5cbiAgICAgICl9XG4gICAgPC9Cb3g+XG4gIClcbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU8sS0FBS0EsS0FBSyxNQUFNLE9BQU87QUFDOUIsU0FBU0MsWUFBWSxRQUFRLHlCQUF5QjtBQUN0RCxTQUFTQyxHQUFHLEVBQUVDLElBQUksUUFBUSxXQUFXO0FBQ3JDLGNBQWNDLE1BQU0sUUFBUSxvQkFBb0I7QUFDaEQsY0FBY0MscUJBQXFCLFFBQVEscUJBQXFCO0FBQ2hFLFNBQVNDLGtCQUFrQixRQUFRLHNCQUFzQjtBQUN6RCxTQUFTQyx3QkFBd0IsUUFBUSwrQkFBK0I7QUFDeEUsU0FBU0MsZUFBZSxRQUFRLHNCQUFzQjtBQUV0RCxLQUFLQyxLQUFLLEdBQUc7RUFDWEMsT0FBTyxFQUFFTCxxQkFBcUI7RUFDOUJNLE1BQU0sRUFBRVAsTUFBTTtBQUNoQixDQUFDO0FBRUQsT0FBTyxTQUFBUSxlQUFBQyxFQUFBO0VBQUEsTUFBQUMsQ0FBQSxHQUFBQyxFQUFBO0VBQXdCO0lBQUFMLE9BQUE7SUFBQUM7RUFBQSxJQUFBRSxFQUEwQjtFQUN2RCxNQUFBRyxnQkFBQSxHQUF5QkwsTUFBTSxLQUFLLFlBQVk7RUFBQSxJQUFBTSxFQUFBO0VBQUEsSUFBQUgsQ0FBQSxRQUFBSixPQUFBO0lBQzVCTyxFQUFBLEdBQUFYLGtCQUFrQixDQUFDSSxPQUFhLENBQUMsSUFBakMsRUFBaUM7SUFBQUksQ0FBQSxNQUFBSixPQUFBO0lBQUFJLENBQUEsTUFBQUcsRUFBQTtFQUFBO0lBQUFBLEVBQUEsR0FBQUgsQ0FBQTtFQUFBO0VBQXJELE1BQUFJLFdBQUEsR0FBb0JELEVBQWlDO0VBQ3JELE1BQUFFLFFBQUEsR0FBaUJULE9BQU8sQ0FBQVUsaUJBQWtCO0VBRzFDLElB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 \ No newline at end of file diff --git a/src/components/ConfigurableShortcutHint.tsx b/src/components/ConfigurableShortcutHint.tsx new file mode 100644 index 0000000..e783da5 --- /dev/null +++ b/src/components/ConfigurableShortcutHint.tsx @@ -0,0 +1,57 @@ +import { c as _c } from "react/compiler-runtime"; +import * as React from 'react'; +import type { KeybindingAction, KeybindingContextName } from '../keybindings/types.js'; +import { useShortcutDisplay } from '../keybindings/useShortcutDisplay.js'; +import { KeyboardShortcutHint } from './design-system/KeyboardShortcutHint.js'; +type Props = { + /** The keybinding action (e.g., 'app:toggleTranscript') */ + action: KeybindingAction; + /** The keybinding context (e.g., 'Global') */ + context: KeybindingContextName; + /** Default shortcut if keybinding not configured */ + fallback: string; + /** The action description text (e.g., 'expand') */ + description: string; + /** Whether to wrap in parentheses */ + parens?: boolean; + /** Whether to show in bold */ + bold?: boolean; +}; + +/** + * KeyboardShortcutHint that displays the user-configured shortcut. + * Falls back to default if keybinding context is not available. + * + * @example + * + */ +export function ConfigurableShortcutHint(t0) { + const $ = _c(5); + const { + action, + context, + fallback, + description, + parens, + bold + } = t0; + const shortcut = useShortcutDisplay(action, context, fallback); + let t1; + if ($[0] !== bold || $[1] !== description || $[2] !== parens || $[3] !== shortcut) { + t1 = ; + $[0] = bold; + $[1] = description; + $[2] = parens; + $[3] = shortcut; + $[4] = t1; + } else { + t1 = $[4]; + } + return t1; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIktleWJpbmRpbmdBY3Rpb24iLCJLZXliaW5kaW5nQ29udGV4dE5hbWUiLCJ1c2VTaG9ydGN1dERpc3BsYXkiLCJLZXlib2FyZFNob3J0Y3V0SGludCIsIlByb3BzIiwiYWN0aW9uIiwiY29udGV4dCIsImZhbGxiYWNrIiwiZGVzY3JpcHRpb24iLCJwYXJlbnMiLCJib2xkIiwiQ29uZmlndXJhYmxlU2hvcnRjdXRIaW50IiwidDAiLCIkIiwiX2MiLCJzaG9ydGN1dCIsInQxIl0sInNvdXJjZXMiOlsiQ29uZmlndXJhYmxlU2hvcnRjdXRIaW50LnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB0eXBlIHtcbiAgS2V5YmluZGluZ0FjdGlvbixcbiAgS2V5YmluZGluZ0NvbnRleHROYW1lLFxufSBmcm9tICcuLi9rZXliaW5kaW5ncy90eXBlcy5qcydcbmltcG9ydCB7IHVzZVNob3J0Y3V0RGlzcGxheSB9IGZyb20gJy4uL2tleWJpbmRpbmdzL3VzZVNob3J0Y3V0RGlzcGxheS5qcydcbmltcG9ydCB7IEtleWJvYXJkU2hvcnRjdXRIaW50IH0gZnJvbSAnLi9kZXNpZ24tc3lzdGVtL0tleWJvYXJkU2hvcnRjdXRIaW50LmpzJ1xuXG50eXBlIFByb3BzID0ge1xuICAvKiogVGhlIGtleWJpbmRpbmcgYWN0aW9uIChlLmcuLCAnYXBwOnRvZ2dsZVRyYW5zY3JpcHQnKSAqL1xuICBhY3Rpb246IEtleWJpbmRpbmdBY3Rpb25cbiAgLyoqIFRoZSBrZXliaW5kaW5nIGNvbnRleHQgKGUuZy4sICdHbG9iYWwnKSAqL1xuICBjb250ZXh0OiBLZXliaW5kaW5nQ29udGV4dE5hbWVcbiAgLyoqIERlZmF1bHQgc2hvcnRjdXQgaWYga2V5YmluZGluZyBub3QgY29uZmlndXJlZCAqL1xuICBmYWxsYmFjazogc3RyaW5nXG4gIC8qKiBUaGUgYWN0aW9uIGRlc2NyaXB0aW9uIHRleHQgKGUuZy4sICdleHBhbmQnKSAqL1xuICBkZXNjcmlwdGlvbjogc3RyaW5nXG4gIC8qKiBXaGV0aGVyIHRvIHdyYXAgaW4gcGFyZW50aGVzZXMgKi9cbiAgcGFyZW5zPzogYm9vbGVhblxuICAvKiogV2hldGhlciB0byBzaG93IGluIGJvbGQgKi9cbiAgYm9sZD86IGJvb2xlYW5cbn1cblxuLyoqXG4gKiBLZXlib2FyZFNob3J0Y3V0SGludCB0aGF0IGRpc3BsYXlzIHRoZSB1c2VyLWNvbmZpZ3VyZWQgc2hvcnRjdXQuXG4gKiBGYWxscyBiYWNrIHRvIGRlZmF1bHQgaWYga2V5YmluZGluZyBjb250ZXh0IGlzIG5vdCBhdmFpbGFibGUuXG4gKlxuICogQGV4YW1wbGVcbiAqIDxDb25maWd1cmFibGVTaG9ydGN1dEhpbnRcbiAqICAgYWN0aW9uPVwiYXBwOnRvZ2dsZVRyYW5zY3JpcHRcIlxuICogICBjb250ZXh0PVwiR2xvYmFsXCJcbiAqICAgZmFsbGJhY2s9XCJjdHJsK29cIlxuICogICBkZXNjcmlwdGlvbj1cImV4cGFuZFwiXG4gKiAvPlxuICovXG5leHBvcnQgZnVuY3Rpb24gQ29uZmlndXJhYmxlU2hvcnRjdXRIaW50KHtcbiAgYWN0aW9uLFxuICBjb250ZXh0LFxuICBmYWxsYmFjayxcbiAgZGVzY3JpcHRpb24sXG4gIHBhcmVucyxcbiAgYm9sZCxcbn06IFByb3BzKTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgY29uc3Qgc2hvcnRjdXQgPSB1c2VTaG9ydGN1dERpc3BsYXkoYWN0aW9uLCBjb250ZXh0LCBmYWxsYmFjaylcbiAgcmV0dXJuIChcbiAgICA8S2V5Ym9hcmRTaG9ydGN1dEhpbnRcbiAgICAgIHNob3J0Y3V0PXtzaG9ydGN1dH1cbiAgICAgIGFjdGlvbj17ZGVzY3JpcHRpb259XG4gICAgICBwYXJlbnM9e3BhcmVuc31cbiAgICAgIGJvbGQ9e2JvbGR9XG4gICAgLz5cbiAgKVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsT0FBTyxLQUFLQSxLQUFLLE1BQU0sT0FBTztBQUM5QixjQUNFQyxnQkFBZ0IsRUFDaEJDLHFCQUFxQixRQUNoQix5QkFBeUI7QUFDaEMsU0FBU0Msa0JBQWtCLFFBQVEsc0NBQXNDO0FBQ3pFLFNBQVNDLG9CQUFvQixRQUFRLHlDQUF5QztBQUU5RSxLQUFLQyxLQUFLLEdBQUc7RUFDWDtFQUNBQyxNQUFNLEVBQUVMLGdCQUFnQjtFQUN4QjtFQUNBTSxPQUFPLEVBQUVMLHFCQUFxQjtFQUM5QjtFQUNBTSxRQUFRLEVBQUUsTUFBTTtFQUNoQjtFQUNBQyxXQUFXLEVBQUUsTUFBTTtFQUNuQjtFQUNBQyxNQUFNLENBQUMsRUFBRSxPQUFPO0VBQ2hCO0VBQ0FDLElBQUksQ0FBQyxFQUFFLE9BQU87QUFDaEIsQ0FBQzs7QUFFRDtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxPQUFPLFNBQUFDLHlCQUFBQyxFQUFBO0VBQUEsTUFBQUMsQ0FBQSxHQUFBQyxFQUFBO0VBQWtDO0lBQUFULE1BQUE7SUFBQUMsT0FBQTtJQUFBQyxRQUFBO0lBQUFDLFdBQUE7SUFBQUMsTUFBQTtJQUFBQztFQUFBLElBQUFFLEVBT2pDO0VBQ04sTUFBQUcsUUFBQSxHQUFpQmIsa0JBQWtCLENBQUNHLE1BQU0sRUFBRUMsT0FBTyxFQUFFQyxRQUFRLENBQUM7RUFBQSxJQUFBUyxFQUFBO0VBQUEsSUFBQUgsQ0FBQSxRQUFBSCxJQUFBLElBQUFHLENBQUEsUUFBQUwsV0FBQSxJQUFBSyxDQUFBLFFBQUFKLE1BQUEsSUFBQUksQ0FBQSxRQUFBRSxRQUFBO0lBRTVEQyxFQUFBLElBQUMsb0JBQW9CLENBQ1RELFFBQVEsQ0FBUkEsU0FBTyxDQUFDLENBQ1ZQLE1BQVcsQ0FBWEEsWUFBVSxDQUFDLENBQ1hDLE1BQU0sQ0FBTkEsT0FBSyxDQUFDLENBQ1JDLElBQUksQ0FBSkEsS0FBRyxDQUFDLEdBQ1Y7SUFBQUcsQ0FBQSxNQUFBSCxJQUFBO0lBQUFHLENBQUEsTUFBQUwsV0FBQTtJQUFBSyxDQUFBLE1BQUFKLE1BQUE7SUFBQUksQ0FBQSxNQUFBRSxRQUFBO0lBQUFGLENBQUEsTUFBQUcsRUFBQTtFQUFBO0lBQUFBLEVBQUEsR0FBQUgsQ0FBQTtFQUFBO0VBQUEsT0FMRkcsRUFLRTtBQUFBIiwiaWdub3JlTGlzdCI6W119 \ No newline at end of file diff --git a/src/components/ConsoleOAuthFlow.tsx b/src/components/ConsoleOAuthFlow.tsx new file mode 100644 index 0000000..717697f --- /dev/null +++ b/src/components/ConsoleOAuthFlow.tsx @@ -0,0 +1,631 @@ +import { c as _c } from "react/compiler-runtime"; +import React, { useCallback, useEffect, useRef, useState } from 'react'; +import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from 'src/services/analytics/index.js'; +import { installOAuthTokens } from '../cli/handlers/auth.js'; +import { useTerminalSize } from '../hooks/useTerminalSize.js'; +import { setClipboard } from '../ink/termio/osc.js'; +import { useTerminalNotification } from '../ink/useTerminalNotification.js'; +import { Box, Link, Text } from '../ink.js'; +import { useKeybinding } from '../keybindings/useKeybinding.js'; +import { getSSLErrorHint } from '../services/api/errorUtils.js'; +import { sendNotification } from '../services/notifier.js'; +import { OAuthService } from '../services/oauth/index.js'; +import { getOauthAccountInfo, validateForceLoginOrg } from '../utils/auth.js'; +import { logError } from '../utils/log.js'; +import { getSettings_DEPRECATED } from '../utils/settings/settings.js'; +import { Select } from './CustomSelect/select.js'; +import { KeyboardShortcutHint } from './design-system/KeyboardShortcutHint.js'; +import { Spinner } from './Spinner.js'; +import TextInput from './TextInput.js'; +type Props = { + onDone(): void; + startingMessage?: string; + mode?: 'login' | 'setup-token'; + forceLoginMethod?: 'claudeai' | 'console'; +}; +type OAuthStatus = { + state: 'idle'; +} // Initial state, waiting to select login method +| { + state: 'platform_setup'; +} // Show platform setup info (Bedrock/Vertex/Foundry) +| { + state: 'ready_to_start'; +} // Flow started, waiting for browser to open +| { + state: 'waiting_for_login'; + url: string; +} // Browser opened, waiting for user to login +| { + state: 'creating_api_key'; +} // Got access token, creating API key +| { + state: 'about_to_retry'; + nextState: OAuthStatus; +} | { + state: 'success'; + token?: string; +} | { + state: 'error'; + message: string; + toRetry?: OAuthStatus; +}; +const PASTE_HERE_MSG = 'Paste code here if prompted > '; +export function ConsoleOAuthFlow({ + onDone, + startingMessage, + mode = 'login', + forceLoginMethod: forceLoginMethodProp +}: Props): React.ReactNode { + const settings = getSettings_DEPRECATED() || {}; + const forceLoginMethod = forceLoginMethodProp ?? settings.forceLoginMethod; + const orgUUID = settings.forceLoginOrgUUID; + const forcedMethodMessage = forceLoginMethod === 'claudeai' ? 'Login method pre-selected: Subscription Plan (Claude Pro/Max)' : forceLoginMethod === 'console' ? 'Login method pre-selected: API Usage Billing (Anthropic Console)' : null; + const terminal = useTerminalNotification(); + const [oauthStatus, setOAuthStatus] = useState(() => { + if (mode === 'setup-token') { + return { + state: 'ready_to_start' + }; + } + if (forceLoginMethod === 'claudeai' || forceLoginMethod === 'console') { + return { + state: 'ready_to_start' + }; + } + return { + state: 'idle' + }; + }); + const [pastedCode, setPastedCode] = useState(''); + const [cursorOffset, setCursorOffset] = useState(0); + const [oauthService] = useState(() => new OAuthService()); + const [loginWithClaudeAi, setLoginWithClaudeAi] = useState(() => { + // Use Claude AI auth for setup-token mode to support user:inference scope + return mode === 'setup-token' || forceLoginMethod === 'claudeai'; + }); + // After a few seconds we suggest the user to copy/paste url if the + // browser did not open automatically. In this flow we expect the user to + // copy the code from the browser and paste it in the terminal + const [showPastePrompt, setShowPastePrompt] = useState(false); + const [urlCopied, setUrlCopied] = useState(false); + const textInputColumns = useTerminalSize().columns - PASTE_HERE_MSG.length - 1; + + // Log forced login method on mount + useEffect(() => { + if (forceLoginMethod === 'claudeai') { + logEvent('tengu_oauth_claudeai_forced', {}); + } else if (forceLoginMethod === 'console') { + logEvent('tengu_oauth_console_forced', {}); + } + }, [forceLoginMethod]); + + // Retry logic + useEffect(() => { + if (oauthStatus.state === 'about_to_retry') { + const timer = setTimeout(setOAuthStatus, 1000, oauthStatus.nextState); + return () => clearTimeout(timer); + } + }, [oauthStatus]); + + // Handle Enter to continue on success state + useKeybinding('confirm:yes', () => { + logEvent('tengu_oauth_success', { + loginWithClaudeAi + }); + onDone(); + }, { + context: 'Confirmation', + isActive: oauthStatus.state === 'success' && mode !== 'setup-token' + }); + + // Handle Enter to continue from platform setup + useKeybinding('confirm:yes', () => { + setOAuthStatus({ + state: 'idle' + }); + }, { + context: 'Confirmation', + isActive: oauthStatus.state === 'platform_setup' + }); + + // Handle Enter to retry on error state + useKeybinding('confirm:yes', () => { + if (oauthStatus.state === 'error' && oauthStatus.toRetry) { + setPastedCode(''); + setOAuthStatus({ + state: 'about_to_retry', + nextState: oauthStatus.toRetry + }); + } + }, { + context: 'Confirmation', + isActive: oauthStatus.state === 'error' && !!oauthStatus.toRetry + }); + useEffect(() => { + if (pastedCode === 'c' && oauthStatus.state === 'waiting_for_login' && showPastePrompt && !urlCopied) { + void setClipboard(oauthStatus.url).then(raw => { + if (raw) process.stdout.write(raw); + setUrlCopied(true); + setTimeout(setUrlCopied, 2000, false); + }); + setPastedCode(''); + } + }, [pastedCode, oauthStatus, showPastePrompt, urlCopied]); + async function handleSubmitCode(value: string, url: string) { + try { + // Expecting format "authorizationCode#state" from the authorization callback URL + const [authorizationCode, state] = value.split('#'); + if (!authorizationCode || !state) { + setOAuthStatus({ + state: 'error', + message: 'Invalid code. Please make sure the full code was copied', + toRetry: { + state: 'waiting_for_login', + url + } + }); + return; + } + + // Track which path the user is taking (manual code entry) + logEvent('tengu_oauth_manual_entry', {}); + oauthService.handleManualAuthCodeInput({ + authorizationCode, + state + }); + } catch (err: unknown) { + logError(err); + setOAuthStatus({ + state: 'error', + message: (err as Error).message, + toRetry: { + state: 'waiting_for_login', + url + } + }); + } + } + const startOAuth = useCallback(async () => { + try { + logEvent('tengu_oauth_flow_start', { + loginWithClaudeAi + }); + const result = await oauthService.startOAuthFlow(async url_0 => { + setOAuthStatus({ + state: 'waiting_for_login', + url: url_0 + }); + setTimeout(setShowPastePrompt, 3000, true); + }, { + loginWithClaudeAi, + inferenceOnly: mode === 'setup-token', + expiresIn: mode === 'setup-token' ? 365 * 24 * 60 * 60 : undefined, + // 1 year for setup-token + orgUUID + }).catch(err_1 => { + const isTokenExchangeError = err_1.message.includes('Token exchange failed'); + // Enterprise TLS proxies (Zscaler et al.) intercept the token + // exchange POST and cause cryptic SSL errors. Surface an + // actionable hint so the user isn't stuck in a login loop. + const sslHint_0 = getSSLErrorHint(err_1); + setOAuthStatus({ + state: 'error', + message: sslHint_0 ?? (isTokenExchangeError ? 'Failed to exchange authorization code for access token. Please try again.' : err_1.message), + toRetry: mode === 'setup-token' ? { + state: 'ready_to_start' + } : { + state: 'idle' + } + }); + logEvent('tengu_oauth_token_exchange_error', { + error: err_1.message, + ssl_error: sslHint_0 !== null + }); + throw err_1; + }); + if (mode === 'setup-token') { + // For setup-token mode, return the OAuth access token directly (it can be used as an API key) + // Don't save to keychain - the token is displayed for manual use with CLAUDE_CODE_OAUTH_TOKEN + setOAuthStatus({ + state: 'success', + token: result.accessToken + }); + } else { + await installOAuthTokens(result); + const orgResult = await validateForceLoginOrg(); + if (!orgResult.valid) { + throw new Error(orgResult.message); + } + setOAuthStatus({ + state: 'success' + }); + void sendNotification({ + message: 'Claude Code login successful', + notificationType: 'auth_success' + }, terminal); + } + } catch (err_0) { + const errorMessage = (err_0 as Error).message; + const sslHint = getSSLErrorHint(err_0); + setOAuthStatus({ + state: 'error', + message: sslHint ?? errorMessage, + toRetry: { + state: mode === 'setup-token' ? 'ready_to_start' : 'idle' + } + }); + logEvent('tengu_oauth_error', { + error: errorMessage as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + ssl_error: sslHint !== null + }); + } + }, [oauthService, setShowPastePrompt, loginWithClaudeAi, mode, orgUUID]); + const pendingOAuthStartRef = useRef(false); + useEffect(() => { + if (oauthStatus.state === 'ready_to_start' && !pendingOAuthStartRef.current) { + pendingOAuthStartRef.current = true; + process.nextTick((startOAuth_0: () => Promise, pendingOAuthStartRef_0: React.MutableRefObject) => { + void startOAuth_0(); + pendingOAuthStartRef_0.current = false; + }, startOAuth, pendingOAuthStartRef); + } + }, [oauthStatus.state, startOAuth]); + + // Auto-exit for setup-token mode + useEffect(() => { + if (mode === 'setup-token' && oauthStatus.state === 'success') { + // Delay to ensure static content is fully rendered before exiting + const timer_0 = setTimeout((loginWithClaudeAi_0, onDone_0) => { + logEvent('tengu_oauth_success', { + loginWithClaudeAi: loginWithClaudeAi_0 + }); + // Don't clear terminal so the token remains visible + onDone_0(); + }, 500, loginWithClaudeAi, onDone); + return () => clearTimeout(timer_0); + } + }, [mode, oauthStatus, loginWithClaudeAi, onDone]); + + // Cleanup OAuth service when component unmounts + useEffect(() => { + return () => { + oauthService.cleanup(); + }; + }, [oauthService]); + return + {oauthStatus.state === 'waiting_for_login' && showPastePrompt && + + + Browser didn't open? Use the url below to sign in{' '} + + {urlCopied ? (Copied!) : + + } + + + {oauthStatus.url} + + } + {mode === 'setup-token' && oauthStatus.state === 'success' && oauthStatus.token && + + ✓ Long-lived authentication token created successfully! + + + Your OAuth token (valid for 1 year): + {oauthStatus.token} + + Store this token securely. You won't be able to see it + again. + + + Use this token by setting: export + CLAUDE_CODE_OAUTH_TOKEN=<token> + + + } + + + + ; +} +type OAuthStatusMessageProps = { + oauthStatus: OAuthStatus; + mode: 'login' | 'setup-token'; + startingMessage: string | undefined; + forcedMethodMessage: string | null; + showPastePrompt: boolean; + pastedCode: string; + setPastedCode: (value: string) => void; + cursorOffset: number; + setCursorOffset: (offset: number) => void; + textInputColumns: number; + handleSubmitCode: (value: string, url: string) => void; + setOAuthStatus: (status: OAuthStatus) => void; + setLoginWithClaudeAi: (value: boolean) => void; +}; +function OAuthStatusMessage(t0) { + const $ = _c(51); + const { + oauthStatus, + mode, + startingMessage, + forcedMethodMessage, + showPastePrompt, + pastedCode, + setPastedCode, + cursorOffset, + setCursorOffset, + textInputColumns, + handleSubmitCode, + setOAuthStatus, + setLoginWithClaudeAi + } = t0; + switch (oauthStatus.state) { + case "idle": + { + const t1 = startingMessage ? startingMessage : "Claude Code can be used with your Claude subscription or billed based on API usage through your Console account."; + let t2; + if ($[0] !== t1) { + t2 = {t1}; + $[0] = t1; + $[1] = t2; + } else { + t2 = $[1]; + } + let t3; + if ($[2] === Symbol.for("react.memo_cache_sentinel")) { + t3 = Select login method:; + $[2] = t3; + } else { + t3 = $[2]; + } + let t4; + if ($[3] === Symbol.for("react.memo_cache_sentinel")) { + t4 = { + label: Claude account with subscription ·{" "}Pro, Max, Team, or Enterprise{false && {"\n"}[ANT-ONLY]{" "}Please use this option unless you need to login to a special org for accessing sensitive data (e.g. customer data, HIPI data) with the Console option}{"\n"}, + value: "claudeai" + }; + $[3] = t4; + } else { + t4 = $[3]; + } + let t5; + if ($[4] === Symbol.for("react.memo_cache_sentinel")) { + t5 = { + label: Anthropic Console account ·{" "}API usage billing{"\n"}, + value: "console" + }; + $[4] = t5; + } else { + t5 = $[4]; + } + let t6; + if ($[5] === Symbol.for("react.memo_cache_sentinel")) { + t6 = [t4, t5, { + label: 3rd-party platform ·{" "}Amazon Bedrock, Microsoft Foundry, or Vertex AI{"\n"}, + value: "platform" + }]; + $[5] = t6; + } else { + t6 = $[5]; + } + let t7; + if ($[6] !== setLoginWithClaudeAi || $[7] !== setOAuthStatus) { + t7 = ; + $[2] = onDone; + $[3] = t3; + } else { + t3 = $[3]; + } + let t4; + if ($[4] !== onDone || $[5] !== t3) { + t4 = {t1}{t3}; + $[4] = onDone; + $[5] = t3; + $[6] = t4; + } else { + t4 = $[6]; + } + return t4; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkJveCIsIkxpbmsiLCJUZXh0IiwiU2VsZWN0IiwiRGlhbG9nIiwiUHJvcHMiLCJvbkRvbmUiLCJDb3N0VGhyZXNob2xkRGlhbG9nIiwidDAiLCIkIiwiX2MiLCJ0MSIsIlN5bWJvbCIsImZvciIsInQyIiwidmFsdWUiLCJsYWJlbCIsInQzIiwidDQiXSwic291cmNlcyI6WyJDb3N0VGhyZXNob2xkRGlhbG9nLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBCb3gsIExpbmssIFRleHQgfSBmcm9tICcuLi9pbmsuanMnXG5pbXBvcnQgeyBTZWxlY3QgfSBmcm9tICcuL0N1c3RvbVNlbGVjdC9pbmRleC5qcydcbmltcG9ydCB7IERpYWxvZyB9IGZyb20gJy4vZGVzaWduLXN5c3RlbS9EaWFsb2cuanMnXG5cbnR5cGUgUHJvcHMgPSB7XG4gIG9uRG9uZTogKCkgPT4gdm9pZFxufVxuXG5leHBvcnQgZnVuY3Rpb24gQ29zdFRocmVzaG9sZERpYWxvZyh7IG9uRG9uZSB9OiBQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIHJldHVybiAoXG4gICAgPERpYWxvZ1xuICAgICAgdGl0bGU9XCJZb3UndmUgc3BlbnQgJDUgb24gdGhlIEFudGhyb3BpYyBBUEkgdGhpcyBzZXNzaW9uLlwiXG4gICAgICBvbkNhbmNlbD17b25Eb25lfVxuICAgID5cbiAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiPlxuICAgICAgICA8VGV4dD5MZWFybiBtb3JlIGFib3V0IGhvdyB0byBtb25pdG9yIHlvdXIgc3BlbmRpbmc6PC9UZXh0PlxuICAgICAgICA8TGluayB1cmw9XCJodHRwczovL2NvZGUuY2xhdWRlLmNvbS9kb2NzL2VuL2Nvc3RzXCIgLz5cbiAgICAgIDwvQm94PlxuICAgICAgPFNlbGVjdFxuICAgICAgICBvcHRpb25zPXtbXG4gICAgICAgICAge1xuICAgICAgICAgICAgdmFsdWU6ICdvaycsXG4gICAgICAgICAgICBsYWJlbDogJ0dvdCBpdCwgdGhhbmtzIScsXG4gICAgICAgICAgfSxcbiAgICAgICAgXX1cbiAgICAgICAgb25DaGFuZ2U9e29uRG9uZX1cbiAgICAgIC8+XG4gICAgPC9EaWFsb2c+XG4gIClcbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU9BLEtBQUssTUFBTSxPQUFPO0FBQ3pCLFNBQVNDLEdBQUcsRUFBRUMsSUFBSSxFQUFFQyxJQUFJLFFBQVEsV0FBVztBQUMzQyxTQUFTQyxNQUFNLFFBQVEseUJBQXlCO0FBQ2hELFNBQVNDLE1BQU0sUUFBUSwyQkFBMkI7QUFFbEQsS0FBS0MsS0FBSyxHQUFHO0VBQ1hDLE1BQU0sRUFBRSxHQUFHLEdBQUcsSUFBSTtBQUNwQixDQUFDO0FBRUQsT0FBTyxTQUFBQyxvQkFBQUMsRUFBQTtFQUFBLE1BQUFDLENBQUEsR0FBQUMsRUFBQTtFQUE2QjtJQUFBSjtFQUFBLElBQUFFLEVBQWlCO0VBQUEsSUFBQUcsRUFBQTtFQUFBLElBQUFGLENBQUEsUUFBQUcsTUFBQSxDQUFBQyxHQUFBO0lBTS9DRixFQUFBLElBQUMsR0FBRyxDQUFlLGFBQVEsQ0FBUixRQUFRLENBQ3pCLENBQUMsSUFBSSxDQUFDLDhDQUE4QyxFQUFuRCxJQUFJLENBQ0wsQ0FBQyxJQUFJLENBQUssR0FBdUMsQ0FBdkMsdUNBQXVDLEdBQ25ELEVBSEMsR0FBRyxDQUdFO0lBQUFGLENBQUEsTUFBQUUsRUFBQTtFQUFBO0lBQUFBLEVBQUEsR0FBQUYsQ0FBQTtFQUFBO0VBQUEsSUFBQUssRUFBQTtFQUFBLElBQUFMLENBQUEsUUFBQUcsTUFBQSxDQUFBQyxHQUFBO0lBRUtDLEVBQUEsSUFDUDtNQUFBQyxLQUFBLEVBQ1MsSUFBSTtNQUFBQyxLQUFBLEVBQ0o7SUFDVCxDQUFDLENBQ0Y7SUFBQVAsQ0FBQSxNQUFBSyxFQUFBO0VBQUE7SUFBQUEsRUFBQSxHQUFBTCxDQUFBO0VBQUE7RUFBQSxJQUFBUSxFQUFBO0VBQUEsSUFBQVIsQ0FBQSxRQUFBSCxNQUFBO0lBTkhXLEVBQUEsSUFBQyxNQUFNLENBQ0ksT0FLUixDQUxRLENBQUFILEVBS1QsQ0FBQyxDQUNTUixRQUFNLENBQU5BLE9BQUssQ0FBQyxHQUNoQjtJQUFBRyxDQUFBLE1BQUFILE1BQUE7SUFBQUcsQ0FBQSxNQUFBUSxFQUFBO0VBQUE7SUFBQUEsRUFBQSxHQUFBUixDQUFBO0VBQUE7RUFBQSxJQUFBUyxFQUFBO0VBQUEsSUFBQVQsQ0FBQSxRQUFBSCxNQUFBLElBQUFHLENBQUEsUUFBQVEsRUFBQTtJQWhCSkMsRUFBQSxJQUFDLE1BQU0sQ0FDQyxLQUFvRCxDQUFwRCxvREFBb0QsQ0FDaERaLFFBQU0sQ0FBTkEsT0FBSyxDQUFDLENBRWhCLENBQUFLLEVBR0ssQ0FDTCxDQUFBTSxFQVFDLENBQ0gsRUFqQkMsTUFBTSxDQWlCRTtJQUFBUixDQUFBLE1BQUFILE1BQUE7SUFBQUcsQ0FBQSxNQUFBUSxFQUFBO0lBQUFSLENBQUEsTUFBQVMsRUFBQTtFQUFBO0lBQUFBLEVBQUEsR0FBQVQsQ0FBQTtFQUFBO0VBQUEsT0FqQlRTLEVBaUJTO0FBQUEiLCJpZ25vcmVMaXN0IjpbXX0= \ No newline at end of file diff --git a/src/components/CtrlOToExpand.tsx b/src/components/CtrlOToExpand.tsx new file mode 100644 index 0000000..3fe799b --- /dev/null +++ b/src/components/CtrlOToExpand.tsx @@ -0,0 +1,51 @@ +import { c as _c } from "react/compiler-runtime"; +import chalk from 'chalk'; +import React, { useContext } from 'react'; +import { Text } from '../ink.js'; +import { getShortcutDisplay } from '../keybindings/shortcutFormat.js'; +import { useShortcutDisplay } from '../keybindings/useShortcutDisplay.js'; +import { KeyboardShortcutHint } from './design-system/KeyboardShortcutHint.js'; +import { InVirtualListContext } from './messageActions.js'; + +// Context to track if we're inside a sub agent +// Similar to MessageResponseContext, this helps us avoid showing +// too many "(ctrl+o to expand)" hints in sub agent output +const SubAgentContext = React.createContext(false); +export function SubAgentProvider(t0) { + const $ = _c(2); + const { + children + } = t0; + let t1; + if ($[0] !== children) { + t1 = {children}; + $[0] = children; + $[1] = t1; + } else { + t1 = $[1]; + } + return t1; +} +export function CtrlOToExpand() { + const $ = _c(2); + const isInSubAgent = useContext(SubAgentContext); + const inVirtualList = useContext(InVirtualListContext); + const expandShortcut = useShortcutDisplay("app:toggleTranscript", "Global", "ctrl+o"); + if (isInSubAgent || inVirtualList) { + return null; + } + let t0; + if ($[0] !== expandShortcut) { + t0 = ; + $[0] = expandShortcut; + $[1] = t0; + } else { + t0 = $[1]; + } + return t0; +} +export function ctrlOToExpand(): string { + const shortcut = getShortcutDisplay('app:toggleTranscript', 'Global', 'ctrl+o'); + return chalk.dim(`(${shortcut} to expand)`); +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJjaGFsayIsIlJlYWN0IiwidXNlQ29udGV4dCIsIlRleHQiLCJnZXRTaG9ydGN1dERpc3BsYXkiLCJ1c2VTaG9ydGN1dERpc3BsYXkiLCJLZXlib2FyZFNob3J0Y3V0SGludCIsIkluVmlydHVhbExpc3RDb250ZXh0IiwiU3ViQWdlbnRDb250ZXh0IiwiY3JlYXRlQ29udGV4dCIsIlN1YkFnZW50UHJvdmlkZXIiLCJ0MCIsIiQiLCJfYyIsImNoaWxkcmVuIiwidDEiLCJDdHJsT1RvRXhwYW5kIiwiaXNJblN1YkFnZW50IiwiaW5WaXJ0dWFsTGlzdCIsImV4cGFuZFNob3J0Y3V0IiwiY3RybE9Ub0V4cGFuZCIsInNob3J0Y3V0IiwiZGltIl0sInNvdXJjZXMiOlsiQ3RybE9Ub0V4cGFuZC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGNoYWxrIGZyb20gJ2NoYWxrJ1xuaW1wb3J0IFJlYWN0LCB7IHVzZUNvbnRleHQgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IFRleHQgfSBmcm9tICcuLi9pbmsuanMnXG5pbXBvcnQgeyBnZXRTaG9ydGN1dERpc3BsYXkgfSBmcm9tICcuLi9rZXliaW5kaW5ncy9zaG9ydGN1dEZvcm1hdC5qcydcbmltcG9ydCB7IHVzZVNob3J0Y3V0RGlzcGxheSB9IGZyb20gJy4uL2tleWJpbmRpbmdzL3VzZVNob3J0Y3V0RGlzcGxheS5qcydcbmltcG9ydCB7IEtleWJvYXJkU2hvcnRjdXRIaW50IH0gZnJvbSAnLi9kZXNpZ24tc3lzdGVtL0tleWJvYXJkU2hvcnRjdXRIaW50LmpzJ1xuaW1wb3J0IHsgSW5WaXJ0dWFsTGlzdENvbnRleHQgfSBmcm9tICcuL21lc3NhZ2VBY3Rpb25zLmpzJ1xuXG4vLyBDb250ZXh0IHRvIHRyYWNrIGlmIHdlJ3JlIGluc2lkZSBhIHN1YiBhZ2VudFxuLy8gU2ltaWxhciB0byBNZXNzYWdlUmVzcG9uc2VDb250ZXh0LCB0aGlzIGhlbHBzIHVzIGF2b2lkIHNob3dpbmdcbi8vIHRvbyBtYW55IFwiKGN0cmwrbyB0byBleHBhbmQpXCIgaGludHMgaW4gc3ViIGFnZW50IG91dHB1dFxuY29uc3QgU3ViQWdlbnRDb250ZXh0ID0gUmVhY3QuY3JlYXRlQ29udGV4dChmYWxzZSlcblxuZXhwb3J0IGZ1bmN0aW9uIFN1YkFnZW50UHJvdmlkZXIoe1xuICBjaGlsZHJlbixcbn06IHtcbiAgY2hpbGRyZW46IFJlYWN0LlJlYWN0Tm9kZVxufSk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIHJldHVybiAoXG4gICAgPFN1YkFnZW50Q29udGV4dC5Qcm92aWRlciB2YWx1ZT17dHJ1ZX0+e2NoaWxkcmVufTwvU3ViQWdlbnRDb250ZXh0LlByb3ZpZGVyPlxuICApXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBDdHJsT1RvRXhwYW5kKCk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IGlzSW5TdWJBZ2VudCA9IHVzZUNvbnRleHQoU3ViQWdlbnRDb250ZXh0KVxuICBjb25zdCBpblZpcnR1YWxMaXN0ID0gdXNlQ29udGV4dChJblZpcnR1YWxMaXN0Q29udGV4dClcbiAgY29uc3QgZXhwYW5kU2hvcnRjdXQgPSB1c2VTaG9ydGN1dERpc3BsYXkoXG4gICAgJ2FwcDp0b2dnbGVUcmFuc2NyaXB0JyxcbiAgICAnR2xvYmFsJyxcbiAgICAnY3RybCtvJyxcbiAgKVxuICBpZiAoaXNJblN1YkFnZW50IHx8IGluVmlydHVhbExpc3QpIHtcbiAgICByZXR1cm4gbnVsbFxuICB9XG4gIHJldHVybiAoXG4gICAgPFRleHQgZGltQ29sb3I+XG4gICAgICA8S2V5Ym9hcmRTaG9ydGN1dEhpbnQgc2hvcnRjdXQ9e2V4cGFuZFNob3J0Y3V0fSBhY3Rpb249XCJleHBhbmRcIiBwYXJlbnMgLz5cbiAgICA8L1RleHQ+XG4gIClcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGN0cmxPVG9FeHBhbmQoKTogc3RyaW5nIHtcbiAgY29uc3Qgc2hvcnRjdXQgPSBnZXRTaG9ydGN1dERpc3BsYXkoXG4gICAgJ2FwcDp0b2dnbGVUcmFuc2NyaXB0JyxcbiAgICAnR2xvYmFsJyxcbiAgICAnY3RybCtvJyxcbiAgKVxuICByZXR1cm4gY2hhbGsuZGltKGAoJHtzaG9ydGN1dH0gdG8gZXhwYW5kKWApXG59XG4iXSwibWFwcGluZ3MiOiI7QUFBQSxPQUFPQSxLQUFLLE1BQU0sT0FBTztBQUN6QixPQUFPQyxLQUFLLElBQUlDLFVBQVUsUUFBUSxPQUFPO0FBQ3pDLFNBQVNDLElBQUksUUFBUSxXQUFXO0FBQ2hDLFNBQVNDLGtCQUFrQixRQUFRLGtDQUFrQztBQUNyRSxTQUFTQyxrQkFBa0IsUUFBUSxzQ0FBc0M7QUFDekUsU0FBU0Msb0JBQW9CLFFBQVEseUNBQXlDO0FBQzlFLFNBQVNDLG9CQUFvQixRQUFRLHFCQUFxQjs7QUFFMUQ7QUFDQTtBQUNBO0FBQ0EsTUFBTUMsZUFBZSxHQUFHUCxLQUFLLENBQUNRLGFBQWEsQ0FBQyxLQUFLLENBQUM7QUFFbEQsT0FBTyxTQUFBQyxpQkFBQUMsRUFBQTtFQUFBLE1BQUFDLENBQUEsR0FBQUMsRUFBQTtFQUEwQjtJQUFBQztFQUFBLElBQUFILEVBSWhDO0VBQUEsSUFBQUksRUFBQTtFQUFBLElBQUFILENBQUEsUUFBQUUsUUFBQTtJQUVHQyxFQUFBLDZCQUFpQyxLQUFJLENBQUosS0FBRyxDQUFDLENBQUdELFNBQU8sQ0FBRSwyQkFBMkI7SUFBQUYsQ0FBQSxNQUFBRSxRQUFBO0lBQUFGLENBQUEsTUFBQUcsRUFBQTtFQUFBO0lBQUFBLEVBQUEsR0FBQUgsQ0FBQTtFQUFBO0VBQUEsT0FBNUVHLEVBQTRFO0FBQUE7QUFJaEYsT0FBTyxTQUFBQyxjQUFBO0VBQUEsTUFBQUosQ0FBQSxHQUFBQyxFQUFBO0VBQ0wsTUFBQUksWUFBQSxHQUFxQmYsVUFBVSxDQUFDTSxlQUFlLENBQUM7RUFDaEQsTUFBQVUsYUFBQSxHQUFzQmhCLFVBQVUsQ0FBQ0ssb0JBQW9CLENBQUM7RUFDdEQsTUFBQVksY0FBQSxHQUF1QmQsa0JBQWtCLENBQ3ZDLHNCQUFzQixFQUN0QixRQUFRLEVBQ1IsUUFDRixDQUFDO0VBQ0QsSUFBSVksWUFBNkIsSUFBN0JDLGFBQTZCO0lBQUEsT0FDeEIsSUFBSTtFQUFBO0VBQ1osSUFBQVAsRUFBQTtFQUFBLElBQUFDLENBQUEsUUFBQU8sY0FBQTtJQUVDUixFQUFBLElBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBUixLQUFPLENBQUMsQ0FDWixDQUFDLG9CQUFvQixDQUFXUSxRQUFjLENBQWRBLGVBQWEsQ0FBQyxDQUFTLE1BQVEsQ0FBUixRQUFRLENBQUMsTUFBTSxDQUFOLEtBQUssQ0FBQyxHQUN4RSxFQUZDLElBQUksQ0FFRTtJQUFBUCxDQUFBLE1BQUFPLGNBQUE7SUFBQVAsQ0FBQSxNQUFBRCxFQUFBO0VBQUE7SUFBQUEsRUFBQSxHQUFBQyxDQUFBO0VBQUE7RUFBQSxPQUZQRCxFQUVPO0FBQUE7QUFJWCxPQUFPLFNBQVNTLGFBQWFBLENBQUEsQ0FBRSxFQUFFLE1BQU0sQ0FBQztFQUN0QyxNQUFNQyxRQUFRLEdBQUdqQixrQkFBa0IsQ0FDakMsc0JBQXNCLEVBQ3RCLFFBQVEsRUFDUixRQUNGLENBQUM7RUFDRCxPQUFPSixLQUFLLENBQUNzQixHQUFHLENBQUMsSUFBSUQsUUFBUSxhQUFhLENBQUM7QUFDN0MiLCJpZ25vcmVMaXN0IjpbXX0= \ No newline at end of file diff --git a/src/components/CustomSelect/SelectMulti.tsx b/src/components/CustomSelect/SelectMulti.tsx new file mode 100644 index 0000000..e757ec3 --- /dev/null +++ b/src/components/CustomSelect/SelectMulti.tsx @@ -0,0 +1,213 @@ +import { c as _c } from "react/compiler-runtime"; +import figures from 'figures'; +import React from 'react'; +import { Box, Text } from '../../ink.js'; +import type { PastedContent } from '../../utils/config.js'; +import type { ImageDimensions } from '../../utils/imageResizer.js'; +import type { OptionWithDescription } from './select.js'; +import { SelectInputOption } from './select-input-option.js'; +import { SelectOption } from './select-option.js'; +import { useMultiSelectState } from './use-multi-select-state.js'; +export type SelectMultiProps = { + readonly isDisabled?: boolean; + readonly visibleOptionCount?: number; + readonly options: OptionWithDescription[]; + readonly defaultValue?: T[]; + readonly onCancel: () => void; + readonly onChange?: (values: T[]) => void; + readonly onFocus?: (value: T) => void; + readonly focusValue?: T; + /** + * Text for the submit button. When provided, a submit button is shown and + * Enter toggles selection (submit only fires when the button is focused). + * When omitted, Enter submits directly and Space toggles selection. + */ + readonly submitButtonText?: string; + /** + * Callback when user submits. Receives the currently selected values. + */ + readonly onSubmit?: (values: T[]) => void; + /** + * When true, hides the numeric indexes next to each option. + */ + readonly hideIndexes?: boolean; + /** + * Callback when user presses down from the last item (submit button). + * If provided, navigation will not wrap to the first item. + */ + readonly onDownFromLastItem?: () => void; + /** + * Callback when user presses up from the first item. + * If provided, navigation will not wrap to the last item. + */ + readonly onUpFromFirstItem?: () => void; + /** + * Focus the last option initially instead of the first. + */ + readonly initialFocusLast?: boolean; + /** + * Callback to open external editor for editing input option values. + * When provided, ctrl+g will trigger this callback in input options + * with the current value and a setter function to update the internal state. + */ + readonly onOpenEditor?: (currentValue: string, setValue: (value: string) => void) => void; + readonly onImagePaste?: (base64Image: string, mediaType?: string, filename?: string, dimensions?: ImageDimensions, sourcePath?: string) => void; + readonly pastedContents?: Record; + readonly onRemoveImage?: (id: number) => void; +}; +export function SelectMulti(t0) { + const $ = _c(44); + const { + isDisabled: t1, + visibleOptionCount: t2, + options, + defaultValue: t3, + onCancel, + onChange, + onFocus, + focusValue, + submitButtonText, + onSubmit, + onDownFromLastItem, + onUpFromFirstItem, + initialFocusLast, + onOpenEditor, + hideIndexes: t4, + onImagePaste, + pastedContents, + onRemoveImage + } = t0; + const isDisabled = t1 === undefined ? false : t1; + const visibleOptionCount = t2 === undefined ? 5 : t2; + let t5; + if ($[0] !== t3) { + t5 = t3 === undefined ? [] : t3; + $[0] = t3; + $[1] = t5; + } else { + t5 = $[1]; + } + const defaultValue = t5; + const hideIndexes = t4 === undefined ? false : t4; + let t6; + if ($[2] !== defaultValue || $[3] !== focusValue || $[4] !== hideIndexes || $[5] !== initialFocusLast || $[6] !== isDisabled || $[7] !== onCancel || $[8] !== onChange || $[9] !== onDownFromLastItem || $[10] !== onFocus || $[11] !== onSubmit || $[12] !== onUpFromFirstItem || $[13] !== options || $[14] !== submitButtonText || $[15] !== visibleOptionCount) { + t6 = { + isDisabled, + visibleOptionCount, + options, + defaultValue, + onChange, + onCancel, + onFocus, + focusValue, + submitButtonText, + onSubmit, + onDownFromLastItem, + onUpFromFirstItem, + initialFocusLast, + hideIndexes + }; + $[2] = defaultValue; + $[3] = focusValue; + $[4] = hideIndexes; + $[5] = initialFocusLast; + $[6] = isDisabled; + $[7] = onCancel; + $[8] = onChange; + $[9] = onDownFromLastItem; + $[10] = onFocus; + $[11] = onSubmit; + $[12] = onUpFromFirstItem; + $[13] = options; + $[14] = submitButtonText; + $[15] = visibleOptionCount; + $[16] = t6; + } else { + t6 = $[16]; + } + const state = useMultiSelectState(t6); + let T0; + let T1; + let t7; + let t8; + let t9; + if ($[17] !== hideIndexes || $[18] !== isDisabled || $[19] !== onCancel || $[20] !== onImagePaste || $[21] !== onOpenEditor || $[22] !== onRemoveImage || $[23] !== options.length || $[24] !== pastedContents || $[25] !== state) { + const maxIndexWidth = options.length.toString().length; + T1 = Box; + t9 = "column"; + T0 = Box; + t7 = "column"; + t8 = state.visibleOptions.map((option, index) => { + const isOptionFocused = !isDisabled && state.focusedValue === option.value && !state.isSubmitFocused; + const isSelected = state.selectedValues.includes(option.value); + const isFirstVisibleOption = option.index === state.visibleFromIndex; + const isLastVisibleOption = option.index === state.visibleToIndex - 1; + const areMoreOptionsBelow = state.visibleToIndex < options.length; + const areMoreOptionsAbove = state.visibleFromIndex > 0; + const i = state.visibleFromIndex + index + 1; + if (option.type === "input") { + const inputValue = state.inputValues.get(option.value) || ""; + return { + state.updateInputValue(option.value, value); + }} onSubmit={_temp} onExit={() => { + onCancel(); + }} layout="compact" onOpenEditor={onOpenEditor} onImagePaste={onImagePaste} pastedContents={pastedContents} onRemoveImage={onRemoveImage}>[{isSelected ? figures.tick : " "}]{" "}; + } + return {!hideIndexes && {`${i}.`.padEnd(maxIndexWidth)}}[{isSelected ? figures.tick : " "}]{option.label}; + }); + $[17] = hideIndexes; + $[18] = isDisabled; + $[19] = onCancel; + $[20] = onImagePaste; + $[21] = onOpenEditor; + $[22] = onRemoveImage; + $[23] = options.length; + $[24] = pastedContents; + $[25] = state; + $[26] = T0; + $[27] = T1; + $[28] = t7; + $[29] = t8; + $[30] = t9; + } else { + T0 = $[26]; + T1 = $[27]; + t7 = $[28]; + t8 = $[29]; + t9 = $[30]; + } + let t10; + if ($[31] !== T0 || $[32] !== t7 || $[33] !== t8) { + t10 = {t8}; + $[31] = T0; + $[32] = t7; + $[33] = t8; + $[34] = t10; + } else { + t10 = $[34]; + } + let t11; + if ($[35] !== onSubmit || $[36] !== state.isSubmitFocused || $[37] !== submitButtonText) { + t11 = submitButtonText && onSubmit && {state.isSubmitFocused ? {figures.pointer} : }{submitButtonText}; + $[35] = onSubmit; + $[36] = state.isSubmitFocused; + $[37] = submitButtonText; + $[38] = t11; + } else { + t11 = $[38]; + } + let t12; + if ($[39] !== T1 || $[40] !== t10 || $[41] !== t11 || $[42] !== t9) { + t12 = {t10}{t11}; + $[39] = T1; + $[40] = t10; + $[41] = t11; + $[42] = t9; + $[43] = t12; + } else { + t12 = $[43]; + } + return t12; +} +function _temp() {} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJmaWd1cmVzIiwiUmVhY3QiLCJCb3giLCJUZXh0IiwiUGFzdGVkQ29udGVudCIsIkltYWdlRGltZW5zaW9ucyIsIk9wdGlvbldpdGhEZXNjcmlwdGlvbiIsIlNlbGVjdElucHV0T3B0aW9uIiwiU2VsZWN0T3B0aW9uIiwidXNlTXVsdGlTZWxlY3RTdGF0ZSIsIlNlbGVjdE11bHRpUHJvcHMiLCJpc0Rpc2FibGVkIiwidmlzaWJsZU9wdGlvbkNvdW50Iiwib3B0aW9ucyIsIlQiLCJkZWZhdWx0VmFsdWUiLCJvbkNhbmNlbCIsIm9uQ2hhbmdlIiwidmFsdWVzIiwib25Gb2N1cyIsInZhbHVlIiwiZm9jdXNWYWx1ZSIsInN1Ym1pdEJ1dHRvblRleHQiLCJvblN1Ym1pdCIsImhpZGVJbmRleGVzIiwib25Eb3duRnJvbUxhc3RJdGVtIiwib25VcEZyb21GaXJzdEl0ZW0iLCJpbml0aWFsRm9jdXNMYXN0Iiwib25PcGVuRWRpdG9yIiwiY3VycmVudFZhbHVlIiwic2V0VmFsdWUiLCJvbkltYWdlUGFzdGUiLCJiYXNlNjRJbWFnZSIsIm1lZGlhVHlwZSIsImZpbGVuYW1lIiwiZGltZW5zaW9ucyIsInNvdXJjZVBhdGgiLCJwYXN0ZWRDb250ZW50cyIsIlJlY29yZCIsIm9uUmVtb3ZlSW1hZ2UiLCJpZCIsIlNlbGVjdE11bHRpIiwidDAiLCIkIiwiX2MiLCJ0MSIsInQyIiwidDMiLCJ0NCIsInVuZGVmaW5lZCIsInQ1IiwidDYiLCJzdGF0ZSIsIlQwIiwiVDEiLCJ0NyIsInQ4IiwidDkiLCJsZW5ndGgiLCJtYXhJbmRleFdpZHRoIiwidG9TdHJpbmciLCJ2aXNpYmxlT3B0aW9ucyIsIm1hcCIsIm9wdGlvbiIsImluZGV4IiwiaXNPcHRpb25Gb2N1c2VkIiwiZm9jdXNlZFZhbHVlIiwiaXNTdWJtaXRGb2N1c2VkIiwiaXNTZWxlY3RlZCIsInNlbGVjdGVkVmFsdWVzIiwiaW5jbHVkZXMiLCJpc0ZpcnN0VmlzaWJsZU9wdGlvbiIsInZpc2libGVGcm9tSW5kZXgiLCJpc0xhc3RWaXNpYmxlT3B0aW9uIiwidmlzaWJsZVRvSW5kZXgiLCJhcmVNb3JlT3B0aW9uc0JlbG93IiwiYXJlTW9yZU9wdGlvbnNBYm92ZSIsImkiLCJ0eXBlIiwiaW5wdXRWYWx1ZSIsImlucHV0VmFsdWVzIiwiZ2V0IiwiU3RyaW5nIiwidXBkYXRlSW5wdXRWYWx1ZSIsIl90ZW1wIiwidGljayIsImRlc2NyaXB0aW9uIiwicGFkRW5kIiwibGFiZWwiLCJ0MTAiLCJ0MTEiLCJwb2ludGVyIiwidDEyIl0sInNvdXJjZXMiOlsiU2VsZWN0TXVsdGkudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBmaWd1cmVzIGZyb20gJ2ZpZ3VyZXMnXG5pbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBCb3gsIFRleHQgfSBmcm9tICcuLi8uLi9pbmsuanMnXG5pbXBvcnQgdHlwZSB7IFBhc3RlZENvbnRlbnQgfSBmcm9tICcuLi8uLi91dGlscy9jb25maWcuanMnXG5pbXBvcnQgdHlwZSB7IEltYWdlRGltZW5zaW9ucyB9IGZyb20gJy4uLy4uL3V0aWxzL2ltYWdlUmVzaXplci5qcydcbmltcG9ydCB0eXBlIHsgT3B0aW9uV2l0aERlc2NyaXB0aW9uIH0gZnJvbSAnLi9zZWxlY3QuanMnXG5pbXBvcnQgeyBTZWxlY3RJbnB1dE9wdGlvbiB9IGZyb20gJy4vc2VsZWN0LWlucHV0LW9wdGlvbi5qcydcbmltcG9ydCB7IFNlbGVjdE9wdGlvbiB9IGZyb20gJy4vc2VsZWN0LW9wdGlvbi5qcydcbmltcG9ydCB7IHVzZU11bHRpU2VsZWN0U3RhdGUgfSBmcm9tICcuL3VzZS1tdWx0aS1zZWxlY3Qtc3RhdGUuanMnXG5cbmV4cG9ydCB0eXBlIFNlbGVjdE11bHRpUHJvcHM8VD4gPSB7XG4gIHJlYWRvbmx5IGlzRGlzYWJsZWQ/OiBib29sZWFuXG4gIHJlYWRvbmx5IHZpc2libGVPcHRpb25Db3VudD86IG51bWJlclxuICByZWFkb25seSBvcHRpb25zOiBPcHRpb25XaXRoRGVzY3JpcHRpb248VD5bXVxuICByZWFkb25seSBkZWZhdWx0VmFsdWU/OiBUW11cbiAgcmVhZG9ubHkgb25DYW5jZWw6ICgpID0+IHZvaWRcbiAgcmVhZG9ubHkgb25DaGFuZ2U/OiAodmFsdWVzOiBUW10pID0+IHZvaWRcbiAgcmVhZG9ubHkgb25Gb2N1cz86ICh2YWx1ZTogVCkgPT4gdm9pZFxuICByZWFkb25seSBmb2N1c1ZhbHVlPzogVFxuICAvKipcbiAgICogVGV4dCBmb3IgdGhlIHN1Ym1pdCBidXR0b24uIFdoZW4gcHJvdmlkZWQsIGEgc3VibWl0IGJ1dHRvbiBpcyBzaG93biBhbmRcbiAgICogRW50ZXIgdG9nZ2xlcyBzZWxlY3Rpb24gKHN1Ym1pdCBvbmx5IGZpcmVzIHdoZW4gdGhlIGJ1dHRvbiBpcyBmb2N1c2VkKS5cbiAgICogV2hlbiBvbWl0dGVkLCBFbnRlciBzdWJtaXRzIGRpcmVjdGx5IGFuZCBTcGFjZSB0b2dnbGVzIHNlbGVjdGlvbi5cbiAgICovXG4gIHJlYWRvbmx5IHN1Ym1pdEJ1dHRvblRleHQ/OiBzdHJpbmdcbiAgLyoqXG4gICAqIENhbGxiYWNrIHdoZW4gdXNlciBzdWJtaXRzLiBSZWNlaXZlcyB0aGUgY3VycmVudGx5IHNlbGVjdGVkIHZhbHVlcy5cbiAgICovXG4gIHJlYWRvbmx5IG9uU3VibWl0PzogKHZhbHVlczogVFtdKSA9PiB2b2lkXG4gIC8qKlxuICAgKiBXaGVuIHRydWUsIGhpZGVzIHRoZSBudW1lcmljIGluZGV4ZXMgbmV4dCB0byBlYWNoIG9wdGlvbi5cbiAgICovXG4gIHJlYWRvbmx5IGhpZGVJbmRleGVzPzogYm9vbGVhblxuICAvKipcbiAgICogQ2FsbGJhY2sgd2hlbiB1c2VyIHByZXNzZXMgZG93biBmcm9tIHRoZSBsYXN0IGl0ZW0gKHN1Ym1pdCBidXR0b24pLlxuICAgKiBJZiBwcm92aWRlZCwgbmF2aWdhdGlvbiB3aWxsIG5vdCB3cmFwIHRvIHRoZSBmaXJzdCBpdGVtLlxuICAgKi9cbiAgcmVhZG9ubHkgb25Eb3duRnJvbUxhc3RJdGVtPzogKCkgPT4gdm9pZFxuICAvKipcbiAgICogQ2FsbGJhY2sgd2hlbiB1c2VyIHByZXNzZXMgdXAgZnJvbSB0aGUgZmlyc3QgaXRlbS5cbiAgICogSWYgcHJvdmlkZWQsIG5hdmlnYXRpb24gd2lsbCBub3Qgd3JhcCB0byB0aGUgbGFzdCBpdGVtLlxuICAgKi9cbiAgcmVhZG9ubHkgb25VcEZyb21GaXJzdEl0ZW0/OiAoKSA9PiB2b2lkXG4gIC8qKlxuICAgKiBGb2N1cyB0aGUgbGFzdCBvcHRpb24gaW5pdGlhbGx5IGluc3RlYWQgb2YgdGhlIGZpcnN0LlxuICAgKi9cbiAgcmVhZG9ubHkgaW5pdGlhbEZvY3VzTGFzdD86IGJvb2xlYW5cbiAgLyoqXG4gICAqIENhbGxiYWNrIHRvIG9wZW4gZXh0ZXJuYWwgZWRpdG9yIGZvciBlZGl0aW5nIGlucHV0IG9wdGlvbiB2YWx1ZXMuXG4gICAqIFdoZW4gcHJvdmlkZWQsIGN0cmwrZyB3aWxsIHRyaWdnZXIgdGhpcyBjYWxsYmFjayBpbiBpbnB1dCBvcHRpb25zXG4gICAqIHdpdGggdGhlIGN1cnJlbnQgdmFsdWUgYW5kIGEgc2V0dGVyIGZ1bmN0aW9uIHRvIHVwZGF0ZSB0aGUgaW50ZXJuYWwgc3RhdGUuXG4gICAqL1xuICByZWFkb25seSBvbk9wZW5FZGl0b3I/OiAoXG4gICAgY3VycmVudFZhbHVlOiBzdHJpbmcsXG4gICAgc2V0VmFsdWU6ICh2YWx1ZTogc3RyaW5nKSA9PiB2b2lkLFxuICApID0+IHZvaWRcbiAgcmVhZG9ubHkgb25JbWFnZVBhc3RlPzogKFxuICAgIGJhc2U2NEltYWdlOiBzdHJpbmcsXG4gICAgbWVkaWFUeXBlPzogc3RyaW5nLFxuICAgIGZpbGVuYW1lPzogc3RyaW5nLFxuICAgIGRpbWVuc2lvbnM/OiBJbWFnZURpbWVuc2lvbnMsXG4gICAgc291cmNlUGF0aD86IHN0cmluZyxcbiAgKSA9PiB2b2lkXG4gIHJlYWRvbmx5IHBhc3RlZENvbnRlbnRzPzogUmVjb3JkPG51bWJlciwgUGFzdGVkQ29udGVudD5cbiAgcmVhZG9ubHkgb25SZW1vdmVJbWFnZT86IChpZDogbnVtYmVyKSA9PiB2b2lkXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBTZWxlY3RNdWx0aTxUPih7XG4gIGlzRGlzYWJsZWQgPSBmYWxzZSxcbiAgdmlzaWJsZU9wdGlvbkNvdW50ID0gNSxcbiAgb3B0aW9ucyxcbiAgZGVmYXVsdFZhbHVlID0gW10sXG4gIG9uQ2FuY2VsLFxuICBvbkNoYW5nZSxcbiAgb25Gb2N1cyxcbiAgZm9jdXNWYWx1ZSxcbiAgc3VibWl0QnV0dG9uVGV4dCxcbiAgb25TdWJtaXQsXG4gIG9uRG93bkZyb21MYXN0SXRlbSxcbiAgb25VcEZyb21GaXJzdEl0ZW0sXG4gIGluaXRpYWxGb2N1c0xhc3QsXG4gIG9uT3BlbkVkaXRvcixcbiAgaGlkZUluZGV4ZXMgPSBmYWxzZSxcbiAgb25JbWFnZVBhc3RlLFxuICBwYXN0ZWRDb250ZW50cyxcbiAgb25SZW1vdmVJbWFnZSxcbn06IFNlbGVjdE11bHRpUHJvcHM8VD4pOiBSZWFjdC5SZWFjdE5vZGUge1xuICBjb25zdCBzdGF0ZSA9IHVzZU11bHRpU2VsZWN0U3RhdGU8VD4oe1xuICAgIGlzRGlzYWJsZWQsXG4gICAgdmlzaWJsZU9wdGlvbkNvdW50LFxuICAgIG9wdGlvbnMsXG4gICAgZGVmYXVsdFZhbHVlLFxuICAgIG9uQ2hhbmdlLFxuICAgIG9uQ2FuY2VsLFxuICAgIG9uRm9jdXMsXG4gICAgZm9jdXNWYWx1ZSxcbiAgICBzdWJtaXRCdXR0b25UZXh0LFxuICAgIG9uU3VibWl0LFxuICAgIG9uRG93bkZyb21MYXN0SXRlbSxcbiAgICBvblVwRnJvbUZpcnN0SXRlbSxcbiAgICBpbml0aWFsRm9jdXNMYXN0LFxuICAgIGhpZGVJbmRleGVzLFxuICB9KVxuXG4gIGNvbnN0IG1heEluZGV4V2lkdGggPSBvcHRpb25zLmxlbmd0aC50b1N0cmluZygpLmxlbmd0aFxuXG4gIHJldHVybiAoXG4gICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCI+XG4gICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIj5cbiAgICAgICAge3N0YXRlLnZpc2libGVPcHRpb25zLm1hcCgob3B0aW9uLCBpbmRleCkgPT4ge1xuICAgICAgICAgIGNvbnN0IGlzT3B0aW9uRm9jdXNlZCA9XG4gICAgICAgICAgICAhaXNEaXNhYmxlZCAmJlxuICAgICAgICAgICAgc3RhdGUuZm9jdXNlZFZhbHVlID09PSBvcHRpb24udmFsdWUgJiZcbiAgICAgICAgICAgICFzdGF0ZS5pc1N1Ym1pdEZvY3VzZWRcbiAgICAgICAgICBjb25zdCBpc1NlbGVjdGVkID0gc3RhdGUuc2VsZWN0ZWRWYWx1ZXMuaW5jbHVkZXMob3B0aW9uLnZhbHVlKVxuXG4gICAgICAgICAgY29uc3QgaXNGaXJzdFZpc2libGVPcHRpb24gPSBvcHRpb24uaW5kZXggPT09IHN0YXRlLnZpc2libGVGcm9tSW5kZXhcbiAgICAgICAgICBjb25zdCBpc0xhc3RWaXNpYmxlT3B0aW9uID0gb3B0aW9uLmluZGV4ID09PSBzdGF0ZS52aXNpYmxlVG9JbmRleCAtIDFcbiAgICAgICAgICBjb25zdCBhcmVNb3JlT3B0aW9uc0JlbG93ID0gc3RhdGUudmlzaWJsZVRvSW5kZXggPCBvcHRpb25zLmxlbmd0aFxuICAgICAgICAgIGNvbnN0IGFyZU1vcmVPcHRpb25zQWJvdmUgPSBzdGF0ZS52aXNpYmxlRnJvbUluZGV4ID4gMFxuXG4gICAgICAgICAgY29uc3QgaSA9IHN0YXRlLnZpc2libGVGcm9tSW5kZXggKyBpbmRleCArIDFcblxuICAgICAgICAgIGlmIChvcHRpb24udHlwZSA9PT0gJ2lucHV0Jykge1xuICAgICAgICAgICAgY29uc3QgaW5wdXRWYWx1ZSA9IHN0YXRlLmlucHV0VmFsdWVzLmdldChvcHRpb24udmFsdWUpIHx8ICcnXG5cbiAgICAgICAgICAgIHJldHVybiAoXG4gICAgICAgICAgICAgIDxCb3gga2V5PXtTdHJpbmcob3B0aW9uLnZhbHVlKX0gZ2FwPXsxfT5cbiAgICAgICAgICAgICAgICA8U2VsZWN0SW5wdXRPcHRpb25cbiAgICAgICAgICAgICAgICAgIG9wdGlvbj17b3B0aW9ufVxuICAgICAgICAgICAgICAgICAgaXNGb2N1c2VkPXtpc09wdGlvbkZvY3VzZWR9XG4gICAgICAgICAgICAgICAgICBpc1NlbGVjdGVkPXtcbiAgICAgICAgICAgICAgICAgICAgZmFsc2UgLyogV2Ugc2hvdyBzZWxlY3Rpb24gc3RhdGUgZGlmZmVyZW50bHkgZm9yIG11bHRpLXNlbGVjdCAqL1xuICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgc2hvdWxkU2hvd0Rvd25BcnJvdz17XG4gICAgICAgICAgICAgICAgICAgIGFyZU1vcmVPcHRpb25zQmVsb3cgJiYgaXNMYXN0VmlzaWJsZU9wdGlvblxuICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgc2hvdWxkU2hvd1VwQXJyb3c9e1xuICAgICAgICAgICAgICAgICAgICBhcmVNb3JlT3B0aW9uc0Fib3ZlICYmIGlzRmlyc3RWaXNpYmxlT3B0aW9uXG4gICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICBtYXhJbmRleFdpZHRoPXttYXhJbmRleFdpZHRofVxuICAgICAgICAgICAgICAgICAgaW5kZXg9e2l9XG4gICAgICAgICAgICAgICAgICBpbnB1dFZhbHVlPXtpbnB1dFZhbHVlfVxuICAgICAgICAgICAgICAgICAgb25JbnB1dENoYW5nZT17dmFsdWUgPT4ge1xuICAgICAgICAgICAgICAgICAgICBzdGF0ZS51cGRhdGVJbnB1dFZhbHVlKG9wdGlvbi52YWx1ZSwgdmFsdWUpXG4gICAgICAgICAgICAgICAgICB9fVxuICAgICAgICAgICAgICAgICAgb25TdWJtaXQ9eygpID0+IHt9fSAvKiBXZSBoYW5kbGUgc3VibWl0IGhpZ2hlciB1cCAqL1xuICAgICAgICAgICAgICAgICAgb25FeGl0PXsoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgIG9uQ2FuY2VsKClcbiAgICAgICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgICAgICBsYXlvdXQ9XCJjb21wYWN0XCJcbiAgICAgICAgICAgICAgICAgIG9uT3BlbkVkaXRvcj17b25PcGVuRWRpdG9yfVxuICAgICAgICAgICAgICAgICAgb25JbWFnZVBhc3RlPXtvbkltYWdlUGFzdGV9XG4gICAgICAgICAgICAgICAgICBwYXN0ZWRDb250ZW50cz17cGFzdGVkQ29udGVudHN9XG4gICAgICAgICAgICAgICAgICBvblJlbW92ZUltYWdlPXtvblJlbW92ZUltYWdlfVxuICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgIDxUZXh0IGNvbG9yPXtpc1NlbGVjdGVkID8gJ3N1Y2Nlc3MnIDogdW5kZWZpbmVkfT5cbiAgICAgICAgICAgICAgICAgICAgW3tpc1NlbGVjdGVkID8gZmlndXJlcy50aWNrIDogJyAnfV17JyAnfVxuICAgICAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgICAgIDwvU2VsZWN0SW5wdXRPcHRpb24+XG4gICAgICAgICAgICAgIDwvQm94PlxuICAgICAgICAgICAgKVxuICAgICAgICAgIH1cblxuICAgICAgICAgIHJldHVybiAoXG4gICAgICAgICAgICA8Qm94IGtleT17U3RyaW5nKG9wdGlvbi52YWx1ZSl9IGdhcD17MX0+XG4gICAgICAgICAgICAgIDxTZWxlY3RPcHRpb25cbiAgICAgICAgICAgICAgICBpc0ZvY3VzZWQ9e2lzT3B0aW9uRm9jdXNlZH1cbiAgICAgICAgICAgICAgICBpc1NlbGVjdGVkPXtcbiAgICAgICAgICAgICAgICAgIGZhbHNlIC8qIFdlIHNob3cgc2VsZWN0aW9uIHN0YXRlIGRpZmZlcmVudGx5IGZvciBtdWx0aS1zZWxlY3QgKi9cbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgc2hvdWxkU2hvd0Rvd25BcnJvdz17YXJlTW9yZU9wdGlvbnNCZWxvdyAmJiBpc0xhc3RWaXNpYmxlT3B0aW9ufVxuICAgICAgICAgICAgICAgIHNob3VsZFNob3dVcEFycm93PXthcmVNb3JlT3B0aW9uc0Fib3ZlICYmIGlzRmlyc3RWaXNpYmxlT3B0aW9ufVxuICAgICAgICAgICAgICAgIGRlc2NyaXB0aW9uPXtvcHRpb24uZGVzY3JpcHRpb259XG4gICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICB7IWhpZGVJbmRleGVzICYmIChcbiAgICAgICAgICAgICAgICAgIDxUZXh0IGRpbUNvbG9yPntgJHtpfS5gLnBhZEVuZChtYXhJbmRleFdpZHRoKX08L1RleHQ+XG4gICAgICAgICAgICAgICAgKX1cbiAgICAgICAgICAgICAgICA8VGV4dCBjb2xvcj17aXNTZWxlY3RlZCA/ICdzdWNjZXNzJyA6IHVuZGVmaW5lZH0+XG4gICAgICAgICAgICAgICAgICBbe2lzU2VsZWN0ZWQgPyBmaWd1cmVzLnRpY2sgOiAnICd9XVxuICAgICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgICA8VGV4dCBjb2xvcj17aXNPcHRpb25Gb2N1c2VkID8gJ3N1Z2dlc3Rpb24nIDogdW5kZWZpbmVkfT5cbiAgICAgICAgICAgICAgICAgIHtvcHRpb24ubGFiZWx9XG4gICAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgICA8L1NlbGVjdE9wdGlvbj5cbiAgICAgICAgICAgIDwvQm94PlxuICAgICAgICAgIClcbiAgICAgICAgfSl9XG4gICAgICA8L0JveD5cbiAgICAgIHtzdWJtaXRCdXR0b25UZXh0ICYmIG9uU3VibWl0ICYmIChcbiAgICAgICAgPEJveCBtYXJnaW5Ub3A9ezB9IGdhcD17MX0+XG4gICAgICAgICAge3N0YXRlLmlzU3VibWl0Rm9jdXNlZCA/IChcbiAgICAgICAgICAgIDxUZXh0IGNvbG9yPVwic3VnZ2VzdGlvblwiPntmaWd1cmVzLnBvaW50ZXJ9PC9UZXh0PlxuICAgICAgICAgICkgOiAoXG4gICAgICAgICAgICA8VGV4dD4gPC9UZXh0PlxuICAgICAgICAgICl9XG4gICAgICAgICAgPEJveCBtYXJnaW5MZWZ0PXszfT5cbiAgICAgICAgICAgIDxUZXh0XG4gICAgICAgICAgICAgIGNvbG9yPXtzdGF0ZS5pc1N1Ym1pdEZvY3VzZWQgPyAnc3VnZ2VzdGlvbicgOiB1bmRlZmluZWR9XG4gICAgICAgICAgICAgIGJvbGQ9e3RydWV9XG4gICAgICAgICAgICA+XG4gICAgICAgICAgICAgIHtzdWJtaXRCdXR0b25UZXh0fVxuICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgIDwvQm94PlxuICAgICAgICA8L0JveD5cbiAgICAgICl9XG4gICAgPC9Cb3g+XG4gIClcbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU9BLE9BQU8sTUFBTSxTQUFTO0FBQzdCLE9BQU9DLEtBQUssTUFBTSxPQUFPO0FBQ3pCLFNBQVNDLEdBQUcsRUFBRUMsSUFBSSxRQUFRLGNBQWM7QUFDeEMsY0FBY0MsYUFBYSxRQUFRLHVCQUF1QjtBQUMxRCxjQUFjQyxlQUFlLFFBQVEsNkJBQTZCO0FBQ2xFLGNBQWNDLHFCQUFxQixRQUFRLGFBQWE7QUFDeEQsU0FBU0MsaUJBQWlCLFFBQVEsMEJBQTBCO0FBQzVELFNBQVNDLFlBQVksUUFBUSxvQkFBb0I7QUFDakQsU0FBU0MsbUJBQW1CLFFBQVEsNkJBQTZCO0FBRWpFLE9BQU8sS0FBS0MsZ0JBQWdCLENBQUMsQ0FBQyxDQUFDLEdBQUc7RUFDaEMsU0FBU0MsVUFBVSxDQUFDLEVBQUUsT0FBTztFQUM3QixTQUFTQyxrQkFBa0IsQ0FBQyxFQUFFLE1BQU07RUFDcEMsU0FBU0MsT0FBTyxFQUFFUCxxQkFBcUIsQ0FBQ1EsQ0FBQyxDQUFDLEVBQUU7RUFDNUMsU0FBU0MsWUFBWSxDQUFDLEVBQUVELENBQUMsRUFBRTtFQUMzQixTQUFTRSxRQUFRLEVBQUUsR0FBRyxHQUFHLElBQUk7RUFDN0IsU0FBU0MsUUFBUSxDQUFDLEVBQUUsQ0FBQ0MsTUFBTSxFQUFFSixDQUFDLEVBQUUsRUFBRSxHQUFHLElBQUk7RUFDekMsU0FBU0ssT0FBTyxDQUFDLEVBQUUsQ0FBQ0MsS0FBSyxFQUFFTixDQUFDLEVBQUUsR0FBRyxJQUFJO0VBQ3JDLFNBQVNPLFVBQVUsQ0FBQyxFQUFFUCxDQUFDO0VBQ3ZCO0FBQ0Y7QUFDQTtBQUNBO0FBQ0E7RUFDRSxTQUFTUSxnQkFBZ0IsQ0FBQyxFQUFFLE1BQU07RUFDbEM7QUFDRjtBQUNBO0VBQ0UsU0FBU0MsUUFBUSxDQUFDLEVBQUUsQ0FBQ0wsTUFBTSxFQUFFSixDQUFDLEVBQUUsRUFBRSxHQUFHLElBQUk7RUFDekM7QUFDRjtBQUNBO0VBQ0UsU0FBU1UsV0FBVyxDQUFDLEVBQUUsT0FBTztFQUM5QjtBQUNGO0FBQ0E7QUFDQTtFQUNFLFNBQVNDLGtCQUFrQixDQUFDLEVBQUUsR0FBRyxHQUFHLElBQUk7RUFDeEM7QUFDRjtBQUNBO0FBQ0E7RUFDRSxTQUFTQyxpQkFBaUIsQ0FBQyxFQUFFLEdBQUcsR0FBRyxJQUFJO0VBQ3ZDO0FBQ0Y7QUFDQTtFQUNFLFNBQVNDLGdCQUFnQixDQUFDLEVBQUUsT0FBTztFQUNuQztBQUNGO0FBQ0E7QUFDQTtBQUNBO0VBQ0UsU0FBU0MsWUFBWSxDQUFDLEVBQUUsQ0FDdEJDLFlBQVksRUFBRSxNQUFNLEVBQ3BCQyxRQUFRLEVBQUUsQ0FBQ1YsS0FBSyxFQUFFLE1BQU0sRUFBRSxHQUFHLElBQUksRUFDakMsR0FBRyxJQUFJO0VBQ1QsU0FBU1csWUFBWSxDQUFDLEVBQUUsQ0FDdEJDLFdBQVcsRUFBRSxNQUFNLEVBQ25CQyxTQUFrQixDQUFSLEVBQUUsTUFBTSxFQUNsQkMsUUFBaUIsQ0FBUixFQUFFLE1BQU0sRUFDakJDLFVBQTRCLENBQWpCLEVBQUU5QixlQUFlLEVBQzVCK0IsVUFBbUIsQ0FBUixFQUFFLE1BQU0sRUFDbkIsR0FBRyxJQUFJO0VBQ1QsU0FBU0MsY0FBYyxDQUFDLEVBQUVDLE1BQU0sQ0FBQyxNQUFNLEVBQUVsQyxhQUFhLENBQUM7RUFDdkQsU0FBU21DLGFBQWEsQ0FBQyxFQUFFLENBQUNDLEVBQUUsRUFBRSxNQUFNLEVBQUUsR0FBRyxJQUFJO0FBQy9DLENBQUM7QUFFRCxPQUFPLFNBQUFDLFlBQUFDLEVBQUE7RUFBQSxNQUFBQyxDQUFBLEdBQUFDLEVBQUE7RUFBd0I7SUFBQWpDLFVBQUEsRUFBQWtDLEVBQUE7SUFBQWpDLGtCQUFBLEVBQUFrQyxFQUFBO0lBQUFqQyxPQUFBO0lBQUFFLFlBQUEsRUFBQWdDLEVBQUE7SUFBQS9CLFFBQUE7SUFBQUMsUUFBQTtJQUFBRSxPQUFBO0lBQUFFLFVBQUE7SUFBQUMsZ0JBQUE7SUFBQUMsUUFBQTtJQUFBRSxrQkFBQTtJQUFBQyxpQkFBQTtJQUFBQyxnQkFBQTtJQUFBQyxZQUFBO0lBQUFKLFdBQUEsRUFBQXdCLEVBQUE7SUFBQWpCLFlBQUE7SUFBQU0sY0FBQTtJQUFBRTtFQUFBLElBQUFHLEVBbUJUO0VBbEJwQixNQUFBL0IsVUFBQSxHQUFBa0MsRUFBa0IsS0FBbEJJLFNBQWtCLEdBQWxCLEtBQWtCLEdBQWxCSixFQUFrQjtFQUNsQixNQUFBakMsa0JBQUEsR0FBQWtDLEVBQXNCLEtBQXRCRyxTQUFzQixHQUF0QixDQUFzQixHQUF0QkgsRUFBc0I7RUFBQSxJQUFBSSxFQUFBO0VBQUEsSUFBQVAsQ0FBQSxRQUFBSSxFQUFBO0lBRXRCRyxFQUFBLEdBQUFILEVBQWlCLEtBQWpCRSxTQUFpQixHQUFqQixFQUFpQixHQUFqQkYsRUFBaUI7SUFBQUosQ0FBQSxNQUFBSSxFQUFBO0lBQUFKLENBQUEsTUFBQU8sRUFBQTtFQUFBO0lBQUFBLEVBQUEsR0FBQVAsQ0FBQTtFQUFBO0VBQWpCLE1BQUE1QixZQUFBLEdBQUFtQyxFQUFpQjtFQVdqQixNQUFBMUIsV0FBQSxHQUFBd0IsRUFBbUIsS0FBbkJDLFNBQW1CLEdBQW5CLEtBQW1CLEdBQW5CRCxFQUFtQjtFQUFBLElBQUFHLEVBQUE7RUFBQSxJQUFBUixDQUFBLFFBQUE1QixZQUFBLElBQUE0QixDQUFBLFFBQUF0QixVQUFBLElBQUFzQixDQUFBLFFBQUFuQixXQUFBLElBQUFtQixDQUFBLFFBQUFoQixnQkFBQSxJQUFBZ0IsQ0FBQSxRQUFBaEMsVUFBQSxJQUFBZ0MsQ0FBQSxRQUFBM0IsUUFBQSxJQUFBMkIsQ0FBQSxRQUFBMUIsUUFBQSxJQUFBMEIsQ0FBQSxRQUFBbEIsa0JBQUEsSUFBQWtCLENBQUEsU0FBQXhCLE9BQUEsSUFBQXdCLENBQUEsU0FBQXBCLFFBQUEsSUFBQW9CLENBQUEsU0FBQWpCLGlCQUFBLElBQUFpQixDQUFBLFNBQUE5QixPQUFBLElBQUE4QixDQUFBLFNBQUFyQixnQkFBQSxJQUFBcUIsQ0FBQSxTQUFBL0Isa0JBQUE7SUFLa0J1QyxFQUFBO01BQUF4QyxVQUFBO01BQUFDLGtCQUFBO01BQUFDLE9BQUE7TUFBQUUsWUFBQTtNQUFBRSxRQUFBO01BQUFELFFBQUE7TUFBQUcsT0FBQTtNQUFBRSxVQUFBO01BQUFDLGdCQUFBO01BQUFDLFFBQUE7TUFBQUUsa0JBQUE7TUFBQUMsaUJBQUE7TUFBQUMsZ0JBQUE7TUFBQUg7SUFlckMsQ0FBQztJQUFBbUIsQ0FBQSxNQUFBNUIsWUFBQTtJQUFBNEIsQ0FBQSxNQUFBdEIsVUFBQTtJQUFBc0IsQ0FBQSxNQUFBbkIsV0FBQTtJQUFBbUIsQ0FBQSxNQUFBaEIsZ0JBQUE7SUFBQWdCLENBQUEsTUFBQWhDLFVBQUE7SUFBQWdDLENBQUEsTUFBQTNCLFFBQUE7SUFBQTJCLENBQUEsTUFBQTFCLFFBQUE7SUFBQTBCLENBQUEsTUFBQWxCLGtCQUFBO0lBQUFrQixDQUFBLE9BQUF4QixPQUFBO0lBQUF3QixDQUFBLE9BQUFwQixRQUFBO0lBQUFvQixDQUFBLE9BQUFqQixpQkFBQTtJQUFBaUIsQ0FBQSxPQUFBOUIsT0FBQTtJQUFBOEIsQ0FBQSxPQUFBckIsZ0JBQUE7SUFBQXFCLENBQUEsT0FBQS9CLGtCQUFBO0lBQUErQixDQUFBLE9BQUFRLEVBQUE7RUFBQTtJQUFBQSxFQUFBLEdBQUFSLENBQUE7RUFBQTtFQWZELE1BQUFTLEtBQUEsR0FBYzNDLG1CQUFtQixDQUFJMEMsRUFlcEMsQ0FBQztFQUFBLElBQUFFLEVBQUE7RUFBQSxJQUFBQyxFQUFBO0VBQUEsSUFBQUMsRUFBQTtFQUFBLElBQUFDLEVBQUE7RUFBQSxJQUFBQyxFQUFBO0VBQUEsSUFBQWQsQ0FBQSxTQUFBbkIsV0FBQSxJQUFBbUIsQ0FBQSxTQUFBaEMsVUFBQSxJQUFBZ0MsQ0FBQSxTQUFBM0IsUUFBQSxJQUFBMkIsQ0FBQSxTQUFBWixZQUFBLElBQUFZLENBQUEsU0FBQWYsWUFBQSxJQUFBZSxDQUFBLFNBQUFKLGFBQUEsSUFBQUksQ0FBQSxTQUFBOUIsT0FBQSxDQUFBNkMsTUFBQSxJQUFBZixDQUFBLFNBQUFOLGNBQUEsSUFBQU0sQ0FBQSxTQUFBUyxLQUFBO0lBRUYsTUFBQU8sYUFBQSxHQUFzQjlDLE9BQU8sQ0FBQTZDLE1BQU8sQ0FBQUUsUUFBUyxDQUFDLENBQUMsQ0FBQUYsTUFBTztJQUduREosRUFBQSxHQUFBcEQsR0FBRztJQUFldUQsRUFBQSxXQUFRO0lBQ3hCSixFQUFBLEdBQUFuRCxHQUFHO0lBQWVxRCxFQUFBLFdBQVE7SUFDeEJDLEVBQUEsR0FBQUosS0FBSyxDQUFBUyxjQUFlLENBQUFDLEdBQUksQ0FBQyxDQUFBQyxNQUFBLEVBQUFDLEtBQUE7TUFDeEIsTUFBQUMsZUFBQSxHQUNFLENBQUN0RCxVQUNrQyxJQUFuQ3lDLEtBQUssQ0FBQWMsWUFBYSxLQUFLSCxNQUFNLENBQUEzQyxLQUNQLElBRnRCLENBRUNnQyxLQUFLLENBQUFlLGVBQWdCO01BQ3hCLE1BQUFDLFVBQUEsR0FBbUJoQixLQUFLLENBQUFpQixjQUFlLENBQUFDLFFBQVMsQ0FBQ1AsTUFBTSxDQUFBM0MsS0FBTSxDQUFDO01BRTlELE1BQUFtRCxvQkFBQSxHQUE2QlIsTUFBTSxDQUFBQyxLQUFNLEtBQUtaLEtBQUssQ0FBQW9CLGdCQUFpQjtNQUNwRSxNQUFBQyxtQkFBQSxHQUE0QlYsTUFBTSxDQUFBQyxLQUFNLEtBQUtaLEtBQUssQ0FBQXNCLGNBQWUsR0FBRyxDQUFDO01BQ3JFLE1BQUFDLG1CQUFBLEdBQTRCdkIsS0FBSyxDQUFBc0IsY0FBZSxHQUFHN0QsT0FBTyxDQUFBNkMsTUFBTztNQUNqRSxNQUFBa0IsbUJBQUEsR0FBNEJ4QixLQUFLLENBQUFvQixnQkFBaUIsR0FBRyxDQUFDO01BRXRELE1BQUFLLENBQUEsR0FBVXpCLEtBQUssQ0FBQW9CLGdCQUFpQixHQUFHUixLQUFLLEdBQUcsQ0FBQztNQUU1QyxJQUFJRCxNQUFNLENBQUFlLElBQUssS0FBSyxPQUFPO1FBQ3pCLE1BQUFDLFVBQUEsR0FBbUIzQixLQUFLLENBQUE0QixXQUFZLENBQUFDLEdBQUksQ0FBQ2xCLE1BQU0sQ0FBQTNDLEtBQVksQ0FBQyxJQUF6QyxFQUF5QztRQUFBLE9BRzFELENBQUMsR0FBRyxDQUFNLEdBQW9CLENBQXBCLENBQUE4RCxNQUFNLENBQUNuQixNQUFNLENBQUEzQyxLQUFNLEVBQUMsQ0FBTyxHQUFDLENBQUQsR0FBQyxDQUNwQyxDQUFDLGlCQUFpQixDQUNSMkMsTUFBTSxDQUFOQSxPQUFLLENBQUMsQ0FDSEUsU0FBZSxDQUFmQSxnQkFBYyxDQUFDLENBRXhCLFVBQUssQ0FBTCxNQUFJLENBQUMsQ0FHTCxtQkFBMEMsQ0FBMUMsQ0FBQVUsbUJBQTBDLElBQTFDRixtQkFBeUMsQ0FBQyxDQUcxQyxpQkFBMkMsQ0FBM0MsQ0FBQUcsbUJBQTJDLElBQTNDTCxvQkFBMEMsQ0FBQyxDQUU5QlosYUFBYSxDQUFiQSxjQUFZLENBQUMsQ0FDckJrQixLQUFDLENBQURBLEVBQUEsQ0FBQyxDQUNJRSxVQUFVLENBQVZBLFdBQVMsQ0FBQyxDQUNQLGFBRWQsQ0FGYyxDQUFBM0QsS0FBQTtZQUNiZ0MsS0FBSyxDQUFBK0IsZ0JBQWlCLENBQUNwQixNQUFNLENBQUEzQyxLQUFNLEVBQUVBLEtBQUssQ0FBQztVQUFBLENBQzdDLENBQUMsQ0FDUyxRQUFRLENBQVIsQ0FBQWdFLEtBQU8sQ0FBQyxDQUNWLE1BRVAsQ0FGTztZQUNOcEUsUUFBUSxDQUFDLENBQUM7VUFBQSxDQUNaLENBQUMsQ0FDTSxNQUFTLENBQVQsU0FBUyxDQUNGWSxZQUFZLENBQVpBLGFBQVcsQ0FBQyxDQUNaRyxZQUFZLENBQVpBLGFBQVcsQ0FBQyxDQUNWTSxjQUFjLENBQWRBLGVBQWEsQ0FBQyxDQUNmRSxhQUFhLENBQWJBLGNBQVksQ0FBQyxDQUU1QixDQUFDLElBQUksQ0FBUSxLQUFrQyxDQUFsQyxDQUFBNkIsVUFBVSxHQUFWLFNBQWtDLEdBQWxDbkIsU0FBaUMsQ0FBQyxDQUFFLENBQzdDLENBQUFtQixVQUFVLEdBQUdwRSxPQUFPLENBQUFxRixJQUFXLEdBQS9CLEdBQThCLENBQUUsQ0FBRSxJQUFFLENBQ3hDLEVBRkMsSUFBSSxDQUdQLEVBL0JDLGlCQUFpQixDQWdDcEIsRUFqQ0MsR0FBRyxDQWlDRTtNQUFBO01BRVQsT0FHQyxDQUFDLEdBQUcsQ0FBTSxHQUFvQixDQUFwQixDQUFBSCxNQUFNLENBQUNuQixNQUFNLENBQUEzQyxLQUFNLEVBQUMsQ0FBTyxHQUFDLENBQUQsR0FBQyxDQUNwQyxDQUFDLFlBQVksQ0FDQTZDLFNBQWUsQ0FBZkEsZ0JBQWMsQ0FBQyxDQUV4QixVQUFLLENBQUwsTUFBSSxDQUFDLENBRWMsbUJBQTBDLENBQTFDLENBQUFVLG1CQUEwQyxJQUExQ0YsbUJBQXlDLENBQUMsQ0FDNUMsaUJBQTJDLENBQTNDLENBQUFHLG1CQUEyQyxJQUEzQ0wsb0JBQTBDLENBQUMsQ0FDakQsV0FBa0IsQ0FBbEIsQ0FBQVIsTUFBTSxDQUFBdUIsV0FBVyxDQUFDLENBRTlCLEVBQUM5RCxXQUVELElBREMsQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFSLEtBQU8sQ0FBQyxDQUFFLElBQUdxRCxDQUFDLEdBQUcsQ0FBQVUsTUFBTyxDQUFDNUIsYUFBYSxFQUFFLEVBQTdDLElBQUksQ0FDUCxDQUNBLENBQUMsSUFBSSxDQUFRLEtBQWtDLENBQWxDLENBQUFTLFVBQVUsR0FBVixTQUFrQyxHQUFsQ25CLFNBQWlDLENBQUMsQ0FBRSxDQUM3QyxDQUFBbUIsVUFBVSxHQUFHcEUsT0FBTyxDQUFBcUYsSUFBVyxHQUEvQixHQUE4QixDQUFFLENBQ3BDLEVBRkMsSUFBSSxDQUdMLENBQUMsSUFBSSxDQUFRLEtBQTBDLENBQTFDLENBQUFwQixlQUFlLEdBQWYsWUFBMEMsR0FBMUNoQixTQUF5QyxDQUFDLENBQ3BELENBQUFjLE1BQU0sQ0FBQXlCLEtBQUssQ0FDZCxFQUZDLElBQUksQ0FHUCxFQWxCQyxZQUFZLENBbUJmLEVBcEJDLEdBQUcsQ0FvQkU7SUFBQSxDQUVULENBQUM7SUFBQTdDLENBQUEsT0FBQW5CLFdBQUE7SUFBQW1CLENBQUEsT0FBQWhDLFVBQUE7SUFBQWdDLENBQUEsT0FBQTNCLFFBQUE7SUFBQTJCLENBQUEsT0FBQVosWUFBQTtJQUFBWSxDQUFBLE9BQUFmLFlBQUE7SUFBQWUsQ0FBQSxPQUFBSixhQUFBO0lBQUFJLENBQUEsT0FBQTlCLE9BQUEsQ0FBQTZDLE1BQUE7SUFBQWYsQ0FBQSxPQUFBTixjQUFBO0lBQUFNLENBQUEsT0FBQVMsS0FBQTtJQUFBVCxDQUFBLE9BQUFVLEVBQUE7SUFBQVYsQ0FBQSxPQUFBVyxFQUFBO0lBQUFYLENBQUEsT0FBQVksRUFBQTtJQUFBWixDQUFBLE9BQUFhLEVBQUE7SUFBQWIsQ0FBQSxPQUFBYyxFQUFBO0VBQUE7SUFBQUosRUFBQSxHQUFBVixDQUFBO0lBQUFXLEVBQUEsR0FBQVgsQ0FBQTtJQUFBWSxFQUFBLEdBQUFaLENBQUE7SUFBQWEsRUFBQSxHQUFBYixDQUFBO0lBQUFjLEVBQUEsR0FBQWQsQ0FBQTtFQUFBO0VBQUEsSUFBQThDLEdBQUE7RUFBQSxJQUFBOUMsQ0FBQSxTQUFBVSxFQUFBLElBQUFWLENBQUEsU0FBQVksRUFBQSxJQUFBWixDQUFBLFNBQUFhLEVBQUE7SUEvRUppQyxHQUFBLElBQUMsRUFBRyxDQUFlLGFBQVEsQ0FBUixDQUFBbEMsRUFBTyxDQUFDLENBQ3hCLENBQUFDLEVBOEVBLENBQ0gsRUFoRkMsRUFBRyxDQWdGRTtJQUFBYixDQUFBLE9BQUFVLEVBQUE7SUFBQVYsQ0FBQSxPQUFBWSxFQUFBO0lBQUFaLENBQUEsT0FBQWEsRUFBQTtJQUFBYixDQUFBLE9BQUE4QyxHQUFBO0VBQUE7SUFBQUEsR0FBQSxHQUFBOUMsQ0FBQTtFQUFBO0VBQUEsSUFBQStDLEdBQUE7RUFBQSxJQUFBL0MsQ0FBQSxTQUFBcEIsUUFBQSxJQUFBb0IsQ0FBQSxTQUFBUyxLQUFBLENBQUFlLGVBQUEsSUFBQXhCLENBQUEsU0FBQXJCLGdCQUFBO0lBQ0xvRSxHQUFBLEdBQUFwRSxnQkFBNEIsSUFBNUJDLFFBZ0JBLElBZkMsQ0FBQyxHQUFHLENBQVksU0FBQyxDQUFELEdBQUMsQ0FBTyxHQUFDLENBQUQsR0FBQyxDQUN0QixDQUFBNkIsS0FBSyxDQUFBZSxlQUlMLEdBSEMsQ0FBQyxJQUFJLENBQU8sS0FBWSxDQUFaLFlBQVksQ0FBRSxDQUFBbkUsT0FBTyxDQUFBMkYsT0FBTyxDQUFFLEVBQXpDLElBQUksQ0FHTixHQURDLENBQUMsSUFBSSxDQUFDLENBQUMsRUFBTixJQUFJLENBQ1AsQ0FDQSxDQUFDLEdBQUcsQ0FBYSxVQUFDLENBQUQsR0FBQyxDQUNoQixDQUFDLElBQUksQ0FDSSxLQUFnRCxDQUFoRCxDQUFBdkMsS0FBSyxDQUFBZSxlQUEyQyxHQUFoRCxZQUFnRCxHQUFoRGxCLFNBQStDLENBQUMsQ0FDakQsSUFBSSxDQUFKLEtBQUcsQ0FBQyxDQUVUM0IsaUJBQWUsQ0FDbEIsRUFMQyxJQUFJLENBTVAsRUFQQyxHQUFHLENBUU4sRUFkQyxHQUFHLENBZUw7SUFBQXFCLENBQUEsT0FBQXBCLFFBQUE7SUFBQW9CLENBQUEsT0FBQVMsS0FBQSxDQUFBZSxlQUFBO0lBQUF4QixDQUFBLE9BQUFyQixnQkFBQTtJQUFBcUIsQ0FBQSxPQUFBK0MsR0FBQTtFQUFBO0lBQUFBLEdBQUEsR0FBQS9DLENBQUE7RUFBQTtFQUFBLElBQUFpRCxHQUFBO0VBQUEsSUFBQWpELENBQUEsU0FBQVcsRUFBQSxJQUFBWCxDQUFBLFNBQUE4QyxHQUFBLElBQUE5QyxDQUFBLFNBQUErQyxHQUFBLElBQUEvQyxDQUFBLFNBQUFjLEVBQUE7SUFsR0htQyxHQUFBLElBQUMsRUFBRyxDQUFlLGFBQVEsQ0FBUixDQUFBbkMsRUFBTyxDQUFDLENBQ3pCLENBQUFnQyxHQWdGSyxDQUNKLENBQUFDLEdBZ0JELENBQ0YsRUFuR0MsRUFBRyxDQW1HRTtJQUFBL0MsQ0FBQSxPQUFBVyxFQUFBO0lBQUFYLENBQUEsT0FBQThDLEdBQUE7SUFBQTlDLENBQUEsT0FBQStDLEdBQUE7SUFBQS9DLENBQUEsT0FBQWMsRUFBQTtJQUFBZCxDQUFBLE9BQUFpRCxHQUFBO0VBQUE7SUFBQUEsR0FBQSxHQUFBakQsQ0FBQTtFQUFBO0VBQUEsT0FuR05pRCxHQW1HTTtBQUFBO0FBM0lILFNBQUFSLE1BQUEiLCJpZ25vcmVMaXN0IjpbXX0= \ No newline at end of file diff --git a/src/components/CustomSelect/index.ts b/src/components/CustomSelect/index.ts new file mode 100644 index 0000000..fee30a5 --- /dev/null +++ b/src/components/CustomSelect/index.ts @@ -0,0 +1,3 @@ +export * from './SelectMulti.js' +export type { OptionWithDescription } from './select.js' +export * from './select.js' diff --git a/src/components/CustomSelect/option-map.ts b/src/components/CustomSelect/option-map.ts new file mode 100644 index 0000000..ef51c5b --- /dev/null +++ b/src/components/CustomSelect/option-map.ts @@ -0,0 +1,50 @@ +import type { ReactNode } from 'react' +import type { OptionWithDescription } from './select.js' + +type OptionMapItem = { + label: ReactNode + value: T + description?: string + previous: OptionMapItem | undefined + next: OptionMapItem | undefined + index: number +} + +export default class OptionMap extends Map> { + readonly first: OptionMapItem | undefined + readonly last: OptionMapItem | undefined + + constructor(options: OptionWithDescription[]) { + const items: Array<[T, OptionMapItem]> = [] + let firstItem: OptionMapItem | undefined + let lastItem: OptionMapItem | undefined + let previous: OptionMapItem | undefined + let index = 0 + + for (const option of options) { + const item = { + label: option.label, + value: option.value, + description: option.description, + previous, + next: undefined, + index, + } + + if (previous) { + previous.next = item + } + + firstItem ||= item + lastItem = item + + items.push([option.value, item]) + index++ + previous = item + } + + super(items) + this.first = firstItem + this.last = lastItem + } +} diff --git a/src/components/CustomSelect/select-input-option.tsx b/src/components/CustomSelect/select-input-option.tsx new file mode 100644 index 0000000..51e60ce --- /dev/null +++ b/src/components/CustomSelect/select-input-option.tsx @@ -0,0 +1,488 @@ +import { c as _c } from "react/compiler-runtime"; +import React, { type ReactNode, useEffect, useRef, useState } from 'react'; +// eslint-disable-next-line custom-rules/prefer-use-keybindings -- UP arrow exit not in Attachments bindings +import { Box, Text, useInput } from '../../ink.js'; +import { useKeybinding, useKeybindings } from '../../keybindings/useKeybinding.js'; +import type { PastedContent } from '../../utils/config.js'; +import { getImageFromClipboard } from '../../utils/imagePaste.js'; +import type { ImageDimensions } from '../../utils/imageResizer.js'; +import { ClickableImageRef } from '../ClickableImageRef.js'; +import { ConfigurableShortcutHint } from '../ConfigurableShortcutHint.js'; +import { Byline } from '../design-system/Byline.js'; +import TextInput from '../TextInput.js'; +import type { OptionWithDescription } from './select.js'; +import { SelectOption } from './select-option.js'; +type Props = { + option: Extract, { + type: 'input'; + }>; + isFocused: boolean; + isSelected: boolean; + shouldShowDownArrow: boolean; + shouldShowUpArrow: boolean; + maxIndexWidth: number; + index: number; + inputValue: string; + onInputChange: (value: string) => void; + onSubmit: (value: string) => void; + onExit?: () => void; + layout: 'compact' | 'expanded'; + children?: ReactNode; + /** + * When true, shows the label before the input field. + * When false (default), uses the label as the placeholder. + */ + showLabel?: boolean; + /** + * Callback to open external editor for editing the input value. + * When provided, ctrl+g will trigger this callback with the current value + * and a setter function to update the internal state. + */ + onOpenEditor?: (currentValue: string, setValue: (value: string) => void) => void; + /** + * When true, automatically reset cursor to end of line when: + * - Option becomes focused + * - Input value changes + * This prevents cursor position bugs when the input value updates asynchronously. + */ + resetCursorOnUpdate?: boolean; + /** + * Optional callback when an image is pasted into the input. + */ + onImagePaste?: (base64Image: string, mediaType?: string, filename?: string, dimensions?: ImageDimensions, sourcePath?: string) => void; + /** + * Pasted content to display inline above the input when focused. + */ + pastedContents?: Record; + /** + * Callback to remove a pasted image by its ID. + */ + onRemoveImage?: (id: number) => void; + /** + * Whether image selection mode is active. + */ + imagesSelected?: boolean; + /** + * Currently selected image index within the image attachments array. + */ + selectedImageIndex?: number; + /** + * Callback to set image selection mode on/off. + */ + onImagesSelectedChange?: (selected: boolean) => void; + /** + * Callback to change the selected image index. + */ + onSelectedImageIndexChange?: (index: number) => void; +}; +export function SelectInputOption(t0) { + const $ = _c(100); + const { + option, + isFocused, + isSelected, + shouldShowDownArrow, + shouldShowUpArrow, + maxIndexWidth, + index, + inputValue, + onInputChange, + onSubmit, + onExit, + layout, + children, + showLabel: t1, + onOpenEditor, + resetCursorOnUpdate: t2, + onImagePaste, + pastedContents, + onRemoveImage, + imagesSelected, + selectedImageIndex: t3, + onImagesSelectedChange, + onSelectedImageIndexChange + } = t0; + const showLabelProp = t1 === undefined ? false : t1; + const resetCursorOnUpdate = t2 === undefined ? false : t2; + const selectedImageIndex = t3 === undefined ? 0 : t3; + let t4; + if ($[0] !== pastedContents) { + t4 = pastedContents ? Object.values(pastedContents).filter(_temp) : []; + $[0] = pastedContents; + $[1] = t4; + } else { + t4 = $[1]; + } + const imageAttachments = t4; + const showLabel = showLabelProp || option.showLabelWithValue === true; + const [cursorOffset, setCursorOffset] = useState(inputValue.length); + const isUserEditing = useRef(false); + let t5; + if ($[2] !== inputValue.length || $[3] !== isFocused || $[4] !== resetCursorOnUpdate) { + t5 = () => { + if (resetCursorOnUpdate && isFocused) { + if (isUserEditing.current) { + isUserEditing.current = false; + } else { + setCursorOffset(inputValue.length); + } + } + }; + $[2] = inputValue.length; + $[3] = isFocused; + $[4] = resetCursorOnUpdate; + $[5] = t5; + } else { + t5 = $[5]; + } + let t6; + if ($[6] !== inputValue || $[7] !== isFocused || $[8] !== resetCursorOnUpdate) { + t6 = [resetCursorOnUpdate, isFocused, inputValue]; + $[6] = inputValue; + $[7] = isFocused; + $[8] = resetCursorOnUpdate; + $[9] = t6; + } else { + t6 = $[9]; + } + useEffect(t5, t6); + let t7; + if ($[10] !== inputValue || $[11] !== onInputChange || $[12] !== onOpenEditor) { + t7 = () => { + onOpenEditor?.(inputValue, onInputChange); + }; + $[10] = inputValue; + $[11] = onInputChange; + $[12] = onOpenEditor; + $[13] = t7; + } else { + t7 = $[13]; + } + const t8 = isFocused && !!onOpenEditor; + let t9; + if ($[14] !== t8) { + t9 = { + context: "Chat", + isActive: t8 + }; + $[14] = t8; + $[15] = t9; + } else { + t9 = $[15]; + } + useKeybinding("chat:externalEditor", t7, t9); + let t10; + if ($[16] !== onImagePaste) { + t10 = () => { + if (!onImagePaste) { + return; + } + getImageFromClipboard().then(imageData => { + if (imageData) { + onImagePaste(imageData.base64, imageData.mediaType, undefined, imageData.dimensions); + } + }); + }; + $[16] = onImagePaste; + $[17] = t10; + } else { + t10 = $[17]; + } + const t11 = isFocused && !!onImagePaste; + let t12; + if ($[18] !== t11) { + t12 = { + context: "Chat", + isActive: t11 + }; + $[18] = t11; + $[19] = t12; + } else { + t12 = $[19]; + } + useKeybinding("chat:imagePaste", t10, t12); + let t13; + if ($[20] !== imageAttachments || $[21] !== onRemoveImage) { + t13 = () => { + if (imageAttachments.length > 0 && onRemoveImage) { + onRemoveImage(imageAttachments.at(-1).id); + } + }; + $[20] = imageAttachments; + $[21] = onRemoveImage; + $[22] = t13; + } else { + t13 = $[22]; + } + const t14 = isFocused && !imagesSelected && inputValue === "" && imageAttachments.length > 0 && !!onRemoveImage; + let t15; + if ($[23] !== t14) { + t15 = { + context: "Attachments", + isActive: t14 + }; + $[23] = t14; + $[24] = t15; + } else { + t15 = $[24]; + } + useKeybinding("attachments:remove", t13, t15); + let t16; + let t17; + if ($[25] !== imageAttachments.length || $[26] !== onSelectedImageIndexChange || $[27] !== selectedImageIndex) { + t16 = () => { + if (imageAttachments.length > 1) { + onSelectedImageIndexChange?.((selectedImageIndex + 1) % imageAttachments.length); + } + }; + t17 = () => { + if (imageAttachments.length > 1) { + onSelectedImageIndexChange?.((selectedImageIndex - 1 + imageAttachments.length) % imageAttachments.length); + } + }; + $[25] = imageAttachments.length; + $[26] = onSelectedImageIndexChange; + $[27] = selectedImageIndex; + $[28] = t16; + $[29] = t17; + } else { + t16 = $[28]; + t17 = $[29]; + } + let t18; + if ($[30] !== imageAttachments || $[31] !== onImagesSelectedChange || $[32] !== onRemoveImage || $[33] !== onSelectedImageIndexChange || $[34] !== selectedImageIndex) { + t18 = () => { + const img = imageAttachments[selectedImageIndex]; + if (img && onRemoveImage) { + onRemoveImage(img.id); + if (imageAttachments.length <= 1) { + onImagesSelectedChange?.(false); + } else { + onSelectedImageIndexChange?.(Math.min(selectedImageIndex, imageAttachments.length - 2)); + } + } + }; + $[30] = imageAttachments; + $[31] = onImagesSelectedChange; + $[32] = onRemoveImage; + $[33] = onSelectedImageIndexChange; + $[34] = selectedImageIndex; + $[35] = t18; + } else { + t18 = $[35]; + } + let t19; + if ($[36] !== onImagesSelectedChange) { + t19 = () => { + onImagesSelectedChange?.(false); + }; + $[36] = onImagesSelectedChange; + $[37] = t19; + } else { + t19 = $[37]; + } + let t20; + if ($[38] !== t16 || $[39] !== t17 || $[40] !== t18 || $[41] !== t19) { + t20 = { + "attachments:next": t16, + "attachments:previous": t17, + "attachments:remove": t18, + "attachments:exit": t19 + }; + $[38] = t16; + $[39] = t17; + $[40] = t18; + $[41] = t19; + $[42] = t20; + } else { + t20 = $[42]; + } + const t21 = isFocused && !!imagesSelected; + let t22; + if ($[43] !== t21) { + t22 = { + context: "Attachments", + isActive: t21 + }; + $[43] = t21; + $[44] = t22; + } else { + t22 = $[44]; + } + useKeybindings(t20, t22); + let t23; + if ($[45] !== onImagesSelectedChange) { + t23 = (_input, key) => { + if (key.upArrow) { + onImagesSelectedChange?.(false); + } + }; + $[45] = onImagesSelectedChange; + $[46] = t23; + } else { + t23 = $[46]; + } + const t24 = isFocused && !!imagesSelected; + let t25; + if ($[47] !== t24) { + t25 = { + isActive: t24 + }; + $[47] = t24; + $[48] = t25; + } else { + t25 = $[48]; + } + useInput(t23, t25); + let t26; + let t27; + if ($[49] !== imagesSelected || $[50] !== isFocused || $[51] !== onImagesSelectedChange) { + t26 = () => { + if (!isFocused && imagesSelected) { + onImagesSelectedChange?.(false); + } + }; + t27 = [isFocused, imagesSelected, onImagesSelectedChange]; + $[49] = imagesSelected; + $[50] = isFocused; + $[51] = onImagesSelectedChange; + $[52] = t26; + $[53] = t27; + } else { + t26 = $[52]; + t27 = $[53]; + } + useEffect(t26, t27); + const descriptionPaddingLeft = layout === "expanded" ? maxIndexWidth + 3 : maxIndexWidth + 4; + const t28 = layout === "compact" ? 0 : undefined; + const t29 = `${index}.`; + let t30; + if ($[54] !== maxIndexWidth || $[55] !== t29) { + t30 = t29.padEnd(maxIndexWidth + 2); + $[54] = maxIndexWidth; + $[55] = t29; + $[56] = t30; + } else { + t30 = $[56]; + } + let t31; + if ($[57] !== t30) { + t31 = {t30}; + $[57] = t30; + $[58] = t31; + } else { + t31 = $[58]; + } + let t32; + if ($[59] !== cursorOffset || $[60] !== imagesSelected || $[61] !== inputValue || $[62] !== isFocused || $[63] !== onExit || $[64] !== onImagePaste || $[65] !== onInputChange || $[66] !== onSubmit || $[67] !== option || $[68] !== showLabel) { + t32 = showLabel ? <>{option.label}{isFocused ? <>{option.labelValueSeparator ?? ", "} { + isUserEditing.current = true; + onInputChange(value); + option.onChange(value); + }} onSubmit={onSubmit} onExit={onExit} placeholder={option.placeholder} focus={!imagesSelected} showCursor={true} multiline={true} cursorOffset={cursorOffset} onChangeCursorOffset={setCursorOffset} columns={80} onImagePaste={onImagePaste} onPaste={pastedText => { + isUserEditing.current = true; + const before = inputValue.slice(0, cursorOffset); + const after = inputValue.slice(cursorOffset); + const newValue = before + pastedText + after; + onInputChange(newValue); + option.onChange(newValue); + setCursorOffset(before.length + pastedText.length); + }} /> : inputValue && {option.labelValueSeparator ?? ", "}{inputValue}} : isFocused ? { + isUserEditing.current = true; + onInputChange(value_0); + option.onChange(value_0); + }} onSubmit={onSubmit} onExit={onExit} placeholder={option.placeholder || (typeof option.label === "string" ? option.label : undefined)} focus={!imagesSelected} showCursor={true} multiline={true} cursorOffset={cursorOffset} onChangeCursorOffset={setCursorOffset} columns={80} onImagePaste={onImagePaste} onPaste={pastedText_0 => { + isUserEditing.current = true; + const before_0 = inputValue.slice(0, cursorOffset); + const after_0 = inputValue.slice(cursorOffset); + const newValue_0 = before_0 + pastedText_0 + after_0; + onInputChange(newValue_0); + option.onChange(newValue_0); + setCursorOffset(before_0.length + pastedText_0.length); + }} /> : {inputValue || option.placeholder || option.label}; + $[59] = cursorOffset; + $[60] = imagesSelected; + $[61] = inputValue; + $[62] = isFocused; + $[63] = onExit; + $[64] = onImagePaste; + $[65] = onInputChange; + $[66] = onSubmit; + $[67] = option; + $[68] = showLabel; + $[69] = t32; + } else { + t32 = $[69]; + } + let t33; + if ($[70] !== children || $[71] !== t28 || $[72] !== t31 || $[73] !== t32) { + t33 = {t31}{children}{t32}; + $[70] = children; + $[71] = t28; + $[72] = t31; + $[73] = t32; + $[74] = t33; + } else { + t33 = $[74]; + } + let t34; + if ($[75] !== isFocused || $[76] !== isSelected || $[77] !== shouldShowDownArrow || $[78] !== shouldShowUpArrow || $[79] !== t33) { + t34 = {t33}; + $[75] = isFocused; + $[76] = isSelected; + $[77] = shouldShowDownArrow; + $[78] = shouldShowUpArrow; + $[79] = t33; + $[80] = t34; + } else { + t34 = $[80]; + } + let t35; + if ($[81] !== descriptionPaddingLeft || $[82] !== isFocused || $[83] !== isSelected || $[84] !== option.description || $[85] !== option.dimDescription) { + t35 = option.description && {option.description}; + $[81] = descriptionPaddingLeft; + $[82] = isFocused; + $[83] = isSelected; + $[84] = option.description; + $[85] = option.dimDescription; + $[86] = t35; + } else { + t35 = $[86]; + } + let t36; + if ($[87] !== descriptionPaddingLeft || $[88] !== imageAttachments || $[89] !== imagesSelected || $[90] !== isFocused || $[91] !== selectedImageIndex) { + t36 = imageAttachments.length > 0 && {imageAttachments.map((img_0, idx) => )}{imagesSelected ? {imageAttachments.length > 1 && <>} : isFocused ? "(\u2193 to select)" : null}; + $[87] = descriptionPaddingLeft; + $[88] = imageAttachments; + $[89] = imagesSelected; + $[90] = isFocused; + $[91] = selectedImageIndex; + $[92] = t36; + } else { + t36 = $[92]; + } + let t37; + if ($[93] !== layout) { + t37 = layout === "expanded" && ; + $[93] = layout; + $[94] = t37; + } else { + t37 = $[94]; + } + let t38; + if ($[95] !== t34 || $[96] !== t35 || $[97] !== t36 || $[98] !== t37) { + t38 = {t34}{t35}{t36}{t37}; + $[95] = t34; + $[96] = t35; + $[97] = t36; + $[98] = t37; + $[99] = t38; + } else { + t38 = $[99]; + } + return t38; +} +function _temp(c) { + return c.type === "image"; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIlJlYWN0Tm9kZSIsInVzZUVmZmVjdCIsInVzZVJlZiIsInVzZVN0YXRlIiwiQm94IiwiVGV4dCIsInVzZUlucHV0IiwidXNlS2V5YmluZGluZyIsInVzZUtleWJpbmRpbmdzIiwiUGFzdGVkQ29udGVudCIsImdldEltYWdlRnJvbUNsaXBib2FyZCIsIkltYWdlRGltZW5zaW9ucyIsIkNsaWNrYWJsZUltYWdlUmVmIiwiQ29uZmlndXJhYmxlU2hvcnRjdXRIaW50IiwiQnlsaW5lIiwiVGV4dElucHV0IiwiT3B0aW9uV2l0aERlc2NyaXB0aW9uIiwiU2VsZWN0T3B0aW9uIiwiUHJvcHMiLCJvcHRpb24iLCJFeHRyYWN0IiwiVCIsInR5cGUiLCJpc0ZvY3VzZWQiLCJpc1NlbGVjdGVkIiwic2hvdWxkU2hvd0Rvd25BcnJvdyIsInNob3VsZFNob3dVcEFycm93IiwibWF4SW5kZXhXaWR0aCIsImluZGV4IiwiaW5wdXRWYWx1ZSIsIm9uSW5wdXRDaGFuZ2UiLCJ2YWx1ZSIsIm9uU3VibWl0Iiwib25FeGl0IiwibGF5b3V0IiwiY2hpbGRyZW4iLCJzaG93TGFiZWwiLCJvbk9wZW5FZGl0b3IiLCJjdXJyZW50VmFsdWUiLCJzZXRWYWx1ZSIsInJlc2V0Q3Vyc29yT25VcGRhdGUiLCJvbkltYWdlUGFzdGUiLCJiYXNlNjRJbWFnZSIsIm1lZGlhVHlwZSIsImZpbGVuYW1lIiwiZGltZW5zaW9ucyIsInNvdXJjZVBhdGgiLCJwYXN0ZWRDb250ZW50cyIsIlJlY29yZCIsIm9uUmVtb3ZlSW1hZ2UiLCJpZCIsImltYWdlc1NlbGVjdGVkIiwic2VsZWN0ZWRJbWFnZUluZGV4Iiwib25JbWFnZXNTZWxlY3RlZENoYW5nZSIsInNlbGVjdGVkIiwib25TZWxlY3RlZEltYWdlSW5kZXhDaGFuZ2UiLCJTZWxlY3RJbnB1dE9wdGlvbiIsInQwIiwiJCIsIl9jIiwidDEiLCJ0MiIsInQzIiwic2hvd0xhYmVsUHJvcCIsInVuZGVmaW5lZCIsInQ0IiwiT2JqZWN0IiwidmFsdWVzIiwiZmlsdGVyIiwiX3RlbXAiLCJpbWFnZUF0dGFjaG1lbnRzIiwic2hvd0xhYmVsV2l0aFZhbHVlIiwiY3Vyc29yT2Zmc2V0Iiwic2V0Q3Vyc29yT2Zmc2V0IiwibGVuZ3RoIiwiaXNVc2VyRWRpdGluZyIsInQ1IiwiY3VycmVudCIsInQ2IiwidDciLCJ0OCIsInQ5IiwiY29udGV4dCIsImlzQWN0aXZlIiwidDEwIiwidGhlbiIsImltYWdlRGF0YSIsImJhc2U2NCIsInQxMSIsInQxMiIsInQxMyIsImF0IiwidDE0IiwidDE1IiwidDE2IiwidDE3IiwidDE4IiwiaW1nIiwiTWF0aCIsIm1pbiIsInQxOSIsInQyMCIsInQyMSIsInQyMiIsInQyMyIsIl9pbnB1dCIsImtleSIsInVwQXJyb3ciLCJ0MjQiLCJ0MjUiLCJ0MjYiLCJ0MjciLCJkZXNjcmlwdGlvblBhZGRpbmdMZWZ0IiwidDI4IiwidDI5IiwidDMwIiwicGFkRW5kIiwidDMxIiwidDMyIiwibGFiZWwiLCJsYWJlbFZhbHVlU2VwYXJhdG9yIiwib25DaGFuZ2UiLCJwbGFjZWhvbGRlciIsInBhc3RlZFRleHQiLCJiZWZvcmUiLCJzbGljZSIsImFmdGVyIiwibmV3VmFsdWUiLCJ2YWx1ZV8wIiwicGFzdGVkVGV4dF8wIiwiYmVmb3JlXzAiLCJhZnRlcl8wIiwibmV3VmFsdWVfMCIsInQzMyIsInQzNCIsInQzNSIsImRlc2NyaXB0aW9uIiwiZGltRGVzY3JpcHRpb24iLCJ0MzYiLCJtYXAiLCJpbWdfMCIsImlkeCIsInQzNyIsInQzOCIsImMiXSwic291cmNlcyI6WyJzZWxlY3QtaW5wdXQtb3B0aW9uLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgUmVhY3QsIHsgdHlwZSBSZWFjdE5vZGUsIHVzZUVmZmVjdCwgdXNlUmVmLCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIGN1c3RvbS1ydWxlcy9wcmVmZXItdXNlLWtleWJpbmRpbmdzIC0tIFVQIGFycm93IGV4aXQgbm90IGluIEF0dGFjaG1lbnRzIGJpbmRpbmdzXG5pbXBvcnQgeyBCb3gsIFRleHQsIHVzZUlucHV0IH0gZnJvbSAnLi4vLi4vaW5rLmpzJ1xuaW1wb3J0IHtcbiAgdXNlS2V5YmluZGluZyxcbiAgdXNlS2V5YmluZGluZ3MsXG59IGZyb20gJy4uLy4uL2tleWJpbmRpbmdzL3VzZUtleWJpbmRpbmcuanMnXG5pbXBvcnQgdHlwZSB7IFBhc3RlZENvbnRlbnQgfSBmcm9tICcuLi8uLi91dGlscy9jb25maWcuanMnXG5pbXBvcnQgeyBnZXRJbWFnZUZyb21DbGlwYm9hcmQgfSBmcm9tICcuLi8uLi91dGlscy9pbWFnZVBhc3RlLmpzJ1xuaW1wb3J0IHR5cGUgeyBJbWFnZURpbWVuc2lvbnMgfSBmcm9tICcuLi8uLi91dGlscy9pbWFnZVJlc2l6ZXIuanMnXG5pbXBvcnQgeyBDbGlja2FibGVJbWFnZVJlZiB9IGZyb20gJy4uL0NsaWNrYWJsZUltYWdlUmVmLmpzJ1xuaW1wb3J0IHsgQ29uZmlndXJhYmxlU2hvcnRjdXRIaW50IH0gZnJvbSAnLi4vQ29uZmlndXJhYmxlU2hvcnRjdXRIaW50LmpzJ1xuaW1wb3J0IHsgQnlsaW5lIH0gZnJvbSAnLi4vZGVzaWduLXN5c3RlbS9CeWxpbmUuanMnXG5pbXBvcnQgVGV4dElucHV0IGZyb20gJy4uL1RleHRJbnB1dC5qcydcbmltcG9ydCB0eXBlIHsgT3B0aW9uV2l0aERlc2NyaXB0aW9uIH0gZnJvbSAnLi9zZWxlY3QuanMnXG5pbXBvcnQgeyBTZWxlY3RPcHRpb24gfSBmcm9tICcuL3NlbGVjdC1vcHRpb24uanMnXG5cbnR5cGUgUHJvcHM8VD4gPSB7XG4gIG9wdGlvbjogRXh0cmFjdDxPcHRpb25XaXRoRGVzY3JpcHRpb248VD4sIHsgdHlwZTogJ2lucHV0JyB9PlxuICBpc0ZvY3VzZWQ6IGJvb2xlYW5cbiAgaXNTZWxlY3RlZDogYm9vbGVhblxuICBzaG91bGRTaG93RG93bkFycm93OiBib29sZWFuXG4gIHNob3VsZFNob3dVcEFycm93OiBib29sZWFuXG4gIG1heEluZGV4V2lkdGg6IG51bWJlclxuICBpbmRleDogbnVtYmVyXG4gIGlucHV0VmFsdWU6IHN0cmluZ1xuICBvbklucHV0Q2hhbmdlOiAodmFsdWU6IHN0cmluZykgPT4gdm9pZFxuICBvblN1Ym1pdDogKHZhbHVlOiBzdHJpbmcpID0+IHZvaWRcbiAgb25FeGl0PzogKCkgPT4gdm9pZFxuICBsYXlvdXQ6ICdjb21wYWN0JyB8ICdleHBhbmRlZCdcbiAgY2hpbGRyZW4/OiBSZWFjdE5vZGVcbiAgLyoqXG4gICAqIFdoZW4gdHJ1ZSwgc2hvd3MgdGhlIGxhYmVsIGJlZm9yZSB0aGUgaW5wdXQgZmllbGQuXG4gICAqIFdoZW4gZmFsc2UgKGRlZmF1bHQpLCB1c2VzIHRoZSBsYWJlbCBhcyB0aGUgcGxhY2Vob2xkZXIuXG4gICAqL1xuICBzaG93TGFiZWw/OiBib29sZWFuXG4gIC8qKlxuICAgKiBDYWxsYmFjayB0byBvcGVuIGV4dGVybmFsIGVkaXRvciBmb3IgZWRpdGluZyB0aGUgaW5wdXQgdmFsdWUuXG4gICAqIFdoZW4gcHJvdmlkZWQsIGN0cmwrZyB3aWxsIHRyaWdnZXIgdGhpcyBjYWxsYmFjayB3aXRoIHRoZSBjdXJyZW50IHZhbHVlXG4gICAqIGFuZCBhIHNldHRlciBmdW5jdGlvbiB0byB1cGRhdGUgdGhlIGludGVybmFsIHN0YXRlLlxuICAgKi9cbiAgb25PcGVuRWRpdG9yPzogKFxuICAgIGN1cnJlbnRWYWx1ZTogc3RyaW5nLFxuICAgIHNldFZhbHVlOiAodmFsdWU6IHN0cmluZykgPT4gdm9pZCxcbiAgKSA9PiB2b2lkXG4gIC8qKlxuICAgKiBXaGVuIHRydWUsIGF1dG9tYXRpY2FsbHkgcmVzZXQgY3Vyc29yIHRvIGVuZCBvZiBsaW5lIHdoZW46XG4gICAqIC0gT3B0aW9uIGJlY29tZXMgZm9jdXNlZFxuICAgKiAtIElucHV0IHZhbHVlIGNoYW5nZXNcbiAgICogVGhpcyBwcmV2ZW50cyBjdXJzb3IgcG9zaXRpb24gYnVncyB3aGVuIHRoZSBpbnB1dCB2YWx1ZSB1cGRhdGVzIGFzeW5jaHJvbm91c2x5LlxuICAgKi9cbiAgcmVzZXRDdXJzb3JPblVwZGF0ZT86IGJvb2xlYW5cbiAgLyoqXG4gICAqIE9wdGlvbmFsIGNhbGxiYWNrIHdoZW4gYW4gaW1hZ2UgaXMgcGFzdGVkIGludG8gdGhlIGlucHV0LlxuICAgKi9cbiAgb25JbWFnZVBhc3RlPzogKFxuICAgIGJhc2U2NEltYWdlOiBzdHJpbmcsXG4gICAgbWVkaWFUeXBlPzogc3RyaW5nLFxuICAgIGZpbGVuYW1lPzogc3RyaW5nLFxuICAgIGRpbWVuc2lvbnM/OiBJbWFnZURpbWVuc2lvbnMsXG4gICAgc291cmNlUGF0aD86IHN0cmluZyxcbiAgKSA9PiB2b2lkXG4gIC8qKlxuICAgKiBQYXN0ZWQgY29udGVudCB0byBkaXNwbGF5IGlubGluZSBhYm92ZSB0aGUgaW5wdXQgd2hlbiBmb2N1c2VkLlxuICAgKi9cbiAgcGFzdGVkQ29udGVudHM/OiBSZWNvcmQ8bnVtYmVyLCBQYXN0ZWRDb250ZW50PlxuICAvKipcbiAgICogQ2FsbGJhY2sgdG8gcmVtb3ZlIGEgcGFzdGVkIGltYWdlIGJ5IGl0cyBJRC5cbiAgICovXG4gIG9uUmVtb3ZlSW1hZ2U/OiAoaWQ6IG51bWJlcikgPT4gdm9pZFxuICAvKipcbiAgICogV2hldGhlciBpbWFnZSBzZWxlY3Rpb24gbW9kZSBpcyBhY3RpdmUuXG4gICAqL1xuICBpbWFnZXNTZWxlY3RlZD86IGJvb2xlYW5cbiAgLyoqXG4gICAqIEN1cnJlbnRseSBzZWxlY3RlZCBpbWFnZSBpbmRleCB3aXRoaW4gdGhlIGltYWdlIGF0dGFjaG1lbnRzIGFycmF5LlxuICAgKi9cbiAgc2VsZWN0ZWRJbWFnZUluZGV4PzogbnVtYmVyXG4gIC8qKlxuICAgKiBDYWxsYmFjayB0byBzZXQgaW1hZ2Ugc2VsZWN0aW9uIG1vZGUgb24vb2ZmLlxuICAgKi9cbiAgb25JbWFnZXNTZWxlY3RlZENoYW5nZT86IChzZWxlY3RlZDogYm9vbGVhbikgPT4gdm9pZFxuICAvKipcbiAgICogQ2FsbGJhY2sgdG8gY2hhbmdlIHRoZSBzZWxlY3RlZCBpbWFnZSBpbmRleC5cbiAgICovXG4gIG9uU2VsZWN0ZWRJbWFnZUluZGV4Q2hhbmdlPzogKGluZGV4OiBudW1iZXIpID0+IHZvaWRcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIFNlbGVjdElucHV0T3B0aW9uPFQ+KHtcbiAgb3B0aW9uLFxuICBpc0ZvY3VzZWQsXG4gIGlzU2VsZWN0ZWQsXG4gIHNob3VsZFNob3dEb3duQXJyb3csXG4gIHNob3VsZFNob3dVcEFycm93LFxuICBtYXhJbmRleFdpZHRoLFxuICBpbmRleCxcbiAgaW5wdXRWYWx1ZSxcbiAgb25JbnB1dENoYW5nZSxcbiAgb25TdWJtaXQsXG4gIG9uRXhpdCxcbiAgbGF5b3V0LFxuICBjaGlsZHJlbixcbiAgc2hvd0xhYmVsOiBzaG93TGFiZWxQcm9wID0gZmFsc2UsXG4gIG9uT3BlbkVkaXRvcixcbiAgcmVzZXRDdXJzb3JPblVwZGF0ZSA9IGZhbHNlLFxuICBvbkltYWdlUGFzdGUsXG4gIHBhc3RlZENvbnRlbnRzLFxuICBvblJlbW92ZUltYWdlLFxuICBpbWFnZXNTZWxlY3RlZCxcbiAgc2VsZWN0ZWRJbWFnZUluZGV4ID0gMCxcbiAgb25JbWFnZXNTZWxlY3RlZENoYW5nZSxcbiAgb25TZWxlY3RlZEltYWdlSW5kZXhDaGFuZ2UsXG59OiBQcm9wczxUPik6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IGltYWdlQXR0YWNobWVudHMgPSBwYXN0ZWRDb250ZW50c1xuICAgID8gT2JqZWN0LnZhbHVlcyhwYXN0ZWRDb250ZW50cykuZmlsdGVyKGMgPT4gYy50eXBlID09PSAnaW1hZ2UnKVxuICAgIDogW11cblxuICAvLyBBbGxvdyBpbmRpdmlkdWFsIG9wdGlvbnMgdG8gZm9yY2Ugc2hvd2luZyB0aGUgbGFiZWwgdmlhIHNob3dMYWJlbFdpdGhWYWx1ZVxuICBjb25zdCBzaG93TGFiZWwgPSBzaG93TGFiZWxQcm9wIHx8IG9wdGlvbi5zaG93TGFiZWxXaXRoVmFsdWUgPT09IHRydWVcbiAgY29uc3QgW2N1cnNvck9mZnNldCwgc2V0Q3Vyc29yT2Zmc2V0XSA9IHVzZVN0YXRlKGlucHV0VmFsdWUubGVuZ3RoKVxuXG4gIC8vIFRyYWNrIHdoZXRoZXIgdGhlIGxhdGVzdCBpbnB1dFZhbHVlIGNoYW5nZSB3YXMgZnJvbSB1c2VyIHR5cGluZy9wYXN0aW5nLFxuICAvLyBzbyB3ZSBjYW4gc2tpcCByZXNldHRpbmcgY3Vyc29yIHRvIGVuZCBvbiB1c2VyLWluaXRpYXRlZCBjaGFuZ2VzLlxuICBjb25zdCBpc1VzZXJFZGl0aW5nID0gdXNlUmVmKGZhbHNlKVxuXG4gIC8vIFJlc2V0IGN1cnNvciB0byBlbmQgb2YgbGluZSB3aGVuOlxuICAvLyAxLiBPcHRpb24gYmVjb21lcyBmb2N1c2VkICh1c2VyIG5hdmlnYXRlcyB0byBpdClcbiAgLy8gMi4gSW5wdXQgdmFsdWUgY2hhbmdlcyBleHRlcm5hbGx5IChlLmcuLCBhc3luYyBjbGFzc2lmaWVyIGRlc2NyaXB0aW9uIHVwZGF0ZXMpXG4gIC8vIFNraXAgcmVzZXQgd2hlbiB0aGUgY2hhbmdlIHdhcyBmcm9tIHVzZXIgdHlwaW5nICh3aGljaCBzZXRzIGlzVXNlckVkaXRpbmcgcmVmKVxuICAvLyBPbmx5IGVuYWJsZWQgd2hlbiByZXNldEN1cnNvck9uVXBkYXRlIHByb3AgaXMgdHJ1ZVxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmIChyZXNldEN1cnNvck9uVXBkYXRlICYmIGlzRm9jdXNlZCkge1xuICAgICAgaWYgKGlzVXNlckVkaXRpbmcuY3VycmVudCkge1xuICAgICAgICBpc1VzZXJFZGl0aW5nLmN1cnJlbnQgPSBmYWxzZVxuICAgICAgfSBlbHNlIHtcbiAgICAgICAgc2V0Q3Vyc29yT2Zmc2V0KGlucHV0VmFsdWUubGVuZ3RoKVxuICAgICAgfVxuICAgIH1cbiAgfSwgW3Jlc2V0Q3Vyc29yT25VcGRhdGUsIGlzRm9jdXNlZCwgaW5wdXRWYWx1ZV0pXG5cbiAgLy8gY3RybCtnIHRvIG9wZW4gZXh0ZXJuYWwgZWRpdG9yIChyZXVzZXMgY2hhdDpleHRlcm5hbEVkaXRvciBrZXliaW5kaW5nKVxuICB1c2VLZXliaW5kaW5nKFxuICAgICdjaGF0OmV4dGVybmFsRWRpdG9yJyxcbiAgICAoKSA9PiB7XG4gICAgICBvbk9wZW5FZGl0b3I/LihpbnB1dFZhbHVlLCBvbklucHV0Q2hhbmdlKVxuICAgIH0sXG4gICAgeyBjb250ZXh0OiAnQ2hhdCcsIGlzQWN0aXZlOiBpc0ZvY3VzZWQgJiYgISFvbk9wZW5FZGl0b3IgfSxcbiAgKVxuXG4gIC8vIGN0cmwrdiB0byBwYXN0ZSBpbWFnZSBmcm9tIGNsaXBib2FyZCAoc2FtZSBhcyBQcm9tcHRJbnB1dClcbiAgdXNlS2V5YmluZGluZyhcbiAgICAnY2hhdDppbWFnZVBhc3RlJyxcbiAgICAoKSA9PiB7XG4gICAgICBpZiAoIW9uSW1hZ2VQYXN0ZSkgcmV0dXJuXG4gICAgICB2b2lkIGdldEltYWdlRnJvbUNsaXBib2FyZCgpLnRoZW4oaW1hZ2VEYXRhID0+IHtcbiAgICAgICAgaWYgKGltYWdlRGF0YSkge1xuICAgICAgICAgIG9uSW1hZ2VQYXN0ZShcbiAgICAgICAgICAgIGltYWdlRGF0YS5iYXNlNjQsXG4gICAgICAgICAgICBpbWFnZURhdGEubWVkaWFUeXBlLFxuICAgICAgICAgICAgdW5kZWZpbmVkLFxuICAgICAgICAgICAgaW1hZ2VEYXRhLmRpbWVuc2lvbnMsXG4gICAgICAgICAgKVxuICAgICAgICB9XG4gICAgICB9KVxuICAgIH0sXG4gICAgeyBjb250ZXh0OiAnQ2hhdCcsIGlzQWN0aXZlOiBpc0ZvY3VzZWQgJiYgISFvbkltYWdlUGFzdGUgfSxcbiAgKVxuXG4gIC8vIEJhY2tzcGFjZSB3aXRoIGVtcHR5IGlucHV0IHJlbW92ZXMgdGhlIGxhc3QgcGFzdGVkIGltYWdlIChub24taW1hZ2Utc2VsZWN0aW9uIG1vZGUpXG4gIHVzZUtleWJpbmRpbmcoXG4gICAgJ2F0dGFjaG1lbnRzOnJlbW92ZScsXG4gICAgKCkgPT4ge1xuICAgICAgaWYgKGltYWdlQXR0YWNobWVudHMubGVuZ3RoID4gMCAmJiBvblJlbW92ZUltYWdlKSB7XG4gICAgICAgIG9uUmVtb3ZlSW1hZ2UoaW1hZ2VBdHRhY2htZW50cy5hdCgtMSkhLmlkKVxuICAgICAgfVxuICAgIH0sXG4gICAge1xuICAgICAgY29udGV4dDogJ0F0dGFjaG1lbnRzJyxcbiAgICAgIGlzQWN0aXZlOlxuICAgICAgICBpc0ZvY3VzZWQgJiZcbiAgICAgICAgIWltYWdlc1NlbGVjdGVkICYmXG4gICAgICAgIGlucHV0VmFsdWUgPT09ICcnICYmXG4gICAgICAgIGltYWdlQXR0YWNobWVudHMubGVuZ3RoID4gMCAmJlxuICAgICAgICAhIW9uUmVtb3ZlSW1hZ2UsXG4gICAgfSxcbiAgKVxuXG4gIC8vIEltYWdlIHNlbGVjdGlvbiBtb2RlIGtleWJpbmRpbmdzIOKAlCByZXVzZXMgZXhpc3RpbmcgQXR0YWNobWVudHMgYWN0aW9uc1xuICB1c2VLZXliaW5kaW5ncyhcbiAgICB7XG4gICAgICAnYXR0YWNobWVudHM6bmV4dCc6ICgpID0+IHtcbiAgICAgICAgaWYgKGltYWdlQXR0YWNobWVudHMubGVuZ3RoID4gMSkge1xuICAgICAgICAgIG9uU2VsZWN0ZWRJbWFnZUluZGV4Q2hhbmdlPy4oXG4gICAgICAgICAgICAoc2VsZWN0ZWRJbWFnZUluZGV4ICsgMSkgJSBpbWFnZUF0dGFjaG1lbnRzLmxlbmd0aCxcbiAgICAgICAgICApXG4gICAgICAgIH1cbiAgICAgIH0sXG4gICAgICAnYXR0YWNobWVudHM6cHJldmlvdXMnOiAoKSA9PiB7XG4gICAgICAgIGlmIChpbWFnZUF0dGFjaG1lbnRzLmxlbmd0aCA+IDEpIHtcbiAgICAgICAgICBvblNlbGVjdGVkSW1hZ2VJbmRleENoYW5nZT8uKFxuICAgICAgICAgICAgKHNlbGVjdGVkSW1hZ2VJbmRleCAtIDEgKyBpbWFnZUF0dGFjaG1lbnRzLmxlbmd0aCkgJVxuICAgICAgICAgICAgICBpbWFnZUF0dGFjaG1lbnRzLmxlbmd0aCxcbiAgICAgICAgICApXG4gICAgICAgIH1cbiAgICAgIH0sXG4gICAgICAnYXR0YWNobWVudHM6cmVtb3ZlJzogKCkgPT4ge1xuICAgICAgICBjb25zdCBpbWcgPSBpbWFnZUF0dGFjaG1lbnRzW3NlbGVjdGVkSW1hZ2VJbmRleF1cbiAgICAgICAgaWYgKGltZyAmJiBvblJlbW92ZUltYWdlKSB7XG4gICAgICAgICAgb25SZW1vdmVJbWFnZShpbWcuaWQpXG4gICAgICAgICAgLy8gSWYgbm8gaW1hZ2VzIGxlZnQgYWZ0ZXIgcmVtb3ZhbCwgZXhpdCBpbWFnZSBzZWxlY3Rpb25cbiAgICAgICAgICBpZiAoaW1hZ2VBdHRhY2htZW50cy5sZW5ndGggPD0gMSkge1xuICAgICAgICAgICAgb25JbWFnZXNTZWxlY3RlZENoYW5nZT8uKGZhbHNlKVxuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAvLyBBZGp1c3QgaW5kZXggaWYgd2UgZGVsZXRlZCB0aGUgbGFzdCBpbWFnZVxuICAgICAgICAgICAgb25TZWxlY3RlZEltYWdlSW5kZXhDaGFuZ2U/LihcbiAgICAgICAgICAgICAgTWF0aC5taW4oc2VsZWN0ZWRJbWFnZUluZGV4LCBpbWFnZUF0dGFjaG1lbnRzLmxlbmd0aCAtIDIpLFxuICAgICAgICAgICAgKVxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfSxcbiAgICAgICdhdHRhY2htZW50czpleGl0JzogKCkgPT4ge1xuICAgICAgICBvbkltYWdlc1NlbGVjdGVkQ2hhbmdlPy4oZmFsc2UpXG4gICAgICB9LFxuICAgIH0sXG4gICAgeyBjb250ZXh0OiAnQXR0YWNobWVudHMnLCBpc0FjdGl2ZTogaXNGb2N1c2VkICYmICEhaW1hZ2VzU2VsZWN0ZWQgfSxcbiAgKVxuXG4gIC8vIFVQIGFycm93IGV4aXRzIGltYWdlIHNlbGVjdGlvbiBtb2RlIChVUCBpc24ndCBib3VuZCB0byBhdHRhY2htZW50czpleGl0KVxuICB1c2VJbnB1dChcbiAgICAoX2lucHV0LCBrZXkpID0+IHtcbiAgICAgIGlmIChrZXkudXBBcnJvdykge1xuICAgICAgICBvbkltYWdlc1NlbGVjdGVkQ2hhbmdlPy4oZmFsc2UpXG4gICAgICB9XG4gICAgfSxcbiAgICB7IGlzQWN0aXZlOiBpc0ZvY3VzZWQgJiYgISFpbWFnZXNTZWxlY3RlZCB9LFxuICApXG5cbiAgLy8gRXhpdCBpbWFnZSBtb2RlIHdoZW4gb3B0aW9uIGxvc2VzIGZvY3VzXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKCFpc0ZvY3VzZWQgJiYgaW1hZ2VzU2VsZWN0ZWQpIHtcbiAgICAgIG9uSW1hZ2VzU2VsZWN0ZWRDaGFuZ2U/LihmYWxzZSlcbiAgICB9XG4gIH0sIFtpc0ZvY3VzZWQsIGltYWdlc1NlbGVjdGVkLCBvbkltYWdlc1NlbGVjdGVkQ2hhbmdlXSlcblxuICBjb25zdCBkZXNjcmlwdGlvblBhZGRpbmdMZWZ0ID1cbiAgICBsYXlvdXQgPT09ICdleHBhbmRlZCcgPyBtYXhJbmRleFdpZHRoICsgMyA6IG1heEluZGV4V2lkdGggKyA0XG5cbiAgcmV0dXJuIChcbiAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIiBmbGV4U2hyaW5rPXswfT5cbiAgICAgIDxTZWxlY3RPcHRpb25cbiAgICAgICAgaXNGb2N1c2VkPXtpc0ZvY3VzZWR9XG4gICAgICAgIGlzU2VsZWN0ZWQ9e2lzU2VsZWN0ZWR9XG4gICAgICAgIHNob3VsZFNob3dEb3duQXJyb3c9e3Nob3VsZFNob3dEb3duQXJyb3d9XG4gICAgICAgIHNob3VsZFNob3dVcEFycm93PXtzaG91bGRTaG93VXBBcnJvd31cbiAgICAgICAgZGVjbGFyZUN1cnNvcj17ZmFsc2V9XG4gICAgICA+XG4gICAgICAgIDxCb3hcbiAgICAgICAgICBmbGV4RGlyZWN0aW9uPVwicm93XCJcbiAgICAgICAgICBmbGV4U2hyaW5rPXtsYXlvdXQgPT09ICdjb21wYWN0JyA/IDAgOiB1bmRlZmluZWR9XG4gICAgICAgID5cbiAgICAgICAgICA8VGV4dCBkaW1Db2xvcj57YCR7aW5kZXh9LmAucGFkRW5kKG1heEluZGV4V2lkdGggKyAyKX08L1RleHQ+XG4gICAgICAgICAge2NoaWxkcmVufVxuICAgICAgICAgIHtzaG93TGFiZWwgPyAoXG4gICAgICAgICAgICA8PlxuICAgICAgICAgICAgICA8VGV4dCBjb2xvcj17aXNGb2N1c2VkID8gJ3N1Z2dlc3Rpb24nIDogdW5kZWZpbmVkfT5cbiAgICAgICAgICAgICAgICB7b3B0aW9uLmxhYmVsfVxuICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICAgIHtpc0ZvY3VzZWQgPyAoXG4gICAgICAgICAgICAgICAgPD5cbiAgICAgICAgICAgICAgICAgIDxUZXh0IGNvbG9yPVwic3VnZ2VzdGlvblwiPlxuICAgICAgICAgICAgICAgICAgICB7b3B0aW9uLmxhYmVsVmFsdWVTZXBhcmF0b3IgPz8gJywgJ31cbiAgICAgICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgICAgIDxUZXh0SW5wdXRcbiAgICAgICAgICAgICAgICAgICAgdmFsdWU9e2lucHV0VmFsdWV9XG4gICAgICAgICAgICAgICAgICAgIG9uQ2hhbmdlPXt2YWx1ZSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgaXNVc2VyRWRpdGluZy5jdXJyZW50ID0gdHJ1ZVxuICAgICAgICAgICAgICAgICAgICAgIG9uSW5wdXRDaGFuZ2UodmFsdWUpXG4gICAgICAgICAgICAgICAgICAgICAgb3B0aW9uLm9uQ2hhbmdlKHZhbHVlKVxuICAgICAgICAgICAgICAgICAgICB9fVxuICAgICAgICAgICAgICAgICAgICBvblN1Ym1pdD17b25TdWJtaXR9XG4gICAgICAgICAgICAgICAgICAgIG9uRXhpdD17b25FeGl0fVxuICAgICAgICAgICAgICAgICAgICBwbGFjZWhvbGRlcj17b3B0aW9uLnBsYWNlaG9sZGVyfVxuICAgICAgICAgICAgICAgICAgICBmb2N1cz17IWltYWdlc1NlbGVjdGVkfVxuICAgICAgICAgICAgICAgICAgICBzaG93Q3Vyc29yPXt0cnVlfVxuICAgICAgICAgICAgICAgICAgICBtdWx0aWxpbmU9e3RydWV9XG4gICAgICAgICAgICAgICAgICAgIGN1cnNvck9mZnNldD17Y3Vyc29yT2Zmc2V0fVxuICAgICAgICAgICAgICAgICAgICBvbkNoYW5nZUN1cnNvck9mZnNldD17c2V0Q3Vyc29yT2Zmc2V0fVxuICAgICAgICAgICAgICAgICAgICBjb2x1bW5zPXs4MH1cbiAgICAgICAgICAgICAgICAgICAgb25JbWFnZVBhc3RlPXtvbkltYWdlUGFzdGV9XG4gICAgICAgICAgICAgICAgICAgIG9uUGFzdGU9eyhwYXN0ZWRUZXh0OiBzdHJpbmcpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICBpc1VzZXJFZGl0aW5nLmN1cnJlbnQgPSB0cnVlXG4gICAgICAgICAgICAgICAgICAgICAgY29uc3QgYmVmb3JlID0gaW5wdXRWYWx1ZS5zbGljZSgwLCBjdXJzb3JPZmZzZXQpXG4gICAgICAgICAgICAgICAgICAgICAgY29uc3QgYWZ0ZXIgPSBpbnB1dFZhbHVlLnNsaWNlKGN1cnNvck9mZnNldClcbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCBuZXdWYWx1ZSA9IGJlZm9yZSArIHBhc3RlZFRleHQgKyBhZnRlclxuICAgICAgICAgICAgICAgICAgICAgIG9uSW5wdXRDaGFuZ2UobmV3VmFsdWUpXG4gICAgICAgICAgICAgICAgICAgICAgb3B0aW9uLm9uQ2hhbmdlKG5ld1ZhbHVlKVxuICAgICAgICAgICAgICAgICAgICAgIHNldEN1cnNvck9mZnNldChiZWZvcmUubGVuZ3RoICsgcGFzdGVkVGV4dC5sZW5ndGgpXG4gICAgICAgICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgICAgIDwvPlxuICAgICAgICAgICAgICApIDogKFxuICAgICAgICAgICAgICAgIGlucHV0VmFsdWUgJiYgKFxuICAgICAgICAgICAgICAgICAgPFRleHQ+XG4gICAgICAgICAgICAgICAgICAgIHtvcHRpb24ubGFiZWxWYWx1ZVNlcGFyYXRvciA/PyAnLCAnfVxuICAgICAgICAgICAgICAgICAgICB7aW5wdXRWYWx1ZX1cbiAgICAgICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgICl9XG4gICAgICAgICAgICA8Lz5cbiAgICAgICAgICApIDogaXNGb2N1c2VkID8gKFxuICAgICAgICAgICAgPFRleHRJbnB1dFxuICAgICAgICAgICAgICB2YWx1ZT17aW5wdXRWYWx1ZX1cbiAgICAgICAgICAgICAgb25DaGFuZ2U9e3ZhbHVlID0+IHtcbiAgICAgICAgICAgICAgICBpc1VzZXJFZGl0aW5nLmN1cnJlbnQgPSB0cnVlXG4gICAgICAgICAgICAgICAgb25JbnB1dENoYW5nZSh2YWx1ZSlcbiAgICAgICAgICAgICAgICBvcHRpb24ub25DaGFuZ2UodmFsdWUpXG4gICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgIG9uU3VibWl0PXtvblN1Ym1pdH1cbiAgICAgICAgICAgICAgb25FeGl0PXtvbkV4aXR9XG4gICAgICAgICAgICAgIHBsYWNlaG9sZGVyPXtcbiAgICAgICAgICAgICAgICBvcHRpb24ucGxhY2Vob2xkZXIgfHxcbiAgICAgICAgICAgICAgICAodHlwZW9mIG9wdGlvbi5sYWJlbCA9PT0gJ3N0cmluZycgPyBvcHRpb24ubGFiZWwgOiB1bmRlZmluZWQpXG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgZm9jdXM9eyFpbWFnZXNTZWxlY3RlZH1cbiAgICAgICAgICAgICAgc2hvd0N1cnNvcj17dHJ1ZX1cbiAgICAgICAgICAgICAgbXVsdGlsaW5lPXt0cnVlfVxuICAgICAgICAgICAgICBjdXJzb3JPZmZzZXQ9e2N1cnNvck9mZnNldH1cbiAgICAgICAgICAgICAgb25DaGFuZ2VDdXJzb3JPZmZzZXQ9e3NldEN1cnNvck9mZnNldH1cbiAgICAgICAgICAgICAgY29sdW1ucz17ODB9XG4gICAgICAgICAgICAgIG9uSW1hZ2VQYXN0ZT17b25JbWFnZVBhc3RlfVxuICAgICAgICAgICAgICBvblBhc3RlPXsocGFzdGVkVGV4dDogc3RyaW5nKSA9PiB7XG4gICAgICAgICAgICAgICAgaXNVc2VyRWRpdGluZy5jdXJyZW50ID0gdHJ1ZVxuICAgICAgICAgICAgICAgIGNvbnN0IGJlZm9yZSA9IGlucHV0VmFsdWUuc2xpY2UoMCwgY3Vyc29yT2Zmc2V0KVxuICAgICAgICAgICAgICAgIGNvbnN0IGFmdGVyID0gaW5wdXRWYWx1ZS5zbGljZShjdXJzb3JPZmZzZXQpXG4gICAgICAgICAgICAgICAgY29uc3QgbmV3VmFsdWUgPSBiZWZvcmUgKyBwYXN0ZWRUZXh0ICsgYWZ0ZXJcbiAgICAgICAgICAgICAgICBvbklucHV0Q2hhbmdlKG5ld1ZhbHVlKVxuICAgICAgICAgICAgICAgIG9wdGlvbi5vbkNoYW5nZShuZXdWYWx1ZSlcbiAgICAgICAgICAgICAgICBzZXRDdXJzb3JPZmZzZXQoYmVmb3JlLmxlbmd0aCArIHBhc3RlZFRleHQubGVuZ3RoKVxuICAgICAgICAgICAgICB9fVxuICAgICAgICAgICAgLz5cbiAgICAgICAgICApIDogKFxuICAgICAgICAgICAgPFRleHQgY29sb3I9e2lucHV0VmFsdWUgPyB1bmRlZmluZWQgOiAnaW5hY3RpdmUnfT5cbiAgICAgICAgICAgICAge2lucHV0VmFsdWUgfHwgb3B0aW9uLnBsYWNlaG9sZGVyIHx8IG9wdGlvbi5sYWJlbH1cbiAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICApfVxuICAgICAgICA8L0JveD5cbiAgICAgIDwvU2VsZWN0T3B0aW9uPlxuICAgICAge29wdGlvbi5kZXNjcmlwdGlvbiAmJiAoXG4gICAgICAgIDxCb3ggcGFkZGluZ0xlZnQ9e2Rlc2NyaXB0aW9uUGFkZGluZ0xlZnR9PlxuICAgICAgICAgIDxUZXh0XG4gICAgICAgICAgICBkaW1Db2xvcj17b3B0aW9uLmRpbURlc2NyaXB0aW9uICE9PSBmYWxzZX1cbiAgICAgICAgICAgIGNvbG9yPXtcbiAgICAgICAgICAgICAgaXNTZWxlY3RlZCA/ICdzdWNjZXNzJyA6IGlzRm9jdXNlZCA/ICdzdWdnZXN0aW9uJyA6IHVuZGVmaW5lZFxuICAgICAgICAgICAgfVxuICAgICAgICAgID5cbiAgICAgICAgICAgIHtvcHRpb24uZGVzY3JpcHRpb259XG4gICAgICAgICAgPC9UZXh0PlxuICAgICAgICA8L0JveD5cbiAgICAgICl9XG4gICAgICB7aW1hZ2VBdHRhY2htZW50cy5sZW5ndGggPiAwICYmIChcbiAgICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwicm93XCIgZ2FwPXsxfSBwYWRkaW5nTGVmdD17ZGVzY3JpcHRpb25QYWRkaW5nTGVmdH0+XG4gICAgICAgICAge2ltYWdlQXR0YWNobWVudHMubWFwKChpbWcsIGlkeCkgPT4gKFxuICAgICAgICAgICAgPENsaWNrYWJsZUltYWdlUmVmXG4gICAgICAgICAgICAgIGtleT17aW1nLmlkfVxuICAgICAgICAgICAgICBpbWFnZUlkPXtpbWcuaWR9XG4gICAgICAgICAgICAgIGlzU2VsZWN0ZWQ9eyEhaW1hZ2VzU2VsZWN0ZWQgJiYgaWR4ID09PSBzZWxlY3RlZEltYWdlSW5kZXh9XG4gICAgICAgICAgICAvPlxuICAgICAgICAgICkpfVxuICAgICAgICAgIDxCb3ggZmxleEdyb3c9ezF9IGp1c3RpZnlDb250ZW50PVwiZmxleC1zdGFydFwiIGZsZXhEaXJlY3Rpb249XCJyb3dcIj5cbiAgICAgICAgICAgIDxUZXh0IGRpbUNvbG9yPlxuICAgICAgICAgICAgICB7aW1hZ2VzU2VsZWN0ZWQgPyAoXG4gICAgICAgICAgICAgICAgPEJ5bGluZT5cbiAgICAgICAgICAgICAgICAgIHtpbWFnZUF0dGFjaG1lbnRzLmxlbmd0aCA+IDEgJiYgKFxuICAgICAgICAgICAgICAgICAgICA8PlxuICAgICAgICAgICAgICAgICAgICAgIDxDb25maWd1cmFibGVTaG9ydGN1dEhpbnRcbiAgICAgICAgICAgICAgICAgICAgICAgIGFjdGlvbj1cImF0dGFjaG1lbnRzOm5leHRcIlxuICAgICAgICAgICAgICAgICAgICAgICAgY29udGV4dD1cIkF0dGFjaG1lbnRzXCJcbiAgICAgICAgICAgICAgICAgICAgICAgIGZhbGxiYWNrPVwi4oaSXCJcbiAgICAgICAgICAgICAgICAgICAgICAgIGRlc2NyaXB0aW9uPVwibmV4dFwiXG4gICAgICAgICAgICAgICAgICAgICAgLz5cbiAgICAgICAgICAgICAgICAgICAgICA8Q29uZmlndXJhYmxlU2hvcnRjdXRIaW50XG4gICAgICAgICAgICAgICAgICAgICAgICBhY3Rpb249XCJhdHRhY2htZW50czpwcmV2aW91c1wiXG4gICAgICAgICAgICAgICAgICAgICAgICBjb250ZXh0PVwiQXR0YWNobWVudHNcIlxuICAgICAgICAgICAgICAgICAgICAgICAgZmFsbGJhY2s9XCLihpBcIlxuICAgICAgICAgICAgICAgICAgICAgICAgZGVzY3JpcHRpb249XCJwcmV2XCJcbiAgICAgICAgICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgICAgICAgICA8Lz5cbiAgICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgICAgICA8Q29uZmlndXJhYmxlU2hvcnRjdXRIaW50XG4gICAgICAgICAgICAgICAgICAgIGFjdGlvbj1cImF0dGFjaG1lbnRzOnJlbW92ZVwiXG4gICAgICAgICAgICAgICAgICAgIGNvbnRleHQ9XCJBdHRhY2htZW50c1wiXG4gICAgICAgICAgICAgICAgICAgIGZhbGxiYWNrPVwiYmFja3NwYWNlXCJcbiAgICAgICAgICAgICAgICAgICAgZGVzY3JpcHRpb249XCJyZW1vdmVcIlxuICAgICAgICAgICAgICAgICAgLz5cbiAgICAgICAgICAgICAgICAgIDxDb25maWd1cmFibGVTaG9ydGN1dEhpbnRcbiAgICAgICAgICAgICAgICAgICAgYWN0aW9uPVwiYXR0YWNobWVudHM6ZXhpdFwiXG4gICAgICAgICAgICAgICAgICAgIGNvbnRleHQ9XCJBdHRhY2htZW50c1wiXG4gICAgICAgICAgICAgICAgICAgIGZhbGxiYWNrPVwiZXNjXCJcbiAgICAgICAgICAgICAgICAgICAgZGVzY3JpcHRpb249XCJjYW5jZWxcIlxuICAgICAgICAgICAgICAgICAgLz5cbiAgICAgICAgICAgICAgICA8L0J5bGluZT5cbiAgICAgICAgICAgICAgKSA6IGlzRm9jdXNlZCA/IChcbiAgICAgICAgICAgICAgICAnKOKGkyB0byBzZWxlY3QpJ1xuICAgICAgICAgICAgICApIDogbnVsbH1cbiAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICA8L0JveD5cbiAgICAgICAgPC9Cb3g+XG4gICAgICApfVxuICAgICAge2xheW91dCA9PT0gJ2V4cGFuZGVkJyAmJiA8VGV4dD4gPC9UZXh0Pn1cbiAgICA8L0JveD5cbiAgKVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsT0FBT0EsS0FBSyxJQUFJLEtBQUtDLFNBQVMsRUFBRUMsU0FBUyxFQUFFQyxNQUFNLEVBQUVDLFFBQVEsUUFBUSxPQUFPO0FBQzFFO0FBQ0EsU0FBU0MsR0FBRyxFQUFFQyxJQUFJLEVBQUVDLFFBQVEsUUFBUSxjQUFjO0FBQ2xELFNBQ0VDLGFBQWEsRUFDYkMsY0FBYyxRQUNULG9DQUFvQztBQUMzQyxjQUFjQyxhQUFhLFFBQVEsdUJBQXVCO0FBQzFELFNBQVNDLHFCQUFxQixRQUFRLDJCQUEyQjtBQUNqRSxjQUFjQyxlQUFlLFFBQVEsNkJBQTZCO0FBQ2xFLFNBQVNDLGlCQUFpQixRQUFRLHlCQUF5QjtBQUMzRCxTQUFTQyx3QkFBd0IsUUFBUSxnQ0FBZ0M7QUFDekUsU0FBU0MsTUFBTSxRQUFRLDRCQUE0QjtBQUNuRCxPQUFPQyxTQUFTLE1BQU0saUJBQWlCO0FBQ3ZDLGNBQWNDLHFCQUFxQixRQUFRLGFBQWE7QUFDeEQsU0FBU0MsWUFBWSxRQUFRLG9CQUFvQjtBQUVqRCxLQUFLQyxLQUFLLENBQUMsQ0FBQyxDQUFDLEdBQUc7RUFDZEMsTUFBTSxFQUFFQyxPQUFPLENBQUNKLHFCQUFxQixDQUFDSyxDQUFDLENBQUMsRUFBRTtJQUFFQyxJQUFJLEVBQUUsT0FBTztFQUFDLENBQUMsQ0FBQztFQUM1REMsU0FBUyxFQUFFLE9BQU87RUFDbEJDLFVBQVUsRUFBRSxPQUFPO0VBQ25CQyxtQkFBbUIsRUFBRSxPQUFPO0VBQzVCQyxp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 \ No newline at end of file diff --git a/src/components/CustomSelect/select-option.tsx b/src/components/CustomSelect/select-option.tsx new file mode 100644 index 0000000..e3a98d6 --- /dev/null +++ b/src/components/CustomSelect/select-option.tsx @@ -0,0 +1,68 @@ +import { c as _c } from "react/compiler-runtime"; +import React, { type ReactNode } from 'react'; +import { ListItem } from '../design-system/ListItem.js'; +export type SelectOptionProps = { + /** + * Determines if option is focused. + */ + readonly isFocused: boolean; + + /** + * Determines if option is selected. + */ + readonly isSelected: boolean; + + /** + * Option label. + */ + readonly children: ReactNode; + + /** + * Optional description to display below the label. + */ + readonly description?: string; + + /** + * Determines if the down arrow should be shown. + */ + readonly shouldShowDownArrow?: boolean; + + /** + * Determines if the up arrow should be shown. + */ + readonly shouldShowUpArrow?: boolean; + + /** + * Whether ListItem should declare the terminal cursor position. + * Set false when a child declares its own cursor (e.g. BaseTextInput). + */ + readonly declareCursor?: boolean; +}; +export function SelectOption(t0) { + const $ = _c(8); + const { + isFocused, + isSelected, + children, + description, + shouldShowDownArrow, + shouldShowUpArrow, + declareCursor + } = t0; + let t1; + if ($[0] !== children || $[1] !== declareCursor || $[2] !== description || $[3] !== isFocused || $[4] !== isSelected || $[5] !== shouldShowDownArrow || $[6] !== shouldShowUpArrow) { + t1 = {children}; + $[0] = children; + $[1] = declareCursor; + $[2] = description; + $[3] = isFocused; + $[4] = isSelected; + $[5] = shouldShowDownArrow; + $[6] = shouldShowUpArrow; + $[7] = t1; + } else { + t1 = $[7]; + } + return t1; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIlJlYWN0Tm9kZSIsIkxpc3RJdGVtIiwiU2VsZWN0T3B0aW9uUHJvcHMiLCJpc0ZvY3VzZWQiLCJpc1NlbGVjdGVkIiwiY2hpbGRyZW4iLCJkZXNjcmlwdGlvbiIsInNob3VsZFNob3dEb3duQXJyb3ciLCJzaG91bGRTaG93VXBBcnJvdyIsImRlY2xhcmVDdXJzb3IiLCJTZWxlY3RPcHRpb24iLCJ0MCIsIiQiLCJfYyIsInQxIl0sInNvdXJjZXMiOlsic2VsZWN0LW9wdGlvbi50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFJlYWN0LCB7IHR5cGUgUmVhY3ROb2RlIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBMaXN0SXRlbSB9IGZyb20gJy4uL2Rlc2lnbi1zeXN0ZW0vTGlzdEl0ZW0uanMnXG5cbmV4cG9ydCB0eXBlIFNlbGVjdE9wdGlvblByb3BzID0ge1xuICAvKipcbiAgICogRGV0ZXJtaW5lcyBpZiBvcHRpb24gaXMgZm9jdXNlZC5cbiAgICovXG4gIHJlYWRvbmx5IGlzRm9jdXNlZDogYm9vbGVhblxuXG4gIC8qKlxuICAgKiBEZXRlcm1pbmVzIGlmIG9wdGlvbiBpcyBzZWxlY3RlZC5cbiAgICovXG4gIHJlYWRvbmx5IGlzU2VsZWN0ZWQ6IGJvb2xlYW5cblxuICAvKipcbiAgICogT3B0aW9uIGxhYmVsLlxuICAgKi9cbiAgcmVhZG9ubHkgY2hpbGRyZW46IFJlYWN0Tm9kZVxuXG4gIC8qKlxuICAgKiBPcHRpb25hbCBkZXNjcmlwdGlvbiB0byBkaXNwbGF5IGJlbG93IHRoZSBsYWJlbC5cbiAgICovXG4gIHJlYWRvbmx5IGRlc2NyaXB0aW9uPzogc3RyaW5nXG5cbiAgLyoqXG4gICAqIERldGVybWluZXMgaWYgdGhlIGRvd24gYXJyb3cgc2hvdWxkIGJlIHNob3duLlxuICAgKi9cbiAgcmVhZG9ubHkgc2hvdWxkU2hvd0Rvd25BcnJvdz86IGJvb2xlYW5cblxuICAvKipcbiAgICogRGV0ZXJtaW5lcyBpZiB0aGUgdXAgYXJyb3cgc2hvdWxkIGJlIHNob3duLlxuICAgKi9cbiAgcmVhZG9ubHkgc2hvdWxkU2hvd1VwQXJyb3c/OiBib29sZWFuXG5cbiAgLyoqXG4gICAqIFdoZXRoZXIgTGlzdEl0ZW0gc2hvdWxkIGRlY2xhcmUgdGhlIHRlcm1pbmFsIGN1cnNvciBwb3NpdGlvbi5cbiAgICogU2V0IGZhbHNlIHdoZW4gYSBjaGlsZCBkZWNsYXJlcyBpdHMgb3duIGN1cnNvciAoZS5nLiBCYXNlVGV4dElucHV0KS5cbiAgICovXG4gIHJlYWRvbmx5IGRlY2xhcmVDdXJzb3I/OiBib29sZWFuXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBTZWxlY3RPcHRpb24oe1xuICBpc0ZvY3VzZWQsXG4gIGlzU2VsZWN0ZWQsXG4gIGNoaWxkcmVuLFxuICBkZXNjcmlwdGlvbixcbiAgc2hvdWxkU2hvd0Rvd25BcnJvdyxcbiAgc2hvdWxkU2hvd1VwQXJyb3csXG4gIGRlY2xhcmVDdXJzb3IsXG59OiBTZWxlY3RPcHRpb25Qcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIHJldHVybiAoXG4gICAgPExpc3RJdGVtXG4gICAgICBpc0ZvY3VzZWQ9e2lzRm9jdXNlZH1cbiAgICAgIGlzU2VsZWN0ZWQ9e2lzU2VsZWN0ZWR9XG4gICAgICBkZXNjcmlwdGlvbj17ZGVzY3JpcHRpb259XG4gICAgICBzaG93U2Nyb2xsRG93bj17c2hvdWxkU2hvd0Rvd25BcnJvd31cbiAgICAgIHNob3dTY3JvbGxVcD17c2hvdWxkU2hvd1VwQXJyb3d9XG4gICAgICBzdHlsZWQ9e2ZhbHNlfVxuICAgICAgZGVjbGFyZUN1cnNvcj17ZGVjbGFyZUN1cnNvcn1cbiAgICA+XG4gICAgICB7Y2hpbGRyZW59XG4gICAgPC9MaXN0SXRlbT5cbiAgKVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsT0FBT0EsS0FBSyxJQUFJLEtBQUtDLFNBQVMsUUFBUSxPQUFPO0FBQzdDLFNBQVNDLFFBQVEsUUFBUSw4QkFBOEI7QUFFdkQsT0FBTyxLQUFLQyxpQkFBaUIsR0FBRztFQUM5QjtBQUNGO0FBQ0E7RUFDRSxTQUFTQyxTQUFTLEVBQUUsT0FBTzs7RUFFM0I7QUFDRjtBQUNBO0VBQ0UsU0FBU0MsVUFBVSxFQUFFLE9BQU87O0VBRTVCO0FBQ0Y7QUFDQTtFQUNFLFNBQVNDLFFBQVEsRUFBRUwsU0FBUzs7RUFFNUI7QUFDRjtBQUNBO0VBQ0UsU0FBU00sV0FBVyxDQUFDLEVBQUUsTUFBTTs7RUFFN0I7QUFDRjtBQUNBO0VBQ0UsU0FBU0MsbUJBQW1CLENBQUMsRUFBRSxPQUFPOztFQUV0QztBQUNGO0FBQ0E7RUFDRSxTQUFTQyxpQkFBaUIsQ0FBQyxFQUFFLE9BQU87O0VBRXBDO0FBQ0Y7QUFDQTtBQUNBO0VBQ0UsU0FBU0MsYUFBYSxDQUFDLEVBQUUsT0FBTztBQUNsQyxDQUFDO0FBRUQsT0FBTyxTQUFBQyxhQUFBQyxFQUFBO0VBQUEsTUFBQUMsQ0FBQSxHQUFBQyxFQUFBO0VBQXNCO0lBQUFWLFNBQUE7SUFBQUMsVUFBQTtJQUFBQyxRQUFBO0lBQUFDLFdBQUE7SUFBQUMsbUJBQUE7SUFBQUMsaUJBQUE7SUFBQUM7RUFBQSxJQUFBRSxFQVFUO0VBQUEsSUFBQUcsRUFBQTtFQUFBLElBQUFGLENBQUEsUUFBQVAsUUFBQSxJQUFBTyxDQUFBLFFBQUFILGFBQUEsSUFBQUcsQ0FBQSxRQUFBTixXQUFBLElBQUFNLENBQUEsUUFBQVQsU0FBQSxJQUFBUyxDQUFBLFFBQUFSLFVBQUEsSUFBQVEsQ0FBQSxRQUFBTCxtQkFBQSxJQUFBSyxDQUFBLFFBQUFKLGlCQUFBO0lBRWhCTSxFQUFBLElBQUMsUUFBUSxDQUNJWCxTQUFTLENBQVRBLFVBQVEsQ0FBQyxDQUNSQyxVQUFVLENBQVZBLFdBQVMsQ0FBQyxDQUNURSxXQUFXLENBQVhBLFlBQVUsQ0FBQyxDQUNSQyxjQUFtQixDQUFuQkEsb0JBQWtCLENBQUMsQ0FDckJDLFlBQWlCLENBQWpCQSxrQkFBZ0IsQ0FBQyxDQUN2QixNQUFLLENBQUwsTUFBSSxDQUFDLENBQ0VDLGFBQWEsQ0FBYkEsY0FBWSxDQUFDLENBRTNCSixTQUFPLENBQ1YsRUFWQyxRQUFRLENBVUU7SUFBQU8sQ0FBQSxNQUFBUCxRQUFBO0lBQUFPLENBQUEsTUFBQUgsYUFBQTtJQUFBRyxDQUFBLE1BQUFOLFdBQUE7SUFBQU0sQ0FBQSxNQUFBVCxTQUFBO0lBQUFTLENBQUEsTUFBQVIsVUFBQTtJQUFBUSxDQUFBLE1BQUFMLG1CQUFBO0lBQUFLLENBQUEsTUFBQUosaUJBQUE7SUFBQUksQ0FBQSxNQUFBRSxFQUFBO0VBQUE7SUFBQUEsRUFBQSxHQUFBRixDQUFBO0VBQUE7RUFBQSxPQVZYRSxFQVVXO0FBQUEiLCJpZ25vcmVMaXN0IjpbXX0= \ No newline at end of file diff --git a/src/components/CustomSelect/select.tsx b/src/components/CustomSelect/select.tsx new file mode 100644 index 0000000..134de48 --- /dev/null +++ b/src/components/CustomSelect/select.tsx @@ -0,0 +1,690 @@ +import { c as _c } from "react/compiler-runtime"; +import figures from 'figures'; +import React, { type ReactNode, useEffect, useRef, useState } from 'react'; +import { useDeclaredCursor } from '../../ink/hooks/use-declared-cursor.js'; +import { stringWidth } from '../../ink/stringWidth.js'; +import { Ansi, Box, Text } from '../../ink.js'; +import { count } from '../../utils/array.js'; +import type { PastedContent } from '../../utils/config.js'; +import type { ImageDimensions } from '../../utils/imageResizer.js'; +import { SelectInputOption } from './select-input-option.js'; +import { SelectOption } from './select-option.js'; +import { useSelectInput } from './use-select-input.js'; +import { useSelectState } from './use-select-state.js'; + +// Extract text content from ReactNode for width calculation +function getTextContent(node: ReactNode): string { + if (typeof node === 'string') return node; + if (typeof node === 'number') return String(node); + if (!node) return ''; + if (Array.isArray(node)) return node.map(getTextContent).join(''); + if (React.isValidElement<{ + children?: ReactNode; + }>(node)) { + return getTextContent(node.props.children); + } + return ''; +} +type BaseOption = { + description?: string; + dimDescription?: boolean; + label: ReactNode; + value: T; + disabled?: boolean; +}; +export type OptionWithDescription = (BaseOption & { + type?: 'text'; +}) | (BaseOption & { + type: 'input'; + onChange: (value: string) => void; + placeholder?: string; + initialValue?: string; + /** + * Controls behavior when submitting with empty input: + * - true: calls onChange (treats empty as valid submission) + * - false (default): calls onCancel (treats empty as cancellation) + * + * Also affects initial Enter press: when true, submits immediately; + * when false, enters input mode first so user can type. + */ + allowEmptySubmitToCancel?: boolean; + /** + * When true, always shows the label alongside the input value, regardless of + * the global inlineDescriptions/showLabel setting. Use this when the label + * provides important context that should always be visible (e.g., "Yes, and allow..."). + */ + showLabelWithValue?: boolean; + /** + * Custom separator between label and value when showLabel is true. + * Defaults to ", ". Use ": " for labels that read better with a colon. + */ + labelValueSeparator?: string; + /** + * When true, automatically reset cursor to end of line when: + * - Option becomes focused + * - Input value changes + * This prevents cursor position bugs when the input value updates asynchronously. + */ + resetCursorOnUpdate?: boolean; +}); +export type SelectProps = { + /** + * When disabled, user input is ignored. + * + * @default false + */ + readonly isDisabled?: boolean; + + /** + * When true, prevents selection on Enter but allows scrolling. + * + * @default false + */ + readonly disableSelection?: boolean; + + /** + * When true, hides the numeric indexes next to each option. + * + * @default false + */ + readonly hideIndexes?: boolean; + + /** + * Number of visible options. + * + * @default 5 + */ + readonly visibleOptionCount?: number; + + /** + * Highlight text in option labels. + */ + readonly highlightText?: string; + + /** + * Options. + */ + readonly options: OptionWithDescription[]; + + /** + * Default value. + */ + readonly defaultValue?: T; + + /** + * Callback when cancel is pressed. + */ + readonly onCancel?: () => void; + + /** + * Callback when selected option changes. + */ + readonly onChange?: (value: T) => void; + + /** + * Callback when focused option changes. + * Note: This is for one-way notification only. Avoid combining with focusValue + * for bidirectional sync, as this can cause feedback loops. + */ + readonly onFocus?: (value: T) => void; + + /** + * Initial value to focus. This is used to set focus when the component mounts. + */ + readonly defaultFocusValue?: T; + + /** + * Layout of the options. + * - `compact` (default) tries to use one line per option + * - `expanded` uses multiple lines and an empty line between options + * - `compact-vertical` uses compact index formatting with descriptions below labels + */ + readonly layout?: 'compact' | 'expanded' | 'compact-vertical'; + + /** + * When true, descriptions are rendered inline after the label instead of + * in a separate column. Use this for short descriptions like hints. + * + * @default false + */ + readonly inlineDescriptions?: boolean; + + /** + * Callback when user presses up from the first item. + * If provided, navigation will not wrap to the last item. + */ + readonly onUpFromFirstItem?: () => void; + + /** + * Callback when user presses down from the last item. + * If provided, navigation will not wrap to the first item. + */ + readonly onDownFromLastItem?: () => void; + + /** + * Callback when input mode should be toggled for an option. + * Called when Tab is pressed (to enter or exit input mode). + */ + readonly onInputModeToggle?: (value: T) => void; + + /** + * Callback to open external editor for editing input option values. + * When provided, ctrl+g will trigger this callback in input options + * with the current value and a setter function to update the internal state. + */ + readonly onOpenEditor?: (currentValue: string, setValue: (value: string) => void) => void; + + /** + * Optional callback when an image is pasted into an input option. + */ + readonly onImagePaste?: (base64Image: string, mediaType?: string, filename?: string, dimensions?: ImageDimensions, sourcePath?: string) => void; + + /** + * Pasted content to display inline in input options. + */ + readonly pastedContents?: Record; + + /** + * Callback to remove a pasted image by its ID. + */ + readonly onRemoveImage?: (id: number) => void; +}; +export function Select(t0) { + const $ = _c(72); + const { + isDisabled: t1, + hideIndexes: t2, + visibleOptionCount: t3, + highlightText, + options, + defaultValue, + onCancel, + onChange, + onFocus, + defaultFocusValue, + layout: t4, + disableSelection: t5, + inlineDescriptions: t6, + onUpFromFirstItem, + onDownFromLastItem, + onInputModeToggle, + onOpenEditor, + onImagePaste, + pastedContents, + onRemoveImage + } = t0; + const isDisabled = t1 === undefined ? false : t1; + const hideIndexes = t2 === undefined ? false : t2; + const visibleOptionCount = t3 === undefined ? 5 : t3; + const layout = t4 === undefined ? "compact" : t4; + const disableSelection = t5 === undefined ? false : t5; + const inlineDescriptions = t6 === undefined ? false : t6; + const [imagesSelected, setImagesSelected] = useState(false); + const [selectedImageIndex, setSelectedImageIndex] = useState(0); + let t7; + if ($[0] !== options) { + t7 = () => { + const initialMap = new Map(); + options.forEach(option => { + if (option.type === "input" && option.initialValue) { + initialMap.set(option.value, option.initialValue); + } + }); + return initialMap; + }; + $[0] = options; + $[1] = t7; + } else { + t7 = $[1]; + } + const [inputValues, setInputValues] = useState(t7); + let t8; + if ($[2] === Symbol.for("react.memo_cache_sentinel")) { + t8 = new Map(); + $[2] = t8; + } else { + t8 = $[2]; + } + const lastInitialValues = useRef(t8); + let t10; + let t9; + if ($[3] !== inputValues || $[4] !== options) { + t9 = () => { + for (const option_0 of options) { + if (option_0.type === "input" && option_0.initialValue !== undefined) { + const lastInitial = lastInitialValues.current.get(option_0.value) ?? ""; + const currentValue = inputValues.get(option_0.value) ?? ""; + const newInitial = option_0.initialValue; + if (newInitial !== lastInitial && currentValue === lastInitial) { + setInputValues(prev => { + const next = new Map(prev); + next.set(option_0.value, newInitial); + return next; + }); + } + lastInitialValues.current.set(option_0.value, newInitial); + } + } + }; + t10 = [options, inputValues]; + $[3] = inputValues; + $[4] = options; + $[5] = t10; + $[6] = t9; + } else { + t10 = $[5]; + t9 = $[6]; + } + useEffect(t9, t10); + let t11; + if ($[7] !== defaultFocusValue || $[8] !== defaultValue || $[9] !== onCancel || $[10] !== onChange || $[11] !== onFocus || $[12] !== options || $[13] !== visibleOptionCount) { + t11 = { + visibleOptionCount, + options, + defaultValue, + onChange, + onCancel, + onFocus, + focusValue: defaultFocusValue + }; + $[7] = defaultFocusValue; + $[8] = defaultValue; + $[9] = onCancel; + $[10] = onChange; + $[11] = onFocus; + $[12] = options; + $[13] = visibleOptionCount; + $[14] = t11; + } else { + t11 = $[14]; + } + const state = useSelectState(t11); + const t12 = disableSelection || (hideIndexes ? "numeric" : false); + let t13; + if ($[15] !== pastedContents) { + t13 = () => { + if (pastedContents && Object.values(pastedContents).some(_temp)) { + const imageCount = count(Object.values(pastedContents), _temp2); + setImagesSelected(true); + setSelectedImageIndex(imageCount - 1); + return true; + } + return false; + }; + $[15] = pastedContents; + $[16] = t13; + } else { + t13 = $[16]; + } + let t14; + if ($[17] !== imagesSelected || $[18] !== inputValues || $[19] !== isDisabled || $[20] !== onDownFromLastItem || $[21] !== onInputModeToggle || $[22] !== onUpFromFirstItem || $[23] !== options || $[24] !== state || $[25] !== t12 || $[26] !== t13) { + t14 = { + isDisabled, + disableSelection: t12, + state, + options, + isMultiSelect: false, + onUpFromFirstItem, + onDownFromLastItem, + onInputModeToggle, + inputValues, + imagesSelected, + onEnterImageSelection: t13 + }; + $[17] = imagesSelected; + $[18] = inputValues; + $[19] = isDisabled; + $[20] = onDownFromLastItem; + $[21] = onInputModeToggle; + $[22] = onUpFromFirstItem; + $[23] = options; + $[24] = state; + $[25] = t12; + $[26] = t13; + $[27] = t14; + } else { + t14 = $[27]; + } + useSelectInput(t14); + let T0; + let t15; + let t16; + let t17; + if ($[28] !== hideIndexes || $[29] !== highlightText || $[30] !== imagesSelected || $[31] !== inlineDescriptions || $[32] !== inputValues || $[33] !== isDisabled || $[34] !== layout || $[35] !== onCancel || $[36] !== onChange || $[37] !== onImagePaste || $[38] !== onOpenEditor || $[39] !== onRemoveImage || $[40] !== options.length || $[41] !== pastedContents || $[42] !== selectedImageIndex || $[43] !== state.focusedValue || $[44] !== state.options || $[45] !== state.value || $[46] !== state.visibleFromIndex || $[47] !== state.visibleOptions || $[48] !== state.visibleToIndex) { + t17 = Symbol.for("react.early_return_sentinel"); + bb0: { + const styles = { + container: _temp3, + highlightedText: _temp4 + }; + if (layout === "expanded") { + let t18; + if ($[53] !== state.options.length) { + t18 = state.options.length.toString(); + $[53] = state.options.length; + $[54] = t18; + } else { + t18 = $[54]; + } + const maxIndexWidth = t18.length; + t17 = {state.visibleOptions.map((option_1, index) => { + const isFirstVisibleOption = option_1.index === state.visibleFromIndex; + const isLastVisibleOption = option_1.index === state.visibleToIndex - 1; + const areMoreOptionsBelow = state.visibleToIndex < options.length; + const areMoreOptionsAbove = state.visibleFromIndex > 0; + const i = state.visibleFromIndex + index + 1; + const isFocused = !isDisabled && state.focusedValue === option_1.value; + const isSelected = state.value === option_1.value; + if (option_1.type === "input") { + const inputValue = inputValues.has(option_1.value) ? inputValues.get(option_1.value) : option_1.initialValue || ""; + return { + setInputValues(prev_0 => { + const next_0 = new Map(prev_0); + next_0.set(option_1.value, value); + return next_0; + }); + }} onSubmit={value_0 => { + const hasImageAttachments = pastedContents && Object.values(pastedContents).some(_temp5); + if (value_0.trim() || hasImageAttachments || option_1.allowEmptySubmitToCancel) { + onChange?.(option_1.value); + } else { + onCancel?.(); + } + }} onExit={onCancel} layout="expanded" showLabel={inlineDescriptions} onOpenEditor={onOpenEditor} resetCursorOnUpdate={option_1.resetCursorOnUpdate} onImagePaste={onImagePaste} pastedContents={pastedContents} onRemoveImage={onRemoveImage} imagesSelected={imagesSelected} selectedImageIndex={selectedImageIndex} onImagesSelectedChange={setImagesSelected} onSelectedImageIndexChange={setSelectedImageIndex} />; + } + let label = option_1.label; + if (typeof option_1.label === "string" && highlightText && option_1.label.includes(highlightText)) { + const labelText = option_1.label; + const index_0 = labelText.indexOf(highlightText); + label = <>{labelText.slice(0, index_0)}{highlightText}{labelText.slice(index_0 + highlightText.length)}; + } + const isOptionDisabled = option_1.disabled === true; + const optionColor = isOptionDisabled ? undefined : isSelected ? "success" : isFocused ? "suggestion" : undefined; + return {label}{option_1.description && {option_1.description}} ; + })}; + break bb0; + } + if (layout === "compact-vertical") { + let t18; + if ($[55] !== hideIndexes || $[56] !== state.options) { + t18 = hideIndexes ? 0 : state.options.length.toString().length; + $[55] = hideIndexes; + $[56] = state.options; + $[57] = t18; + } else { + t18 = $[57]; + } + const maxIndexWidth_0 = t18; + t17 = {state.visibleOptions.map((option_2, index_1) => { + const isFirstVisibleOption_0 = option_2.index === state.visibleFromIndex; + const isLastVisibleOption_0 = option_2.index === state.visibleToIndex - 1; + const areMoreOptionsBelow_0 = state.visibleToIndex < options.length; + const areMoreOptionsAbove_0 = state.visibleFromIndex > 0; + const i_0 = state.visibleFromIndex + index_1 + 1; + const isFocused_0 = !isDisabled && state.focusedValue === option_2.value; + const isSelected_0 = state.value === option_2.value; + if (option_2.type === "input") { + const inputValue_0 = inputValues.has(option_2.value) ? inputValues.get(option_2.value) : option_2.initialValue || ""; + return { + setInputValues(prev_1 => { + const next_1 = new Map(prev_1); + next_1.set(option_2.value, value_1); + return next_1; + }); + }} onSubmit={value_2 => { + const hasImageAttachments_0 = pastedContents && Object.values(pastedContents).some(_temp6); + if (value_2.trim() || hasImageAttachments_0 || option_2.allowEmptySubmitToCancel) { + onChange?.(option_2.value); + } else { + onCancel?.(); + } + }} onExit={onCancel} layout="compact" showLabel={inlineDescriptions} onOpenEditor={onOpenEditor} resetCursorOnUpdate={option_2.resetCursorOnUpdate} onImagePaste={onImagePaste} pastedContents={pastedContents} onRemoveImage={onRemoveImage} imagesSelected={imagesSelected} selectedImageIndex={selectedImageIndex} onImagesSelectedChange={setImagesSelected} onSelectedImageIndexChange={setSelectedImageIndex} />; + } + let label_0 = option_2.label; + if (typeof option_2.label === "string" && highlightText && option_2.label.includes(highlightText)) { + const labelText_0 = option_2.label; + const index_2 = labelText_0.indexOf(highlightText); + label_0 = <>{labelText_0.slice(0, index_2)}{highlightText}{labelText_0.slice(index_2 + highlightText.length)}; + } + const isOptionDisabled_0 = option_2.disabled === true; + return <>{!hideIndexes && {`${i_0}.`.padEnd(maxIndexWidth_0 + 1)}}{label_0}{option_2.description && {option_2.description}}; + })}; + break bb0; + } + let t18; + if ($[58] !== hideIndexes || $[59] !== state.options) { + t18 = hideIndexes ? 0 : state.options.length.toString().length; + $[58] = hideIndexes; + $[59] = state.options; + $[60] = t18; + } else { + t18 = $[60]; + } + const maxIndexWidth_1 = t18; + const hasInputOptions = state.visibleOptions.some(_temp7); + const hasDescriptions = !inlineDescriptions && !hasInputOptions && state.visibleOptions.some(_temp8); + const optionData = state.visibleOptions.map((option_3, index_3) => { + const isFirstVisibleOption_1 = option_3.index === state.visibleFromIndex; + const isLastVisibleOption_1 = option_3.index === state.visibleToIndex - 1; + const areMoreOptionsBelow_1 = state.visibleToIndex < options.length; + const areMoreOptionsAbove_1 = state.visibleFromIndex > 0; + const i_1 = state.visibleFromIndex + index_3 + 1; + const isFocused_1 = !isDisabled && state.focusedValue === option_3.value; + const isSelected_1 = state.value === option_3.value; + const isOptionDisabled_1 = option_3.disabled === true; + let label_1 = option_3.label; + if (typeof option_3.label === "string" && highlightText && option_3.label.includes(highlightText)) { + const labelText_1 = option_3.label; + const idx = labelText_1.indexOf(highlightText); + label_1 = <>{labelText_1.slice(0, idx)}{highlightText}{labelText_1.slice(idx + highlightText.length)}; + } + return { + option: option_3, + index: i_1, + label: label_1, + isFocused: isFocused_1, + isSelected: isSelected_1, + isOptionDisabled: isOptionDisabled_1, + shouldShowDownArrow: areMoreOptionsBelow_1 && isLastVisibleOption_1, + shouldShowUpArrow: areMoreOptionsAbove_1 && isFirstVisibleOption_1 + }; + }); + if (hasDescriptions) { + let t19; + if ($[61] !== hideIndexes || $[62] !== maxIndexWidth_1) { + t19 = data => { + if (data.option.type === "input") { + return 0; + } + const labelText_2 = getTextContent(data.option.label); + const indexWidth = hideIndexes ? 0 : maxIndexWidth_1 + 2; + const checkmarkWidth = data.isSelected ? 2 : 0; + return 2 + indexWidth + stringWidth(labelText_2) + checkmarkWidth; + }; + $[61] = hideIndexes; + $[62] = maxIndexWidth_1; + $[63] = t19; + } else { + t19 = $[63]; + } + const maxLabelWidth = Math.max(...optionData.map(t19)); + let t20; + if ($[64] !== hideIndexes || $[65] !== maxIndexWidth_1 || $[66] !== maxLabelWidth) { + t20 = data_0 => { + if (data_0.option.type === "input") { + return null; + } + const labelText_3 = getTextContent(data_0.option.label); + const indexWidth_0 = hideIndexes ? 0 : maxIndexWidth_1 + 2; + const checkmarkWidth_0 = data_0.isSelected ? 2 : 0; + const currentLabelWidth = 2 + indexWidth_0 + stringWidth(labelText_3) + checkmarkWidth_0; + const padding = maxLabelWidth - currentLabelWidth; + return {data_0.isFocused ? {figures.pointer} : data_0.shouldShowDownArrow ? {figures.arrowDown} : data_0.shouldShowUpArrow ? {figures.arrowUp} : } {!hideIndexes && {`${data_0.index}.`.padEnd(maxIndexWidth_1 + 2)}}{data_0.label}{data_0.isSelected && {figures.tick}}{padding > 0 && {" ".repeat(padding)}}{data_0.option.description || " "}; + }; + $[64] = hideIndexes; + $[65] = maxIndexWidth_1; + $[66] = maxLabelWidth; + $[67] = t20; + } else { + t20 = $[67]; + } + t17 = {optionData.map(t20)}; + break bb0; + } + T0 = Box; + t15 = styles.container(); + t16 = state.visibleOptions.map((option_4, index_4) => { + if (option_4.type === "input") { + const inputValue_1 = inputValues.has(option_4.value) ? inputValues.get(option_4.value) : option_4.initialValue || ""; + const isFirstVisibleOption_2 = option_4.index === state.visibleFromIndex; + const isLastVisibleOption_2 = option_4.index === state.visibleToIndex - 1; + const areMoreOptionsBelow_2 = state.visibleToIndex < options.length; + const areMoreOptionsAbove_2 = state.visibleFromIndex > 0; + const i_2 = state.visibleFromIndex + index_4 + 1; + const isFocused_2 = !isDisabled && state.focusedValue === option_4.value; + const isSelected_2 = state.value === option_4.value; + return { + setInputValues(prev_2 => { + const next_2 = new Map(prev_2); + next_2.set(option_4.value, value_3); + return next_2; + }); + }} onSubmit={value_4 => { + const hasImageAttachments_1 = pastedContents && Object.values(pastedContents).some(_temp9); + if (value_4.trim() || hasImageAttachments_1 || option_4.allowEmptySubmitToCancel) { + onChange?.(option_4.value); + } else { + onCancel?.(); + } + }} onExit={onCancel} layout="compact" showLabel={inlineDescriptions} onOpenEditor={onOpenEditor} resetCursorOnUpdate={option_4.resetCursorOnUpdate} onImagePaste={onImagePaste} pastedContents={pastedContents} onRemoveImage={onRemoveImage} imagesSelected={imagesSelected} selectedImageIndex={selectedImageIndex} onImagesSelectedChange={setImagesSelected} onSelectedImageIndexChange={setSelectedImageIndex} />; + } + let label_2 = option_4.label; + if (typeof option_4.label === "string" && highlightText && option_4.label.includes(highlightText)) { + const labelText_4 = option_4.label; + const index_5 = labelText_4.indexOf(highlightText); + label_2 = <>{labelText_4.slice(0, index_5)}{highlightText}{labelText_4.slice(index_5 + highlightText.length)}; + } + const isFirstVisibleOption_3 = option_4.index === state.visibleFromIndex; + const isLastVisibleOption_3 = option_4.index === state.visibleToIndex - 1; + const areMoreOptionsBelow_3 = state.visibleToIndex < options.length; + const areMoreOptionsAbove_3 = state.visibleFromIndex > 0; + const i_3 = state.visibleFromIndex + index_4 + 1; + const isFocused_3 = !isDisabled && state.focusedValue === option_4.value; + const isSelected_3 = state.value === option_4.value; + const isOptionDisabled_2 = option_4.disabled === true; + return {!hideIndexes && {`${i_3}.`.padEnd(maxIndexWidth_1 + 2)}}{label_2}{inlineDescriptions && option_4.description && {" "}{option_4.description}}{!inlineDescriptions && option_4.description && {option_4.description}}; + }); + } + $[28] = hideIndexes; + $[29] = highlightText; + $[30] = imagesSelected; + $[31] = inlineDescriptions; + $[32] = inputValues; + $[33] = isDisabled; + $[34] = layout; + $[35] = onCancel; + $[36] = onChange; + $[37] = onImagePaste; + $[38] = onOpenEditor; + $[39] = onRemoveImage; + $[40] = options.length; + $[41] = pastedContents; + $[42] = selectedImageIndex; + $[43] = state.focusedValue; + $[44] = state.options; + $[45] = state.value; + $[46] = state.visibleFromIndex; + $[47] = state.visibleOptions; + $[48] = state.visibleToIndex; + $[49] = T0; + $[50] = t15; + $[51] = t16; + $[52] = t17; + } else { + T0 = $[49]; + t15 = $[50]; + t16 = $[51]; + t17 = $[52]; + } + if (t17 !== Symbol.for("react.early_return_sentinel")) { + return t17; + } + let t18; + if ($[68] !== T0 || $[69] !== t15 || $[70] !== t16) { + t18 = {t16}; + $[68] = T0; + $[69] = t15; + $[70] = t16; + $[71] = t18; + } else { + t18 = $[71]; + } + return t18; +} + +// Row container for the two-column (label + description) layout. Unlike +// the other Select layouts, this one doesn't render through SelectOption → +// ListItem, so it declares the native cursor directly. Parks the cursor +// on the pointer indicator so screen readers / magnifiers track focus. +function _temp9(c_3) { + return c_3.type === "image"; +} +function _temp8(opt_0) { + return opt_0.description; +} +function _temp7(opt) { + return opt.type === "input"; +} +function _temp6(c_2) { + return c_2.type === "image"; +} +function _temp5(c_1) { + return c_1.type === "image"; +} +function _temp4() { + return { + bold: true + }; +} +function _temp3() { + return { + flexDirection: "column" as const + }; +} +function _temp2(c) { + return c.type === "image"; +} +function _temp(c_0) { + return c_0.type === "image"; +} +function TwoColumnRow(t0) { + const $ = _c(5); + const { + isFocused, + children + } = t0; + let t1; + if ($[0] !== isFocused) { + t1 = { + line: 0, + column: 0, + active: isFocused + }; + $[0] = isFocused; + $[1] = t1; + } else { + t1 = $[1]; + } + const cursorRef = useDeclaredCursor(t1); + let t2; + if ($[2] !== children || $[3] !== cursorRef) { + t2 = {children}; + $[2] = children; + $[3] = cursorRef; + $[4] = t2; + } else { + t2 = $[4]; + } + return t2; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJmaWd1cmVzIiwiUmVhY3QiLCJSZWFjdE5vZGUiLCJ1c2VFZmZlY3QiLCJ1c2VSZWYiLCJ1c2VTdGF0ZSIsInVzZURlY2xhcmVkQ3Vyc29yIiwic3RyaW5nV2lkdGgiLCJBbnNpIiwiQm94IiwiVGV4dCIsImNvdW50IiwiUGFzdGVkQ29udGVudCIsIkltYWdlRGltZW5zaW9ucyIsIlNlbGVjdElucHV0T3B0aW9uIiwiU2VsZWN0T3B0aW9uIiwidXNlU2VsZWN0SW5wdXQiLCJ1c2VTZWxlY3RTdGF0ZSIsImdldFRleHRDb250ZW50Iiwibm9kZSIsIlN0cmluZyIsIkFycmF5IiwiaXNBcnJheSIsIm1hcCIsImpvaW4iLCJpc1ZhbGlkRWxlbWVudCIsImNoaWxkcmVuIiwicHJvcHMiLCJCYXNlT3B0aW9uIiwiZGVzY3JpcHRpb24iLCJkaW1EZXNjcmlwdGlvbiIsImxhYmVsIiwidmFsdWUiLCJUIiwiZGlzYWJsZWQiLCJPcHRpb25XaXRoRGVzY3JpcHRpb24iLCJ0eXBlIiwib25DaGFuZ2UiLCJwbGFjZWhvbGRlciIsImluaXRpYWxWYWx1ZSIsImFsbG93RW1wdHlTdWJtaXRUb0NhbmNlbCIsInNob3dMYWJlbFdpdGhWYWx1ZSIsImxhYmVsVmFsdWVTZXBhcmF0b3IiLCJyZXNldEN1cnNvck9uVXBkYXRlIiwiU2VsZWN0UHJvcHMiLCJpc0Rpc2FibGVkIiwiZGlzYWJsZVNlbGVjdGlvbiIsImhpZGVJbmRleGVzIiwidmlzaWJsZU9wdGlvbkNvdW50IiwiaGlnaGxpZ2h0VGV4dCIsIm9wdGlvbnMiLCJkZWZhdWx0VmFsdWUiLCJvbkNhbmNlbCIsIm9uRm9jdXMiLCJkZWZhdWx0Rm9jdXNWYWx1ZSIsImxheW91dCIsImlubGluZURlc2NyaXB0aW9ucyIsIm9uVXBGcm9tRmlyc3RJdGVtIiwib25Eb3duRnJvbUxhc3RJdGVtIiwib25JbnB1dE1vZGVUb2dnbGUiLCJvbk9wZW5FZGl0b3IiLCJjdXJyZW50VmFsdWUiLCJzZXRWYWx1ZSIsIm9uSW1hZ2VQYXN0ZSIsImJhc2U2NEltYWdlIiwibWVkaWFUeXBlIiwiZmlsZW5hbWUiLCJkaW1lbnNpb25zIiwic291cmNlUGF0aCIsInBhc3RlZENvbnRlbnRzIiwiUmVjb3JkIiwib25SZW1vdmVJbWFnZSIsImlkIiwiU2VsZWN0IiwidDAiLCIkIiwiX2MiLCJ0MSIsInQyIiwidDMiLCJ0NCIsInQ1IiwidDYiLCJ1bmRlZmluZWQiLCJpbWFnZXNTZWxlY3RlZCIsInNldEltYWdlc1NlbGVjdGVkIiwic2VsZWN0ZWRJbWFnZUluZGV4Iiwic2V0U2VsZWN0ZWRJbWFnZUluZGV4IiwidDciLCJpbml0aWFsTWFwIiwiTWFwIiwiZm9yRWFjaCIsIm9wdGlvbiIsInNldCIsImlucHV0VmFsdWVzIiwic2V0SW5wdXRWYWx1ZXMiLCJ0OCIsIlN5bWJvbCIsImZvciIsImxhc3RJbml0aWFsVmFsdWVzIiwidDEwIiwidDkiLCJvcHRpb25fMCIsImxhc3RJbml0aWFsIiwiY3VycmVudCIsImdldCIsIm5ld0luaXRpYWwiLCJwcmV2IiwibmV4dCIsInQxMSIsImZvY3VzVmFsdWUiLCJzdGF0ZSIsInQxMiIsInQxMyIsIk9iamVjdCIsInZhbHVlcyIsInNvbWUiLCJfdGVtcCIsImltYWdlQ291bnQiLCJfdGVtcDIiLCJ0MTQiLCJpc011bHRpU2VsZWN0Iiwib25FbnRlckltYWdlU2VsZWN0aW9uIiwiVDAiLCJ0MTUiLCJ0MTYiLCJ0MTciLCJsZW5ndGgiLCJmb2N1c2VkVmFsdWUiLCJ2aXNpYmxlRnJvbUluZGV4IiwidmlzaWJsZU9wdGlvbnMiLCJ2aXNpYmxlVG9JbmRleCIsImJiMCIsInN0eWxlcyIsImNvbnRhaW5lciIsIl90ZW1wMyIsImhpZ2hsaWdodGVkVGV4dCIsIl90ZW1wNCIsInQxOCIsInRvU3RyaW5nIiwibWF4SW5kZXhXaWR0aCIsIm9wdGlvbl8xIiwiaW5kZXgiLCJpc0ZpcnN0VmlzaWJsZU9wdGlvbiIsImlzTGFzdFZpc2libGVPcHRpb24iLCJhcmVNb3JlT3B0aW9uc0JlbG93IiwiYXJlTW9yZU9wdGlvbnNBYm92ZSIsImkiLCJpc0ZvY3VzZWQiLCJpc1NlbGVjdGVkIiwiaW5wdXRWYWx1ZSIsImhhcyIsInByZXZfMCIsIm5leHRfMCIsInZhbHVlXzAiLCJoYXNJbWFnZUF0dGFjaG1lbnRzIiwiX3RlbXA1IiwidHJpbSIsImluY2x1ZGVzIiwibGFiZWxUZXh0IiwiaW5kZXhfMCIsImluZGV4T2YiLCJzbGljZSIsImlzT3B0aW9uRGlzYWJsZWQiLCJvcHRpb25Db2xvciIsIm1heEluZGV4V2lkdGhfMCIsIm9wdGlvbl8yIiwiaW5kZXhfMSIsImlzRmlyc3RWaXNpYmxlT3B0aW9uXzAiLCJpc0xhc3RWaXNpYmxlT3B0aW9uXzAiLCJhcmVNb3JlT3B0aW9uc0JlbG93XzAiLCJhcmVNb3JlT3B0aW9uc0Fib3ZlXzAiLCJpXzAiLCJpc0ZvY3VzZWRfMCIsImlzU2VsZWN0ZWRfMCIsImlucHV0VmFsdWVfMCIsInZhbHVlXzEiLCJwcmV2XzEiLCJuZXh0XzEiLCJ2YWx1ZV8yIiwiaGFzSW1hZ2VBdHRhY2htZW50c18wIiwiX3RlbXA2IiwibGFiZWxfMCIsImxhYmVsVGV4dF8wIiwiaW5kZXhfMiIsImlzT3B0aW9uRGlzYWJsZWRfMCIsInBhZEVuZCIsIm1heEluZGV4V2lkdGhfMSIsImhhc0lucHV0T3B0aW9ucyIsIl90ZW1wNyIsImhhc0Rlc2NyaXB0aW9ucyIsIl90ZW1wOCIsIm9wdGlvbkRhdGEiLCJvcHRpb25fMyIsImluZGV4XzMiLCJpc0ZpcnN0VmlzaWJsZU9wdGlvbl8xIiwiaXNMYXN0VmlzaWJsZU9wdGlvbl8xIiwiYXJlTW9yZU9wdGlvbnNCZWxvd18xIiwiYXJlTW9yZU9wdGlvbnNBYm92ZV8xIiwiaV8xIiwiaXNGb2N1c2VkXzEiLCJpc1NlbGVjdGVkXzEiLCJpc09wdGlvbkRpc2FibGVkXzEiLCJsYWJlbF8xIiwibGFiZWxUZXh0XzEiLCJpZHgiLCJzaG91bGRTaG93RG93bkFycm93Iiwic2hvdWxkU2hvd1VwQXJyb3ciLCJ0MTkiLCJkYXRhIiwibGFiZWxUZXh0XzIiLCJpbmRleFdpZHRoIiwiY2hlY2ttYXJrV2lkdGgiLCJtYXhMYWJlbFdpZHRoIiwiTWF0aCIsIm1heCIsInQyMCIsImRhdGFfMCIsImxhYmVsVGV4dF8zIiwiaW5kZXhXaWR0aF8wIiwiY2hlY2ttYXJrV2lkdGhfMCIsImN1cnJlbnRMYWJlbFdpZHRoIiwicGFkZGluZyIsInBvaW50ZXIiLCJhcnJvd0Rvd24iLCJhcnJvd1VwIiwidGljayIsInJlcGVhdCIsIm9wdGlvbl80IiwiaW5kZXhfNCIsImlucHV0VmFsdWVfMSIsImlzRmlyc3RWaXNpYmxlT3B0aW9uXzIiLCJpc0xhc3RWaXNpYmxlT3B0aW9uXzIiLCJhcmVNb3JlT3B0aW9uc0JlbG93XzIiLCJhcmVNb3JlT3B0aW9uc0Fib3ZlXzIiLCJpXzIiLCJpc0ZvY3VzZWRfMiIsImlzU2VsZWN0ZWRfMiIsInZhbHVlXzMiLCJwcmV2XzIiLCJuZXh0XzIiLCJ2YWx1ZV80IiwiaGFzSW1hZ2VBdHRhY2htZW50c18xIiwiX3RlbXA5IiwibGFiZWxfMiIsImxhYmVsVGV4dF80IiwiaW5kZXhfNSIsImlzRmlyc3RWaXNpYmxlT3B0aW9uXzMiLCJpc0xhc3RWaXNpYmxlT3B0aW9uXzMiLCJhcmVNb3JlT3B0aW9uc0JlbG93XzMiLCJhcmVNb3JlT3B0aW9uc0Fib3ZlXzMiLCJpXzMiLCJpc0ZvY3VzZWRfMyIsImlzU2VsZWN0ZWRfMyIsImlzT3B0aW9uRGlzYWJsZWRfMiIsImNfMyIsImMiLCJvcHRfMCIsIm9wdCIsImNfMiIsImNfMSIsImJvbGQiLCJmbGV4RGlyZWN0aW9uIiwiY29uc3QiLCJjXzAiLCJUd29Db2x1bW5Sb3ciLCJsaW5lIiwiY29sdW1uIiwiYWN0aXZlIiwiY3Vyc29yUmVmIl0sInNvdXJjZXMiOlsic2VsZWN0LnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgZmlndXJlcyBmcm9tICdmaWd1cmVzJ1xuaW1wb3J0IFJlYWN0LCB7IHR5cGUgUmVhY3ROb2RlLCB1c2VFZmZlY3QsIHVzZVJlZiwgdXNlU3RhdGUgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZURlY2xhcmVkQ3Vyc29yIH0gZnJvbSAnLi4vLi4vaW5rL2hvb2tzL3VzZS1kZWNsYXJlZC1jdXJzb3IuanMnXG5pbXBvcnQgeyBzdHJpbmdXaWR0aCB9IGZyb20gJy4uLy4uL2luay9zdHJpbmdXaWR0aC5qcydcbmltcG9ydCB7IEFuc2ksIEJveCwgVGV4dCB9IGZyb20gJy4uLy4uL2luay5qcydcbmltcG9ydCB7IGNvdW50IH0gZnJvbSAnLi4vLi4vdXRpbHMvYXJyYXkuanMnXG5pbXBvcnQgdHlwZSB7IFBhc3RlZENvbnRlbnQgfSBmcm9tICcuLi8uLi91dGlscy9jb25maWcuanMnXG5pbXBvcnQgdHlwZSB7IEltYWdlRGltZW5zaW9ucyB9IGZyb20gJy4uLy4uL3V0aWxzL2ltYWdlUmVzaXplci5qcydcbmltcG9ydCB7IFNlbGVjdElucHV0T3B0aW9uIH0gZnJvbSAnLi9zZWxlY3QtaW5wdXQtb3B0aW9uLmpzJ1xuaW1wb3J0IHsgU2VsZWN0T3B0aW9uIH0gZnJvbSAnLi9zZWxlY3Qtb3B0aW9uLmpzJ1xuaW1wb3J0IHsgdXNlU2VsZWN0SW5wdXQgfSBmcm9tICcuL3VzZS1zZWxlY3QtaW5wdXQuanMnXG5pbXBvcnQgeyB1c2VTZWxlY3RTdGF0ZSB9IGZyb20gJy4vdXNlLXNlbGVjdC1zdGF0ZS5qcydcblxuLy8gRXh0cmFjdCB0ZXh0IGNvbnRlbnQgZnJvbSBSZWFjdE5vZGUgZm9yIHdpZHRoIGNhbGN1bGF0aW9uXG5mdW5jdGlvbiBnZXRUZXh0Q29udGVudChub2RlOiBSZWFjdE5vZGUpOiBzdHJpbmcge1xuICBpZiAodHlwZW9mIG5vZGUgPT09ICdzdHJpbmcnKSByZXR1cm4gbm9kZVxuICBpZiAodHlwZW9mIG5vZGUgPT09ICdudW1iZXInKSByZXR1cm4gU3RyaW5nKG5vZGUpXG4gIGlmICghbm9kZSkgcmV0dXJuICcnXG4gIGlmIChBcnJheS5pc0FycmF5KG5vZGUpKSByZXR1cm4gbm9kZS5tYXAoZ2V0VGV4dENvbnRlbnQpLmpvaW4oJycpXG4gIGlmIChSZWFjdC5pc1ZhbGlkRWxlbWVudDx7IGNoaWxkcmVuPzogUmVhY3ROb2RlIH0+KG5vZGUpKSB7XG4gICAgcmV0dXJuIGdldFRleHRDb250ZW50KG5vZGUucHJvcHMuY2hpbGRyZW4pXG4gIH1cbiAgcmV0dXJuICcnXG59XG5cbnR5cGUgQmFzZU9wdGlvbjxUPiA9IHtcbiAgZGVzY3JpcHRpb24/OiBzdHJpbmdcbiAgZGltRGVzY3JpcHRpb24/OiBib29sZWFuXG4gIGxhYmVsOiBSZWFjdE5vZGVcbiAgdmFsdWU6IFRcbiAgZGlzYWJsZWQ/OiBib29sZWFuXG59XG5cbmV4cG9ydCB0eXBlIE9wdGlvbldpdGhEZXNjcmlwdGlvbjxUID0gc3RyaW5nPiA9XG4gIHwgKEJhc2VPcHRpb248VD4gJiB7XG4gICAgICB0eXBlPzogJ3RleHQnXG4gICAgfSlcbiAgfCAoQmFzZU9wdGlvbjxUPiAmIHtcbiAgICAgIHR5cGU6ICdpbnB1dCdcbiAgICAgIG9uQ2hhbmdlOiAodmFsdWU6IHN0cmluZykgPT4gdm9pZFxuICAgICAgcGxhY2Vob2xkZXI/OiBzdHJpbmdcbiAgICAgIGluaXRpYWxWYWx1ZT86IHN0cmluZ1xuICAgICAgLyoqXG4gICAgICAgKiBDb250cm9scyBiZWhhdmlvciB3aGVuIHN1Ym1pdHRpbmcgd2l0aCBlbXB0eSBpbnB1dDpcbiAgICAgICAqIC0gdHJ1ZTogY2FsbHMgb25DaGFuZ2UgKHRyZWF0cyBlbXB0eSBhcyB2YWxpZCBzdWJtaXNzaW9uKVxuICAgICAgICogLSBmYWxzZSAoZGVmYXVsdCk6IGNhbGxzIG9uQ2FuY2VsICh0cmVhdHMgZW1wdHkgYXMgY2FuY2VsbGF0aW9uKVxuICAgICAgICpcbiAgICAgICAqIEFsc28gYWZmZWN0cyBpbml0aWFsIEVudGVyIHByZXNzOiB3aGVuIHRydWUsIHN1Ym1pdHMgaW1tZWRpYXRlbHk7XG4gICAgICAgKiB3aGVuIGZhbHNlLCBlbnRlcnMgaW5wdXQgbW9kZSBmaXJzdCBzbyB1c2VyIGNhbiB0eXBlLlxuICAgICAgICovXG4gICAgICBhbGxvd0VtcHR5U3VibWl0VG9DYW5jZWw/OiBib29sZWFuXG4gICAgICAvKipcbiAgICAgICAqIFdoZW4gdHJ1ZSwgYWx3YXlzIHNob3dzIHRoZSBsYWJlbCBhbG9uZ3NpZGUgdGhlIGlucHV0IHZhbHVlLCByZWdhcmRsZXNzIG9mXG4gICAgICAgKiB0aGUgZ2xvYmFsIGlubGluZURlc2NyaXB0aW9ucy9zaG93TGFiZWwgc2V0dGluZy4gVXNlIHRoaXMgd2hlbiB0aGUgbGFiZWxcbiAgICAgICAqIHByb3ZpZGVzIGltcG9ydGFudCBjb250ZXh0IHRoYXQgc2hvdWxkIGFsd2F5cyBiZSB2aXNpYmxlIChlLmcuLCBcIlllcywgYW5kIGFsbG93Li4uXCIpLlxuICAgICAgICovXG4gICAgICBzaG93TGFiZWxXaXRoVmFsdWU/OiBib29sZWFuXG4gICAgICAvKipcbiAgICAgICAqIEN1c3RvbSBzZXBhcmF0b3IgYmV0d2VlbiBsYWJlbCBhbmQgdmFsdWUgd2hlbiBzaG93TGFiZWwgaXMgdHJ1ZS5cbiAgICAgICAqIERlZmF1bHRzIHRvIFwiLCBcIi4gVXNlIFwiOiBcIiBmb3IgbGFiZWxzIHRoYXQgcmVhZCBiZXR0ZXIgd2l0aCBhIGNvbG9uLlxuICAgICAgICovXG4gICAgICBsYWJlbFZhbHVlU2VwYXJhdG9yPzogc3RyaW5nXG4gICAgICAvKipcbiAgICAgICAqIFdoZW4gdHJ1ZSwgYXV0b21hdGljYWxseSByZXNldCBjdXJzb3IgdG8gZW5kIG9mIGxpbmUgd2hlbjpcbiAgICAgICAqIC0gT3B0aW9uIGJlY29tZXMgZm9jdXNlZFxuICAgICAgICogLSBJbnB1dCB2YWx1ZSBjaGFuZ2VzXG4gICAgICAgKiBUaGlzIHByZXZlbnRzIGN1cnNvciBwb3NpdGlvbiBidWdzIHdoZW4gdGhlIGlucHV0IHZhbHVlIHVwZGF0ZXMgYXN5bmNocm9ub3VzbHkuXG4gICAgICAgKi9cbiAgICAgIHJlc2V0Q3Vyc29yT25VcGRhdGU/OiBib29sZWFuXG4gICAgfSlcblxuZXhwb3J0IHR5cGUgU2VsZWN0UHJvcHM8VD4gPSB7XG4gIC8qKlxuICAgKiBXaGVuIGRpc2FibGVkLCB1c2VyIGlucHV0IGlzIGlnbm9yZWQuXG4gICAqXG4gICAqIEBkZWZhdWx0IGZhbHNlXG4gICAqL1xuICByZWFkb25seSBpc0Rpc2FibGVkPzogYm9vbGVhblxuXG4gIC8qKlxuICAgKiBXaGVuIHRydWUsIHByZXZlbnRzIHNlbGVjdGlvbiBvbiBFbnRlciBidXQgYWxsb3dzIHNjcm9sbGluZy5cbiAgICpcbiAgICogQGRlZmF1bHQgZmFsc2VcbiAgICovXG4gIHJlYWRvbmx5IGRpc2FibGVTZWxlY3Rpb24/OiBib29sZWFuXG5cbiAgLyoqXG4gICAqIFdoZW4gdHJ1ZSwgaGlkZXMgdGhlIG51bWVyaWMgaW5kZXhlcyBuZXh0IHRvIGVhY2ggb3B0aW9uLlxuICAgKlxuICAgKiBAZGVmYXVsdCBmYWxzZVxuICAgKi9cbiAgcmVhZG9ubHkgaGlkZUluZGV4ZXM/OiBib29sZWFuXG5cbiAgLyoqXG4gICAqIE51bWJlciBvZiB2aXNpYmxlIG9wdGlvbnMuXG4gICAqXG4gICAqIEBkZWZhdWx0IDVcbiAgICovXG4gIHJlYWRvbmx5IHZpc2libGVPcHRpb25Db3VudD86IG51bWJlclxuXG4gIC8qKlxuICAgKiBIaWdobGlnaHQgdGV4dCBpbiBvcHRpb24gbGFiZWxzLlxuICAgKi9cbiAgcmVhZG9ubHkgaGlnaGxpZ2h0VGV4dD86IHN0cmluZ1xuXG4gIC8qKlxuICAgKiBPcHRpb25zLlxuICAgKi9cbiAgcmVhZG9ubHkgb3B0aW9uczogT3B0aW9uV2l0aERlc2NyaXB0aW9uPFQ+W11cblxuICAvKipcbiAgICogRGVmYXVsdCB2YWx1ZS5cbiAgICovXG4gIHJlYWRvbmx5IGRlZmF1bHRWYWx1ZT86IFRcblxuICAvKipcbiAgICogQ2FsbGJhY2sgd2hlbiBjYW5jZWwgaXMgcHJlc3NlZC5cbiAgICovXG4gIHJlYWRvbmx5IG9uQ2FuY2VsPzogKCkgPT4gdm9pZFxuXG4gIC8qKlxuICAgKiBDYWxsYmFjayB3aGVuIHNlbGVjdGVkIG9wdGlvbiBjaGFuZ2VzLlxuICAgKi9cbiAgcmVhZG9ubHkgb25DaGFuZ2U/OiAodmFsdWU6IFQpID0+IHZvaWRcblxuICAvKipcbiAgICogQ2FsbGJhY2sgd2hlbiBmb2N1c2VkIG9wdGlvbiBjaGFuZ2VzLlxuICAgKiBOb3RlOiBUaGlzIGlzIGZvciBvbmUtd2F5IG5vdGlmaWNhdGlvbiBvbmx5LiBBdm9pZCBjb21iaW5pbmcgd2l0aCBmb2N1c1ZhbHVlXG4gICAqIGZvciBiaWRpcmVjdGlvbmFsIHN5bmMsIGFzIHRoaXMgY2FuIGNhdXNlIGZlZWRiYWNrIGxvb3BzLlxuICAgKi9cbiAgcmVhZG9ubHkgb25Gb2N1cz86ICh2YWx1ZTogVCkgPT4gdm9pZFxuXG4gIC8qKlxuICAgKiBJbml0aWFsIHZhbHVlIHRvIGZvY3VzLiBUaGlzIGlzIHVzZWQgdG8gc2V0IGZvY3VzIHdoZW4gdGhlIGNvbXBvbmVudCBtb3VudHMuXG4gICAqL1xuICByZWFkb25seSBkZWZhdWx0Rm9jdXNWYWx1ZT86IFRcblxuICAvKipcbiAgICogTGF5b3V0IG9mIHRoZSBvcHRpb25zLlxuICAgKiAtIGBjb21wYWN0YCAoZGVmYXVsdCkgdHJpZXMgdG8gdXNlIG9uZSBsaW5lIHBlciBvcHRpb25cbiAgICogLSBgZXhwYW5kZWRgIHVzZXMgbXVsdGlwbGUgbGluZXMgYW5kIGFuIGVtcHR5IGxpbmUgYmV0d2VlbiBvcHRpb25zXG4gICAqIC0gYGNvbXBhY3QtdmVydGljYWxgIHVzZXMgY29tcGFjdCBpbmRleCBmb3JtYXR0aW5nIHdpdGggZGVzY3JpcHRpb25zIGJlbG93IGxhYmVsc1xuICAgKi9cbiAgcmVhZG9ubHkgbGF5b3V0PzogJ2NvbXBhY3QnIHwgJ2V4cGFuZGVkJyB8ICdjb21wYWN0LXZlcnRpY2FsJ1xuXG4gIC8qKlxuICAgKiBXaGVuIHRydWUsIGRlc2NyaXB0aW9ucyBhcmUgcmVuZGVyZWQgaW5saW5lIGFmdGVyIHRoZSBsYWJlbCBpbnN0ZWFkIG9mXG4gICAqIGluIGEgc2VwYXJhdGUgY29sdW1uLiBVc2UgdGhpcyBmb3Igc2hvcnQgZGVzY3JpcHRpb25zIGxpa2UgaGludHMuXG4gICAqXG4gICAqIEBkZWZhdWx0IGZhbHNlXG4gICAqL1xuICByZWFkb25seSBpbmxpbmVEZXNjcmlwdGlvbnM/OiBib29sZWFuXG5cbiAgLyoqXG4gICAqIENhbGxiYWNrIHdoZW4gdXNlciBwcmVzc2VzIHVwIGZyb20gdGhlIGZpcnN0IGl0ZW0uXG4gICAqIElmIHByb3ZpZGVkLCBuYXZpZ2F0aW9uIHdpbGwgbm90IHdyYXAgdG8gdGhlIGxhc3QgaXRlbS5cbiAgICovXG4gIHJlYWRvbmx5IG9uVXBGcm9tRmlyc3RJdGVtPzogKCkgPT4gdm9pZFxuXG4gIC8qKlxuICAgKiBDYWxsYmFjayB3aGVuIHVzZXIgcHJlc3NlcyBkb3duIGZyb20gdGhlIGxhc3QgaXRlbS5cbiAgICogSWYgcHJvdmlkZWQsIG5hdmlnYXRpb24gd2lsbCBub3Qgd3JhcCB0byB0aGUgZmlyc3QgaXRlbS5cbiAgICovXG4gIHJlYWRvbmx5IG9uRG93bkZyb21MYXN0SXRlbT86ICgpID0+IHZvaWRcblxuICAvKipcbiAgICogQ2FsbGJhY2sgd2hlbiBpbnB1dCBtb2RlIHNob3VsZCBiZSB0b2dnbGVkIGZvciBhbiBvcHRpb24uXG4gICAqIENhbGxlZCB3aGVuIFRhYiBpcyBwcmVzc2VkICh0byBlbnRlciBvciBleGl0IGlucHV0IG1vZGUpLlxuICAgKi9cbiAgcmVhZG9ubHkgb25JbnB1dE1vZGVUb2dnbGU/OiAodmFsdWU6IFQpID0+IHZvaWRcblxuICAvKipcbiAgICogQ2FsbGJhY2sgdG8gb3BlbiBleHRlcm5hbCBlZGl0b3IgZm9yIGVkaXRpbmcgaW5wdXQgb3B0aW9uIHZhbHVlcy5cbiAgICogV2hlbiBwcm92aWRlZCwgY3RybCtnIHdpbGwgdHJpZ2dlciB0aGlzIGNhbGxiYWNrIGluIGlucHV0IG9wdGlvbnNcbiAgICogd2l0aCB0aGUgY3VycmVudCB2YWx1ZSBhbmQgYSBzZXR0ZXIgZnVuY3Rpb24gdG8gdXBkYXRlIHRoZSBpbnRlcm5hbCBzdGF0ZS5cbiAgICovXG4gIHJlYWRvbmx5IG9uT3BlbkVkaXRvcj86IChcbiAgICBjdXJyZW50VmFsdWU6IHN0cmluZyxcbiAgICBzZXRWYWx1ZTogKHZhbHVlOiBzdHJpbmcpID0+IHZvaWQsXG4gICkgPT4gdm9pZFxuXG4gIC8qKlxuICAgKiBPcHRpb25hbCBjYWxsYmFjayB3aGVuIGFuIGltYWdlIGlzIHBhc3RlZCBpbnRvIGFuIGlucHV0IG9wdGlvbi5cbiAgICovXG4gIHJlYWRvbmx5IG9uSW1hZ2VQYXN0ZT86IChcbiAgICBiYXNlNjRJbWFnZTogc3RyaW5nLFxuICAgIG1lZGlhVHlwZT86IHN0cmluZyxcbiAgICBmaWxlbmFtZT86IHN0cmluZyxcbiAgICBkaW1lbnNpb25zPzogSW1hZ2VEaW1lbnNpb25zLFxuICAgIHNvdXJjZVBhdGg/OiBzdHJpbmcsXG4gICkgPT4gdm9pZFxuXG4gIC8qKlxuICAgKiBQYXN0ZWQgY29udGVudCB0byBkaXNwbGF5IGlubGluZSBpbiBpbnB1dCBvcHRpb25zLlxuICAgKi9cbiAgcmVhZG9ubHkgcGFzdGVkQ29udGVudHM/OiBSZWNvcmQ8bnVtYmVyLCBQYXN0ZWRDb250ZW50PlxuXG4gIC8qKlxuICAgKiBDYWxsYmFjayB0byByZW1vdmUgYSBwYXN0ZWQgaW1hZ2UgYnkgaXRzIElELlxuICAgKi9cbiAgcmVhZG9ubHkgb25SZW1vdmVJbWFnZT86IChpZDogbnVtYmVyKSA9PiB2b2lkXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBTZWxlY3Q8VD4oe1xuICBpc0Rpc2FibGVkID0gZmFsc2UsXG4gIGhpZGVJbmRleGVzID0gZmFsc2UsXG4gIHZpc2libGVPcHRpb25Db3VudCA9IDUsXG4gIGhpZ2hsaWdodFRleHQsXG4gIG9wdGlvbnMsXG4gIGRlZmF1bHRWYWx1ZSxcbiAgb25DYW5jZWwsXG4gIG9uQ2hhbmdlLFxuICBvbkZvY3VzLFxuICBkZWZhdWx0Rm9jdXNWYWx1ZSxcbiAgbGF5b3V0ID0gJ2NvbXBhY3QnLFxuICBkaXNhYmxlU2VsZWN0aW9uID0gZmFsc2UsXG4gIGlubGluZURlc2NyaXB0aW9ucyA9IGZhbHNlLFxuICBvblVwRnJvbUZpcnN0SXRlbSxcbiAgb25Eb3duRnJvbUxhc3RJdGVtLFxuICBvbklucHV0TW9kZVRvZ2dsZSxcbiAgb25PcGVuRWRpdG9yLFxuICBvbkltYWdlUGFzdGUsXG4gIHBhc3RlZENvbnRlbnRzLFxuICBvblJlbW92ZUltYWdlLFxufTogU2VsZWN0UHJvcHM8VD4pOiBSZWFjdC5SZWFjdE5vZGUge1xuICAvLyBJbWFnZSBzZWxlY3Rpb24gbW9kZSBzdGF0ZVxuICBjb25zdCBbaW1hZ2VzU2VsZWN0ZWQsIHNldEltYWdlc1NlbGVjdGVkXSA9IHVzZVN0YXRlKGZhbHNlKVxuICBjb25zdCBbc2VsZWN0ZWRJbWFnZUluZGV4LCBzZXRTZWxlY3RlZEltYWdlSW5kZXhdID0gdXNlU3RhdGUoMClcblxuICAvLyBTdGF0ZSBmb3IgaW5wdXQgdHlwZSBvcHRpb25zXG4gIGNvbnN0IFtpbnB1dFZhbHVlcywgc2V0SW5wdXRWYWx1ZXNdID0gdXNlU3RhdGU8TWFwPFQsIHN0cmluZz4+KCgpID0+IHtcbiAgICBjb25zdCBpbml0aWFsTWFwID0gbmV3IE1hcDxULCBzdHJpbmc+KClcbiAgICBvcHRpb25zLmZvckVhY2gob3B0aW9uID0+IHtcbiAgICAgIGlmIChvcHRpb24udHlwZSA9PT0gJ2lucHV0JyAmJiBvcHRpb24uaW5pdGlhbFZhbHVlKSB7XG4gICAgICAgIGluaXRpYWxNYXAuc2V0KG9wdGlvbi52YWx1ZSwgb3B0aW9uLmluaXRpYWxWYWx1ZSlcbiAgICAgIH1cbiAgICB9KVxuICAgIHJldHVybiBpbml0aWFsTWFwXG4gIH0pXG5cbiAgLy8gVHJhY2sgdGhlIGxhc3QgaW5pdGlhbFZhbHVlIHdlIHN5bmNlZCwgc28gd2UgY2FuIGRldGVjdCB1c2VyIGVkaXRzXG4gIGNvbnN0IGxhc3RJbml0aWFsVmFsdWVzID0gdXNlUmVmPE1hcDxULCBzdHJpbmc+PihuZXcgTWFwKCkpXG5cbiAgLy8gU3luYyBpbml0aWFsVmFsdWUgY2hhbmdlcyB0byBpbnB1dFZhbHVlcyBzdGF0ZSwgYnV0IG9ubHkgaWYgdXNlciBoYXNuJ3QgZWRpdGVkXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgZm9yIChjb25zdCBvcHRpb24gb2Ygb3B0aW9ucykge1xuICAgICAgaWYgKG9wdGlvbi50eXBlID09PSAnaW5wdXQnICYmIG9wdGlvbi5pbml0aWFsVmFsdWUgIT09IHVuZGVmaW5lZCkge1xuICAgICAgICBjb25zdCBsYXN0SW5pdGlhbCA9IGxhc3RJbml0aWFsVmFsdWVzLmN1cnJlbnQuZ2V0KG9wdGlvbi52YWx1ZSkgPz8gJydcbiAgICAgICAgY29uc3QgY3VycmVudFZhbHVlID0gaW5wdXRWYWx1ZXMuZ2V0KG9wdGlvbi52YWx1ZSkgPz8gJydcbiAgICAgICAgY29uc3QgbmV3SW5pdGlhbCA9IG9wdGlvbi5pbml0aWFsVmFsdWVcblxuICAgICAgICAvLyBPbmx5IHVwZGF0ZSBpZjpcbiAgICAgICAgLy8gMS4gVGhlIGluaXRpYWxWYWx1ZSBoYXMgY2hhbmdlZFxuICAgICAgICAvLyAyLiBUaGUgdXNlciBoYXNuJ3QgZWRpdGVkIChjdXJyZW50IHZhbHVlIHN0aWxsIG1hdGNoZXMgdGhlIGxhc3QgaW5pdGlhbFZhbHVlIHdlIHNldClcbiAgICAgICAgaWYgKG5ld0luaXRpYWwgIT09IGxhc3RJbml0aWFsICYmIGN1cnJlbnRWYWx1ZSA9PT0gbGFzdEluaXRpYWwpIHtcbiAgICAgICAgICBzZXRJbnB1dFZhbHVlcyhwcmV2ID0+IHtcbiAgICAgICAgICAgIGNvbnN0IG5leHQgPSBuZXcgTWFwKHByZXYpXG4gICAgICAgICAgICBuZXh0LnNldChvcHRpb24udmFsdWUsIG5ld0luaXRpYWwpXG4gICAgICAgICAgICByZXR1cm4gbmV4dFxuICAgICAgICAgIH0pXG4gICAgICAgIH1cblxuICAgICAgICAvLyBBbHdheXMgdHJhY2sgdGhlIGxhdGVzdCBpbml0aWFsVmFsdWVcbiAgICAgICAgbGFzdEluaXRpYWxWYWx1ZXMuY3VycmVudC5zZXQob3B0aW9uLnZhbHVlLCBuZXdJbml0aWFsKVxuICAgICAgfVxuICAgIH1cbiAgfSwgW29wdGlvbnMsIGlucHV0VmFsdWVzXSlcblxuICBjb25zdCBzdGF0ZSA9IHVzZVNlbGVjdFN0YXRlKHtcbiAgICB2aXNpYmxlT3B0aW9uQ291bnQsXG4gICAgb3B0aW9ucyxcbiAgICBkZWZhdWx0VmFsdWUsXG4gICAgb25DaGFuZ2UsXG4gICAgb25DYW5jZWwsXG4gICAgb25Gb2N1cyxcbiAgICBmb2N1c1ZhbHVlOiBkZWZhdWx0Rm9jdXNWYWx1ZSxcbiAgfSlcblxuICB1c2VTZWxlY3RJbnB1dCh7XG4gICAgaXNEaXNhYmxlZCxcbiAgICBkaXNhYmxlU2VsZWN0aW9uOiBkaXNhYmxlU2VsZWN0aW9uIHx8IChoaWRlSW5kZXhlcyA/ICdudW1lcmljJyA6IGZhbHNlKSxcbiAgICBzdGF0ZSxcbiAgICBvcHRpb25zLFxuICAgIGlzTXVsdGlTZWxlY3Q6IGZhbHNlLCAvLyBTZWxlY3QgaXMgYWx3YXlzIHNpbmdsZS1jaG9pY2VcbiAgICBvblVwRnJvbUZpcnN0SXRlbSxcbiAgICBvbkRvd25Gcm9tTGFzdEl0ZW0sXG4gICAgb25JbnB1dE1vZGVUb2dnbGUsXG4gICAgaW5wdXRWYWx1ZXMsXG4gICAgaW1hZ2VzU2VsZWN0ZWQsXG4gICAgb25FbnRlckltYWdlU2VsZWN0aW9uOiAoKSA9PiB7XG4gICAgICBpZiAoXG4gICAgICAgIHBhc3RlZENvbnRlbnRzICYmXG4gICAgICAgIE9iamVjdC52YWx1ZXMocGFzdGVkQ29udGVudHMpLnNvbWUoYyA9PiBjLnR5cGUgPT09ICdpbWFnZScpXG4gICAgICApIHtcbiAgICAgICAgY29uc3QgaW1hZ2VDb3VudCA9IGNvdW50KFxuICAgICAgICAgIE9iamVjdC52YWx1ZXMocGFzdGVkQ29udGVudHMpLFxuICAgICAgICAgIGMgPT4gYy50eXBlID09PSAnaW1hZ2UnLFxuICAgICAgICApXG4gICAgICAgIHNldEltYWdlc1NlbGVjdGVkKHRydWUpXG4gICAgICAgIHNldFNlbGVjdGVkSW1hZ2VJbmRleChpbWFnZUNvdW50IC0gMSlcbiAgICAgICAgcmV0dXJuIHRydWVcbiAgICAgIH1cbiAgICAgIHJldHVybiBmYWxzZVxuICAgIH0sXG4gIH0pXG5cbiAgY29uc3Qgc3R5bGVzID0ge1xuICAgIGNvbnRhaW5lcjogKCkgPT4gKHsgZmxleERpcmVjdGlvbjogJ2NvbHVtbicgYXMgY29uc3QgfSksXG4gICAgaGlnaGxpZ2h0ZWRUZXh0OiAoKSA9PiAoeyBib2xkOiB0cnVlIH0pLFxuICB9XG5cbiAgaWYgKGxheW91dCA9PT0gJ2V4cGFuZGVkJykge1xuICAgIGNvbnN0IG1heEluZGV4V2lkdGggPSBzdGF0ZS5vcHRpb25zLmxlbmd0aC50b1N0cmluZygpLmxlbmd0aFxuXG4gICAgcmV0dXJuIChcbiAgICAgIDxCb3ggey4uLnN0eWxlcy5jb250YWluZXIoKX0+XG4gICAgICAgIHtzdGF0ZS52aXNpYmxlT3B0aW9ucy5tYXAoKG9wdGlvbiwgaW5kZXgpID0+IHtcbiAgICAgICAgICBjb25zdCBpc0ZpcnN0VmlzaWJsZU9wdGlvbiA9IG9wdGlvbi5pbmRleCA9PT0gc3RhdGUudmlzaWJsZUZyb21JbmRleFxuICAgICAgICAgIGNvbnN0IGlzTGFzdFZpc2libGVPcHRpb24gPSBvcHRpb24uaW5kZXggPT09IHN0YXRlLnZpc2libGVUb0luZGV4IC0gMVxuICAgICAgICAgIGNvbnN0IGFyZU1vcmVPcHRpb25zQmVsb3cgPSBzdGF0ZS52aXNpYmxlVG9JbmRleCA8IG9wdGlvbnMubGVuZ3RoXG4gICAgICAgICAgY29uc3QgYXJlTW9yZU9wdGlvbnNBYm92ZSA9IHN0YXRlLnZpc2libGVGcm9tSW5kZXggPiAwXG5cbiAgICAgICAgICBjb25zdCBpID0gc3RhdGUudmlzaWJsZUZyb21JbmRleCArIGluZGV4ICsgMVxuXG4gICAgICAgICAgY29uc3QgaXNGb2N1c2VkID0gIWlzRGlzYWJsZWQgJiYgc3RhdGUuZm9jdXNlZFZhbHVlID09PSBvcHRpb24udmFsdWVcbiAgICAgICAgICBjb25zdCBpc1NlbGVjdGVkID0gc3RhdGUudmFsdWUgPT09IG9wdGlvbi52YWx1ZVxuXG4gICAgICAgICAgLy8gSGFuZGxlIGlucHV0IHR5cGUgb3B0aW9uc1xuICAgICAgICAgIGlmIChvcHRpb24udHlwZSA9PT0gJ2lucHV0Jykge1xuICAgICAgICAgICAgY29uc3QgaW5wdXRWYWx1ZSA9IGlucHV0VmFsdWVzLmhhcyhvcHRpb24udmFsdWUpXG4gICAgICAgICAgICAgID8gaW5wdXRWYWx1ZXMuZ2V0KG9wdGlvbi52YWx1ZSkhXG4gICAgICAgICAgICAgIDogb3B0aW9uLmluaXRpYWxWYWx1ZSB8fCAnJ1xuXG4gICAgICAgICAgICByZXR1cm4gKFxuICAgICAgICAgICAgICA8U2VsZWN0SW5wdXRPcHRpb25cbiAgICAgICAgICAgICAgICBrZXk9e1N0cmluZyhvcHRpb24udmFsdWUpfVxuICAgICAgICAgICAgICAgIG9wdGlvbj17b3B0aW9ufVxuICAgICAgICAgICAgICAgIGlzRm9jdXNlZD17aXNGb2N1c2VkfVxuICAgICAgICAgICAgICAgIGlzU2VsZWN0ZWQ9e2lzU2VsZWN0ZWR9XG4gICAgICAgICAgICAgICAgc2hvdWxkU2hvd0Rvd25BcnJvdz17YXJlTW9yZU9wdGlvbnNCZWxvdyAmJiBpc0xhc3RWaXNpYmxlT3B0aW9ufVxuICAgICAgICAgICAgICAgIHNob3VsZFNob3dVcEFycm93PXthcmVNb3JlT3B0aW9uc0Fib3ZlICYmIGlzRmlyc3RWaXNpYmxlT3B0aW9ufVxuICAgICAgICAgICAgICAgIG1heEluZGV4V2lkdGg9e21heEluZGV4V2lkdGh9XG4gICAgICAgICAgICAgICAgaW5kZXg9e2l9XG4gICAgICAgICAgICAgICAgaW5wdXRWYWx1ZT17aW5wdXRWYWx1ZX1cbiAgICAgICAgICAgICAgICBvbklucHV0Q2hhbmdlPXt2YWx1ZSA9PiB7XG4gICAgICAgICAgICAgICAgICBzZXRJbnB1dFZhbHVlcyhwcmV2ID0+IHtcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgbmV4dCA9IG5ldyBNYXAocHJldilcbiAgICAgICAgICAgICAgICAgICAgbmV4dC5zZXQob3B0aW9uLnZhbHVlLCB2YWx1ZSlcbiAgICAgICAgICAgICAgICAgICAgcmV0dXJuIG5leHRcbiAgICAgICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICBvblN1Ym1pdD17KHZhbHVlOiBzdHJpbmcpID0+IHtcbiAgICAgICAgICAgICAgICAgIGNvbnN0IGhhc0ltYWdlQXR0YWNobWVudHMgPVxuICAgICAgICAgICAgICAgICAgICBwYXN0ZWRDb250ZW50cyAmJlxuICAgICAgICAgICAgICAgICAgICBPYmplY3QudmFsdWVzKHBhc3RlZENvbnRlbnRzKS5zb21lKGMgPT4gYy50eXBlID09PSAnaW1hZ2UnKVxuICAgICAgICAgICAgICAgICAgaWYgKFxuICAgICAgICAgICAgICAgICAgICB2YWx1ZS50cmltKCkgfHxcbiAgICAgICAgICAgICAgICAgICAgaGFzSW1hZ2VBdHRhY2htZW50cyB8fFxuICAgICAgICAgICAgICAgICAgICBvcHRpb24uYWxsb3dFbXB0eVN1Ym1pdFRvQ2FuY2VsXG4gICAgICAgICAgICAgICAgICApIHtcbiAgICAgICAgICAgICAgICAgICAgb25DaGFuZ2U/LihvcHRpb24udmFsdWUpXG4gICAgICAgICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgICAgICBvbkNhbmNlbD8uKClcbiAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICB9fVxuICAgICAgICAgICAgICAgIG9uRXhpdD17b25DYW5jZWx9XG4gICAgICAgICAgICAgICAgbGF5b3V0PVwiZXhwYW5kZWRcIlxuICAgICAgICAgICAgICAgIHNob3dMYWJlbD17aW5saW5lRGVzY3JpcHRpb25zfVxuICAgICAgICAgICAgICAgIG9uT3BlbkVkaXRvcj17b25PcGVuRWRpdG9yfVxuICAgICAgICAgICAgICAgIHJlc2V0Q3Vyc29yT25VcGRhdGU9e29wdGlvbi5yZXNldEN1cnNvck9uVXBkYXRlfVxuICAgICAgICAgICAgICAgIG9uSW1hZ2VQYXN0ZT17b25JbWFnZVBhc3RlfVxuICAgICAgICAgICAgICAgIHBhc3RlZENvbnRlbnRzPXtwYXN0ZWRDb250ZW50c31cbiAgICAgICAgICAgICAgICBvblJlbW92ZUltYWdlPXtvblJlbW92ZUltYWdlfVxuICAgICAgICAgICAgICAgIGltYWdlc1NlbGVjdGVkPXtpbWFnZXNTZWxlY3RlZH1cbiAgICAgICAgICAgICAgICBzZWxlY3RlZEltYWdlSW5kZXg9e3NlbGVjdGVkSW1hZ2VJbmRleH1cbiAgICAgICAgICAgICAgICBvbkltYWdlc1NlbGVjdGVkQ2hhbmdlPXtzZXRJbWFnZXNTZWxlY3RlZH1cbiAgICAgICAgICAgICAgICBvblNlbGVjdGVkSW1hZ2VJbmRleENoYW5nZT17c2V0U2VsZWN0ZWRJbWFnZUluZGV4fVxuICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgKVxuICAgICAgICAgIH1cblxuICAgICAgICAgIC8vIEhhbmRsZSB0ZXh0IHR5cGUgb3B0aW9uc1xuICAgICAgICAgIGxldCBsYWJlbDogUmVhY3ROb2RlID0gb3B0aW9uLmxhYmVsXG5cbiAgICAgICAgICAvLyBPbmx5IGFwcGx5IGhpZ2hsaWdodCB3aGVuIGxhYmVsIGlzIGEgc3RyaW5nXG4gICAgICAgICAgaWYgKFxuICAgICAgICAgICAgdHlwZW9mIG9wdGlvbi5sYWJlbCA9PT0gJ3N0cmluZycgJiZcbiAgICAgICAgICAgIGhpZ2hsaWdodFRleHQgJiZcbiAgICAgICAgICAgIG9wdGlvbi5sYWJlbC5pbmNsdWRlcyhoaWdobGlnaHRUZXh0KVxuICAgICAgICAgICkge1xuICAgICAgICAgICAgY29uc3QgbGFiZWxUZXh0ID0gb3B0aW9uLmxhYmVsXG4gICAgICAgICAgICBjb25zdCBpbmRleCA9IGxhYmVsVGV4dC5pbmRleE9mKGhpZ2hsaWdodFRleHQpXG5cbiAgICAgICAgICAgIGxhYmVsID0gKFxuICAgICAgICAgICAgICA8PlxuICAgICAgICAgICAgICAgIHtsYWJlbFRleHQuc2xpY2UoMCwgaW5kZXgpfVxuICAgICAgICAgICAgICAgIDxUZXh0IHsuLi5zdHlsZXMuaGlnaGxpZ2h0ZWRUZXh0KCl9PntoaWdobGlnaHRUZXh0fTwvVGV4dD5cbiAgICAgICAgICAgICAgICB7bGFiZWxUZXh0LnNsaWNlKGluZGV4ICsgaGlnaGxpZ2h0VGV4dC5sZW5ndGgpfVxuICAgICAgICAgICAgICA8Lz5cbiAgICAgICAgICAgIClcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBjb25zdCBpc09wdGlvbkRpc2FibGVkID0gb3B0aW9uLmRpc2FibGVkID09PSB0cnVlXG4gICAgICAgICAgY29uc3Qgb3B0aW9uQ29sb3IgPSBpc09wdGlvbkRpc2FibGVkXG4gICAgICAgICAgICA/IHVuZGVmaW5lZFxuICAgICAgICAgICAgOiBpc1NlbGVjdGVkXG4gICAgICAgICAgICAgID8gJ3N1Y2Nlc3MnXG4gICAgICAgICAgICAgIDogaXNGb2N1c2VkXG4gICAgICAgICAgICAgICAgPyAnc3VnZ2VzdGlvbidcbiAgICAgICAgICAgICAgICA6IHVuZGVmaW5lZFxuXG4gICAgICAgICAgcmV0dXJuIChcbiAgICAgICAgICAgIDxCb3hcbiAgICAgICAgICAgICAga2V5PXtTdHJpbmcob3B0aW9uLnZhbHVlKX1cbiAgICAgICAgICAgICAgZmxleERpcmVjdGlvbj1cImNvbHVtblwiXG4gICAgICAgICAgICAgIGZsZXhTaHJpbms9ezB9XG4gICAgICAgICAgICA+XG4gICAgICAgICAgICAgIDxTZWxlY3RPcHRpb25cbiAgICAgICAgICAgICAgICBpc0ZvY3VzZWQ9e2lzRm9jdXNlZH1cbiAgICAgICAgICAgICAgICBpc1NlbGVjdGVkPXtpc1NlbGVjdGVkfVxuICAgICAgICAgICAgICAgIHNob3VsZFNob3dEb3duQXJyb3c9e2FyZU1vcmVPcHRpb25zQmVsb3cgJiYgaXNMYXN0VmlzaWJsZU9wdGlvbn1cbiAgICAgICAgICAgICAgICBzaG91bGRTaG93VXBBcnJvdz17YXJlTW9yZU9wdGlvbnNBYm92ZSAmJiBpc0ZpcnN0VmlzaWJsZU9wdGlvbn1cbiAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgIDxUZXh0IGRpbUNvbG9yPXtpc09wdGlvbkRpc2FibGVkfSBjb2xvcj17b3B0aW9uQ29sb3J9PlxuICAgICAgICAgICAgICAgICAge2xhYmVsfVxuICAgICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgPC9TZWxlY3RPcHRpb24+XG4gICAgICAgICAgICAgIHtvcHRpb24uZGVzY3JpcHRpb24gJiYgKFxuICAgICAgICAgICAgICAgIDxCb3ggcGFkZGluZ0xlZnQ9ezJ9PlxuICAgICAgICAgICAgICAgICAgPFRleHRcbiAgICAgICAgICAgICAgICAgICAgZGltQ29sb3I9e1xuICAgICAgICAgICAgICAgICAgICAgIGlzT3B0aW9uRGlzYWJsZWQgfHwgb3B0aW9uLmRpbURlc2NyaXB0aW9uICE9PSBmYWxzZVxuICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgIGNvbG9yPXtvcHRpb25Db2xvcn1cbiAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAgPEFuc2k+e29wdGlvbi5kZXNjcmlwdGlvbn08L0Fuc2k+XG4gICAgICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICAgICAgPC9Cb3g+XG4gICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgIDxUZXh0PiA8L1RleHQ+XG4gICAgICAgICAgICA8L0JveD5cbiAgICAgICAgICApXG4gICAgICAgIH0pfVxuICAgICAgPC9Cb3g+XG4gICAgKVxuICB9XG5cbiAgaWYgKGxheW91dCA9PT0gJ2NvbXBhY3QtdmVydGljYWwnKSB7XG4gICAgY29uc3QgbWF4SW5kZXhXaWR0aCA9IGhpZGVJbmRleGVzXG4gICAgICA/IDBcbiAgICAgIDogc3RhdGUub3B0aW9ucy5sZW5ndGgudG9TdHJpbmcoKS5sZW5ndGhcblxuICAgIHJldHVybiAoXG4gICAgICA8Qm94IHsuLi5zdHlsZXMuY29udGFpbmVyKCl9PlxuICAgICAgICB7c3RhdGUudmlzaWJsZU9wdGlvbnMubWFwKChvcHRpb24sIGluZGV4KSA9PiB7XG4gICAgICAgICAgY29uc3QgaXNGaXJzdFZpc2libGVPcHRpb24gPSBvcHRpb24uaW5kZXggPT09IHN0YXRlLnZpc2libGVGcm9tSW5kZXhcbiAgICAgICAgICBjb25zdCBpc0xhc3RWaXNpYmxlT3B0aW9uID0gb3B0aW9uLmluZGV4ID09PSBzdGF0ZS52aXNpYmxlVG9JbmRleCAtIDFcbiAgICAgICAgICBjb25zdCBhcmVNb3JlT3B0aW9uc0JlbG93ID0gc3RhdGUudmlzaWJsZVRvSW5kZXggPCBvcHRpb25zLmxlbmd0aFxuICAgICAgICAgIGNvbnN0IGFyZU1vcmVPcHRpb25zQWJvdmUgPSBzdGF0ZS52aXNpYmxlRnJvbUluZGV4ID4gMFxuXG4gICAgICAgICAgY29uc3QgaSA9IHN0YXRlLnZpc2libGVGcm9tSW5kZXggKyBpbmRleCArIDFcblxuICAgICAgICAgIGNvbnN0IGlzRm9jdXNlZCA9ICFpc0Rpc2FibGVkICYmIHN0YXRlLmZvY3VzZWRWYWx1ZSA9PT0gb3B0aW9uLnZhbHVlXG4gICAgICAgICAgY29uc3QgaXNTZWxlY3RlZCA9IHN0YXRlLnZhbHVlID09PSBvcHRpb24udmFsdWVcblxuICAgICAgICAgIC8vIEhhbmRsZSBpbnB1dCB0eXBlIG9wdGlvbnNcbiAgICAgICAgICBpZiAob3B0aW9uLnR5cGUgPT09ICdpbnB1dCcpIHtcbiAgICAgICAgICAgIGNvbnN0IGlucHV0VmFsdWUgPSBpbnB1dFZhbHVlcy5oYXMob3B0aW9uLnZhbHVlKVxuICAgICAgICAgICAgICA/IGlucHV0VmFsdWVzLmdldChvcHRpb24udmFsdWUpIVxuICAgICAgICAgICAgICA6IG9wdGlvbi5pbml0aWFsVmFsdWUgfHwgJydcblxuICAgICAgICAgICAgcmV0dXJuIChcbiAgICAgICAgICAgICAgPFNlbGVjdElucHV0T3B0aW9uXG4gICAgICAgICAgICAgICAga2V5PXtTdHJpbmcob3B0aW9uLnZhbHVlKX1cbiAgICAgICAgICAgICAgICBvcHRpb249e29wdGlvbn1cbiAgICAgICAgICAgICAgICBpc0ZvY3VzZWQ9e2lzRm9jdXNlZH1cbiAgICAgICAgICAgICAgICBpc1NlbGVjdGVkPXtpc1NlbGVjdGVkfVxuICAgICAgICAgICAgICAgIHNob3VsZFNob3dEb3duQXJyb3c9e2FyZU1vcmVPcHRpb25zQmVsb3cgJiYgaXNMYXN0VmlzaWJsZU9wdGlvbn1cbiAgICAgICAgICAgICAgICBzaG91bGRTaG93VXBBcnJvdz17YXJlTW9yZU9wdGlvbnNBYm92ZSAmJiBpc0ZpcnN0VmlzaWJsZU9wdGlvbn1cbiAgICAgICAgICAgICAgICBtYXhJbmRleFdpZHRoPXttYXhJbmRleFdpZHRofVxuICAgICAgICAgICAgICAgIGluZGV4PXtpfVxuICAgICAgICAgICAgICAgIGlucHV0VmFsdWU9e2lucHV0VmFsdWV9XG4gICAgICAgICAgICAgICAgb25JbnB1dENoYW5nZT17dmFsdWUgPT4ge1xuICAgICAgICAgICAgICAgICAgc2V0SW5wdXRWYWx1ZXMocHJldiA9PiB7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IG5leHQgPSBuZXcgTWFwKHByZXYpXG4gICAgICAgICAgICAgICAgICAgIG5leHQuc2V0KG9wdGlvbi52YWx1ZSwgdmFsdWUpXG4gICAgICAgICAgICAgICAgICAgIHJldHVybiBuZXh0XG4gICAgICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgICAgb25TdWJtaXQ9eyh2YWx1ZTogc3RyaW5nKSA9PiB7XG4gICAgICAgICAgICAgICAgICBjb25zdCBoYXNJbWFnZUF0dGFjaG1lbnRzID1cbiAgICAgICAgICAgICAgICAgICAgcGFzdGVkQ29udGVudHMgJiZcbiAgICAgICAgICAgICAgICAgICAgT2JqZWN0LnZhbHVlcyhwYXN0ZWRDb250ZW50cykuc29tZShjID0+IGMudHlwZSA9PT0gJ2ltYWdlJylcbiAgICAgICAgICAgICAgICAgIGlmIChcbiAgICAgICAgICAgICAgICAgICAgdmFsdWUudHJpbSgpIHx8XG4gICAgICAgICAgICAgICAgICAgIGhhc0ltYWdlQXR0YWNobWVudHMgfHxcbiAgICAgICAgICAgICAgICAgICAgb3B0aW9uLmFsbG93RW1wdHlTdWJtaXRUb0NhbmNlbFxuICAgICAgICAgICAgICAgICAgKSB7XG4gICAgICAgICAgICAgICAgICAgIG9uQ2hhbmdlPy4ob3B0aW9uLnZhbHVlKVxuICAgICAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAgICAgb25DYW5jZWw/LigpXG4gICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICBvbkV4aXQ9e29uQ2FuY2VsfVxuICAgICAgICAgICAgICAgIGxheW91dD1cImNvbXBhY3RcIlxuICAgICAgICAgICAgICAgIHNob3dMYWJlbD17aW5saW5lRGVzY3JpcHRpb25zfVxuICAgICAgICAgICAgICAgIG9uT3BlbkVkaXRvcj17b25PcGVuRWRpdG9yfVxuICAgICAgICAgICAgICAgIHJlc2V0Q3Vyc29yT25VcGRhdGU9e29wdGlvbi5yZXNldEN1cnNvck9uVXBkYXRlfVxuICAgICAgICAgICAgICAgIG9uSW1hZ2VQYXN0ZT17b25JbWFnZVBhc3RlfVxuICAgICAgICAgICAgICAgIHBhc3RlZENvbnRlbnRzPXtwYXN0ZWRDb250ZW50c31cbiAgICAgICAgICAgICAgICBvblJlbW92ZUltYWdlPXtvblJlbW92ZUltYWdlfVxuICAgICAgICAgICAgICAgIGltYWdlc1NlbGVjdGVkPXtpbWFnZXNTZWxlY3RlZH1cbiAgICAgICAgICAgICAgICBzZWxlY3RlZEltYWdlSW5kZXg9e3NlbGVjdGVkSW1hZ2VJbmRleH1cbiAgICAgICAgICAgICAgICBvbkltYWdlc1NlbGVjdGVkQ2hhbmdlPXtzZXRJbWFnZXNTZWxlY3RlZH1cbiAgICAgICAgICAgICAgICBvblNlbGVjdGVkSW1hZ2VJbmRleENoYW5nZT17c2V0U2VsZWN0ZWRJbWFnZUluZGV4fVxuICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgKVxuICAgICAgICAgIH1cblxuICAgICAgICAgIC8vIEhhbmRsZSB0ZXh0IHR5cGUgb3B0aW9uc1xuICAgICAgICAgIGxldCBsYWJlbDogUmVhY3ROb2RlID0gb3B0aW9uLmxhYmVsXG5cbiAgICAgICAgICAvLyBPbmx5IGFwcGx5IGhpZ2hsaWdodCB3aGVuIGxhYmVsIGlzIGEgc3RyaW5nXG4gICAgICAgICAgaWYgKFxuICAgICAgICAgICAgdHlwZW9mIG9wdGlvbi5sYWJlbCA9PT0gJ3N0cmluZycgJiZcbiAgICAgICAgICAgIGhpZ2hsaWdodFRleHQgJiZcbiAgICAgICAgICAgIG9wdGlvbi5sYWJlbC5pbmNsdWRlcyhoaWdobGlnaHRUZXh0KVxuICAgICAgICAgICkge1xuICAgICAgICAgICAgY29uc3QgbGFiZWxUZXh0ID0gb3B0aW9uLmxhYmVsXG4gICAgICAgICAgICBjb25zdCBpbmRleCA9IGxhYmVsVGV4dC5pbmRleE9mKGhpZ2hsaWdodFRleHQpXG5cbiAgICAgICAgICAgIGxhYmVsID0gKFxuICAgICAgICAgICAgICA8PlxuICAgICAgICAgICAgICAgIHtsYWJlbFRleHQuc2xpY2UoMCwgaW5kZXgpfVxuICAgICAgICAgICAgICAgIDxUZXh0IHsuLi5zdHlsZXMuaGlnaGxpZ2h0ZWRUZXh0KCl9PntoaWdobGlnaHRUZXh0fTwvVGV4dD5cbiAgICAgICAgICAgICAgICB7bGFiZWxUZXh0LnNsaWNlKGluZGV4ICsgaGlnaGxpZ2h0VGV4dC5sZW5ndGgpfVxuICAgICAgICAgICAgICA8Lz5cbiAgICAgICAgICAgIClcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBjb25zdCBpc09wdGlvbkRpc2FibGVkID0gb3B0aW9uLmRpc2FibGVkID09PSB0cnVlXG5cbiAgICAgICAgICByZXR1cm4gKFxuICAgICAgICAgICAgPEJveFxuICAgICAgICAgICAgICBrZXk9e1N0cmluZyhvcHRpb24udmFsdWUpfVxuICAgICAgICAgICAgICBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCJcbiAgICAgICAgICAgICAgZmxleFNocmluaz17MH1cbiAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgPFNlbGVjdE9wdGlvblxuICAgICAgICAgICAgICAgIGlzRm9jdXNlZD17aXNGb2N1c2VkfVxuICAgICAgICAgICAgICAgIGlzU2VsZWN0ZWQ9e2lzU2VsZWN0ZWR9XG4gICAgICAgICAgICAgICAgc2hvdWxkU2hvd0Rvd25BcnJvdz17YXJlTW9yZU9wdGlvbnNCZWxvdyAmJiBpc0xhc3RWaXNpYmxlT3B0aW9ufVxuICAgICAgICAgICAgICAgIHNob3VsZFNob3dVcEFycm93PXthcmVNb3JlT3B0aW9uc0Fib3ZlICYmIGlzRmlyc3RWaXNpYmxlT3B0aW9ufVxuICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgPD5cbiAgICAgICAgICAgICAgICAgIHshaGlkZUluZGV4ZXMgJiYgKFxuICAgICAgICAgICAgICAgICAgICA8VGV4dCBkaW1Db2xvcj57YCR7aX0uYC5wYWRFbmQobWF4SW5kZXhXaWR0aCArIDEpfTwvVGV4dD5cbiAgICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgICAgICA8VGV4dFxuICAgICAgICAgICAgICAgICAgICBkaW1Db2xvcj17aXNPcHRpb25EaXNhYmxlZH1cbiAgICAgICAgICAgICAgICAgICAgY29sb3I9e1xuICAgICAgICAgICAgICAgICAgICAgIGlzT3B0aW9uRGlzYWJsZWRcbiAgICAgICAgICAgICAgICAgICAgICAgID8gdW5kZWZpbmVkXG4gICAgICAgICAgICAgICAgICAgICAgICA6IGlzU2VsZWN0ZWRcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPyAnc3VjY2VzcydcbiAgICAgICAgICAgICAgICAgICAgICAgICAgOiBpc0ZvY3VzZWRcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICA/ICdzdWdnZXN0aW9uJ1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIDogdW5kZWZpbmVkXG4gICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAge2xhYmVsfVxuICAgICAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgICAgIDwvPlxuICAgICAgICAgICAgICA8L1NlbGVjdE9wdGlvbj5cbiAgICAgICAgICAgICAge29wdGlvbi5kZXNjcmlwdGlvbiAmJiAoXG4gICAgICAgICAgICAgICAgPEJveCBwYWRkaW5nTGVmdD17aGlkZUluZGV4ZXMgPyA0IDogbWF4SW5kZXhXaWR0aCArIDR9PlxuICAgICAgICAgICAgICAgICAgPFRleHRcbiAgICAgICAgICAgICAgICAgICAgZGltQ29sb3I9e1xuICAgICAgICAgICAgICAgICAgICAgIGlzT3B0aW9uRGlzYWJsZWQgfHwgb3B0aW9uLmRpbURlc2NyaXB0aW9uICE9PSBmYWxzZVxuICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgIGNvbG9yPXtcbiAgICAgICAgICAgICAgICAgICAgICBpc09wdGlvbkRpc2FibGVkXG4gICAgICAgICAgICAgICAgICAgICAgICA/IHVuZGVmaW5lZFxuICAgICAgICAgICAgICAgICAgICAgICAgOiBpc1NlbGVjdGVkXG4gICAgICAgICAgICAgICAgICAgICAgICAgID8gJ3N1Y2Nlc3MnXG4gICAgICAgICAgICAgICAgICAgICAgICAgIDogaXNGb2N1c2VkXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgPyAnc3VnZ2VzdGlvbidcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICA6IHVuZGVmaW5lZFxuICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgIDxBbnNpPntvcHRpb24uZGVzY3JpcHRpb259PC9BbnNpPlxuICAgICAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgICAgIDwvQm94PlxuICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgPC9Cb3g+XG4gICAgICAgICAgKVxuICAgICAgICB9KX1cbiAgICAgIDwvQm94PlxuICAgIClcbiAgfVxuXG4gIGNvbnN0IG1heEluZGV4V2lkdGggPSBoaWRlSW5kZXhlcyA/IDAgOiBzdGF0ZS5vcHRpb25zLmxlbmd0aC50b1N0cmluZygpLmxlbmd0aFxuXG4gIC8vIENoZWNrIGlmIGFueSB2aXNpYmxlIG9wdGlvbnMgaGF2ZSBkZXNjcmlwdGlvbnMgKGZvciB0d28tY29sdW1uIGxheW91dClcbiAgLy8gQWxzbyBjaGVjayB0aGF0IHRoZXJlIGFyZSBOTyBpbnB1dCBvcHRpb25zLCBzaW5jZSB0aGV5J3JlIG5vdCBzdXBwb3J0ZWQgaW4gdHdvLWNvbHVtbiBsYXlvdXRcbiAgLy8gU2tpcCB0d28tY29sdW1uIGxheW91dCB3aGVuIGlubGluZURlc2NyaXB0aW9ucyBpcyBlbmFibGVkXG4gIGNvbnN0IGhhc0lucHV0T3B0aW9ucyA9IHN0YXRlLnZpc2libGVPcHRpb25zLnNvbWUob3B0ID0+IG9wdC50eXBlID09PSAnaW5wdXQnKVxuICBjb25zdCBoYXNEZXNjcmlwdGlvbnMgPVxuICAgICFpbmxpbmVEZXNjcmlwdGlvbnMgJiZcbiAgICAhaGFzSW5wdXRPcHRpb25zICYmXG4gICAgc3RhdGUudmlzaWJsZU9wdGlvbnMuc29tZShvcHQgPT4gb3B0LmRlc2NyaXB0aW9uKVxuXG4gIC8vIFByZS1jb21wdXRlIG9wdGlvbiBkYXRhIGZvciB0d28tY29sdW1uIGxheW91dFxuICBjb25zdCBvcHRpb25EYXRhID0gc3RhdGUudmlzaWJsZU9wdGlvbnMubWFwKChvcHRpb24sIGluZGV4KSA9PiB7XG4gICAgY29uc3QgaXNGaXJzdFZpc2libGVPcHRpb24gPSBvcHRpb24uaW5kZXggPT09IHN0YXRlLnZpc2libGVGcm9tSW5kZXhcbiAgICBjb25zdCBpc0xhc3RWaXNpYmxlT3B0aW9uID0gb3B0aW9uLmluZGV4ID09PSBzdGF0ZS52aXNpYmxlVG9JbmRleCAtIDFcbiAgICBjb25zdCBhcmVNb3JlT3B0aW9uc0JlbG93ID0gc3RhdGUudmlzaWJsZVRvSW5kZXggPCBvcHRpb25zLmxlbmd0aFxuICAgIGNvbnN0IGFyZU1vcmVPcHRpb25zQWJvdmUgPSBzdGF0ZS52aXNpYmxlRnJvbUluZGV4ID4gMFxuICAgIGNvbnN0IGkgPSBzdGF0ZS52aXNpYmxlRnJvbUluZGV4ICsgaW5kZXggKyAxXG4gICAgY29uc3QgaXNGb2N1c2VkID0gIWlzRGlzYWJsZWQgJiYgc3RhdGUuZm9jdXNlZFZhbHVlID09PSBvcHRpb24udmFsdWVcbiAgICBjb25zdCBpc1NlbGVjdGVkID0gc3RhdGUudmFsdWUgPT09IG9wdGlvbi52YWx1ZVxuICAgIGNvbnN0IGlzT3B0aW9uRGlzYWJsZWQgPSBvcHRpb24uZGlzYWJsZWQgPT09IHRydWVcblxuICAgIGxldCBsYWJlbDogUmVhY3ROb2RlID0gb3B0aW9uLmxhYmVsXG4gICAgaWYgKFxuICAgICAgdHlwZW9mIG9wdGlvbi5sYWJlbCA9PT0gJ3N0cmluZycgJiZcbiAgICAgIGhpZ2hsaWdodFRleHQgJiZcbiAgICAgIG9wdGlvbi5sYWJlbC5pbmNsdWRlcyhoaWdobGlnaHRUZXh0KVxuICAgICkge1xuICAgICAgY29uc3QgbGFiZWxUZXh0ID0gb3B0aW9uLmxhYmVsXG4gICAgICBjb25zdCBpZHggPSBsYWJlbFRleHQuaW5kZXhPZihoaWdobGlnaHRUZXh0KVxuICAgICAgbGFiZWwgPSAoXG4gICAgICAgIDw+XG4gICAgICAgICAge2xhYmVsVGV4dC5zbGljZSgwLCBpZHgpfVxuICAgICAgICAgIDxUZXh0IHsuLi5zdHlsZXMuaGlnaGxpZ2h0ZWRUZXh0KCl9PntoaWdobGlnaHRUZXh0fTwvVGV4dD5cbiAgICAgICAgICB7bGFiZWxUZXh0LnNsaWNlKGlkeCArIGhpZ2hsaWdodFRleHQubGVuZ3RoKX1cbiAgICAgICAgPC8+XG4gICAgICApXG4gICAgfVxuXG4gICAgcmV0dXJuIHtcbiAgICAgIG9wdGlvbixcbiAgICAgIGluZGV4OiBpLFxuICAgICAgbGFiZWwsXG4gICAgICBpc0ZvY3VzZWQsXG4gICAgICBpc1NlbGVjdGVkLFxuICAgICAgaXNPcHRpb25EaXNhYmxlZCxcbiAgICAgIHNob3VsZFNob3dEb3duQXJyb3c6IGFyZU1vcmVPcHRpb25zQmVsb3cgJiYgaXNMYXN0VmlzaWJsZU9wdGlvbixcbiAgICAgIHNob3VsZFNob3dVcEFycm93OiBhcmVNb3JlT3B0aW9uc0Fib3ZlICYmIGlzRmlyc3RWaXNpYmxlT3B0aW9uLFxuICAgIH1cbiAgfSlcblxuICAvLyBDYWxjdWxhdGUgbWF4IGxhYmVsIHdpZHRoIGZvciBhbGlnbm1lbnQgd2hlbiBkZXNjcmlwdGlvbnMgZXhpc3RcbiAgaWYgKGhhc0Rlc2NyaXB0aW9ucykge1xuICAgIGNvbnN0IG1heExhYmVsV2lkdGggPSBNYXRoLm1heChcbiAgICAgIC4uLm9wdGlvbkRhdGEubWFwKGRhdGEgPT4ge1xuICAgICAgICBpZiAoZGF0YS5vcHRpb24udHlwZSA9PT0gJ2lucHV0JykgcmV0dXJuIDBcbiAgICAgICAgY29uc3QgbGFiZWxUZXh0ID0gZ2V0VGV4dENvbnRlbnQoZGF0YS5vcHRpb24ubGFiZWwpXG4gICAgICAgIC8vIFdpZHRoOiBpbmRpY2F0b3IgKDEpICsgc3BhY2UgKDEpICsgaW5kZXggKyBsYWJlbCArIHNwYWNlICsgY2hlY2ttYXJrICgxKVxuICAgICAgICBjb25zdCBpbmRleFdpZHRoID0gaGlkZUluZGV4ZXMgPyAwIDogbWF4SW5kZXhXaWR0aCArIDJcbiAgICAgICAgY29uc3QgY2hlY2ttYXJrV2lkdGggPSBkYXRhLmlzU2VsZWN0ZWQgPyAyIDogMFxuICAgICAgICByZXR1cm4gMiArIGluZGV4V2lkdGggKyBzdHJpbmdXaWR0aChsYWJlbFRleHQpICsgY2hlY2ttYXJrV2lkdGhcbiAgICAgIH0pLFxuICAgIClcblxuICAgIHJldHVybiAoXG4gICAgICA8Qm94IHsuLi5zdHlsZXMuY29udGFpbmVyKCl9PlxuICAgICAgICB7b3B0aW9uRGF0YS5tYXAoZGF0YSA9PiB7XG4gICAgICAgICAgaWYgKGRhdGEub3B0aW9uLnR5cGUgPT09ICdpbnB1dCcpIHtcbiAgICAgICAgICAgIC8vIElucHV0IG9wdGlvbnMgbm90IHN1cHBvcnRlZCBpbiB0d28tY29sdW1uIGxheW91dFxuICAgICAgICAgICAgcmV0dXJuIG51bGxcbiAgICAgICAgICB9XG4gICAgICAgICAgY29uc3QgbGFiZWxUZXh0ID0gZ2V0VGV4dENvbnRlbnQoZGF0YS5vcHRpb24ubGFiZWwpXG4gICAgICAgICAgY29uc3QgaW5kZXhXaWR0aCA9IGhpZGVJbmRleGVzID8gMCA6IG1heEluZGV4V2lkdGggKyAyXG4gICAgICAgICAgY29uc3QgY2hlY2ttYXJrV2lkdGggPSBkYXRhLmlzU2VsZWN0ZWQgPyAyIDogMFxuICAgICAgICAgIGNvbnN0IGN1cnJlbnRMYWJlbFdpZHRoID1cbiAgICAgICAgICAgIDIgKyBpbmRleFdpZHRoICsgc3RyaW5nV2lkdGgobGFiZWxUZXh0KSArIGNoZWNrbWFya1dpZHRoXG4gICAgICAgICAgY29uc3QgcGFkZGluZyA9IG1heExhYmVsV2lkdGggLSBjdXJyZW50TGFiZWxXaWR0aFxuXG4gICAgICAgICAgcmV0dXJuIChcbiAgICAgICAgICAgIDxUd29Db2x1bW5Sb3dcbiAgICAgICAgICAgICAga2V5PXtTdHJpbmcoZGF0YS5vcHRpb24udmFsdWUpfVxuICAgICAgICAgICAgICBpc0ZvY3VzZWQ9e2RhdGEuaXNGb2N1c2VkfVxuICAgICAgICAgICAgPlxuICAgICAgICAgICAgICB7LyogTGFiZWwgcGFydCAtIG5vIGdhcCwgaGFuZGxlIHNwYWNpbmcgZXhwbGljaXRseSAqL31cbiAgICAgICAgICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwicm93XCIgZmxleFNocmluaz17MH0+XG4gICAgICAgICAgICAgICAge2RhdGEuaXNGb2N1c2VkID8gKFxuICAgICAgICAgICAgICAgICAgPFRleHQgY29sb3I9XCJzdWdnZXN0aW9uXCI+e2ZpZ3VyZXMucG9pbnRlcn08L1RleHQ+XG4gICAgICAgICAgICAgICAgKSA6IGRhdGEuc2hvdWxkU2hvd0Rvd25BcnJvdyA/IChcbiAgICAgICAgICAgICAgICAgIDxUZXh0IGRpbUNvbG9yPntmaWd1cmVzLmFycm93RG93bn08L1RleHQ+XG4gICAgICAgICAgICAgICAgKSA6IGRhdGEuc2hvdWxkU2hvd1VwQXJyb3cgPyAoXG4gICAgICAgICAgICAgICAgICA8VGV4dCBkaW1Db2xvcj57ZmlndXJlcy5hcnJvd1VwfTwvVGV4dD5cbiAgICAgICAgICAgICAgICApIDogKFxuICAgICAgICAgICAgICAgICAgPFRleHQ+IDwvVGV4dD5cbiAgICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgICAgIDxUZXh0PiA8L1RleHQ+XG4gICAgICAgICAgICAgICAgPFRleHRcbiAgICAgICAgICAgICAgICAgIGRpbUNvbG9yPXtkYXRhLmlzT3B0aW9uRGlzYWJsZWR9XG4gICAgICAgICAgICAgICAgICBjb2xvcj17XG4gICAgICAgICAgICAgICAgICAgIGRhdGEuaXNPcHRpb25EaXNhYmxlZFxuICAgICAgICAgICAgICAgICAgICAgID8gdW5kZWZpbmVkXG4gICAgICAgICAgICAgICAgICAgICAgOiBkYXRhLmlzU2VsZWN0ZWRcbiAgICAgICAgICAgICAgICAgICAgICAgID8gJ3N1Y2Nlc3MnXG4gICAgICAgICAgICAgICAgICAgICAgICA6IGRhdGEuaXNGb2N1c2VkXG4gICAgICAgICAgICAgICAgICAgICAgICAgID8gJ3N1Z2dlc3Rpb24nXG4gICAgICAgICAgICAgICAgICAgICAgICAgIDogdW5kZWZpbmVkXG4gICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgeyFoaWRlSW5kZXhlcyAmJiAoXG4gICAgICAgICAgICAgICAgICAgIDxUZXh0IGRpbUNvbG9yPlxuICAgICAgICAgICAgICAgICAgICAgIHtgJHtkYXRhLmluZGV4fS5gLnBhZEVuZChtYXhJbmRleFdpZHRoICsgMil9XG4gICAgICAgICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgICAgICB7ZGF0YS5sYWJlbH1cbiAgICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICAgICAge2RhdGEuaXNTZWxlY3RlZCAmJiAoXG4gICAgICAgICAgICAgICAgICA8VGV4dCBjb2xvcj1cInN1Y2Nlc3NcIj4ge2ZpZ3VyZXMudGlja308L1RleHQ+XG4gICAgICAgICAgICAgICAgKX1cbiAgICAgICAgICAgICAgICB7LyogUGFkZGluZyB0byBhbGlnbiBkZXNjcmlwdGlvbnMgKi99XG4gICAgICAgICAgICAgICAge3BhZGRpbmcgPiAwICYmIDxUZXh0PnsnICcucmVwZWF0KHBhZGRpbmcpfTwvVGV4dD59XG4gICAgICAgICAgICAgIDwvQm94PlxuICAgICAgICAgICAgICB7LyogRGVzY3JpcHRpb24gcGFydCAqL31cbiAgICAgICAgICAgICAgPEJveCBmbGV4R3Jvdz17MX0gbWFyZ2luTGVmdD17Mn0+XG4gICAgICAgICAgICAgICAgPFRleHRcbiAgICAgICAgICAgICAgICAgIHdyYXA9XCJ3cmFwXCJcbiAgICAgICAgICAgICAgICAgIGRpbUNvbG9yPXtcbiAgICAgICAgICAgICAgICAgICAgZGF0YS5pc09wdGlvbkRpc2FibGVkIHx8XG4gICAgICAgICAgICAgICAgICAgIGRhdGEub3B0aW9uLmRpbURlc2NyaXB0aW9uICE9PSBmYWxzZVxuICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgY29sb3I9e1xuICAgICAgICAgICAgICAgICAgICBkYXRhLmlzT3B0aW9uRGlzYWJsZWRcbiAgICAgICAgICAgICAgICAgICAgICA/IHVuZGVmaW5lZFxuICAgICAgICAgICAgICAgICAgICAgIDogZGF0YS5pc1NlbGVjdGVkXG4gICAgICAgICAgICAgICAgICAgICAgICA/ICdzdWNjZXNzJ1xuICAgICAgICAgICAgICAgICAgICAgICAgOiBkYXRhLmlzRm9jdXNlZFxuICAgICAgICAgICAgICAgICAgICAgICAgICA/ICdzdWdnZXN0aW9uJ1xuICAgICAgICAgICAgICAgICAgICAgICAgICA6IHVuZGVmaW5lZFxuICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgIDxBbnNpPntkYXRhLm9wdGlvbi5kZXNjcmlwdGlvbiB8fCAnICd9PC9BbnNpPlxuICAgICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgPC9Cb3g+XG4gICAgICAgICAgICA8L1R3b0NvbHVtblJvdz5cbiAgICAgICAgICApXG4gICAgICAgIH0pfVxuICAgICAgPC9Cb3g+XG4gICAgKVxuICB9XG5cbiAgcmV0dXJuIChcbiAgICA8Qm94IHsuLi5zdHlsZXMuY29udGFpbmVyKCl9PlxuICAgICAge3N0YXRlLnZpc2libGVPcHRpb25zLm1hcCgob3B0aW9uLCBpbmRleCkgPT4ge1xuICAgICAgICAvLyBIYW5kbGUgaW5wdXQgdHlwZSBvcHRpb25zXG4gICAgICAgIGlmIChvcHRpb24udHlwZSA9PT0gJ2lucHV0Jykge1xuICAgICAgICAgIGNvbnN0IGlucHV0VmFsdWUgPSBpbnB1dFZhbHVlcy5oYXMob3B0aW9uLnZhbHVlKVxuICAgICAgICAgICAgPyBpbnB1dFZhbHVlcy5nZXQob3B0aW9uLnZhbHVlKSFcbiAgICAgICAgICAgIDogb3B0aW9uLmluaXRpYWxWYWx1ZSB8fCAnJ1xuXG4gICAgICAgICAgY29uc3QgaXNGaXJzdFZpc2libGVPcHRpb24gPSBvcHRpb24uaW5kZXggPT09IHN0YXRlLnZpc2libGVGcm9tSW5kZXhcbiAgICAgICAgICBjb25zdCBpc0xhc3RWaXNpYmxlT3B0aW9uID0gb3B0aW9uLmluZGV4ID09PSBzdGF0ZS52aXNpYmxlVG9JbmRleCAtIDFcbiAgICAgICAgICBjb25zdCBhcmVNb3JlT3B0aW9uc0JlbG93ID0gc3RhdGUudmlzaWJsZVRvSW5kZXggPCBvcHRpb25zLmxlbmd0aFxuICAgICAgICAgIGNvbnN0IGFyZU1vcmVPcHRpb25zQWJvdmUgPSBzdGF0ZS52aXNpYmxlRnJvbUluZGV4ID4gMFxuXG4gICAgICAgICAgY29uc3QgaSA9IHN0YXRlLnZpc2libGVGcm9tSW5kZXggKyBpbmRleCArIDFcblxuICAgICAgICAgIGNvbnN0IGlzRm9jdXNlZCA9ICFpc0Rpc2FibGVkICYmIHN0YXRlLmZvY3VzZWRWYWx1ZSA9PT0gb3B0aW9uLnZhbHVlXG4gICAgICAgICAgY29uc3QgaXNTZWxlY3RlZCA9IHN0YXRlLnZhbHVlID09PSBvcHRpb24udmFsdWVcblxuICAgICAgICAgIHJldHVybiAoXG4gICAgICAgICAgICA8U2VsZWN0SW5wdXRPcHRpb25cbiAgICAgICAgICAgICAga2V5PXtTdHJpbmcob3B0aW9uLnZhbHVlKX1cbiAgICAgICAgICAgICAgb3B0aW9uPXtvcHRpb259XG4gICAgICAgICAgICAgIGlzRm9jdXNlZD17aXNGb2N1c2VkfVxuICAgICAgICAgICAgICBpc1NlbGVjdGVkPXtpc1NlbGVjdGVkfVxuICAgICAgICAgICAgICBzaG91bGRTaG93RG93bkFycm93PXthcmVNb3JlT3B0aW9uc0JlbG93ICYmIGlzTGFzdFZpc2libGVPcHRpb259XG4gICAgICAgICAgICAgIHNob3VsZFNob3dVcEFycm93PXthcmVNb3JlT3B0aW9uc0Fib3ZlICYmIGlzRmlyc3RWaXNpYmxlT3B0aW9ufVxuICAgICAgICAgICAgICBtYXhJbmRleFdpZHRoPXttYXhJbmRleFdpZHRofVxuICAgICAgICAgICAgICBpbmRleD17aX1cbiAgICAgICAgICAgICAgaW5wdXRWYWx1ZT17aW5wdXRWYWx1ZX1cbiAgICAgICAgICAgICAgb25JbnB1dENoYW5nZT17dmFsdWUgPT4ge1xuICAgICAgICAgICAgICAgIHNldElucHV0VmFsdWVzKHByZXYgPT4ge1xuICAgICAgICAgICAgICAgICAgY29uc3QgbmV4dCA9IG5ldyBNYXAocHJldilcbiAgICAgICAgICAgICAgICAgIG5leHQuc2V0KG9wdGlvbi52YWx1ZSwgdmFsdWUpXG4gICAgICAgICAgICAgICAgICByZXR1cm4gbmV4dFxuICAgICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgIG9uU3VibWl0PXsodmFsdWU6IHN0cmluZykgPT4ge1xuICAgICAgICAgICAgICAgIGNvbnN0IGhhc0ltYWdlQXR0YWNobWVudHMgPVxuICAgICAgICAgICAgICAgICAgcGFzdGVkQ29udGVudHMgJiZcbiAgICAgICAgICAgICAgICAgIE9iamVjdC52YWx1ZXMocGFzdGVkQ29udGVudHMpLnNvbWUoYyA9PiBjLnR5cGUgPT09ICdpbWFnZScpXG4gICAgICAgICAgICAgICAgaWYgKFxuICAgICAgICAgICAgICAgICAgdmFsdWUudHJpbSgpIHx8XG4gICAgICAgICAgICAgICAgICBoYXNJbWFnZUF0dGFjaG1lbnRzIHx8XG4gICAgICAgICAgICAgICAgICBvcHRpb24uYWxsb3dFbXB0eVN1Ym1pdFRvQ2FuY2VsXG4gICAgICAgICAgICAgICAgKSB7XG4gICAgICAgICAgICAgICAgICBvbkNoYW5nZT8uKG9wdGlvbi52YWx1ZSlcbiAgICAgICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgICAgb25DYW5jZWw/LigpXG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICB9fVxuICAgICAgICAgICAgICBvbkV4aXQ9e29uQ2FuY2VsfVxuICAgICAgICAgICAgICBsYXlvdXQ9XCJjb21wYWN0XCJcbiAgICAgICAgICAgICAgc2hvd0xhYmVsPXtpbmxpbmVEZXNjcmlwdGlvbnN9XG4gICAgICAgICAgICAgIG9uT3BlbkVkaXRvcj17b25PcGVuRWRpdG9yfVxuICAgICAgICAgICAgICByZXNldEN1cnNvck9uVXBkYXRlPXtvcHRpb24ucmVzZXRDdXJzb3JPblVwZGF0ZX1cbiAgICAgICAgICAgICAgb25JbWFnZVBhc3RlPXtvbkltYWdlUGFzdGV9XG4gICAgICAgICAgICAgIHBhc3RlZENvbnRlbnRzPXtwYXN0ZWRDb250ZW50c31cbiAgICAgICAgICAgICAgb25SZW1vdmVJbWFnZT17b25SZW1vdmVJbWFnZX1cbiAgICAgICAgICAgICAgaW1hZ2VzU2VsZWN0ZWQ9e2ltYWdlc1NlbGVjdGVkfVxuICAgICAgICAgICAgICBzZWxlY3RlZEltYWdlSW5kZXg9e3NlbGVjdGVkSW1hZ2VJbmRleH1cbiAgICAgICAgICAgICAgb25JbWFnZXNTZWxlY3RlZENoYW5nZT17c2V0SW1hZ2VzU2VsZWN0ZWR9XG4gICAgICAgICAgICAgIG9uU2VsZWN0ZWRJbWFnZUluZGV4Q2hhbmdlPXtzZXRTZWxlY3RlZEltYWdlSW5kZXh9XG4gICAgICAgICAgICAvPlxuICAgICAgICAgIClcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIEhhbmRsZSB0ZXh0IHR5cGUgb3B0aW9uc1xuICAgICAgICBsZXQgbGFiZWw6IFJlYWN0Tm9kZSA9IG9wdGlvbi5sYWJlbFxuXG4gICAgICAgIC8vIE9ubHkgYXBwbHkgaGlnaGxpZ2h0IHdoZW4gbGFiZWwgaXMgYSBzdHJpbmdcbiAgICAgICAgaWYgKFxuICAgICAgICAgIHR5cGVvZiBvcHRpb24ubGFiZWwgPT09ICdzdHJpbmcnICYmXG4gICAgICAgICAgaGlnaGxpZ2h0VGV4dCAmJlxuICAgICAgICAgIG9wdGlvbi5sYWJlbC5pbmNsdWRlcyhoaWdobGlnaHRUZXh0KVxuICAgICAgICApIHtcbiAgICAgICAgICBjb25zdCBsYWJlbFRleHQgPSBvcHRpb24ubGFiZWxcbiAgICAgICAgICBjb25zdCBpbmRleCA9IGxhYmVsVGV4dC5pbmRleE9mKGhpZ2hsaWdodFRleHQpXG5cbiAgICAgICAgICBsYWJlbCA9IChcbiAgICAgICAgICAgIDw+XG4gICAgICAgICAgICAgIHtsYWJlbFRleHQuc2xpY2UoMCwgaW5kZXgpfVxuICAgICAgICAgICAgICA8VGV4dCB7Li4uc3R5bGVzLmhpZ2hsaWdodGVkVGV4dCgpfT57aGlnaGxpZ2h0VGV4dH08L1RleHQ+XG4gICAgICAgICAgICAgIHtsYWJlbFRleHQuc2xpY2UoaW5kZXggKyBoaWdobGlnaHRUZXh0Lmxlbmd0aCl9XG4gICAgICAgICAgICA8Lz5cbiAgICAgICAgICApXG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBpc0ZpcnN0VmlzaWJsZU9wdGlvbiA9IG9wdGlvbi5pbmRleCA9PT0gc3RhdGUudmlzaWJsZUZyb21JbmRleFxuICAgICAgICBjb25zdCBpc0xhc3RWaXNpYmxlT3B0aW9uID0gb3B0aW9uLmluZGV4ID09PSBzdGF0ZS52aXNpYmxlVG9JbmRleCAtIDFcbiAgICAgICAgY29uc3QgYXJlTW9yZU9wdGlvbnNCZWxvdyA9IHN0YXRlLnZpc2libGVUb0luZGV4IDwgb3B0aW9ucy5sZW5ndGhcbiAgICAgICAgY29uc3QgYXJlTW9yZU9wdGlvbnNBYm92ZSA9IHN0YXRlLnZpc2libGVGcm9tSW5kZXggPiAwXG5cbiAgICAgICAgY29uc3QgaSA9IHN0YXRlLnZpc2libGVGcm9tSW5kZXggKyBpbmRleCArIDFcblxuICAgICAgICBjb25zdCBpc0ZvY3VzZWQgPSAhaXNEaXNhYmxlZCAmJiBzdGF0ZS5mb2N1c2VkVmFsdWUgPT09IG9wdGlvbi52YWx1ZVxuICAgICAgICBjb25zdCBpc1NlbGVjdGVkID0gc3RhdGUudmFsdWUgPT09IG9wdGlvbi52YWx1ZVxuICAgICAgICBjb25zdCBpc09wdGlvbkRpc2FibGVkID0gb3B0aW9uLmRpc2FibGVkID09PSB0cnVlXG5cbiAgICAgICAgcmV0dXJuIChcbiAgICAgICAgICA8U2VsZWN0T3B0aW9uXG4gICAgICAgICAgICBrZXk9e1N0cmluZyhvcHRpb24udmFsdWUpfVxuICAgICAgICAgICAgaXNGb2N1c2VkPXtpc0ZvY3VzZWR9XG4gICAgICAgICAgICBpc1NlbGVjdGVkPXtpc1NlbGVjdGVkfVxuICAgICAgICAgICAgc2hvdWxkU2hvd0Rvd25BcnJvdz17YXJlTW9yZU9wdGlvbnNCZWxvdyAmJiBpc0xhc3RWaXNpYmxlT3B0aW9ufVxuICAgICAgICAgICAgc2hvdWxkU2hvd1VwQXJyb3c9e2FyZU1vcmVPcHRpb25zQWJvdmUgJiYgaXNGaXJzdFZpc2libGVPcHRpb259XG4gICAgICAgICAgPlxuICAgICAgICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwicm93XCIgZmxleFNocmluaz17MH0+XG4gICAgICAgICAgICAgIHshaGlkZUluZGV4ZXMgJiYgKFxuICAgICAgICAgICAgICAgIDxUZXh0IGRpbUNvbG9yPntgJHtpfS5gLnBhZEVuZChtYXhJbmRleFdpZHRoICsgMil9PC9UZXh0PlxuICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgICA8VGV4dFxuICAgICAgICAgICAgICAgIGRpbUNvbG9yPXtpc09wdGlvbkRpc2FibGVkfVxuICAgICAgICAgICAgICAgIGNvbG9yPXtcbiAgICAgICAgICAgICAgICAgIGlzT3B0aW9uRGlzYWJsZWRcbiAgICAgICAgICAgICAgICAgICAgPyB1bmRlZmluZWRcbiAgICAgICAgICAgICAgICAgICAgOiBpc1NlbGVjdGVkXG4gICAgICAgICAgICAgICAgICAgICAgPyAnc3VjY2VzcydcbiAgICAgICAgICAgICAgICAgICAgICA6IGlzRm9jdXNlZFxuICAgICAgICAgICAgICAgICAgICAgICAgPyAnc3VnZ2VzdGlvbidcbiAgICAgICAgICAgICAgICAgICAgICAgIDogdW5kZWZpbmVkXG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAge2xhYmVsfVxuICAgICAgICAgICAgICAgIHtpbmxpbmVEZXNjcmlwdGlvbnMgJiYgb3B0aW9uLmRlc2NyaXB0aW9uICYmIChcbiAgICAgICAgICAgICAgICAgIDxUZXh0XG4gICAgICAgICAgICAgICAgICAgIGRpbUNvbG9yPXtcbiAgICAgICAgICAgICAgICAgICAgICBpc09wdGlvbkRpc2FibGVkIHx8IG9wdGlvbi5kaW1EZXNjcmlwdGlvbiAhPT0gZmFsc2VcbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgICB7JyAnfVxuICAgICAgICAgICAgICAgICAgICB7b3B0aW9uLmRlc2NyaXB0aW9ufVxuICAgICAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgIDwvQm94PlxuICAgICAgICAgICAgeyFpbmxpbmVEZXNjcmlwdGlvbnMgJiYgb3B0aW9uLmRlc2NyaXB0aW9uICYmIChcbiAgICAgICAgICAgICAgPEJveCBmbGV4U2hyaW5rPXs5OX0gbWFyZ2luTGVmdD17Mn0+XG4gICAgICAgICAgICAgICAgPFRleHRcbiAgICAgICAgICAgICAgICAgIHdyYXA9XCJ3cmFwLXRyaW1cIlxuICAgICAgICAgICAgICAgICAgZGltQ29sb3I9e2lzT3B0aW9uRGlzYWJsZWQgfHwgb3B0aW9uLmRpbURlc2NyaXB0aW9uICE9PSBmYWxzZX1cbiAgICAgICAgICAgICAgICAgIGNvbG9yPXtcbiAgICAgICAgICAgICAgICAgICAgaXNPcHRpb25EaXNhYmxlZFxuICAgICAgICAgICAgICAgICAgICAgID8gdW5kZWZpbmVkXG4gICAgICAgICAgICAgICAgICAgICAgOiBpc1NlbGVjdGVkXG4gICAgICAgICAgICAgICAgICAgICAgICA/ICdzdWNjZXNzJ1xuICAgICAgICAgICAgICAgICAgICAgICAgOiBpc0ZvY3VzZWRcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPyAnc3VnZ2VzdGlvbidcbiAgICAgICAgICAgICAgICAgICAgICAgICAgOiB1bmRlZmluZWRcbiAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICA8QW5zaT57b3B0aW9uLmRlc2NyaXB0aW9ufTwvQW5zaT5cbiAgICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICAgIDwvQm94PlxuICAgICAgICAgICAgKX1cbiAgICAgICAgICA8L1NlbGVjdE9wdGlvbj5cbiAgICAgICAgKVxuICAgICAgfSl9XG4gICAgPC9Cb3g+XG4gIClcbn1cblxuLy8gUm93IGNvbnRhaW5lciBmb3IgdGhlIHR3by1jb2x1bW4gKGxhYmVsICsgZGVzY3JpcHRpb24pIGxheW91dC4gVW5saWtlXG4vLyB0aGUgb3RoZXIgU2VsZWN0IGxheW91dHMsIHRoaXMgb25lIGRvZXNuJ3QgcmVuZGVyIHRocm91Z2ggU2VsZWN0T3B0aW9uIOKGklxuLy8gTGlzdEl0ZW0sIHNvIGl0IGRlY2xhcmVzIHRoZSBuYXRpdmUgY3Vyc29yIGRpcmVjdGx5LiBQYXJrcyB0aGUgY3Vyc29yXG4vLyBvbiB0aGUgcG9pbnRlciBpbmRpY2F0b3Igc28gc2NyZWVuIHJlYWRlcnMgLyBtYWduaWZpZXJzIHRyYWNrIGZvY3VzLlxuZnVuY3Rpb24gVHdvQ29sdW1uUm93KHtcbiAgaXNGb2N1c2VkLFxuICBjaGlsZHJlbixcbn06IHtcbiAgaXNGb2N1c2VkOiBib29sZWFuXG4gIGNoaWxkcmVuOiBSZWFjdE5vZGVcbn0pOiBSZWFjdC5SZWFjdE5vZGUge1xuICBjb25zdCBjdXJzb3JSZWYgPSB1c2VEZWNsYXJlZEN1cnNvcih7XG4gICAgbGluZTogMCxcbiAgICBjb2x1bW46IDAsXG4gICAgYWN0aXZlOiBpc0ZvY3VzZWQsXG4gIH0pXG4gIHJldHVybiAoXG4gICAgPEJveCByZWY9e2N1cnNvclJlZn0gZmxleERpcmVjdGlvbj1cInJvd1wiPlxuICAgICAge2NoaWxkcmVufVxuICAgIDwvQm94PlxuICApXG59XG4iXSwibWFwcGluZ3MiOiI7QUFBQSxPQUFPQSxPQUFPLE1BQU0sU0FBUztBQUM3QixPQUFPQyxLQUFLLElBQUksS0FBS0MsU0FBUyxFQUFFQyxTQUFTLEVBQUVDLE1BQU0sRUFBRUMsUUFBUSxRQUFRLE9BQU87QUFDMUUsU0FBU0MsaUJBQWlCLFFBQVEsd0NBQXdDO0FBQzFFLFNBQVNDLFdBQVcsUUFBUSwwQkFBMEI7QUFDdEQsU0FBU0MsSUFBSSxFQUFFQyxHQUFHLEVBQUVDLElBQUksUUFBUSxjQUFjO0FBQzlDLFNBQVNDLEtBQUssUUFBUSxzQkFBc0I7QUFDNUMsY0FBY0MsYUFBYSxRQUFRLHVCQUF1QjtBQUMxRCxjQUFjQyxlQUFlLFFBQVEsNkJBQTZCO0FBQ2xFLFNBQVNDLGlCQUFpQixRQUFRLDBCQUEwQjtBQUM1RCxTQUFTQyxZQUFZLFFBQVEsb0JBQW9CO0FBQ2pELFNBQVND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 \ No newline at end of file diff --git a/src/components/CustomSelect/use-multi-select-state.ts b/src/components/CustomSelect/use-multi-select-state.ts new file mode 100644 index 0000000..bf2bd8b --- /dev/null +++ b/src/components/CustomSelect/use-multi-select-state.ts @@ -0,0 +1,414 @@ +import { useCallback, useState } from 'react' +import { isDeepStrictEqual } from 'util' +import { useRegisterOverlay } from '../../context/overlayContext.js' +import type { InputEvent } from '../../ink/events/input-event.js' +// eslint-disable-next-line custom-rules/prefer-use-keybindings -- raw space/arrow multiselect input +import { useInput } from '../../ink.js' +import { + normalizeFullWidthDigits, + normalizeFullWidthSpace, +} from '../../utils/stringUtils.js' +import type { OptionWithDescription } from './select.js' +import { useSelectNavigation } from './use-select-navigation.js' + +export type UseMultiSelectStateProps = { + /** + * When disabled, user input is ignored. + * + * @default false + */ + isDisabled?: boolean + + /** + * Number of items to display. + * + * @default 5 + */ + visibleOptionCount?: number + + /** + * Options. + */ + options: OptionWithDescription[] + + /** + * Initially selected values. + */ + defaultValue?: T[] + + /** + * Callback when selection changes. + */ + onChange?: (values: T[]) => void + + /** + * Callback for canceling the select. + */ + onCancel: () => void + + /** + * Callback for focusing an option. + */ + onFocus?: (value: T) => void + + /** + * Value to focus + */ + focusValue?: T + + /** + * Text for the submit button. When provided, a submit button is shown and + * Enter toggles selection (submit only fires when the button is focused). + * When omitted, Enter submits directly and Space toggles selection. + */ + submitButtonText?: string + + /** + * Callback when user submits. Receives the currently selected values. + */ + onSubmit?: (values: T[]) => void + + /** + * Callback when user presses down from the last item (submit button). + * If provided, navigation will not wrap to the first item. + */ + onDownFromLastItem?: () => void + + /** + * Callback when user presses up from the first item. + * If provided, navigation will not wrap to the last item. + */ + onUpFromFirstItem?: () => void + + /** + * Focus the last option initially instead of the first. + */ + initialFocusLast?: boolean + + /** + * When true, numeric keys (1-9) do not toggle options by index. + * Mirrors the rendering layer's hideIndexes: if index labels aren't shown, + * pressing a number shouldn't silently toggle an invisible mapping. + */ + hideIndexes?: boolean +} + +export type MultiSelectState = { + /** + * Value of the currently focused option. + */ + focusedValue: T | undefined + + /** + * Index of the first visible option. + */ + visibleFromIndex: number + + /** + * Index of the last visible option. + */ + visibleToIndex: number + + /** + * All options. + */ + options: OptionWithDescription[] + + /** + * Visible options. + */ + visibleOptions: Array & { index: number }> + + /** + * Whether the focused option is an input type. + */ + isInInput: boolean + + /** + * Currently selected values. + */ + selectedValues: T[] + + /** + * Current input field values. + */ + inputValues: Map + + /** + * Whether the submit button is focused. + */ + isSubmitFocused: boolean + + /** + * Update an input field value. + */ + updateInputValue: (value: T, inputValue: string) => void + + /** + * Callback for canceling the select. + */ + onCancel: () => void +} + +export function useMultiSelectState({ + isDisabled = false, + visibleOptionCount = 5, + options, + defaultValue = [], + onChange, + onCancel, + onFocus, + focusValue, + submitButtonText, + onSubmit, + onDownFromLastItem, + onUpFromFirstItem, + initialFocusLast, + hideIndexes = false, +}: UseMultiSelectStateProps): MultiSelectState { + const [selectedValues, setSelectedValues] = useState(defaultValue) + const [isSubmitFocused, setIsSubmitFocused] = useState(false) + + // Reset selectedValues when options change (e.g. async-loaded data changes + // defaultValue after mount). Mirrors the reset pattern in use-select-navigation.ts + // and the deleted ui/useMultiSelectState.ts — without this, MCPServerDesktopImportDialog + // keeps colliding servers checked after getAllMcpConfigs() resolves. + const [lastOptions, setLastOptions] = useState(options) + if (options !== lastOptions && !isDeepStrictEqual(options, lastOptions)) { + setSelectedValues(defaultValue) + setLastOptions(options) + } + + // State for input type options + const [inputValues, setInputValues] = useState>(() => { + const initialMap = new Map() + options.forEach(option => { + if (option.type === 'input' && option.initialValue) { + initialMap.set(option.value, option.initialValue) + } + }) + return initialMap + }) + + const updateSelectedValues = useCallback( + (values: T[] | ((prev: T[]) => T[])) => { + const newValues = + typeof values === 'function' ? values(selectedValues) : values + setSelectedValues(newValues) + onChange?.(newValues) + }, + [selectedValues, onChange], + ) + + const navigation = useSelectNavigation({ + visibleOptionCount, + options, + initialFocusValue: initialFocusLast + ? options[options.length - 1]?.value + : undefined, + onFocus, + focusValue, + }) + + // Automatically register as an overlay. + // This ensures CancelRequestHandler won't intercept Escape when the multi-select is active. + useRegisterOverlay('multi-select') + + const updateInputValue = useCallback( + (value: T, inputValue: string) => { + setInputValues(prev => { + const next = new Map(prev) + next.set(value, inputValue) + return next + }) + + // Find the option and call its onChange + const option = options.find(opt => opt.value === value) + if (option && option.type === 'input') { + option.onChange(inputValue) + } + + // Update selected values to include/exclude based on input + updateSelectedValues(prev => { + if (inputValue) { + if (!prev.includes(value)) { + return [...prev, value] + } + return prev + } else { + return prev.filter(v => v !== value) + } + }) + }, + [options, updateSelectedValues], + ) + + // Handle all keyboard input + useInput( + (input, key, event: InputEvent) => { + const normalizedInput = normalizeFullWidthDigits(input) + const focusedOption = options.find( + opt => opt.value === navigation.focusedValue, + ) + const isInInput = focusedOption?.type === 'input' + + // When in input field, only allow navigation keys + if (isInInput) { + const isAllowedKey = + key.upArrow || + key.downArrow || + key.escape || + key.tab || + key.return || + (key.ctrl && (input === 'n' || input === 'p' || key.return)) + if (!isAllowedKey) return + } + + const lastOptionValue = options[options.length - 1]?.value + + // Handle Tab to move forward + if (key.tab && !key.shift) { + if ( + submitButtonText && + onSubmit && + navigation.focusedValue === lastOptionValue && + !isSubmitFocused + ) { + setIsSubmitFocused(true) + } else if (!isSubmitFocused) { + navigation.focusNextOption() + } + return + } + + // Handle Shift+Tab to move backward + if (key.tab && key.shift) { + if (submitButtonText && onSubmit && isSubmitFocused) { + setIsSubmitFocused(false) + navigation.focusOption(lastOptionValue) + } else { + navigation.focusPreviousOption() + } + return + } + + // Handle arrow down / Ctrl+N / j + if ( + key.downArrow || + (key.ctrl && input === 'n') || + (!key.ctrl && !key.shift && input === 'j') + ) { + if (isSubmitFocused && onDownFromLastItem) { + onDownFromLastItem() + } else if ( + submitButtonText && + onSubmit && + navigation.focusedValue === lastOptionValue && + !isSubmitFocused + ) { + setIsSubmitFocused(true) + } else if ( + !submitButtonText && + onDownFromLastItem && + navigation.focusedValue === lastOptionValue + ) { + // No submit button — exit from the last option + onDownFromLastItem() + } else if (!isSubmitFocused) { + navigation.focusNextOption() + } + return + } + + // Handle arrow up / Ctrl+P / k + if ( + key.upArrow || + (key.ctrl && input === 'p') || + (!key.ctrl && !key.shift && input === 'k') + ) { + if (submitButtonText && onSubmit && isSubmitFocused) { + setIsSubmitFocused(false) + navigation.focusOption(lastOptionValue) + } else if ( + onUpFromFirstItem && + navigation.focusedValue === options[0]?.value + ) { + onUpFromFirstItem() + } else { + navigation.focusPreviousOption() + } + return + } + + // Handle page navigation + if (key.pageDown) { + navigation.focusNextPage() + return + } + + if (key.pageUp) { + navigation.focusPreviousPage() + return + } + + // Handle Enter or Space for selection/submit + if (key.return || normalizeFullWidthSpace(input) === ' ') { + // Ctrl+Enter from input field submits + if (key.ctrl && key.return && isInInput && onSubmit) { + onSubmit(selectedValues) + return + } + + // Enter on submit button submits + if (isSubmitFocused && onSubmit) { + onSubmit(selectedValues) + return + } + + // No submit button: Enter submits directly, Space still toggles + if (key.return && !submitButtonText && onSubmit) { + onSubmit(selectedValues) + return + } + + // Enter or Space toggles selection (including for input fields) + if (navigation.focusedValue !== undefined) { + const newValues = selectedValues.includes(navigation.focusedValue) + ? selectedValues.filter(v => v !== navigation.focusedValue) + : [...selectedValues, navigation.focusedValue] + updateSelectedValues(newValues) + } + return + } + + // Handle numeric keys (1-9) for direct selection + if (!hideIndexes && /^[0-9]+$/.test(normalizedInput)) { + const index = parseInt(normalizedInput) - 1 + if (index >= 0 && index < options.length) { + const value = options[index]!.value + const newValues = selectedValues.includes(value) + ? selectedValues.filter(v => v !== value) + : [...selectedValues, value] + updateSelectedValues(newValues) + } + return + } + + // Handle Escape + if (key.escape) { + onCancel() + event.stopImmediatePropagation() + } + }, + { isActive: !isDisabled }, + ) + + return { + ...navigation, + selectedValues, + inputValues, + isSubmitFocused, + updateInputValue, + onCancel, + } +} diff --git a/src/components/CustomSelect/use-select-input.ts b/src/components/CustomSelect/use-select-input.ts new file mode 100644 index 0000000..dcafeb4 --- /dev/null +++ b/src/components/CustomSelect/use-select-input.ts @@ -0,0 +1,287 @@ +import { useMemo } from 'react' +import { useRegisterOverlay } from '../../context/overlayContext.js' +import type { InputEvent } from '../../ink/events/input-event.js' +import { useInput } from '../../ink.js' +import { useKeybindings } from '../../keybindings/useKeybinding.js' +import { + normalizeFullWidthDigits, + normalizeFullWidthSpace, +} from '../../utils/stringUtils.js' +import type { OptionWithDescription } from './select.js' +import type { SelectState } from './use-select-state.js' + +export type UseSelectProps = { + /** + * When disabled, user input is ignored. + * + * @default false + */ + isDisabled?: boolean + + /** + * When true, prevents selection on Enter or number keys, but allows + * scrolling. + * When 'numeric', prevents selection on number keys, but allows Enter (and + * scrolling). + * + * @default false + */ + readonly disableSelection?: boolean | 'numeric' + + /** + * Select state. + */ + state: SelectState + + /** + * Options. + */ + options: OptionWithDescription[] + + /** + * Whether this is a multi-select component. + * + * @default false + */ + isMultiSelect?: boolean + + /** + * Callback when user presses up from the first item. + * If provided, navigation will not wrap to the last item. + */ + onUpFromFirstItem?: () => void + + /** + * Callback when user presses down from the last item. + * If provided, navigation will not wrap to the first item. + */ + onDownFromLastItem?: () => void + + /** + * Callback when input mode should be toggled for an option. + * Called when Tab is pressed (to enter or exit input mode). + */ + onInputModeToggle?: (value: T) => void + + /** + * Current input values for input-type options. + * Used to determine if number key should submit an empty input option. + */ + inputValues?: Map + + /** + * Whether image selection mode is active on the focused input option. + * When true, arrow key navigation in useInput is suppressed so that + * Attachments keybindings can handle image navigation instead. + */ + imagesSelected?: boolean + + /** + * Callback to attempt entering image selection mode on DOWN arrow. + * Returns true if image selection was entered (images exist), false otherwise. + */ + onEnterImageSelection?: () => boolean +} + +export const useSelectInput = ({ + isDisabled = false, + disableSelection = false, + state, + options, + isMultiSelect = false, + onUpFromFirstItem, + onDownFromLastItem, + onInputModeToggle, + inputValues, + imagesSelected = false, + onEnterImageSelection, +}: UseSelectProps) => { + // Automatically register as an overlay when onCancel is provided. + // This ensures CancelRequestHandler won't intercept Escape when the select is active. + useRegisterOverlay('select', !!state.onCancel) + + // Determine if the focused option is an input type + const isInInput = useMemo(() => { + const focusedOption = options.find(opt => opt.value === state.focusedValue) + return focusedOption?.type === 'input' + }, [options, state.focusedValue]) + + // Core navigation via keybindings (up/down/enter/escape) + // When in input mode, exclude navigation/accept keybindings so that + // j/k/enter pass through to the TextInput instead of being intercepted. + const keybindingHandlers = useMemo(() => { + const handlers: Record void> = {} + + if (!isInInput) { + handlers['select:next'] = () => { + if (onDownFromLastItem) { + const lastOption = options[options.length - 1] + if (lastOption && state.focusedValue === lastOption.value) { + onDownFromLastItem() + return + } + } + state.focusNextOption() + } + handlers['select:previous'] = () => { + if (onUpFromFirstItem && state.visibleFromIndex === 0) { + const firstOption = options[0] + if (firstOption && state.focusedValue === firstOption.value) { + onUpFromFirstItem() + return + } + } + state.focusPreviousOption() + } + handlers['select:accept'] = () => { + if (disableSelection === true) return + if (state.focusedValue === undefined) return + + const focusedOption = options.find( + opt => opt.value === state.focusedValue, + ) + if (focusedOption?.disabled === true) return + + state.selectFocusedOption?.() + state.onChange?.(state.focusedValue) + } + } + + if (state.onCancel) { + handlers['select:cancel'] = () => { + state.onCancel!() + } + } + + return handlers + }, [ + options, + state, + onDownFromLastItem, + onUpFromFirstItem, + isInInput, + disableSelection, + ]) + + useKeybindings(keybindingHandlers, { + context: 'Select', + isActive: !isDisabled, + }) + + // Remaining keys that stay as useInput: number keys, pageUp/pageDown, tab, space, + // and arrow key navigation when in input mode + useInput( + (input, key, event: InputEvent) => { + const normalizedInput = normalizeFullWidthDigits(input) + const focusedOption = options.find( + opt => opt.value === state.focusedValue, + ) + const currentIsInInput = focusedOption?.type === 'input' + + // Handle Tab key for input mode toggling + if (key.tab && onInputModeToggle && state.focusedValue !== undefined) { + onInputModeToggle(state.focusedValue) + return + } + + if (currentIsInInput) { + // When in image selection mode, suppress all input handling so + // Attachments keybindings can handle navigation/deletion instead + if (imagesSelected) return + + // DOWN arrow enters image selection mode if images exist + if (key.downArrow && onEnterImageSelection?.()) { + event.stopImmediatePropagation() + return + } + + // Arrow keys still navigate the select even while in input mode + if (key.downArrow || (key.ctrl && input === 'n')) { + if (onDownFromLastItem) { + const lastOption = options[options.length - 1] + if (lastOption && state.focusedValue === lastOption.value) { + onDownFromLastItem() + event.stopImmediatePropagation() + return + } + } + state.focusNextOption() + event.stopImmediatePropagation() + return + } + if (key.upArrow || (key.ctrl && input === 'p')) { + if (onUpFromFirstItem && state.visibleFromIndex === 0) { + const firstOption = options[0] + if (firstOption && state.focusedValue === firstOption.value) { + onUpFromFirstItem() + event.stopImmediatePropagation() + return + } + } + state.focusPreviousOption() + event.stopImmediatePropagation() + return + } + + // All other keys (including digits) pass through to TextInput. + // Digits should type literally into the input rather than select + // options — the user has focused a text field and expects typing + // to insert characters, not jump to a different option. + return + } + + if (key.pageDown) { + state.focusNextPage() + } + + if (key.pageUp) { + state.focusPreviousPage() + } + + if (disableSelection !== true) { + // Space for multi-select toggle + if ( + isMultiSelect && + normalizeFullWidthSpace(input) === ' ' && + state.focusedValue !== undefined + ) { + const isFocusedOptionDisabled = focusedOption?.disabled === true + if (!isFocusedOptionDisabled) { + state.selectFocusedOption?.() + state.onChange?.(state.focusedValue) + } + } + + if ( + disableSelection !== 'numeric' && + /^[0-9]+$/.test(normalizedInput) + ) { + const index = parseInt(normalizedInput) - 1 + if (index >= 0 && index < state.options.length) { + const selectedOption = state.options[index]! + if (selectedOption.disabled === true) { + return + } + if (selectedOption.type === 'input') { + const currentValue = inputValues?.get(selectedOption.value) ?? '' + if (currentValue.trim()) { + // Pre-filled input: auto-submit (user can Tab to edit instead) + state.onChange?.(selectedOption.value) + return + } + if (selectedOption.allowEmptySubmitToCancel) { + state.onChange?.(selectedOption.value) + return + } + state.focusOption(selectedOption.value) + return + } + state.onChange?.(selectedOption.value) + return + } + } + } + }, + { isActive: !isDisabled }, + ) +} diff --git a/src/components/CustomSelect/use-select-navigation.ts b/src/components/CustomSelect/use-select-navigation.ts new file mode 100644 index 0000000..7ecb4e7 --- /dev/null +++ b/src/components/CustomSelect/use-select-navigation.ts @@ -0,0 +1,653 @@ +import { + useCallback, + useEffect, + useMemo, + useReducer, + useRef, + useState, +} from 'react' +import { isDeepStrictEqual } from 'util' +import OptionMap from './option-map.js' +import type { OptionWithDescription } from './select.js' + +type State = { + /** + * Map where key is option's value and value is option's index. + */ + optionMap: OptionMap + + /** + * Number of visible options. + */ + visibleOptionCount: number + + /** + * Value of the currently focused option. + */ + focusedValue: T | undefined + + /** + * Index of the first visible option. + */ + visibleFromIndex: number + + /** + * Index of the last visible option. + */ + visibleToIndex: number +} + +type Action = + | FocusNextOptionAction + | FocusPreviousOptionAction + | FocusNextPageAction + | FocusPreviousPageAction + | SetFocusAction + | ResetAction + +type SetFocusAction = { + type: 'set-focus' + value: T +} + +type FocusNextOptionAction = { + type: 'focus-next-option' +} + +type FocusPreviousOptionAction = { + type: 'focus-previous-option' +} + +type FocusNextPageAction = { + type: 'focus-next-page' +} + +type FocusPreviousPageAction = { + type: 'focus-previous-page' +} + +type ResetAction = { + type: 'reset' + state: State +} + +const reducer = (state: State, action: Action): State => { + switch (action.type) { + case 'focus-next-option': { + if (state.focusedValue === undefined) { + return state + } + + const item = state.optionMap.get(state.focusedValue) + + if (!item) { + return state + } + + // Wrap to first item if at the end + const next = item.next || state.optionMap.first + + if (!next) { + return state + } + + // When wrapping to first, reset viewport to start + if (!item.next && next === state.optionMap.first) { + return { + ...state, + focusedValue: next.value, + visibleFromIndex: 0, + visibleToIndex: state.visibleOptionCount, + } + } + + const needsToScroll = next.index >= state.visibleToIndex + + if (!needsToScroll) { + return { + ...state, + focusedValue: next.value, + } + } + + const nextVisibleToIndex = Math.min( + state.optionMap.size, + state.visibleToIndex + 1, + ) + + const nextVisibleFromIndex = nextVisibleToIndex - state.visibleOptionCount + + return { + ...state, + focusedValue: next.value, + visibleFromIndex: nextVisibleFromIndex, + visibleToIndex: nextVisibleToIndex, + } + } + + case 'focus-previous-option': { + if (state.focusedValue === undefined) { + return state + } + + const item = state.optionMap.get(state.focusedValue) + + if (!item) { + return state + } + + // Wrap to last item if at the beginning + const previous = item.previous || state.optionMap.last + + if (!previous) { + return state + } + + // When wrapping to last, reset viewport to end + if (!item.previous && previous === state.optionMap.last) { + const nextVisibleToIndex = state.optionMap.size + const nextVisibleFromIndex = Math.max( + 0, + nextVisibleToIndex - state.visibleOptionCount, + ) + return { + ...state, + focusedValue: previous.value, + visibleFromIndex: nextVisibleFromIndex, + visibleToIndex: nextVisibleToIndex, + } + } + + const needsToScroll = previous.index <= state.visibleFromIndex + + if (!needsToScroll) { + return { + ...state, + focusedValue: previous.value, + } + } + + const nextVisibleFromIndex = Math.max(0, state.visibleFromIndex - 1) + + const nextVisibleToIndex = nextVisibleFromIndex + state.visibleOptionCount + + return { + ...state, + focusedValue: previous.value, + visibleFromIndex: nextVisibleFromIndex, + visibleToIndex: nextVisibleToIndex, + } + } + + case 'focus-next-page': { + if (state.focusedValue === undefined) { + return state + } + + const item = state.optionMap.get(state.focusedValue) + + if (!item) { + return state + } + + // Move by a full page (visibleOptionCount items) + const targetIndex = Math.min( + state.optionMap.size - 1, + item.index + state.visibleOptionCount, + ) + + // Find the item at the target index + let targetItem = state.optionMap.first + while (targetItem && targetItem.index < targetIndex) { + if (targetItem.next) { + targetItem = targetItem.next + } else { + break + } + } + + if (!targetItem) { + return state + } + + // Update the visible range to include the new focused item + const nextVisibleToIndex = Math.min( + state.optionMap.size, + targetItem.index + 1, + ) + const nextVisibleFromIndex = Math.max( + 0, + nextVisibleToIndex - state.visibleOptionCount, + ) + + return { + ...state, + focusedValue: targetItem.value, + visibleFromIndex: nextVisibleFromIndex, + visibleToIndex: nextVisibleToIndex, + } + } + + case 'focus-previous-page': { + if (state.focusedValue === undefined) { + return state + } + + const item = state.optionMap.get(state.focusedValue) + + if (!item) { + return state + } + + // Move by a full page (visibleOptionCount items) + const targetIndex = Math.max(0, item.index - state.visibleOptionCount) + + // Find the item at the target index + let targetItem = state.optionMap.first + while (targetItem && targetItem.index < targetIndex) { + if (targetItem.next) { + targetItem = targetItem.next + } else { + break + } + } + + if (!targetItem) { + return state + } + + // Update the visible range to include the new focused item + const nextVisibleFromIndex = Math.max(0, targetItem.index) + const nextVisibleToIndex = Math.min( + state.optionMap.size, + nextVisibleFromIndex + state.visibleOptionCount, + ) + + return { + ...state, + focusedValue: targetItem.value, + visibleFromIndex: nextVisibleFromIndex, + visibleToIndex: nextVisibleToIndex, + } + } + + case 'reset': { + return action.state + } + + case 'set-focus': { + // Early return if already focused on this value + if (state.focusedValue === action.value) { + return state + } + + const item = state.optionMap.get(action.value) + if (!item) { + return state + } + + // Check if the item is already in view + if ( + item.index >= state.visibleFromIndex && + item.index < state.visibleToIndex + ) { + // Already visible, just update focus + return { + ...state, + focusedValue: action.value, + } + } + + // Need to scroll to make the item visible + // Scroll as little as possible - put item at edge of viewport + let nextVisibleFromIndex: number + let nextVisibleToIndex: number + + if (item.index < state.visibleFromIndex) { + // Item is above viewport - scroll up to put it at the top + nextVisibleFromIndex = item.index + nextVisibleToIndex = Math.min( + state.optionMap.size, + nextVisibleFromIndex + state.visibleOptionCount, + ) + } else { + // Item is below viewport - scroll down to put it at the bottom + nextVisibleToIndex = Math.min(state.optionMap.size, item.index + 1) + nextVisibleFromIndex = Math.max( + 0, + nextVisibleToIndex - state.visibleOptionCount, + ) + } + + return { + ...state, + focusedValue: action.value, + visibleFromIndex: nextVisibleFromIndex, + visibleToIndex: nextVisibleToIndex, + } + } + } +} + +export type UseSelectNavigationProps = { + /** + * Number of items to display. + * + * @default 5 + */ + visibleOptionCount?: number + + /** + * Options. + */ + options: OptionWithDescription[] + + /** + * Initially focused option's value. + */ + initialFocusValue?: T + + /** + * Callback for focusing an option. + */ + onFocus?: (value: T) => void + + /** + * Value to focus + */ + focusValue?: T +} + +export type SelectNavigation = { + /** + * Value of the currently focused option. + */ + focusedValue: T | undefined + + /** + * 1-based index of the focused option in the full list. + * Returns 0 if no option is focused. + */ + focusedIndex: number + + /** + * Index of the first visible option. + */ + visibleFromIndex: number + + /** + * Index of the last visible option. + */ + visibleToIndex: number + + /** + * All options. + */ + options: OptionWithDescription[] + + /** + * Visible options. + */ + visibleOptions: Array & { index: number }> + + /** + * Whether the focused option is an input type. + */ + isInInput: boolean + + /** + * Focus next option and scroll the list down, if needed. + */ + focusNextOption: () => void + + /** + * Focus previous option and scroll the list up, if needed. + */ + focusPreviousOption: () => void + + /** + * Focus next page and scroll the list down by a page. + */ + focusNextPage: () => void + + /** + * Focus previous page and scroll the list up by a page. + */ + focusPreviousPage: () => void + + /** + * Focus a specific option by value. + */ + focusOption: (value: T | undefined) => void +} + +const createDefaultState = ({ + visibleOptionCount: customVisibleOptionCount, + options, + initialFocusValue, + currentViewport, +}: Pick, 'visibleOptionCount' | 'options'> & { + initialFocusValue?: T + currentViewport?: { visibleFromIndex: number; visibleToIndex: number } +}): State => { + const visibleOptionCount = + typeof customVisibleOptionCount === 'number' + ? Math.min(customVisibleOptionCount, options.length) + : options.length + + const optionMap = new OptionMap(options) + const focusedItem = + initialFocusValue !== undefined && optionMap.get(initialFocusValue) + const focusedValue = focusedItem ? initialFocusValue : optionMap.first?.value + + let visibleFromIndex = 0 + let visibleToIndex = visibleOptionCount + + // When there's a valid focused item, adjust viewport to show it + if (focusedItem) { + const focusedIndex = focusedItem.index + + if (currentViewport) { + // If focused item is already in the current viewport range, try to preserve it + if ( + focusedIndex >= currentViewport.visibleFromIndex && + focusedIndex < currentViewport.visibleToIndex + ) { + // Keep the same viewport if it's valid + visibleFromIndex = currentViewport.visibleFromIndex + visibleToIndex = Math.min( + optionMap.size, + currentViewport.visibleToIndex, + ) + } else { + // Need to adjust viewport to show focused item + // Use minimal scrolling - put item at edge of viewport + if (focusedIndex < currentViewport.visibleFromIndex) { + // Item is above current viewport - scroll up to put it at the top + visibleFromIndex = focusedIndex + visibleToIndex = Math.min( + optionMap.size, + visibleFromIndex + visibleOptionCount, + ) + } else { + // Item is below current viewport - scroll down to put it at the bottom + visibleToIndex = Math.min(optionMap.size, focusedIndex + 1) + visibleFromIndex = Math.max(0, visibleToIndex - visibleOptionCount) + } + } + } else if (focusedIndex >= visibleOptionCount) { + // No current viewport but focused item is outside default viewport + // Scroll to show the focused item at the bottom of the viewport + visibleToIndex = Math.min(optionMap.size, focusedIndex + 1) + visibleFromIndex = Math.max(0, visibleToIndex - visibleOptionCount) + } + + // Ensure viewport bounds are valid + visibleFromIndex = Math.max( + 0, + Math.min(visibleFromIndex, optionMap.size - 1), + ) + visibleToIndex = Math.min( + optionMap.size, + Math.max(visibleOptionCount, visibleToIndex), + ) + } + + return { + optionMap, + visibleOptionCount, + focusedValue, + visibleFromIndex, + visibleToIndex, + } +} + +export function useSelectNavigation({ + visibleOptionCount = 5, + options, + initialFocusValue, + onFocus, + focusValue, +}: UseSelectNavigationProps): SelectNavigation { + const [state, dispatch] = useReducer( + reducer, + { + visibleOptionCount, + options, + initialFocusValue: focusValue || initialFocusValue, + } as Parameters>[0], + createDefaultState, + ) + + // Store onFocus in a ref to avoid re-running useEffect when callback changes + const onFocusRef = useRef(onFocus) + onFocusRef.current = onFocus + + const [lastOptions, setLastOptions] = useState(options) + + if (options !== lastOptions && !isDeepStrictEqual(options, lastOptions)) { + dispatch({ + type: 'reset', + state: createDefaultState({ + visibleOptionCount, + options, + initialFocusValue: + focusValue ?? state.focusedValue ?? initialFocusValue, + currentViewport: { + visibleFromIndex: state.visibleFromIndex, + visibleToIndex: state.visibleToIndex, + }, + }), + }) + + setLastOptions(options) + } + + const focusNextOption = useCallback(() => { + dispatch({ + type: 'focus-next-option', + }) + }, []) + + const focusPreviousOption = useCallback(() => { + dispatch({ + type: 'focus-previous-option', + }) + }, []) + + const focusNextPage = useCallback(() => { + dispatch({ + type: 'focus-next-page', + }) + }, []) + + const focusPreviousPage = useCallback(() => { + dispatch({ + type: 'focus-previous-page', + }) + }, []) + + const focusOption = useCallback((value: T | undefined) => { + if (value !== undefined) { + dispatch({ + type: 'set-focus', + value, + }) + } + }, []) + + const visibleOptions = useMemo(() => { + return options + .map((option, index) => ({ + ...option, + index, + })) + .slice(state.visibleFromIndex, state.visibleToIndex) + }, [options, state.visibleFromIndex, state.visibleToIndex]) + + // Validate that focusedValue exists in current options. + // This handles the case where options change during render but the reset + // action hasn't been processed yet - without this, the cursor would disappear + // because focusedValue points to an option that no longer exists. + const validatedFocusedValue = useMemo(() => { + if (state.focusedValue === undefined) { + return undefined + } + const exists = options.some(opt => opt.value === state.focusedValue) + if (exists) { + return state.focusedValue + } + // Fall back to first option if focused value doesn't exist + return options[0]?.value + }, [state.focusedValue, options]) + + const isInInput = useMemo(() => { + const focusedOption = options.find( + opt => opt.value === validatedFocusedValue, + ) + return focusedOption?.type === 'input' + }, [validatedFocusedValue, options]) + + // Call onFocus with the validated value (what's actually displayed), + // not the internal state value which may be stale if options changed. + // Use ref to avoid re-running when callback reference changes. + useEffect(() => { + if (validatedFocusedValue !== undefined) { + onFocusRef.current?.(validatedFocusedValue) + } + }, [validatedFocusedValue]) + + // Allow parent to programmatically set focus via focusValue prop + useEffect(() => { + if (focusValue !== undefined) { + dispatch({ + type: 'set-focus', + value: focusValue, + }) + } + }, [focusValue]) + + // Compute 1-based focused index for scroll position display + const focusedIndex = useMemo(() => { + if (validatedFocusedValue === undefined) { + return 0 + } + const index = options.findIndex(opt => opt.value === validatedFocusedValue) + return index >= 0 ? index + 1 : 0 + }, [validatedFocusedValue, options]) + + return { + focusedValue: validatedFocusedValue, + focusedIndex, + visibleFromIndex: state.visibleFromIndex, + visibleToIndex: state.visibleToIndex, + visibleOptions, + isInInput: isInInput ?? false, + focusNextOption, + focusPreviousOption, + focusNextPage, + focusPreviousPage, + focusOption, + options, + } +} diff --git a/src/components/CustomSelect/use-select-state.ts b/src/components/CustomSelect/use-select-state.ts new file mode 100644 index 0000000..3951d95 --- /dev/null +++ b/src/components/CustomSelect/use-select-state.ts @@ -0,0 +1,157 @@ +import { useCallback, useState } from 'react' +import type { OptionWithDescription } from './select.js' +import { useSelectNavigation } from './use-select-navigation.js' + +export type UseSelectStateProps = { + /** + * Number of items to display. + * + * @default 5 + */ + visibleOptionCount?: number + + /** + * Options. + */ + options: OptionWithDescription[] + + /** + * Initially selected option's value. + */ + defaultValue?: T + + /** + * Callback for selecting an option. + */ + onChange?: (value: T) => void + + /** + * Callback for canceling the select. + */ + onCancel?: () => void + + /** + * Callback for focusing an option. + */ + onFocus?: (value: T) => void + + /** + * Value to focus + */ + focusValue?: T +} + +export type SelectState = { + /** + * Value of the currently focused option. + */ + focusedValue: T | undefined + + /** + * 1-based index of the focused option in the full list. + * Returns 0 if no option is focused. + */ + focusedIndex: number + + /** + * Index of the first visible option. + */ + visibleFromIndex: number + + /** + * Index of the last visible option. + */ + visibleToIndex: number + + /** + * Value of the selected option. + */ + value: T | undefined + + /** + * All options. + */ + options: OptionWithDescription[] + + /** + * Visible options. + */ + visibleOptions: Array & { index: number }> + + /** + * Whether the focused option is an input type. + */ + isInInput: boolean + + /** + * Focus next option and scroll the list down, if needed. + */ + focusNextOption: () => void + + /** + * Focus previous option and scroll the list up, if needed. + */ + focusPreviousOption: () => void + + /** + * Focus next page and scroll the list down by a page. + */ + focusNextPage: () => void + + /** + * Focus previous page and scroll the list up by a page. + */ + focusPreviousPage: () => void + + /** + * Focus a specific option by value. + */ + focusOption: (value: T | undefined) => void + + /** + * Select currently focused option. + */ + selectFocusedOption: () => void + + /** + * Callback for selecting an option. + */ + onChange?: (value: T) => void + + /** + * Callback for canceling the select. + */ + onCancel?: () => void +} + +export function useSelectState({ + visibleOptionCount = 5, + options, + defaultValue, + onChange, + onCancel, + onFocus, + focusValue, +}: UseSelectStateProps): SelectState { + const [value, setValue] = useState(defaultValue) + + const navigation = useSelectNavigation({ + visibleOptionCount, + options, + initialFocusValue: undefined, + onFocus, + focusValue, + }) + + const selectFocusedOption = useCallback(() => { + setValue(navigation.focusedValue) + }, [navigation.focusedValue]) + + return { + ...navigation, + value, + selectFocusedOption, + onChange, + onCancel, + } +} diff --git a/src/components/DesktopHandoff.tsx b/src/components/DesktopHandoff.tsx new file mode 100644 index 0000000..7e70733 --- /dev/null +++ b/src/components/DesktopHandoff.tsx @@ -0,0 +1,193 @@ +import { c as _c } from "react/compiler-runtime"; +import React, { useEffect, useState } from 'react'; +import type { CommandResultDisplay } from '../commands.js'; +// eslint-disable-next-line custom-rules/prefer-use-keybindings -- raw input for "any key" dismiss and y/n prompt +import { Box, Text, useInput } from '../ink.js'; +import { openBrowser } from '../utils/browser.js'; +import { getDesktopInstallStatus, openCurrentSessionInDesktop } from '../utils/desktopDeepLink.js'; +import { errorMessage } from '../utils/errors.js'; +import { gracefulShutdown } from '../utils/gracefulShutdown.js'; +import { flushSessionStorage } from '../utils/sessionStorage.js'; +import { LoadingState } from './design-system/LoadingState.js'; +const DESKTOP_DOCS_URL = 'https://clau.de/desktop'; +export function getDownloadUrl(): string { + switch (process.platform) { + case 'win32': + return 'https://claude.ai/api/desktop/win32/x64/exe/latest/redirect'; + default: + return 'https://claude.ai/api/desktop/darwin/universal/dmg/latest/redirect'; + } +} +type DesktopHandoffState = 'checking' | 'prompt-download' | 'flushing' | 'opening' | 'success' | 'error'; +type Props = { + onDone: (result?: string, options?: { + display?: CommandResultDisplay; + }) => void; +}; +export function DesktopHandoff(t0) { + const $ = _c(20); + const { + onDone + } = t0; + const [state, setState] = useState("checking"); + const [error, setError] = useState(null); + const [downloadMessage, setDownloadMessage] = useState(""); + let t1; + if ($[0] !== error || $[1] !== onDone || $[2] !== state) { + t1 = input => { + if (state === "error") { + onDone(error ?? "Unknown error", { + display: "system" + }); + return; + } + if (state === "prompt-download") { + if (input === "y" || input === "Y") { + openBrowser(getDownloadUrl()).catch(_temp); + onDone(`Starting download. Re-run /desktop once you\u2019ve installed the app.\nLearn more at ${DESKTOP_DOCS_URL}`, { + display: "system" + }); + } else { + if (input === "n" || input === "N") { + onDone(`The desktop app is required for /desktop. Learn more at ${DESKTOP_DOCS_URL}`, { + display: "system" + }); + } + } + } + }; + $[0] = error; + $[1] = onDone; + $[2] = state; + $[3] = t1; + } else { + t1 = $[3]; + } + useInput(t1); + let t2; + let t3; + if ($[4] !== onDone) { + t2 = () => { + const performHandoff = async function performHandoff() { + setState("checking"); + const installStatus = await getDesktopInstallStatus(); + if (installStatus.status === "not-installed") { + setDownloadMessage("Claude Desktop is not installed."); + setState("prompt-download"); + return; + } + if (installStatus.status === "version-too-old") { + setDownloadMessage(`Claude Desktop needs to be updated (found v${installStatus.version}, need v1.1.2396+).`); + setState("prompt-download"); + return; + } + setState("flushing"); + await flushSessionStorage(); + setState("opening"); + const result = await openCurrentSessionInDesktop(); + if (!result.success) { + setError(result.error ?? "Failed to open Claude Desktop"); + setState("error"); + return; + } + setState("success"); + setTimeout(_temp2, 500, onDone); + }; + performHandoff().catch(err => { + setError(errorMessage(err)); + setState("error"); + }); + }; + t3 = [onDone]; + $[4] = onDone; + $[5] = t2; + $[6] = t3; + } else { + t2 = $[5]; + t3 = $[6]; + } + useEffect(t2, t3); + if (state === "error") { + let t4; + if ($[7] !== error) { + t4 = Error: {error}; + $[7] = error; + $[8] = t4; + } else { + t4 = $[8]; + } + let t5; + if ($[9] === Symbol.for("react.memo_cache_sentinel")) { + t5 = Press any key to continue…; + $[9] = t5; + } else { + t5 = $[9]; + } + let t6; + if ($[10] !== t4) { + t6 = {t4}{t5}; + $[10] = t4; + $[11] = t6; + } else { + t6 = $[11]; + } + return t6; + } + if (state === "prompt-download") { + let t4; + if ($[12] !== downloadMessage) { + t4 = {downloadMessage}; + $[12] = downloadMessage; + $[13] = t4; + } else { + t4 = $[13]; + } + let t5; + if ($[14] === Symbol.for("react.memo_cache_sentinel")) { + t5 = Download now? (y/n); + $[14] = t5; + } else { + t5 = $[14]; + } + let t6; + if ($[15] !== t4) { + t6 = {t4}{t5}; + $[15] = t4; + $[16] = t6; + } else { + t6 = $[16]; + } + return t6; + } + let t4; + if ($[17] === Symbol.for("react.memo_cache_sentinel")) { + t4 = { + checking: "Checking for Claude Desktop\u2026", + flushing: "Saving session\u2026", + opening: "Opening Claude Desktop\u2026", + success: "Opening in Claude Desktop\u2026" + }; + $[17] = t4; + } else { + t4 = $[17]; + } + const messages = t4; + const t5 = messages[state]; + let t6; + if ($[18] !== t5) { + t6 = ; + $[18] = t5; + $[19] = t6; + } else { + t6 = $[19]; + } + return t6; +} +async function _temp2(onDone_0) { + onDone_0("Session transferred to Claude Desktop", { + display: "system" + }); + await gracefulShutdown(0, "other"); +} +function _temp() {} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsInVzZUVmZmVjdCIsInVzZVN0YXRlIiwiQ29tbWFuZFJlc3VsdERpc3BsYXkiLCJCb3giLCJUZXh0IiwidXNlSW5wdXQiLCJvcGVuQnJvd3NlciIsImdldERlc2t0b3BJbnN0YWxsU3RhdHVzIiwib3BlbkN1cnJlbnRTZXNzaW9uSW5EZXNrdG9wIiwiZXJyb3JNZXNzYWdlIiwiZ3JhY2VmdWxTaHV0ZG93biIsImZsdXNoU2Vzc2lvblN0b3JhZ2UiLCJMb2FkaW5nU3RhdGUiLCJERVNLVE9QX0RPQ1NfVVJMIiwiZ2V0RG93bmxvYWRVcmwiLCJwcm9jZXNzIiwicGxhdGZvcm0iLCJEZXNrdG9wSGFuZG9mZlN0YXRlIiwiUHJvcHMiLCJvbkRvbmUiLCJyZXN1bHQiLCJvcHRpb25zIiwiZGlzcGxheSIsIkRlc2t0b3BIYW5kb2ZmIiwidDAiLCIkIiwiX2MiLCJzdGF0ZSIsInNldFN0YXRlIiwiZXJyb3IiLCJzZXRFcnJvciIsImRvd25sb2FkTWVzc2FnZSIsInNldERvd25sb2FkTWVzc2FnZSIsInQxIiwiaW5wdXQiLCJjYXRjaCIsIl90ZW1wIiwidDIiLCJ0MyIsInBlcmZvcm1IYW5kb2ZmIiwiaW5zdGFsbFN0YXR1cyIsInN0YXR1cyIsInZlcnNpb24iLCJzdWNjZXNzIiwic2V0VGltZW91dCIsIl90ZW1wMiIsImVyciIsInQ0IiwidDUiLCJTeW1ib2wiLCJmb3IiLCJ0NiIsImNoZWNraW5nIiwiZmx1c2hpbmciLCJvcGVuaW5nIiwibWVzc2FnZXMiLCJvbkRvbmVfMCJdLCJzb3VyY2VzIjpbIkRlc2t0b3BIYW5kb2ZmLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgUmVhY3QsIHsgdXNlRWZmZWN0LCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHR5cGUgeyBDb21tYW5kUmVzdWx0RGlzcGxheSB9IGZyb20gJy4uL2NvbW1hbmRzLmpzJ1xuLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIGN1c3RvbS1ydWxlcy9wcmVmZXItdXNlLWtleWJpbmRpbmdzIC0tIHJhdyBpbnB1dCBmb3IgXCJhbnkga2V5XCIgZGlzbWlzcyBhbmQgeS9uIHByb21wdFxuaW1wb3J0IHsgQm94LCBUZXh0LCB1c2VJbnB1dCB9IGZyb20gJy4uL2luay5qcydcbmltcG9ydCB7IG9wZW5Ccm93c2VyIH0gZnJvbSAnLi4vdXRpbHMvYnJvd3Nlci5qcydcbmltcG9ydCB7XG4gIGdldERlc2t0b3BJbnN0YWxsU3RhdHVzLFxuICBvcGVuQ3VycmVudFNlc3Npb25JbkRlc2t0b3AsXG59IGZyb20gJy4uL3V0aWxzL2Rlc2t0b3BEZWVwTGluay5qcydcbmltcG9ydCB7IGVycm9yTWVzc2FnZSB9IGZyb20gJy4uL3V0aWxzL2Vycm9ycy5qcydcbmltcG9ydCB7IGdyYWNlZnVsU2h1dGRvd24gfSBmcm9tICcuLi91dGlscy9ncmFjZWZ1bFNodXRkb3duLmpzJ1xuaW1wb3J0IHsgZmx1c2hTZXNzaW9uU3RvcmFnZSB9IGZyb20gJy4uL3V0aWxzL3Nlc3Npb25TdG9yYWdlLmpzJ1xuaW1wb3J0IHsgTG9hZGluZ1N0YXRlIH0gZnJvbSAnLi9kZXNpZ24tc3lzdGVtL0xvYWRpbmdTdGF0ZS5qcydcblxuY29uc3QgREVTS1RPUF9ET0NTX1VSTCA9ICdodHRwczovL2NsYXUuZGUvZGVza3RvcCdcblxuZXhwb3J0IGZ1bmN0aW9uIGdldERvd25sb2FkVXJsKCk6IHN0cmluZyB7XG4gIHN3aXRjaCAocHJvY2Vzcy5wbGF0Zm9ybSkge1xuICAgIGNhc2UgJ3dpbjMyJzpcbiAgICAgIHJldHVybiAnaHR0cHM6Ly9jbGF1ZGUuYWkvYXBpL2Rlc2t0b3Avd2luMzIveDY0L2V4ZS9sYXRlc3QvcmVkaXJlY3QnXG4gICAgZGVmYXVsdDpcbiAgICAgIHJldHVybiAnaHR0cHM6Ly9jbGF1ZGUuYWkvYXBpL2Rlc2t0b3AvZGFyd2luL3VuaXZlcnNhbC9kbWcvbGF0ZXN0L3JlZGlyZWN0J1xuICB9XG59XG5cbnR5cGUgRGVza3RvcEhhbmRvZmZTdGF0ZSA9XG4gIHwgJ2NoZWNraW5nJ1xuICB8ICdwcm9tcHQtZG93bmxvYWQnXG4gIHwgJ2ZsdXNoaW5nJ1xuICB8ICdvcGVuaW5nJ1xuICB8ICdzdWNjZXNzJ1xuICB8ICdlcnJvcidcblxudHlwZSBQcm9wcyA9IHtcbiAgb25Eb25lOiAoXG4gICAgcmVzdWx0Pzogc3RyaW5nLFxuICAgIG9wdGlvbnM/OiB7IGRpc3BsYXk/OiBDb21tYW5kUmVzdWx0RGlzcGxheSB9LFxuICApID0+IHZvaWRcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIERlc2t0b3BIYW5kb2ZmKHsgb25Eb25lIH06IFByb3BzKTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgY29uc3QgW3N0YXRlLCBzZXRTdGF0ZV0gPSB1c2VTdGF0ZTxEZXNrdG9wSGFuZG9mZlN0YXRlPignY2hlY2tpbmcnKVxuICBjb25zdCBbZXJyb3IsIHNldEVycm9yXSA9IHVzZVN0YXRlPHN0cmluZyB8IG51bGw+KG51bGwpXG4gIGNvbnN0IFtkb3dubG9hZE1lc3NhZ2UsIHNldERvd25sb2FkTWVzc2FnZV0gPSB1c2VTdGF0ZTxzdHJpbmc+KCcnKVxuXG4gIC8vIEhhbmRsZSBrZXlib2FyZCBpbnB1dCBmb3IgZXJyb3IgYW5kIHByb21wdC1kb3dubG9hZCBzdGF0ZXNcbiAgdXNlSW5wdXQoaW5wdXQgPT4ge1xuICAgIGlmIChzdGF0ZSA9PT0gJ2Vycm9yJykge1xuICAgICAgb25Eb25lKGVycm9yID8/ICdVbmtub3duIGVycm9yJywgeyBkaXNwbGF5OiAnc3lzdGVtJyB9KVxuICAgICAgcmV0dXJuXG4gICAgfVxuICAgIGlmIChzdGF0ZSA9PT0gJ3Byb21wdC1kb3dubG9hZCcpIHtcbiAgICAgIGlmIChpbnB1dCA9PT0gJ3knIHx8IGlucHV0ID09PSAnWScpIHtcbiAgICAgICAgb3BlbkJyb3dzZXIoZ2V0RG93bmxvYWRVcmwoKSkuY2F0Y2goKCkgPT4ge30pXG4gICAgICAgIG9uRG9uZShcbiAgICAgICAgICBgU3RhcnRpbmcgZG93bmxvYWQuIFJlLXJ1biAvZGVza3RvcCBvbmNlIHlvdVxcdTIwMTl2ZSBpbnN0YWxsZWQgdGhlIGFwcC5cXG5MZWFybiBtb3JlIGF0ICR7REVTS1RPUF9ET0NTX1VSTH1gLFxuICAgICAgICAgIHsgZGlzcGxheTogJ3N5c3RlbScgfSxcbiAgICAgICAgKVxuICAgICAgfSBlbHNlIGlmIChpbnB1dCA9PT0gJ24nIHx8IGlucHV0ID09PSAnTicpIHtcbiAgICAgICAgb25Eb25lKFxuICAgICAgICAgIGBUaGUgZGVza3RvcCBhcHAgaXMgcmVxdWlyZWQgZm9yIC9kZXNrdG9wLiBMZWFybiBtb3JlIGF0ICR7REVTS1RPUF9ET0NTX1VSTH1gLFxuICAgICAgICAgIHsgZGlzcGxheTogJ3N5c3RlbScgfSxcbiAgICAgICAgKVxuICAgICAgfVxuICAgIH1cbiAgfSlcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGFzeW5jIGZ1bmN0aW9uIHBlcmZvcm1IYW5kb2ZmKCk6IFByb21pc2U8dm9pZD4ge1xuICAgICAgLy8gQ2hlY2sgRGVza3RvcCBpbnN0YWxsIHN0YXR1c1xuICAgICAgc2V0U3RhdGUoJ2NoZWNraW5nJylcbiAgICAgIGNvbnN0IGluc3RhbGxTdGF0dXMgPSBhd2FpdCBnZXREZXNrdG9wSW5zdGFsbFN0YXR1cygpXG5cbiAgICAgIGlmIChpbnN0YWxsU3RhdHVzLnN0YXR1cyA9PT0gJ25vdC1pbnN0YWxsZWQnKSB7XG4gICAgICAgIHNldERvd25sb2FkTWVzc2FnZSgnQ2xhdWRlIERlc2t0b3AgaXMgbm90IGluc3RhbGxlZC4nKVxuICAgICAgICBzZXRTdGF0ZSgncHJvbXB0LWRvd25sb2FkJylcbiAgICAgICAgcmV0dXJuXG4gICAgICB9XG5cbiAgICAgIGlmIChpbnN0YWxsU3RhdHVzLnN0YXR1cyA9PT0gJ3ZlcnNpb24tdG9vLW9sZCcpIHtcbiAgICAgICAgc2V0RG93bmxvYWRNZXNzYWdlKFxuICAgICAgICAgIGBDbGF1ZGUgRGVza3RvcCBuZWVkcyB0byBiZSB1cGRhdGVkIChmb3VuZCB2JHtpbnN0YWxsU3RhdHVzLnZlcnNpb259LCBuZWVkIHYxLjEuMjM5NispLmAsXG4gICAgICAgIClcbiAgICAgICAgc2V0U3RhdGUoJ3Byb21wdC1kb3dubG9hZCcpXG4gICAgICAgIHJldHVyblxuICAgICAgfVxuXG4gICAgICAvLyBGbHVzaCBzZXNzaW9uIHN0b3JhZ2UgdG8gZW5zdXJlIHRyYW5zY3JpcHQgaXMgZnVsbHkgd3JpdHRlblxuICAgICAgc2V0U3RhdGUoJ2ZsdXNoaW5nJylcbiAgICAgIGF3YWl0IGZsdXNoU2Vzc2lvblN0b3JhZ2UoKVxuXG4gICAgICAvLyBPcGVuIHRoZSBkZWVwIGxpbmsgKHVzZXMgY2xhdWRlLWRldjovLyBpbiBkZXYgbW9kZSlcbiAgICAgIHNldFN0YXRlKCdvcGVuaW5nJylcbiAgICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IG9wZW5DdXJyZW50U2Vzc2lvbkluRGVza3RvcCgpXG5cbiAgICAgIGlmICghcmVzdWx0LnN1Y2Nlc3MpIHtcbiAgICAgICAgc2V0RXJyb3IocmVzdWx0LmVycm9yID8/ICdGYWlsZWQgdG8gb3BlbiBDbGF1ZGUgRGVza3RvcCcpXG4gICAgICAgIHNldFN0YXRlKCdlcnJvcicpXG4gICAgICAgIHJldHVyblxuICAgICAgfVxuXG4gICAgICAvLyBTdWNjZXNzIC0gZXhpdCB0aGUgQ0xJXG4gICAgICBzZXRTdGF0ZSgnc3VjY2VzcycpXG5cbiAgICAgIC8vIEdpdmUgdGhlIHVzZXIgYSBtb21lbnQgdG8gc2VlIHRoZSBzdWNjZXNzIG1lc3NhZ2VcbiAgICAgIHNldFRpbWVvdXQoXG4gICAgICAgIGFzeW5jIChvbkRvbmU6IFByb3BzWydvbkRvbmUnXSkgPT4ge1xuICAgICAgICAgIG9uRG9uZSgnU2Vzc2lvbiB0cmFuc2ZlcnJlZCB0byBDbGF1ZGUgRGVza3RvcCcsIHsgZGlzcGxheTogJ3N5c3RlbScgfSlcbiAgICAgICAgICBhd2FpdCBncmFjZWZ1bFNodXRkb3duKDAsICdvdGhlcicpXG4gICAgICAgIH0sXG4gICAgICAgIDUwMCxcbiAgICAgICAgb25Eb25lLFxuICAgICAgKVxuICAgIH1cblxuICAgIHBlcmZvcm1IYW5kb2ZmKCkuY2F0Y2goZXJyID0+IHtcbiAgICAgIHNldEVycm9yKGVycm9yTWVzc2FnZShlcnIpKVxuICAgICAgc2V0U3RhdGUoJ2Vycm9yJylcbiAgICB9KVxuICB9LCBbb25Eb25lXSlcblxuICBpZiAoc3RhdGUgPT09ICdlcnJvcicpIHtcbiAgICByZXR1cm4gKFxuICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCIgcGFkZGluZ1g9ezJ9PlxuICAgICAgICA8VGV4dCBjb2xvcj1cImVycm9yXCI+RXJyb3I6IHtlcnJvcn08L1RleHQ+XG4gICAgICAgIDxUZXh0IGRpbUNvbG9yPlByZXNzIGFueSBrZXkgdG8gY29udGludWXigKY8L1RleHQ+XG4gICAgICA8L0JveD5cbiAgICApXG4gIH1cblxuICBpZiAoc3RhdGUgPT09ICdwcm9tcHQtZG93bmxvYWQnKSB7XG4gICAgcmV0dXJuIChcbiAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiIHBhZGRpbmdYPXsyfT5cbiAgICAgICAgPFRleHQ+e2Rvd25sb2FkTWVzc2FnZX08L1RleHQ+XG4gICAgICAgIDxUZXh0PkRvd25sb2FkIG5vdz8gKHkvbik8L1RleHQ+XG4gICAgICA8L0JveD5cbiAgICApXG4gIH1cblxuICBjb25zdCBtZXNzYWdlczogUmVjb3JkPFxuICAgIEV4Y2x1ZGU8RGVza3RvcEhhbmRvZmZTdGF0ZSwgJ2Vycm9yJyB8ICdwcm9tcHQtZG93bmxvYWQnPixcbiAgICBzdHJpbmdcbiAgPiA9IHtcbiAgICBjaGVja2luZzogJ0NoZWNraW5nIGZvciBDbGF1ZGUgRGVza3RvcOKApicsXG4gICAgZmx1c2hpbmc6ICdTYXZpbmcgc2Vzc2lvbuKApicsXG4gICAgb3BlbmluZzogJ09wZW5pbmcgQ2xhdWRlIERlc2t0b3DigKYnLFxuICAgIHN1Y2Nlc3M6ICdPcGVuaW5nIGluIENsYXVkZSBEZXNrdG9w4oCmJyxcbiAgfVxuXG4gIHJldHVybiA8TG9hZGluZ1N0YXRlIG1lc3NhZ2U9e21lc3NhZ2VzW3N0YXRl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 \ No newline at end of file diff --git a/src/components/DesktopUpsell/DesktopUpsellStartup.tsx b/src/components/DesktopUpsell/DesktopUpsellStartup.tsx new file mode 100644 index 0000000..e919039 --- /dev/null +++ b/src/components/DesktopUpsell/DesktopUpsellStartup.tsx @@ -0,0 +1,171 @@ +import { c as _c } from "react/compiler-runtime"; +import * as React from 'react'; +import { useEffect, useState } from 'react'; +import { Box, Text } from '../../ink.js'; +import { getDynamicConfig_CACHED_MAY_BE_STALE } from '../../services/analytics/growthbook.js'; +import { logEvent } from '../../services/analytics/index.js'; +import { getGlobalConfig, saveGlobalConfig } from '../../utils/config.js'; +import { Select } from '../CustomSelect/select.js'; +import { DesktopHandoff } from '../DesktopHandoff.js'; +import { PermissionDialog } from '../permissions/PermissionDialog.js'; +type DesktopUpsellConfig = { + enable_shortcut_tip: boolean; + enable_startup_dialog: boolean; +}; +const DESKTOP_UPSELL_DEFAULT: DesktopUpsellConfig = { + enable_shortcut_tip: false, + enable_startup_dialog: false +}; +export function getDesktopUpsellConfig(): DesktopUpsellConfig { + return getDynamicConfig_CACHED_MAY_BE_STALE('tengu_desktop_upsell', DESKTOP_UPSELL_DEFAULT); +} +function isSupportedPlatform(): boolean { + return process.platform === 'darwin' || process.platform === 'win32' && process.arch === 'x64'; +} +export function shouldShowDesktopUpsellStartup(): boolean { + if (!isSupportedPlatform()) return false; + if (!getDesktopUpsellConfig().enable_startup_dialog) return false; + const config = getGlobalConfig(); + if (config.desktopUpsellDismissed) return false; + if ((config.desktopUpsellSeenCount ?? 0) >= 3) return false; + return true; +} +type DesktopUpsellSelection = 'try' | 'not-now' | 'never'; +type Props = { + onDone: () => void; +}; +export function DesktopUpsellStartup(t0) { + const $ = _c(14); + const { + onDone + } = t0; + const [showHandoff, setShowHandoff] = useState(false); + let t1; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t1 = []; + $[0] = t1; + } else { + t1 = $[0]; + } + useEffect(_temp, t1); + if (showHandoff) { + let t2; + if ($[1] !== onDone) { + t2 = onDone()} />; + $[1] = onDone; + $[2] = t2; + } else { + t2 = $[2]; + } + return t2; + } + let t2; + if ($[3] !== onDone) { + t2 = function handleSelect(value) { + switch (value) { + case "try": + { + setShowHandoff(true); + return; + } + case "never": + { + saveGlobalConfig(_temp2); + onDone(); + return; + } + case "not-now": + { + onDone(); + return; + } + } + }; + $[3] = onDone; + $[4] = t2; + } else { + t2 = $[4]; + } + const handleSelect = t2; + let t3; + if ($[5] === Symbol.for("react.memo_cache_sentinel")) { + t3 = { + label: "Open in Claude Code Desktop", + value: "try" as const + }; + $[5] = t3; + } else { + t3 = $[5]; + } + let t4; + if ($[6] === Symbol.for("react.memo_cache_sentinel")) { + t4 = { + label: "Not now", + value: "not-now" as const + }; + $[6] = t4; + } else { + t4 = $[6]; + } + let t5; + if ($[7] === Symbol.for("react.memo_cache_sentinel")) { + t5 = [t3, t4, { + label: "Don't ask again", + value: "never" as const + }]; + $[7] = t5; + } else { + t5 = $[7]; + } + const options = t5; + let t6; + if ($[8] === Symbol.for("react.memo_cache_sentinel")) { + t6 = Same Claude Code with visual diffs, live app preview, parallel sessions, and more.; + $[8] = t6; + } else { + t6 = $[8]; + } + let t7; + if ($[9] !== handleSelect) { + t7 = () => handleSelect("not-now"); + $[9] = handleSelect; + $[10] = t7; + } else { + t7 = $[10]; + } + let t8; + if ($[11] !== handleSelect || $[12] !== t7) { + t8 = {t6} onChange(value_0 as 'accept' | 'exit')} />; + $[9] = onChange; + $[10] = t7; + } else { + t7 = $[10]; + } + let t8; + if ($[11] !== t5 || $[12] !== t7) { + t8 = {t5}{t7}; + $[11] = t5; + $[12] = t7; + $[13] = t8; + } else { + t8 = $[13]; + } + return t8; +} +function _temp2(c) { + return c.kind === "plugin" ? `plugin:${c.name}@${c.marketplace}` : `server:${c.name}`; +} +function _temp() { + gracefulShutdownSync(0); +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsInVzZUNhbGxiYWNrIiwiQ2hhbm5lbEVudHJ5IiwiQm94IiwiVGV4dCIsImdyYWNlZnVsU2h1dGRvd25TeW5jIiwiU2VsZWN0IiwiRGlhbG9nIiwiUHJvcHMiLCJjaGFubmVscyIsIm9uQWNjZXB0IiwiRGV2Q2hhbm5lbHNEaWFsb2ciLCJ0MCIsIiQiLCJfYyIsInQxIiwib25DaGFuZ2UiLCJ2YWx1ZSIsImJiMiIsImhhbmRsZUVzY2FwZSIsIl90ZW1wIiwidDIiLCJ0MyIsIlN5bWJvbCIsImZvciIsInQ0IiwibWFwIiwiX3RlbXAyIiwiam9pbiIsInQ1IiwidDYiLCJsYWJlbCIsInQ3IiwidmFsdWVfMCIsInQ4IiwiYyIsImtpbmQiLCJuYW1lIiwibWFya2V0cGxhY2UiXSwic291cmNlcyI6WyJEZXZDaGFubmVsc0RpYWxvZy50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFJlYWN0LCB7IHVzZUNhbGxiYWNrIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgdHlwZSB7IENoYW5uZWxFbnRyeSB9IGZyb20gJy4uL2Jvb3RzdHJhcC9zdGF0ZS5qcydcbmltcG9ydCB7IEJveCwgVGV4dCB9IGZyb20gJy4uL2luay5qcydcbmltcG9ydCB7IGdyYWNlZnVsU2h1dGRvd25TeW5jIH0gZnJvbSAnLi4vdXRpbHMvZ3JhY2VmdWxTaHV0ZG93bi5qcydcbmltcG9ydCB7IFNlbGVjdCB9IGZyb20gJy4vQ3VzdG9tU2VsZWN0L2luZGV4LmpzJ1xuaW1wb3J0IHsgRGlhbG9nIH0gZnJvbSAnLi9kZXNpZ24tc3lzdGVtL0RpYWxvZy5qcydcblxudHlwZSBQcm9wcyA9IHtcbiAgY2hhbm5lbHM6IENoYW5uZWxFbnRyeVtdXG4gIG9uQWNjZXB0KCk6IHZvaWRcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIERldkNoYW5uZWxzRGlhbG9nKHtcbiAgY2hhbm5lbHMsXG4gIG9uQWNjZXB0LFxufTogUHJvcHMpOiBSZWFjdC5SZWFjdE5vZGUge1xuICBmdW5jdGlvbiBvbkNoYW5nZSh2YWx1ZTogJ2FjY2VwdCcgfCAnZXhpdCcpIHtcbiAgICBzd2l0Y2ggKHZhbHVlKSB7XG4gICAgICBjYXNlICdhY2NlcHQnOlxuICAgICAgICBvbkFjY2VwdCgpXG4gICAgICAgIGJyZWFrXG4gICAgICBjYXNlICdleGl0JzpcbiAgICAgICAgZ3JhY2VmdWxTaHV0ZG93blN5bmMoMSlcbiAgICAgICAgYnJlYWtcbiAgICB9XG4gIH1cblxuICBjb25zdCBoYW5kbGVFc2NhcGUgPSB1c2VDYWxsYmFjaygoKSA9PiB7XG4gICAgZ3JhY2VmdWxTaHV0ZG93blN5bmMoMClcbiAgfSwgW10pXG5cbiAgcmV0dXJuIChcbiAgICA8RGlhbG9nXG4gICAgICB0aXRsZT1cIldBUk5JTkc6IExvYWRpbmcgZGV2ZWxvcG1lbnQgY2hhbm5lbHNcIlxuICAgICAgY29sb3I9XCJlcnJvclwiXG4gICAgICBvbkNhbmNlbD17aGFuZGxlRXNjYXBlfVxuICAgID5cbiAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiIGdhcD17MX0+XG4gICAgICAgIDxUZXh0PlxuICAgICAgICAgIC0tZGFuZ2Vyb3VzbHktbG9hZC1kZXZlbG9wbWVudC1jaGFubmVscyBpcyBmb3IgbG9jYWwgY2hhbm5lbFxuICAgICAgICAgIGRldmVsb3BtZW50IG9ubHkuIERvIG5vdCB1c2UgdGhpcyBvcHRpb24gdG8gcnVuIGNoYW5uZWxzIHlvdSBoYXZlXG4gICAgICAgICAgZG93bmxvYWRlZCBvZmYgdGhlIGludGVybmV0LlxuICAgICAgICA8L1RleHQ+XG4gICAgICAgIDxUZXh0PlBsZWFzZSB1c2UgLS1jaGFubmVscyB0byBydW4gYSBsaXN0IG9mIGFwcHJvdmVkIGNoYW5uZWxzLjwvVGV4dD5cbiAgICAgICAgPFRleHQgZGltQ29sb3I+XG4gICAgICAgICAgQ2hhbm5lbHM6eycgJ31cbiAgICAgICAgICB7Y2hhbm5lbHNcbiAgICAgICAgICAgIC5tYXAoYyA9PlxuICAgICAgICAgICAgICBjLmtpbmQgPT09ICdwbHVnaW4nXG4gICAgICAgICAgICAgICAgPyBgcGx1Z2luOiR7Yy5uYW1lfUAke2MubWFya2V0cGxhY2V9YFxuICAgICAgICAgICAgICAgIDogYHNlcnZlcjoke2MubmFtZX1gLFxuICAgICAgICAgICAgKVxuICAgICAgICAgICAgLmpvaW4oJywgJyl9XG4gICAgICAgIDwvVGV4dD5cbiAgICAgIDwvQm94PlxuXG4gICAgICA8U2VsZWN0XG4gICAgICAgIG9wdGlvbnM9e1tcbiAgICAgICAgICB7IGxhYmVsOiAnSSBhbSB1c2luZyB0aGlzIGZvciBsb2NhbCBkZXZlbG9wbWVudCcsIHZhbHVlOiAnYWNjZXB0JyB9LFxuICAgICAgICAgIHsgbGFiZWw6ICdFeGl0JywgdmFsdWU6ICdleGl0JyB9LFxuICAgICAgICBdfVxuICAgICAgICBvbkNoYW5nZT17dmFsdWUgPT4gb25DaGFuZ2UodmFsdWUgYXMgJ2FjY2VwdCcgfCAnZXhpdCcpfVxuICAgICAgLz5cbiAgICA8L0RpYWxvZz5cbiAgKVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsT0FBT0EsS0FBSyxJQUFJQyxXQUFXLFFBQVEsT0FBTztBQUMxQyxjQUFjQyxZQUFZLFFBQVEsdUJBQXVCO0FBQ3pELFNBQVNDLEdBQUcsRUFBRUMsSUFBSSxRQUFRLFdBQVc7QUFDckMsU0FBU0Msb0JBQW9CLFFBQVEsOEJBQThCO0FBQ25FLFNBQVNDLE1BQU0sUUFBUSx5QkFBeUI7QUFDaEQsU0FBU0MsTUFBTSxRQUFRLDJCQUEyQjtBQUVsRCxLQUFLQyxLQUFLLEdBQUc7RUFDWEMsUUFBUSxFQUFFUCxZQUFZLEVBQUU7RUFDeEJRLFFBQVEsRUFBRSxFQUFFLElBQUk7QUFDbEIsQ0FBQztBQUVELE9BQU8sU0FBQUMsa0JBQUFDLEVBQUE7RUFBQSxNQUFBQyxDQUFBLEdBQUFDLEVBQUE7RUFBMkI7SUFBQUwsUUFBQTtJQUFBQztFQUFBLElBQUFFLEVBRzFCO0VBQUEsSUFBQUcsRUFBQTtFQUFBLElBQUFGLENBQUEsUUFBQUgsUUFBQTtJQUNOSyxFQUFBLFlBQUFDLFNBQUFDLEtBQUE7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 \ No newline at end of file diff --git a/src/components/DiagnosticsDisplay.tsx b/src/components/DiagnosticsDisplay.tsx new file mode 100644 index 0000000..6eb3ad8 --- /dev/null +++ b/src/components/DiagnosticsDisplay.tsx @@ -0,0 +1,95 @@ +import { c as _c } from "react/compiler-runtime"; +import { relative } from 'path'; +import React from 'react'; +import { Box, Text } from '../ink.js'; +import { DiagnosticTrackingService } from '../services/diagnosticTracking.js'; +import type { Attachment } from '../utils/attachments.js'; +import { getCwd } from '../utils/cwd.js'; +import { CtrlOToExpand } from './CtrlOToExpand.js'; +import { MessageResponse } from './MessageResponse.js'; +type DiagnosticsAttachment = Extract; +type DiagnosticsDisplayProps = { + attachment: DiagnosticsAttachment; + verbose: boolean; +}; +export function DiagnosticsDisplay(t0) { + const $ = _c(14); + const { + attachment, + verbose + } = t0; + if (attachment.files.length === 0) { + return null; + } + let t1; + if ($[0] !== attachment.files) { + t1 = attachment.files.reduce(_temp, 0); + $[0] = attachment.files; + $[1] = t1; + } else { + t1 = $[1]; + } + const totalIssues = t1; + const fileCount = attachment.files.length; + if (verbose) { + let t2; + if ($[2] !== attachment.files) { + t2 = attachment.files.map(_temp3); + $[2] = attachment.files; + $[3] = t2; + } else { + t2 = $[3]; + } + let t3; + if ($[4] !== t2) { + t3 = {t2}; + $[4] = t2; + $[5] = t3; + } else { + t3 = $[5]; + } + return t3; + } else { + let t2; + if ($[6] !== totalIssues) { + t2 = {totalIssues}; + $[6] = totalIssues; + $[7] = t2; + } else { + t2 = $[7]; + } + const t3 = totalIssues === 1 ? "issue" : "issues"; + const t4 = fileCount === 1 ? "file" : "files"; + let t5; + if ($[8] === Symbol.for("react.memo_cache_sentinel")) { + t5 = ; + $[8] = t5; + } else { + t5 = $[8]; + } + let t6; + if ($[9] !== fileCount || $[10] !== t2 || $[11] !== t3 || $[12] !== t4) { + t6 = Found {t2} new diagnostic{" "}{t3} in {fileCount}{" "}{t4} {t5}; + $[9] = fileCount; + $[10] = t2; + $[11] = t3; + $[12] = t4; + $[13] = t6; + } else { + t6 = $[13]; + } + return t6; + } +} +function _temp3(file_0, fileIndex) { + return {relative(getCwd(), file_0.uri.replace("file://", "").replace("_claude_fs_right:", ""))}{" "}{file_0.uri.startsWith("file://") ? "(file://)" : file_0.uri.startsWith("_claude_fs_right:") ? "(claude_fs_right)" : `(${file_0.uri.split(":")[0]})`}:{file_0.diagnostics.map(_temp2)}; +} +function _temp2(diagnostic, diagIndex) { + return {" "}{DiagnosticTrackingService.getSeveritySymbol(diagnostic.severity)}{" [Line "}{diagnostic.range.start.line + 1}:{diagnostic.range.start.character + 1}{"] "}{diagnostic.message}{diagnostic.code ? ` [${diagnostic.code}]` : ""}{diagnostic.source ? ` (${diagnostic.source})` : ""}; +} +function _temp(sum, file) { + return sum + file.diagnostics.length; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJyZWxhdGl2ZSIsIlJlYWN0IiwiQm94IiwiVGV4dCIsIkRpYWdub3N0aWNUcmFja2luZ1NlcnZpY2UiLCJBdHRhY2htZW50IiwiZ2V0Q3dkIiwiQ3RybE9Ub0V4cGFuZCIsIk1lc3NhZ2VSZXNwb25zZSIsIkRpYWdub3N0aWNzQXR0YWNobWVudCIsIkV4dHJhY3QiLCJ0eXBlIiwiRGlhZ25vc3RpY3NEaXNwbGF5UHJvcHMiLCJhdHRhY2htZW50IiwidmVyYm9zZSIsIkRpYWdub3N0aWNzRGlzcGxheSIsInQwIiwiJCIsIl9jIiwiZmlsZXMiLCJsZW5ndGgiLCJ0MSIsInJlZHVjZSIsIl90ZW1wIiwidG90YWxJc3N1ZXMiLCJmaWxlQ291bnQiLCJ0MiIsIm1hcCIsIl90ZW1wMyIsInQzIiwidDQiLCJ0NSIsIlN5bWJvbCIsImZvciIsInQ2IiwiZmlsZV8wIiwiZmlsZUluZGV4IiwiZmlsZSIsInVyaSIsInJlcGxhY2UiLCJzdGFydHNXaXRoIiwic3BsaXQiLCJkaWFnbm9zdGljcyIsIl90ZW1wMiIsImRpYWdub3N0aWMiLCJkaWFnSW5kZXgiLCJnZXRTZXZlcml0eVN5bWJvbCIsInNldmVyaXR5IiwicmFuZ2UiLCJzdGFydCIsImxpbmUiLCJjaGFyYWN0ZXIiLCJtZXNzYWdlIiwiY29kZSIsInNvdXJjZSIsInN1bSJdLCJzb3VyY2VzIjpbIkRpYWdub3N0aWNzRGlzcGxheS50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgcmVsYXRpdmUgfSBmcm9tICdwYXRoJ1xuaW1wb3J0IFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgQm94LCBUZXh0IH0gZnJvbSAnLi4vaW5rLmpzJ1xuaW1wb3J0IHsgRGlhZ25vc3RpY1RyYWNraW5nU2VydmljZSB9IGZyb20gJy4uL3NlcnZpY2VzL2RpYWdub3N0aWNUcmFja2luZy5qcydcbmltcG9ydCB0eXBlIHsgQXR0YWNobWVudCB9IGZyb20gJy4uL3V0aWxzL2F0dGFjaG1lbnRzLmpzJ1xuaW1wb3J0IHsgZ2V0Q3dkIH0gZnJvbSAnLi4vdXRpbHMvY3dkLmpzJ1xuaW1wb3J0IHsgQ3RybE9Ub0V4cGFuZCB9IGZyb20gJy4vQ3RybE9Ub0V4cGFuZC5qcydcbmltcG9ydCB7IE1lc3NhZ2VSZXNwb25zZSB9IGZyb20gJy4vTWVzc2FnZVJlc3BvbnNlLmpzJ1xuXG50eXBlIERpYWdub3N0aWNzQXR0YWNobWVudCA9IEV4dHJhY3Q8QXR0YWNobWVudCwgeyB0eXBlOiAnZGlhZ25vc3RpY3MnIH0+XG5cbnR5cGUgRGlhZ25vc3RpY3NEaXNwbGF5UHJvcHMgPSB7XG4gIGF0dGFjaG1lbnQ6IERpYWdub3N0aWNzQXR0YWNobWVudFxuICB2ZXJib3NlOiBib29sZWFuXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBEaWFnbm9zdGljc0Rpc3BsYXkoe1xuICBhdHRhY2htZW50LFxuICB2ZXJib3NlLFxufTogRGlhZ25vc3RpY3NEaXNwbGF5UHJvcHMpOiBSZWFjdC5SZWFjdE5vZGUge1xuICAvLyBPbmx5IHNob3cgaWYgdGhlcmUgYXJlIGRpYWdub3N0aWNzIHRvIHJlcG9ydFxuICBpZiAoYXR0YWNobWVudC5maWxlcy5sZW5ndGggPT09IDApIHJldHVybiBudWxsXG5cbiAgLy8gQ291bnQgdG90YWwgaXNzdWVzXG4gIGNvbnN0IHRvdGFsSXNzdWVzID0gYXR0YWNobWVudC5maWxlcy5yZWR1Y2UoXG4gICAgKHN1bSwgZmlsZSkgPT4gc3VtICsgZmlsZS5kaWFnbm9zdGljcy5sZW5ndGgsXG4gICAgMCxcbiAgKVxuXG4gIGNvbnN0IGZpbGVDb3VudCA9IGF0dGFjaG1lbnQuZmlsZXMubGVuZ3RoXG5cbiAgaWYgKHZlcmJvc2UpIHtcbiAgICAvLyBTaG93IGFsbCBkaWFnbm9zdGljcyBpbiB2ZXJib3NlIG1vZGUgKGN0cmwrbylcbiAgICByZXR1cm4gKFxuICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCI+XG4gICAgICAgIHthdHRhY2htZW50LmZpbGVzLm1hcCgoZmlsZSwgZmlsZUluZGV4KSA9PiAoXG4gICAgICAgICAgPFJlYWN0LkZyYWdtZW50IGtleT17ZmlsZUluZGV4fT5cbiAgICAgICAgICAgIDxNZXNzYWdlUmVzcG9uc2U+XG4gICAgICAgICAgICAgIDxUZXh0IGRpbUNvbG9yIHdyYXA9XCJ3cmFwXCI+XG4gICAgICAgICAgICAgICAgPFRleHQgYm9sZD5cbiAgICAgICAgICAgICAgICAgIHtyZWxhdGl2ZShcbiAgICAgICAgICAgICAgICAgICAgZ2V0Q3dkKCksXG4gICAgICAgICAgICAgICAgICAgIGZpbGUudXJpXG4gICAgICAgICAgICAgICAgICAgICAgLnJlcGxhY2UoJ2ZpbGU6Ly8nLCAnJylcbiAgICAgICAgICAgICAgICAgICAgICAucmVwbGFjZSgnX2NsYXVkZV9mc19yaWdodDonLCAnJyksXG4gICAgICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgICAgIDwvVGV4dD57JyAnfVxuICAgICAgICAgICAgICAgIDxUZXh0IGRpbUNvbG9yPlxuICAgICAgICAgICAgICAgICAge2ZpbGUudXJpLnN0YXJ0c1dpdGgoJ2ZpbGU6Ly8nKVxuICAgICAgICAgICAgICAgICAgICA/ICcoZmlsZTovLyknXG4gICAgICAgICAgICAgICAgICAgIDogZmlsZS51cmkuc3RhcnRzV2l0aCgnX2NsYXVkZV9mc19yaWdodDonKVxuICAgICAgICAgICAgICAgICAgICAgID8gJyhjbGF1ZGVfZnNfcmlnaHQpJ1xuICAgICAgICAgICAgICAgICAgICAgIDogYCgke2ZpbGUudXJpLnNwbGl0KCc6JylbMF19KWB9XG4gICAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgICAgIDpcbiAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgPC9NZXNzYWdlUmVzcG9uc2U+XG4gICAgICAgICAgICB7ZmlsZS5kaWFnbm9zdGljcy5tYXAoKGRpYWdub3N0aWMsIGRpYWdJbmRleCkgPT4gKFxuICAgICAgICAgICAgICA8TWVzc2FnZVJlc3BvbnNlIGtleT17ZGlhZ0luZGV4fT5cbiAgICAgICAgICAgICAgICA8VGV4dCBkaW1Db2xvciB3cmFwPVwid3JhcFwiPlxuICAgICAgICAgICAgICAgICAgeycgICd9XG4gICAgICAgICAgICAgICAgICB7RGlhZ25vc3RpY1RyYWNraW5nU2VydmljZS5nZXRTZXZlcml0eVN5bWJvbChcbiAgICAgICAgICAgICAgICAgICAgZGlhZ25vc3RpYy5zZXZlcml0eSxcbiAgICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgICAgICB7JyBbTGluZSAnfVxuICAgICAgICAgICAgICAgICAge2RpYWdub3N0aWMucmFuZ2Uuc3RhcnQubGluZSArIDF9OlxuICAgICAgICAgICAgICAgICAge2RpYWdub3N0aWMucmFuZ2Uuc3RhcnQuY2hhcmFjdGVyICsgMX1cbiAgICAgICAgICAgICAgICAgIHsnXSAnfVxuICAgICAgICAgICAgICAgICAge2RpYWdub3N0aWMubWVzc2FnZX1cbiAgICAgICAgICAgICAgICAgIHtkaWFnbm9zdGljLmNvZGUgPyBgIFske2RpYWdub3N0aWMuY29kZX1dYCA6ICcnfVxuICAgICAgICAgICAgICAgICAge2RpYWdub3N0aWMuc291cmNlID8gYCAoJHtkaWFnbm9zdGljLnNvdXJjZX0pYCA6ICcnfVxuICAgICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgPC9NZXNzYWdlUmVzcG9uc2U+XG4gICAgICAgICAgICApKX1cbiAgICAgICAgICA8L1JlYWN0LkZyYWdtZW50PlxuICAgICAgICApKX1cbiAgICAgIDwvQm94PlxuICAgIClcbiAgfSBlbHNlIHtcbiAgICAvLyBTaG93IHN1bW1hcnkgaW4gbm9ybWFsIG1vZGVcbiAgICByZXR1cm4gKFxuICAgICAgPE1lc3NhZ2VSZXNwb25zZT5cbiAgICAgICAgPFRleHQgZGltQ29sb3Igd3JhcD1cIndyYXBcIj5cbiAgICAgICAgICBGb3VuZCA8VGV4dCBib2xkPnt0b3RhbElzc3Vlc308L1RleHQ+IG5ldyBkaWFnbm9zdGljeycgJ31cbiAgICAgICAgICB7dG90YWxJc3N1ZXMgPT09IDEgPyAnaXNzdWUnIDogJ2lzc3Vlcyd9IGluIHtmaWxlQ291bnR9eycgJ31cbiAgICAgICAgICB7ZmlsZUNvdW50ID09PSAxID8gJ2ZpbGUnIDogJ2ZpbGVzJ30gPEN0cmxPVG9FeHBhbmQgLz5cbiAgICAgICAgPC9UZXh0PlxuICAgICAgPC9NZXNzYWdlUmVzcG9uc2U+XG4gICAgKVxuICB9XG59XG4iXSwibWFwcGluZ3MiOiI7QUFBQSxTQUFTQSxRQUFRLFFBQVEsTUFBTTtBQUMvQixPQUFPQyxLQUFLLE1BQU0sT0FBTztBQUN6QixTQUFTQyxHQUFHLEVBQUVDLElBQUksUUFBUSxXQUFXO0FBQ3JDLFNBQVNDLHlCQUF5QixRQUFRLG1DQUFtQztBQUM3RSxjQUFjQyxVQUFVLFFBQVEseUJBQXlCO0FBQ3pELFNBQVNDLE1BQU0sUUFBUSxpQkFBaUI7QUFDeEMsU0FBU0MsYUFBYSxRQUFRLG9CQUFvQjtBQUNsRCxTQUFTQyxlQUFlLFFBQVEsc0JBQXNCO0FBRXRELEtBQUtDLHFCQUFxQixHQUFHQyxPQUFPLENBQUNMLFVBQVUsRUFBRTtFQUFFTSxJQUFJLEVBQUUsYUFBYTtBQUFDLENBQUMsQ0FBQztBQUV6RSxLQUFLQyx1QkFBdUIsR0FBRztFQUM3QkMsVUFBVSxFQUFFSixxQkFBcUI7RUFDakNLLE9BQU8sRUFBRSxPQUFPO0FBQ2xCLENBQUM7QUFFRCxPQUFPLFNBQUFDLG1CQUFBQyxFQUFBO0VBQUEsTUFBQUMsQ0FBQSxHQUFBQyxFQUFBO0VBQTRCO0lBQUFMLFVBQUE7SUFBQUM7RUFBQSxJQUFBRSxFQUdUO0VBRXhCLElBQUlILFVBQVUsQ0FBQU0sS0FBTSxDQUFBQyxNQUFPLEtBQUssQ0FBQztJQUFBLE9BQVMsSUFBSTtFQUFBO0VBQUEsSUFBQUMsRUFBQTtFQUFBLElBQUFKLENBQUEsUUFBQUosVUFBQSxDQUFBTSxLQUFBO0lBRzFCRSxFQUFBLEdBQUFSLFVBQVUsQ0FBQU0sS0FBTSxDQUFBRyxNQUFPLENBQ3pDQyxLQUE0QyxFQUM1QyxDQUNGLENBQUM7SUFBQU4sQ0FBQSxNQUFBSixVQUFBLENB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QUNSK0IsTUFBSSxDQUFBQyxHQUFJLENBQUFDLE9BQ0UsQ0FBQyxTQUFTLEVBQUUsRUFBRSxDQUFDLENBQUFBLE9BQ2YsQ0FBQyxtQkFBbUIsRUFBRSxFQUFFLENBQ3BDLEVBQ0YsRUFQQyxJQUFJLENBT0csSUFBRSxDQUNWLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBUixLQUFPLENBQUMsQ0FDWCxDQUFBRixNQUFJLENBQUFDLEdBQUksQ0FBQUUsVUFBVyxDQUFDLFNBSWEsQ0FBQyxHQUpsQyxXQUlrQyxHQUYvQkgsTUFBSSxDQUFBQyxHQUFJLENBQUFFLFVBQVcsQ0FBQyxtQkFFVSxDQUFDLEdBRi9CLG1CQUUrQixHQUYvQixJQUVNSCxNQUFJLENBQUFDLEdBQUksQ0FBQUcsS0FBTSxDQUFDLEdBQUcsQ0FBQyxHQUFHLEdBQUUsQ0FDcEMsRUFOQyxJQUFJLENBTUUsQ0FFVCxFQWpCQyxJQUFJLENBa0JQLEVBbkJDLGVBQWUsQ0FvQmYsQ0FBQUosTUFBSSxDQUFBSyxXQUFZLENBQUFmLEdBQUksQ0FBQ2dCLE1BZ0JyQixFQUNILGlCQUFpQjtBQUFBO0FBMURwQixTQUFBQSxPQUFBQyxVQUFBLEVBQUFDLFNBQUE7RUFBQSxPQTBDTyxDQUFDLGVBQWUsQ0FBTUEsR0FBUyxDQUFUQSxVQUFRLENBQUMsQ0FDN0IsQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFSLEtBQU8sQ0FBQyxDQUFNLElBQU0sQ0FBTixNQUFNLENBQ3ZCLEtBQUcsQ0FDSCxDQUFBekMseUJBQXlCLENBQUEwQyxpQkFBa0IsQ0FDMUNGLFVBQVUsQ0FBQUcsUUFDWixFQUNDLFVBQVEsQ0FDUixDQUFBSCxVQUFVLENBQUFJLEtBQU0sQ0FBQUMsS0FBTSxDQUFBQyxJQUFLLEdBQUcsRUFBRSxDQUNoQyxDQUFBTixVQUFVLENBQUFJLEtBQU0sQ0FBQUMsS0FBTSxDQUFBRSxTQUFVLEdBQUcsRUFDbkMsS0FBRyxDQUNILENBQUFQLFVBQVUsQ0FBQVEsT0FBTyxDQUNqQixDQUFBUixVQUFVLENBQUFTLElBQW9DLEdBQTlDLEtBQXVCVCxVQUFVLENBQUFTLElBQUssR0FBUSxHQUE5QyxFQUE2QyxDQUM3QyxDQUFBVCxVQUFVLENBQUFVLE1BQXdDLEdBQWxELEtBQXlCVixVQUFVLENBQUFVLE1BQU8sR0FBUSxHQUFsRCxFQUFpRCxDQUNwRCxFQVpDLElBQUksQ0FhUCxFQWRDLGVBQWUsQ0FjRTtBQUFBO0FBeER6QixTQUFBL0IsTUFBQWdDLEdBQUEsRUFBQWxCLElBQUE7RUFBQSxPQVNZa0IsR0FBRyxHQUFHbEIsSUFBSSxDQUFBSyxXQUFZLENBQUF0QixNQUFPO0FBQUEiLCJpZ25vcmVMaXN0IjpbXX0= \ No newline at end of file diff --git a/src/components/EffortCallout.tsx b/src/components/EffortCallout.tsx new file mode 100644 index 0000000..68e311f --- /dev/null +++ b/src/components/EffortCallout.tsx @@ -0,0 +1,265 @@ +import { c as _c } from "react/compiler-runtime"; +import React, { useCallback, useEffect, useRef } from 'react'; +import { Box, Text } from '../ink.js'; +import { isMaxSubscriber, isProSubscriber, isTeamSubscriber } from '../utils/auth.js'; +import { getGlobalConfig, saveGlobalConfig } from '../utils/config.js'; +import type { EffortLevel } from '../utils/effort.js'; +import { convertEffortValueToLevel, getDefaultEffortForModel, getOpusDefaultEffortConfig, toPersistableEffort } from '../utils/effort.js'; +import { parseUserSpecifiedModel } from '../utils/model/model.js'; +import { updateSettingsForSource } from '../utils/settings/settings.js'; +import type { OptionWithDescription } from './CustomSelect/select.js'; +import { Select } from './CustomSelect/select.js'; +import { effortLevelToSymbol } from './EffortIndicator.js'; +import { PermissionDialog } from './permissions/PermissionDialog.js'; +type EffortCalloutSelection = EffortLevel | undefined | 'dismiss'; +type Props = { + model: string; + onDone: (selection: EffortCalloutSelection) => void; +}; +const AUTO_DISMISS_MS = 30_000; +export function EffortCallout(t0) { + const $ = _c(18); + const { + model, + onDone + } = t0; + let t1; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t1 = getOpusDefaultEffortConfig(); + $[0] = t1; + } else { + t1 = $[0]; + } + const defaultEffortConfig = t1; + const onDoneRef = useRef(onDone); + let t2; + if ($[1] !== onDone) { + t2 = () => { + onDoneRef.current = onDone; + }; + $[1] = onDone; + $[2] = t2; + } else { + t2 = $[2]; + } + useEffect(t2); + let t3; + if ($[3] === Symbol.for("react.memo_cache_sentinel")) { + t3 = () => { + onDoneRef.current("dismiss"); + }; + $[3] = t3; + } else { + t3 = $[3]; + } + const handleCancel = t3; + let t4; + if ($[4] === Symbol.for("react.memo_cache_sentinel")) { + t4 = []; + $[4] = t4; + } else { + t4 = $[4]; + } + useEffect(_temp, t4); + let t5; + let t6; + if ($[5] === Symbol.for("react.memo_cache_sentinel")) { + t5 = () => { + const timeoutId = setTimeout(handleCancel, AUTO_DISMISS_MS); + return () => clearTimeout(timeoutId); + }; + t6 = [handleCancel]; + $[5] = t5; + $[6] = t6; + } else { + t5 = $[5]; + t6 = $[6]; + } + useEffect(t5, t6); + let t7; + if ($[7] !== model) { + const defaultEffort = getDefaultEffortForModel(model); + t7 = defaultEffort ? convertEffortValueToLevel(defaultEffort) : "high"; + $[7] = model; + $[8] = t7; + } else { + t7 = $[8]; + } + const defaultLevel = t7; + let t8; + if ($[9] !== defaultLevel) { + t8 = value => { + const effortLevel = value === defaultLevel ? undefined : value; + updateSettingsForSource("userSettings", { + effortLevel: toPersistableEffort(effortLevel) + }); + onDoneRef.current(value); + }; + $[9] = defaultLevel; + $[10] = t8; + } else { + t8 = $[10]; + } + const handleSelect = t8; + let t9; + if ($[11] === Symbol.for("react.memo_cache_sentinel")) { + t9 = [{ + label: , + value: "medium" + }, { + label: , + value: "high" + }, { + label: , + value: "low" + }]; + $[11] = t9; + } else { + t9 = $[11]; + } + const options = t9; + let t10; + if ($[12] === Symbol.for("react.memo_cache_sentinel")) { + t10 = {defaultEffortConfig.dialogDescription}; + $[12] = t10; + } else { + t10 = $[12]; + } + let t11; + if ($[13] === Symbol.for("react.memo_cache_sentinel")) { + t11 = ; + $[13] = t11; + } else { + t11 = $[13]; + } + let t12; + if ($[14] === Symbol.for("react.memo_cache_sentinel")) { + t12 = ; + $[14] = t12; + } else { + t12 = $[14]; + } + let t13; + if ($[15] === Symbol.for("react.memo_cache_sentinel")) { + t13 = {t11} low {"\xB7"}{" "}{t12} medium {"\xB7"}{" "} high; + $[15] = t13; + } else { + t13 = $[15]; + } + let t14; + if ($[16] !== handleSelect) { + t14 = {t10}{t13} : + Enter filename: + + > + + + } + ; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJqb2luIiwiUmVhY3QiLCJ1c2VDYWxsYmFjayIsInVzZVN0YXRlIiwiRXhpdFN0YXRlIiwidXNlVGVybWluYWxTaXplIiwic2V0Q2xpcGJvYXJkIiwiQm94IiwiVGV4dCIsInVzZUtleWJpbmRpbmciLCJnZXRDd2QiLCJ3cml0ZUZpbGVTeW5jX0RFUFJFQ0FURUQiLCJDb25maWd1cmFibGVTaG9ydGN1dEhpbnQiLCJTZWxlY3QiLCJCeWxpbmUiLCJEaWFsb2ciLCJLZXlib2FyZFNob3J0Y3V0SGludCIsIlRleHRJbnB1dCIsIkV4cG9ydERpYWxvZ1Byb3BzIiwiY29udGVudCIsImRlZmF1bHRGaWxlbmFtZSIsIm9uRG9uZSIsInJlc3VsdCIsInN1Y2Nlc3MiLCJtZXNzYWdlIiwiRXhwb3J0T3B0aW9uIiwiRXhwb3J0RGlhbG9nIiwiUmVhY3ROb2RlIiwic2V0U2VsZWN0ZWRPcHRpb24iLCJmaWxlbmFtZSIsInNldEZpbGVuYW1lIiwiY3Vyc29yT2Zmc2V0Iiwic2V0Q3Vyc29yT2Zmc2V0IiwibGVuZ3RoIiwic2hvd0ZpbGVuYW1lSW5wdXQiLCJzZXRTaG93RmlsZW5hbWVJbnB1dCIsImNvbHVtbnMiLCJoYW5kbGVHb0JhY2siLCJoYW5kbGVTZWxlY3RPcHRpb24iLCJ2YWx1ZSIsIlByb21pc2UiLCJyYXciLCJwcm9jZXNzIiwic3Rkb3V0Iiwid3JpdGUiLCJoYW5kbGVGaWxlbmFtZVN1Ym1pdCIsImZpbmFsRmlsZW5hbWUiLCJlbmRzV2l0aCIsInJlcGxhY2UiLCJmaWxlcGF0aCIsImVuY29kaW5nIiwiZmx1c2giLCJlcnJvciIsIkVycm9yIiwiaGFuZGxlQ2FuY2VsIiwib3B0aW9ucyIsImxhYmVsIiwiZGVzY3JpcHRpb24iLCJyZW5kZXJJbnB1dEd1aWRlIiwiZXhpdFN0YXRlIiwicGVuZGluZyIsImtleU5hbWUiLCJjb250ZXh0IiwiaXNBY3RpdmUiXSwic291cmNlcyI6WyJFeHBvcnREaWFsb2cudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGpvaW4gfSBmcm9tICdwYXRoJ1xuaW1wb3J0IFJlYWN0LCB7IHVzZUNhbGxiYWNrLCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHR5cGUgeyBFeGl0U3RhdGUgfSBmcm9tICcuLi9ob29rcy91c2VFeGl0T25DdHJsQ0RXaXRoS2V5YmluZGluZ3MuanMnXG5pbXBvcnQgeyB1c2VUZXJtaW5hbFNpemUgfSBmcm9tICcuLi9ob29rcy91c2VUZXJtaW5hbFNpemUuanMnXG5pbXBvcnQgeyBzZXRDbGlwYm9hcmQgfSBmcm9tICcuLi9pbmsvdGVybWlvL29zYy5qcydcbmltcG9ydCB7IEJveCwgVGV4dCB9IGZyb20gJy4uL2luay5qcydcbmltcG9ydCB7IHVzZUtleWJpbmRpbmcgfSBmcm9tICcuLi9rZXliaW5kaW5ncy91c2VLZXliaW5kaW5nLmpzJ1xuaW1wb3J0IHsgZ2V0Q3dkIH0gZnJvbSAnLi4vdXRpbHMvY3dkLmpzJ1xuaW1wb3J0IHsgd3JpdGVGaWxlU3luY19ERVBSRUNBVEVEIH0gZnJvbSAnLi4vdXRpbHMvc2xvd09wZXJhdGlvbnMuanMnXG5pbXBvcnQgeyBDb25maWd1cmFibGVTaG9ydGN1dEhpbnQgfSBmcm9tICcuL0NvbmZpZ3VyYWJsZVNob3J0Y3V0SGludC5qcydcbmltcG9ydCB7IFNlbGVjdCB9IGZyb20gJy4vQ3VzdG9tU2VsZWN0L3NlbGVjdC5qcydcbmltcG9ydCB7IEJ5bGluZSB9IGZyb20gJy4vZGVzaWduLXN5c3RlbS9CeWxpbmUuanMnXG5pbXBvcnQgeyBEaWFsb2cgfSBmcm9tICcuL2Rlc2lnbi1zeXN0ZW0vRGlhbG9nLmpzJ1xuaW1wb3J0IHsgS2V5Ym9hcmRTaG9ydGN1dEhpbnQgfSBmcm9tICcuL2Rlc2lnbi1zeXN0ZW0vS2V5Ym9hcmRTaG9ydGN1dEhpbnQuanMnXG5pbXBvcnQgVGV4dElucHV0IGZyb20gJy4vVGV4dElucHV0LmpzJ1xuXG50eXBlIEV4cG9ydERpYWxvZ1Byb3BzID0ge1xuICBjb250ZW50OiBzdHJpbmdcbiAgZGVmYXVsdEZpbGVuYW1lOiBzdHJpbmdcbiAgb25Eb25lOiAocmVzdWx0OiB7IHN1Y2Nlc3M6IGJvb2xlYW47IG1lc3NhZ2U6IHN0cmluZyB9KSA9PiB2b2lkXG59XG5cbnR5cGUgRXhwb3J0T3B0aW9uID0gJ2NsaXBib2FyZCcgfCAnZmlsZSdcblxuZXhwb3J0IGZ1bmN0aW9uIEV4cG9ydERpYWxvZyh7XG4gIGNvbnRlbnQsXG4gIGRlZmF1bHRGaWxlbmFtZSxcbiAgb25Eb25lLFxufTogRXhwb3J0RGlhbG9nUHJvcHMpOiBSZWFjdC5SZWFjdE5vZGUge1xuICBjb25zdCBbLCBzZXRTZWxlY3RlZE9wdGlvbl0gPSB1c2VTdGF0ZTxFeHBvcnRPcHRpb24gfCBudWxsPihudWxsKVxuICBjb25zdCBbZmlsZW5hbWUsIHNldEZpbGVuYW1lXSA9IHVzZVN0YXRlPHN0cmluZz4oZGVmYXVsdEZpbGVuYW1lKVxuICBjb25zdCBbY3Vyc29yT2Zmc2V0LCBzZXRDdXJzb3JPZmZzZXRdID0gdXNlU3RhdGU8bnVtYmVyPihcbiAgICBkZWZhdWx0RmlsZW5hbWUubGVuZ3RoLFxuICApXG4gIGNvbnN0IFtzaG93RmlsZW5hbWVJbnB1dCwgc2V0U2hvd0ZpbGVuYW1lSW5wdXRdID0gdXNlU3RhdGUoZmFsc2UpXG4gIGNvbnN0IHsgY29sdW1ucyB9ID0gdXNlVGVybWluYWxTaXplKClcblxuICAvLyBIYW5kbGUgZ29pbmcgYmFjayBmcm9tIGZpbGVuYW1lIGlucHV0IHRvIG9wdGlvbiBzZWxlY3Rpb25cbiAgY29uc3QgaGFuZGxlR29CYWNrID0gdXNlQ2FsbGJhY2soKCkgPT4ge1xuICAgIHNldFNob3dGaWxlbmFtZUlucHV0KGZhbHNlKVxuICAgIHNldFNlbGVjdGVkT3B0aW9uKG51bGwpXG4gIH0sIFtdKVxuXG4gIGNvbnN0IGhhbmRsZVNlbGVjdE9wdGlvbiA9IGFzeW5jICh2YWx1ZTogc3RyaW5nKTogUHJvbWlzZTx2b2lkPiA9PiB7XG4gICAgaWYgKHZhbHVlID09PSAnY2xpcGJvYXJkJykge1xuICAgICAgLy8gQ29weSB0byBjbGlwYm9hcmQgaW1tZWRpYXRlbHlcbiAgICAgIGNvbnN0IHJhdyA9IGF3YWl0IHNldENsaXBib2FyZChjb250ZW50KVxuICAgICAgaWYgKHJhdykgcHJvY2Vzcy5zdGRvdXQud3JpdGUocmF3KVxuICAgICAgb25Eb25lKHsgc3VjY2VzczogdHJ1ZSwgbWVzc2FnZTogJ0NvbnZlcnNhdGlvbiBjb3BpZWQgdG8gY2xpcGJvYXJkJyB9KVxuICAgIH0gZWxzZSBpZiAodmFsdWUgPT09ICdmaWxlJykge1xuICAgICAgc2V0U2VsZWN0ZWRPcHRpb24oJ2ZpbGUnKVxuICAgICAgc2V0U2hvd0ZpbGVuYW1lSW5wdXQodHJ1ZSlcbiAgICB9XG4gIH1cblxuICBjb25zdCBoYW5kbGVGaWxlbmFtZVN1Ym1pdCA9ICgpID0+IHtcbiAgICBjb25zdCBmaW5hbEZpbGVuYW1lID0gZmlsZW5hbWUuZW5kc1dpdGgoJy50eHQnKVxuICAgICAgPyBmaWxlbmFtZVxuICAgICAgOiBmaWxlbmFtZS5yZXBsYWNlKC9cXC5bXi5dKyQvLCAnJykgKyAnLnR4dCdcbiAgICBjb25zdCBmaWxlcGF0aCA9IGpvaW4oZ2V0Q3dkKCksIGZpbmFsRmlsZW5hbWUpXG5cbiAgICB0cnkge1xuICAgICAgd3JpdGVGaWxlU3luY19ERVBSRUNBVEVEKGZpbGVwYXRoLCBjb250ZW50LCB7XG4gICAgICAgIGVuY29kaW5nOiAndXRmLTgnLFxuICAgICAgICBmbHVzaDogdHJ1ZSxcbiAgICAgIH0pXG4gICAgICBvbkRvbmUoe1xuICAgICAgICBzdWNjZXNzOiB0cnVlLFxuICAgICAgICBtZXNzYWdlOiBgQ29udmVyc2F0aW9uIGV4cG9ydGVkIHRvOiAke2ZpbGVwYXRofWAsXG4gICAgICB9KVxuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICBvbkRvbmUoe1xuICAgICAgICBzdWNjZXNzOiBmYWxzZSxcbiAgICAgICAgbWVzc2FnZTogYEZhaWxlZCB0byBleHBvcnQgY29udmVyc2F0aW9uOiAke2Vycm9yIGluc3RhbmNlb2YgRXJyb3IgPyBlcnJvci5tZXNzYWdlIDogJ1Vua25vd24gZXJyb3InfWAsXG4gICAgICB9KVxuICAgIH1cbiAgfVxuXG4gIC8vIERpYWxvZyBjYWxscyBvbkNhbmNlbCB3aGVuIEVzY2FwZSBpcyBwcmVzc2VkLiBJZiB3ZSBhcmUgaW4gdGhlIGZpbGVuYW1lXG4gIC8vIGlucHV0IHN1Yi1zY3JlZW4sIGdvIGJhY2sgdG8gdGhlIG9wdGlvbiBsaXN0IGluc3RlYWQgb2YgY2xvc2luZyBlbnRpcmVseS5cbiAgY29uc3QgaGFuZGxlQ2FuY2VsID0gdXNlQ2FsbGJhY2soKCkgPT4ge1xuICAgIGlmIChzaG93RmlsZW5hbWVJbnB1dCkge1xuICAgICAgaGFuZGxlR29CYWNrKClcbiAgICB9IGVsc2Uge1xuICAgICAgb25Eb25lKHsgc3VjY2VzczogZmFsc2UsIG1lc3NhZ2U6ICdFeHBvcnQgY2FuY2VsbGVkJyB9KVxuICAgIH1cbiAgfSwgW3Nob3dGaWxlbmFtZUlucHV0LCBoYW5kbGVHb0JhY2ssIG9uRG9uZV0pXG5cbiAgY29uc3Qgb3B0aW9ucyA9IFtcbiAgICB7XG4gICAgICBsYWJlbDogJ0NvcHkgdG8gY2xpcGJvYXJkJyxcbiAgICAgIHZhbHVlOiAnY2xpcGJvYXJkJyxcbiAgICAgIGRlc2NyaXB0aW9uOiAnQ29weSB0aGUgY29udmVyc2F0aW9uIHRvIHlvdXIgc3lzdGVtIGNsaXBib2FyZCcsXG4gICAgfSxcbiAgICB7XG4gICAgICBsYWJlbDogJ1NhdmUgdG8gZmlsZScsXG4gICAgICB2YWx1ZTogJ2ZpbGUnLFxuICAgICAgZGVzY3JpcHRpb246ICdTYXZlIHRoZSBjb252ZXJzYXRpb24gdG8gYSBmaWxlIGluIHRoZSBjdXJyZW50IGRpcmVjdG9yeScsXG4gICAgfSxcbiAgXVxuXG4gIC8vIEN1c3RvbSBpbnB1dCBndWlkZSB0aGF0IGNoYW5nZXMgYmFzZWQgb24gZGlhbG9nIHN0YXRlXG4gIGZ1bmN0aW9uIHJlbmRlcklucHV0R3VpZGUoZXhpdFN0YXRlOiBFeGl0U3RhdGUpOiBSZWFjdC5SZWFjdE5vZGUge1xuICAgIGlmIChzaG93RmlsZW5hbWVJbnB1dCkge1xuICAgICAgcmV0dXJuIChcbiAgICAgICAgPEJ5bGluZT5cbiAgICAgICAgICA8S2V5Ym9hcmRTaG9ydGN1dEhpbnQgc2hvcnRjdXQ9XCJFbnRlclwiIGFjdGlvbj1cInNhdmVcIiAvPlxuICAgICAgICAgIDxDb25maWd1cmFibGVTaG9ydGN1dEhpbnRcbiAgICAgICAgICAgIGFjdGlvbj1cImNvbmZpcm06bm9cIlxuICAgICAgICAgICAgY29udGV4dD1cIkNvbmZpcm1hdGlvblwiXG4gICAgICAgICAgICBmYWxsYmFjaz1cIkVzY1wiXG4gICAgICAgICAgICBkZXNjcmlwdGlvbj1cImdvIGJhY2tcIlxuICAgICAgICAgIC8+XG4gICAgICAgIDwvQnlsaW5lPlxuICAgICAgKVxuICAgIH1cblxuICAgIGlmIChleGl0U3RhdGUucGVuZGluZykge1xuICAgICAgcmV0dXJuIDxUZXh0PlByZXNzIHtleGl0U3RhdGUua2V5TmFtZX0gYWdhaW4gdG8gZXhpdDwvVGV4dD5cbiAgICB9XG5cbiAgICByZXR1cm4gKFxuICAgICAgPENvbmZpZ3VyYWJsZVNob3J0Y3V0SGludFxuICAgICAgICBhY3Rpb249XCJjb25maXJtOm5vXCJcbiAgICAgICAgY29udGV4dD1cIkNvbmZpcm1hdGlvblwiXG4gICAgICAgIGZhbGxiYWNrPVwiRXNjXCJcbiAgICAgICAgZGVzY3JpcHRpb249XCJjYW5jZWxcIlxuICAgICAgLz5cbiAgICApXG4gIH1cblxuICAvLyBVc2UgU2V0dGluZ3MgY29udGV4dCBzbyAnbicga2V5IGRvZXNuJ3QgY2FuY2VsIChhbGxvd3MgdHlwaW5nICduJyBpbiBmaWxlbmFtZSBpbnB1dClcbiAgdXNlS2V5YmluZGluZygnY29uZmlybTpubycsIGhhbmRsZUNhbmNlbCwge1xuICAgIGNvbnRleHQ6ICdTZXR0aW5ncycsXG4gICAgaXNBY3RpdmU6IHNob3dGaWxlbmFtZUlucHV0LFxuICB9KVxuXG4gIHJldHVybiAoXG4gICAgPERpYWxvZ1xuICAgICAgdGl0bGU9XCJFeHBvcnQgQ29udmVyc2F0aW9uXCJcbiAgICAgIHN1YnRpdGxlPVwiU2VsZWN0IGV4cG9ydCBtZXRob2Q6XCJcbiAgICAgIGNvbG9yPVwicGVybWlzc2lvblwiXG4gICAgICBvbkNhbmNlbD17aGFuZGxlQ2FuY2VsfVxuICAgICAgaW5wdXRHdWlkZT17cmVuZGVySW5wdXRHdWlkZX1cbiAgICAgIGlzQ2FuY2VsQWN0aXZlPXshc2hvd0ZpbGVuYW1lSW5wdXR9XG4gICAgPlxuICAgICAgeyFzaG93RmlsZW5hbWVJbnB1dCA/IChcbiAgICAgICAgPFNlbGVjdFxuICAgICAgICAgIG9wdGlvbnM9e29wdGlvbnN9XG4gICAgICAgICAgb25DaGFuZ2U9e2hhbmRsZVNlbGVjdE9wdGlvbn1cbiAgICAgICAgICBvbkNhbmNlbD17aGFuZGxlQ2FuY2VsfVxuICAgICAgICAvPlxuICAgICAgKSA6IChcbiAgICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCI+XG4gICAgICAgICAgPFRleHQ+RW50ZXIgZmlsZW5hbWU6PC9UZXh0PlxuICAgICAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cInJvd1wiIGdhcD17MX0gbWFyZ2luVG9wPXsxfT5cbiAgICAgICAgICAgIDxUZXh0PiZndDs8L1RleHQ+XG4gICAgICAgICAgICA8VGV4dElucHV0XG4gICAgICAgICAgICAgIHZhbHVlPXtmaWxlbmFtZX1cbiAgICAgICAgICAgICAgb25DaGFuZ2U9e3NldEZpbGVuYW1lfVxuICAgICAgICAgICAgICBvblN1Ym1pdD17aGFuZGxlRmlsZW5hbWVTdWJtaXR9XG4gICAgICAgICAgICAgIGZvY3VzPXt0cnVlfVxuICAgICAgICAgICAgICBzaG93Q3Vyc29yPXt0cnVlfVxuICAgICAgICAgICAgICBjb2x1bW5zPXtjb2x1bW5zfVxuICAgICAgICAgICAgICBjdXJzb3JPZmZzZXQ9e2N1cnNvck9mZnNldH1cbiAgICAgICAgICAgICAgb25DaGFuZ2VDdXJzb3JPZmZzZXQ9e3NldEN1cnNvck9mZnNldH1cbiAgICAgICAgICAgIC8+XG4gICAgICAgICAgPC9Cb3g+XG4gICAgICAgIDwvQm94PlxuICAgICAgKX1cbiAgICA8L0RpYWxvZz5cbiAgKVxufVxuIl0sIm1hcHBpbmdzIjoiQUFBQSxTQUFTQSxJQUFJLFFBQVEsTUFBTTtBQUMzQixPQUFPQyxLQUFLLElBQUlDLFdBQVcsRUFBRUMsUUFBUSxRQUFRLE9BQU87QUFDcEQsY0FBY0MsU0FBUyxRQUFRLDRDQUE0QztBQUMzRSxTQUFTQyxlQUFlLFFBQVEsNkJBQTZCO0FBQzdELFNBQVNDLFlBQVksUUFBUSxzQkFBc0I7QUFDbkQsU0FBU0MsR0FBRyxFQUFFQyxJQUFJLFFBQVEsV0FBVztBQUNyQyxTQUFTQyxhQUFhLFFBQVEsaUNBQWlDO0FBQy9ELFNBQVNDLE1BQU0sUUFBUSxpQkFBaUI7QUFDeEMsU0FBU0Msd0JBQXdCLFFBQVEsNEJBQTRCO0FBQ3JFLFNBQVNDLHdCQUF3QixRQUFRLCtCQUErQjtBQUN4RSxTQUFTQyxNQUFNLFFBQVEsMEJBQTBCO0FBQ2pELFNBQVNDLE1BQU0sUUFBUSwyQkFBMkI7QUFDbEQsU0FBU0MsTUFBTSxRQUFRLDJCQUEyQjtBQUNsRCxTQUFTQyxvQkFBb0IsUUFBUSx5Q0FBeUM7QUFDOUUsT0FBT0MsU0FBUyxNQUFNLGdCQUFnQjtBQUV0QyxLQUFLQyxpQkFBaUIsR0FBRztFQUN2QkMsT0FBTyxFQUFFLE1BQU07RUFDZkMsZUFBZSxFQUFFLE1BQU07RUFDdkJDLE1BQU0sRUFBRSxDQUFDQyxNQUFNLEVBQUU7SUFBRUMsT0FBTyxFQUFFLE9BQU87SUFBRUMsT0FBTyxFQUFFLE1BQU07RUFBQyxDQUFDLEVBQUUsR0FBRyxJQUFJO0FBQ2pFLENBQUM7QUFFRCxLQUFLQyxZQUFZLEdBQUcsV0FBVyxHQUFHLE1BQU07QUFFeEMsT0FBTyxTQUFTQyxZQUFZQSxDQUFDO0VBQzNCUCxPQUFPO0VBQ1BDLGVBQWU7RUFDZkM7QUFDaUIsQ0FBbEIsRUFBRUgsaUJBQWlCLENBQUMsRUFBRWpCLEtBQUssQ0FBQzBCLFNBQVMsQ0FBQztFQUNyQyxNQUFNLEdB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 \ No newline at end of file diff --git a/src/components/FallbackToolUseErrorMessage.tsx b/src/components/FallbackToolUseErrorMessage.tsx new file mode 100644 index 0000000..12d3b39 --- /dev/null +++ b/src/components/FallbackToolUseErrorMessage.tsx @@ -0,0 +1,116 @@ +import { c as _c } from "react/compiler-runtime"; +import type { ToolResultBlockParam } from '@anthropic-ai/sdk/resources/messages/messages.mjs'; +import * as React from 'react'; +import { stripUnderlineAnsi } from 'src/components/shell/OutputLine.js'; +import { extractTag } from 'src/utils/messages.js'; +import { removeSandboxViolationTags } from 'src/utils/sandbox/sandbox-ui-utils.js'; +import { Box, Text } from '../ink.js'; +import { useShortcutDisplay } from '../keybindings/useShortcutDisplay.js'; +import { countCharInString } from '../utils/stringUtils.js'; +import { MessageResponse } from './MessageResponse.js'; +const MAX_RENDERED_LINES = 10; +type Props = { + result: ToolResultBlockParam['content']; + verbose: boolean; +}; +export function FallbackToolUseErrorMessage(t0) { + const $ = _c(25); + const { + result, + verbose + } = t0; + const transcriptShortcut = useShortcutDisplay("app:toggleTranscript", "Global", "ctrl+o"); + let T0; + let T1; + let T2; + let plusLines; + let t1; + let t2; + let t3; + if ($[0] !== result || $[1] !== verbose) { + let error; + if (typeof result !== "string") { + error = "Tool execution failed"; + } else { + const extractedError = extractTag(result, "tool_use_error") ?? result; + const withoutSandboxViolations = removeSandboxViolationTags(extractedError); + const withoutErrorTags = withoutSandboxViolations.replace(/<\/?error>/g, ""); + const trimmed = withoutErrorTags.trim(); + if (!verbose && trimmed.includes("InputValidationError: ")) { + error = "Invalid tool parameters"; + } else { + if (trimmed.startsWith("Error: ") || trimmed.startsWith("Cancelled: ")) { + error = trimmed; + } else { + error = `Error: ${trimmed}`; + } + } + } + plusLines = countCharInString(error, "\n") + 1 - MAX_RENDERED_LINES; + T2 = MessageResponse; + T1 = Box; + t3 = "column"; + T0 = Text; + t1 = "error"; + t2 = stripUnderlineAnsi(verbose ? error : error.split("\n").slice(0, MAX_RENDERED_LINES).join("\n")); + $[0] = result; + $[1] = verbose; + $[2] = T0; + $[3] = T1; + $[4] = T2; + $[5] = plusLines; + $[6] = t1; + $[7] = t2; + $[8] = t3; + } else { + T0 = $[2]; + T1 = $[3]; + T2 = $[4]; + plusLines = $[5]; + t1 = $[6]; + t2 = $[7]; + t3 = $[8]; + } + let t4; + if ($[9] !== T0 || $[10] !== t1 || $[11] !== t2) { + t4 = {t2}; + $[9] = T0; + $[10] = t1; + $[11] = t2; + $[12] = t4; + } else { + t4 = $[12]; + } + let t5; + if ($[13] !== plusLines || $[14] !== transcriptShortcut || $[15] !== verbose) { + t5 = !verbose && plusLines > 0 && … +{plusLines} {plusLines === 1 ? "line" : "lines"} ({transcriptShortcut} to see all); + $[13] = plusLines; + $[14] = transcriptShortcut; + $[15] = verbose; + $[16] = t5; + } else { + t5 = $[16]; + } + let t6; + if ($[17] !== T1 || $[18] !== t3 || $[19] !== t4 || $[20] !== t5) { + t6 = {t4}{t5}; + $[17] = T1; + $[18] = t3; + $[19] = t4; + $[20] = t5; + $[21] = t6; + } else { + t6 = $[21]; + } + let t7; + if ($[22] !== T2 || $[23] !== t6) { + t7 = {t6}; + $[22] = T2; + $[23] = t6; + $[24] = t7; + } else { + t7 = $[24]; + } + return t7; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJUb29sUmVzdWx0QmxvY2tQYXJhbSIsIlJlYWN0Iiwic3RyaXBVbmRlcmxpbmVBbnNpIiwiZXh0cmFjdFRhZyIsInJlbW92ZVNhbmRib3hWaW9sYXRpb25UYWdzIiwiQm94IiwiVGV4dCIsInVzZVNob3J0Y3V0RGlzcGxheSIsImNvdW50Q2hhckluU3RyaW5nIiwiTWVzc2FnZVJlc3BvbnNlIiwiTUFYX1JFTkRFUkVEX0xJTkVTIiwiUHJvcHMiLCJyZXN1bHQiLCJ2ZXJib3NlIiwiRmFsbGJhY2tUb29sVXNlRXJyb3JNZXNzYWdlIiwidDAiLCIkIiwiX2MiLCJ0cmFuc2NyaXB0U2hvcnRjdXQiLCJUMCIsIlQxIiwiVDIiLCJwbHVzTGluZXMiLCJ0MSIsInQyIiwidDMiLCJlcnJvciIsImV4dHJhY3RlZEVycm9yIiwid2l0aG91dFNhbmRib3hWaW9sYXRpb25zIiwid2l0aG91dEVycm9yVGFncyIsInJlcGxhY2UiLCJ0cmltbWVkIiwidHJpbSIsImluY2x1ZGVzIiwic3RhcnRzV2l0aCIsInNwbGl0Iiwic2xpY2UiLCJqb2luIiwidDQiLCJ0NSIsInQ2IiwidDciXSwic291cmNlcyI6WyJGYWxsYmFja1Rvb2xVc2VFcnJvck1lc3NhZ2UudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHsgVG9vbFJlc3VsdEJsb2NrUGFyYW0gfSBmcm9tICdAYW50aHJvcGljLWFpL3Nkay9yZXNvdXJjZXMvbWVzc2FnZXMvbWVzc2FnZXMubWpzJ1xuaW1wb3J0ICogYXMgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBzdHJpcFVuZGVybGluZUFuc2kgfSBmcm9tICdzcmMvY29tcG9uZW50cy9zaGVsbC9PdXRwdXRMaW5lLmpzJ1xuaW1wb3J0IHsgZXh0cmFjdFRhZyB9IGZyb20gJ3NyYy91dGlscy9tZXNzYWdlcy5qcydcbmltcG9ydCB7IHJlbW92ZVNhbmRib3hWaW9sYXRpb25UYWdzIH0gZnJvbSAnc3JjL3V0aWxzL3NhbmRib3gvc2FuZGJveC11aS11dGlscy5qcydcbmltcG9ydCB7IEJveCwgVGV4dCB9IGZyb20gJy4uL2luay5qcydcbmltcG9ydCB7IHVzZVNob3J0Y3V0RGlzcGxheSB9IGZyb20gJy4uL2tleWJpbmRpbmdzL3VzZVNob3J0Y3V0RGlzcGxheS5qcydcbmltcG9ydCB7IGNvdW50Q2hhckluU3RyaW5nIH0gZnJvbSAnLi4vdXRpbHMvc3RyaW5nVXRpbHMuanMnXG5pbXBvcnQgeyBNZXNzYWdlUmVzcG9uc2UgfSBmcm9tICcuL01lc3NhZ2VSZXNwb25zZS5qcydcblxuY29uc3QgTUFYX1JFTkRFUkVEX0xJTkVTID0gMTBcblxudHlwZSBQcm9wcyA9IHtcbiAgcmVzdWx0OiBUb29sUmVzdWx0QmxvY2tQYXJhbVsnY29udGVudCddXG4gIHZlcmJvc2U6IGJvb2xlYW5cbn1cblxuZXhwb3J0IGZ1bmN0aW9uIEZhbGxiYWNrVG9vbFVzZUVycm9yTWVzc2FnZSh7XG4gIHJlc3VsdCxcbiAgdmVyYm9zZSxcbn06IFByb3BzKTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgY29uc3QgdHJhbnNjcmlwdFNob3J0Y3V0ID0gdXNlU2hvcnRjdXREaXNwbGF5KFxuICAgICdhcHA6dG9nZ2xlVHJhbnNjcmlwdCcsXG4gICAgJ0dsb2JhbCcsXG4gICAgJ2N0cmwrbycsXG4gIClcbiAgbGV0IGVycm9yOiBzdHJpbmdcblxuICBpZiAodHlwZW9mIHJlc3VsdCAhPT0gJ3N0cmluZycpIHtcbiAgICBlcnJvciA9ICdUb29sIGV4ZWN1dGlvbiBmYWlsZWQnXG4gIH0gZWxzZSB7XG4gICAgY29uc3QgZXh0cmFjdGVkRXJyb3IgPSBleHRyYWN0VGFnKHJlc3VsdCwgJ3Rvb2xfdXNlX2Vycm9yJykgPz8gcmVzdWx0XG4gICAgLy8gUmVtb3ZlIHNhbmRib3hfdmlvbGF0aW9ucyB0YWdzIGZyb20gZXJyb3IgZGlzcGxheSAoQ2xhdWRlIHN0aWxsIHNlZXMgdGhlbSBpbiB0aGUgdG9vbCByZXN1bHQpXG4gICAgY29uc3Qgd2l0aG91dFNhbmRib3hWaW9sYXRpb25zID0gcmVtb3ZlU2FuZGJveFZpb2xhdGlvblRhZ3MoZXh0cmFjdGVkRXJyb3IpXG4gICAgLy8gU3RyaXAgPGVycm9yPiB0YWdzIGJ1dCBrZWVwIHRoZWlyIGNvbnRlbnQgKHRhZ3MgYXJlIGZvciB0aGUgbW9kZWwsIG5vdCB0aGUgVUkpXG4gICAgY29uc3Qgd2l0aG91dEVycm9yVGFncyA9IHdpdGhvdXRTYW5kYm94VmlvbGF0aW9ucy5yZXBsYWNlKC88XFwvP2Vycm9yPi9nLCAnJylcbiAgICBjb25zdCB0cmltbWVkID0gd2l0aG91dEVycm9yVGFncy50cmltKClcbiAgICBpZiAoIXZlcmJvc2UgJiYgdHJpbW1lZC5pbmNsdWRlcygnSW5wdXRWYWxpZGF0aW9uRXJyb3I6ICcpKSB7XG4gICAgICBlcnJvciA9ICdJbnZhbGlkIHRvb2wgcGFyYW1ldGVycydcbiAgICB9IGVsc2UgaWYgKFxuICAgICAgdHJpbW1lZC5zdGFydHNXaXRoKCdFcnJvcjogJykgfHxcbiAgICAgIHRyaW1tZWQuc3RhcnRzV2l0aCgnQ2FuY2VsbGVkOiAnKVxuICAgICkge1xuICAgICAgZXJyb3IgPSB0cmltbWVkXG4gICAgfSBlbHNlIHtcbiAgICAgIGVycm9yID0gYEVycm9yOiAke3RyaW1tZWR9YFxuICAgIH1cbiAgfVxuXG4gIGNvbnN0IHBsdXNMaW5lcyA9IGNvdW50Q2hhckluU3RyaW5nKGVycm9yLCAnXFxuJykgKyAxIC0gTUFYX1JFTkRFUkVEX0xJTkVTXG5cbiAgcmV0dXJuIChcbiAgICA8TWVzc2FnZVJlc3BvbnNlPlxuICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCI+XG4gICAgICAgIDxUZXh0IGNvbG9yPVwiZXJyb3JcIj5cbiAgICAgICAgICB7c3RyaXBVbmRlcmxpbmVBbnNpKFxuICAgICAgICAgICAgdmVyYm9zZVxuICAgICAgICAgICAgICA/IGVycm9yXG4gICAgICAgICAgICAgIDogZXJyb3Iuc3BsaXQoJ1xcbicpLnNsaWNlKDAsIE1BWF9SRU5ERVJFRF9MSU5FUykuam9pbignXFxuJyksXG4gICAgICAgICAgKX1cbiAgICAgICAgPC9UZXh0PlxuICAgICAgICB7IXZlcmJvc2UgJiYgcGx1c0xpbmVzID4gMCAmJiAoXG4gICAgICAgICAgLy8gVGhlIGNhcmVmdWwgPFRleHQ+IGxheW91dCBpcyBhIHdvcmthcm91bmQgZm9yIHRoZSBkaW0tYm9sZFxuICAgICAgICAgIC8vIHJlbmRlcmluZyBidWdcbiAgICAgICAgICA8Qm94PlxuICAgICAgICAgICAgPFRleHQgZGltQ29sb3I+XG4gICAgICAgICAgICAgIOKApiAre3BsdXNMaW5lc30ge3BsdXNMaW5lcyA9PT0gMSA/ICdsaW5lJyA6ICdsaW5lcyd9IChcbiAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgIDxUZXh0IGRpbUNvbG9yIGJvbGQ+XG4gICAgICAgICAgICAgIHt0cmFuc2NyaXB0U2hvcnRjdXR9XG4gICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICA8VGV4dD4gPC9UZXh0PlxuICAgICAgICAgICAgPFRleHQgZGltQ29sb3I+dG8gc2VlIGFsbCk8L1RleHQ+XG4gICAgICAgICAgPC9Cb3g+XG4gICAgICAgICl9XG4gICAgICA8L0JveD5cbiAgICA8L01lc3NhZ2VSZXNwb25zZT5cbiAgKVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsY0FBY0Esb0JBQW9CLFFBQVEsbURBQW1EO0FBQzdGLE9BQU8sS0FBS0MsS0FBSyxNQUFNLE9BQU87QUFDOUIsU0FBU0Msa0JBQWtCLFFBQVEsb0NBQW9DO0FBQ3ZFLFNBQVNDLFVBQVUsUUFBUSx1QkFBdUI7QUFDbEQsU0FBU0MsMEJBQTBCLFFBQVEsdUNBQXVDO0FBQ2xGLFNBQVNDLEdBQUcsRUFBRUMsSUFBSSxRQUFRLFdBQVc7QUFDckMsU0FBU0Msa0JBQWtCLFFBQVEsc0NBQXNDO0FBQ3pFLFNBQVNDLGlCQUFpQixRQUFRLHlCQUF5QjtBQUMzRCxTQUFTQyxlQUFlLFFBQVEsc0JBQXNCO0FBRXRELE1BQU1DLGtCQUFrQixHQUFHLEVBQUU7QUFFN0IsS0FBS0MsS0FBSyxHQUFHO0VB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 \ No newline at end of file diff --git a/src/components/FallbackToolUseRejectedMessage.tsx b/src/components/FallbackToolUseRejectedMessage.tsx new file mode 100644 index 0000000..3e0d2ca --- /dev/null +++ b/src/components/FallbackToolUseRejectedMessage.tsx @@ -0,0 +1,16 @@ +import { c as _c } from "react/compiler-runtime"; +import * as React from 'react'; +import { InterruptedByUser } from './InterruptedByUser.js'; +import { MessageResponse } from './MessageResponse.js'; +export function FallbackToolUseRejectedMessage() { + const $ = _c(1); + let t0; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t0 = ; + $[0] = t0; + } else { + t0 = $[0]; + } + return t0; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkludGVycnVwdGVkQnlVc2VyIiwiTWVzc2FnZVJlc3BvbnNlIiwiRmFsbGJhY2tUb29sVXNlUmVqZWN0ZWRNZXNzYWdlIiwiJCIsIl9jIiwidDAiLCJTeW1ib2wiLCJmb3IiXSwic291cmNlcyI6WyJGYWxsYmFja1Rvb2xVc2VSZWplY3RlZE1lc3NhZ2UudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgSW50ZXJydXB0ZWRCeVVzZXIgfSBmcm9tICcuL0ludGVycnVwdGVkQnlVc2VyLmpzJ1xuaW1wb3J0IHsgTWVzc2FnZVJlc3BvbnNlIH0gZnJvbSAnLi9NZXNzYWdlUmVzcG9uc2UuanMnXG5cbmV4cG9ydCBmdW5jdGlvbiBGYWxsYmFja1Rvb2xVc2VSZWplY3RlZE1lc3NhZ2UoKTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgcmV0dXJuIChcbiAgICA8TWVzc2FnZVJlc3BvbnNlIGhlaWdodD17MX0+XG4gICAgICA8SW50ZXJydXB0ZWRCeVVzZXIgLz5cbiAgICA8L01lc3NhZ2VSZXNwb25zZT5cbiAgKVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsT0FBTyxLQUFLQSxLQUFLLE1BQU0sT0FBTztBQUM5QixTQUFTQyxpQkFBaUIsUUFBUSx3QkFBd0I7QUFDMUQsU0FBU0MsZUFBZSxRQUFRLHNCQUFzQjtBQUV0RCxPQUFPLFNBQUFDLCtCQUFBO0VBQUEsTUFBQUMsQ0FBQSxHQUFBQyxFQUFBO0VBQUEsSUFBQUMsRUFBQTtFQUFBLElBQUFGLENBQUEsUUFBQUcsTUFBQSxDQUFBQyxHQUFBO0lBRUhGLEVBQUEsSUFBQyxlQUFlLENBQVMsTUFBQyxDQUFELEdBQUMsQ0FDeEIsQ0FBQyxpQkFBaUIsR0FDcEIsRUFGQyxlQUFlLENBRUU7SUFBQUYsQ0FBQSxNQUFBRSxFQUFBO0VBQUE7SUFBQUEsRUFBQSxHQUFBRixDQUFBO0VBQUE7RUFBQSxPQUZsQkUsRUFFa0I7QUFBQSIsImlnbm9yZUxpc3QiOltdfQ== \ No newline at end of file diff --git a/src/components/FastIcon.tsx b/src/components/FastIcon.tsx new file mode 100644 index 0000000..d229a86 --- /dev/null +++ b/src/components/FastIcon.tsx @@ -0,0 +1,46 @@ +import { c as _c } from "react/compiler-runtime"; +import chalk from 'chalk'; +import * as React from 'react'; +import { LIGHTNING_BOLT } from '../constants/figures.js'; +import { Text } from '../ink.js'; +import { getGlobalConfig } from '../utils/config.js'; +import { resolveThemeSetting } from '../utils/systemTheme.js'; +import { color } from './design-system/color.js'; +type Props = { + cooldown?: boolean; +}; +export function FastIcon(t0) { + const $ = _c(2); + const { + cooldown + } = t0; + if (cooldown) { + let t1; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t1 = {LIGHTNING_BOLT}; + $[0] = t1; + } else { + t1 = $[0]; + } + return t1; + } + let t1; + if ($[1] === Symbol.for("react.memo_cache_sentinel")) { + t1 = {LIGHTNING_BOLT}; + $[1] = t1; + } else { + t1 = $[1]; + } + return t1; +} +export function getFastIconString(applyColor = true, cooldown = false): string { + if (!applyColor) { + return LIGHTNING_BOLT; + } + const themeName = resolveThemeSetting(getGlobalConfig().theme); + if (cooldown) { + return chalk.dim(color('promptBorder', themeName)(LIGHTNING_BOLT)); + } + return color('fastMode', themeName)(LIGHTNING_BOLT); +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJjaGFsayIsIlJlYWN0IiwiTElHSFROSU5HX0JPTFQiLCJUZXh0IiwiZ2V0R2xvYmFsQ29uZmlnIiwicmVzb2x2ZVRoZW1lU2V0dGluZyIsImNvbG9yIiwiUHJvcHMiLCJjb29sZG93biIsIkZhc3RJY29uIiwidDAiLCIkIiwiX2MiLCJ0MSIsIlN5bWJvbCIsImZvciIsImdldEZhc3RJY29uU3RyaW5nIiwiYXBwbHlDb2xvciIsInRoZW1lTmFtZSIsInRoZW1lIiwiZGltIl0sInNvdXJjZXMiOlsiRmFzdEljb24udHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBjaGFsayBmcm9tICdjaGFsaydcbmltcG9ydCAqIGFzIFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgTElHSFROSU5HX0JPTFQgfSBmcm9tICcuLi9jb25zdGFudHMvZmlndXJlcy5qcydcbmltcG9ydCB7IFRleHQgfSBmcm9tICcuLi9pbmsuanMnXG5pbXBvcnQgeyBnZXRHbG9iYWxDb25maWcgfSBmcm9tICcuLi91dGlscy9jb25maWcuanMnXG5pbXBvcnQgeyByZXNvbHZlVGhlbWVTZXR0aW5nIH0gZnJvbSAnLi4vdXRpbHMvc3lzdGVtVGhlbWUuanMnXG5pbXBvcnQgeyBjb2xvciB9IGZyb20gJy4vZGVzaWduLXN5c3RlbS9jb2xvci5qcydcblxudHlwZSBQcm9wcyA9IHtcbiAgY29vbGRvd24/OiBib29sZWFuXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBGYXN0SWNvbih7IGNvb2xkb3duIH06IFByb3BzKTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgaWYgKGNvb2xkb3duKSB7XG4gICAgcmV0dXJuIChcbiAgICAgIDxUZXh0IGNvbG9yPVwicHJvbXB0Qm9yZGVyXCIgZGltQ29sb3I+XG4gICAgICAgIHtMSUdIVE5JTkdfQk9MVH1cbiAgICAgIDwvVGV4dD5cbiAgICApXG4gIH1cbiAgcmV0dXJuIDxUZXh0IGNvbG9yPVwiZmFzdE1vZGVcIj57TElHSFROSU5HX0JPTFR9PC9UZXh0PlxufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2V0RmFzdEljb25TdHJpbmcoYXBwbHlDb2xvciA9IHRydWUsIGNvb2xkb3duID0gZmFsc2UpOiBzdHJpbmcge1xuICBpZiAoIWFwcGx5Q29sb3IpIHtcbiAgICByZXR1cm4gTElHSFROSU5HX0JPTFRcbiAgfVxuICBjb25zdCB0aGVtZU5hbWUgPSByZXNvbHZlVGhlbWVTZXR0aW5nKGdldEdsb2JhbENvbmZpZygpLnRoZW1lKVxuICBpZiAoY29vbGRvd24pIHtcbiAgICByZXR1cm4gY2hhbGsuZGltKGNvbG9yKCdwcm9tcHRCb3JkZXInLCB0aGVtZU5hbWUpKExJR0hUTklOR19CT0xUKSlcbiAgfVxuICByZXR1cm4gY29sb3IoJ2Zhc3RNb2RlJywgdGhlbWVOYW1lKShMSUdIVE5JTkdfQk9MVClcbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU9BLEtBQUssTUFBTSxPQUFPO0FBQ3pCLE9BQU8sS0FBS0MsS0FBSyxNQUFNLE9BQU87QUFDOUIsU0FBU0MsY0FBYyxRQUFRLHlCQUF5QjtBQUN4RCxTQUFTQyxJQUFJLFFBQVEsV0FBVztBQUNoQyxTQUFTQyxlQUFlLFFBQVEsb0JBQW9CO0FBQ3BELFNBQVNDLG1CQUFtQixRQUFRLHlCQUF5QjtBQUM3RCxTQUFTQyxLQUFLLFFBQVEsMEJBQTBCO0FBRWhELEtBQUtDLEtBQUssR0FBRztFQUNYQyxRQUFRLENBQUMsRUFBRSxPQUFPO0FBQ3BCLENBQUM7QUFFRCxPQUFPLFNBQUFDLFNBQUFDLEVBQUE7RUFBQSxNQUFBQyxDQUFBLEdBQUFDLEVBQUE7RUFBa0I7SUFBQUo7RUFBQSxJQUFBRSxFQUFtQjtFQUMxQyxJQUFJRixRQUFRO0lBQUEsSUFBQUssRUFBQTtJQUFBLElBQUFGLENBQUEsUUFBQUcsTUFBQSxDQUFBQyxHQUFBO01BRVJGLEVBQUEsSUFBQyxJQUFJLENBQU8sS0FBYyxDQUFkLGNBQWMsQ0FBQyxRQUFRLENBQVIsS0FBTyxDQUFDLENBQ2hDWCxlQUFhLENBQ2hCLEVBRkMsSUFBSSxDQUVFO01BQUFTLENBQUEsTUFBQUUsRUFBQTtJQUFBO01BQUFBLEVBQUEsR0FBQUYsQ0FBQTtJQUFBO0lBQUEsT0FGUEUsRUFFTztFQUFBO0VBRVYsSUFBQUEsRUFBQTtFQUFBLElBQUFGLENBQUEsUUFBQUcsTUFBQSxDQUFBQyxHQUFBO0lBQ01GLEVBQUEsSUFBQyxJQUFJLENBQU8sS0FBVSxDQUFWLFVBQVUsQ0FBRVgsZUFBYSxDQUFFLEVBQXRDLElBQUksQ0FBeUM7SUFBQVMsQ0FBQSxNQUFBRSxFQUFBO0VBQUE7SUFBQUEsRUFBQSxHQUFBRixDQUFBO0VBQUE7RUFBQSxPQUE5Q0UsRUFBOEM7QUFBQTtBQUd2RCxPQUFPLFNBQVNHLGlCQUFpQkEsQ0FBQ0MsVUFBVSxHQUFHLElBQUksRUFBRVQsUUFBUSxHQUFHLEtBQUssQ0FBQyxFQUFFLE1BQU0sQ0FBQztFQUM3RSxJQUFJLENBQUNTLFVBQVUsRUFBRTtJQUNmLE9BQU9mLGNBQWM7RUFDdkI7RUFDQSxNQUFNZ0IsU0FBUyxHQUFHYixtQkFBbUIsQ0FBQ0QsZUFBZSxDQUFDLENBQUMsQ0FBQ2UsS0FBSyxDQUFDO0VBQzlELElBQUlYLFFBQVEsRUFBRTtJQUNaLE9BQU9SLEtBQUssQ0FBQ29CLEdBQUcsQ0FBQ2QsS0FBSyxDQUFDLGNBQWMsRUFBRVksU0FBUyxDQUFDLENBQUNoQixjQUFjLENBQUMsQ0FBQztFQUNwRTtFQUNBLE9BQU9JLEtBQUssQ0FBQyxVQUFVLEVBQUVZLFNBQVMsQ0FBQyxDQUFDaEIsY0FBYyxDQUFDO0FBQ3JEIiwiaWdub3JlTGlzdCI6W119 \ No newline at end of file diff --git a/src/components/Feedback.tsx b/src/components/Feedback.tsx new file mode 100644 index 0000000..8f2fbb1 --- /dev/null +++ b/src/components/Feedback.tsx @@ -0,0 +1,592 @@ +import axios from 'axios'; +import { readFile, stat } from 'fs/promises'; +import * as React from 'react'; +import { useCallback, useEffect, useState } from 'react'; +import { getLastAPIRequest } from 'src/bootstrap/state.js'; +import { logEventTo1P } from 'src/services/analytics/firstPartyEventLogger.js'; +import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from 'src/services/analytics/index.js'; +import { getLastAssistantMessage, normalizeMessagesForAPI } from 'src/utils/messages.js'; +import type { CommandResultDisplay } from '../commands.js'; +import { useTerminalSize } from '../hooks/useTerminalSize.js'; +import { Box, Text, useInput } from '../ink.js'; +import { useKeybinding } from '../keybindings/useKeybinding.js'; +import { queryHaiku } from '../services/api/claude.js'; +import { startsWithApiErrorPrefix } from '../services/api/errors.js'; +import type { Message } from '../types/message.js'; +import { checkAndRefreshOAuthTokenIfNeeded } from '../utils/auth.js'; +import { openBrowser } from '../utils/browser.js'; +import { logForDebugging } from '../utils/debug.js'; +import { env } from '../utils/env.js'; +import { type GitRepoState, getGitState, getIsGit } from '../utils/git.js'; +import { getAuthHeaders, getUserAgent } from '../utils/http.js'; +import { getInMemoryErrors, logError } from '../utils/log.js'; +import { isEssentialTrafficOnly } from '../utils/privacyLevel.js'; +import { extractTeammateTranscriptsFromTasks, getTranscriptPath, loadAllSubagentTranscriptsFromDisk, MAX_TRANSCRIPT_READ_BYTES } from '../utils/sessionStorage.js'; +import { jsonStringify } from '../utils/slowOperations.js'; +import { asSystemPrompt } from '../utils/systemPromptType.js'; +import { ConfigurableShortcutHint } from './ConfigurableShortcutHint.js'; +import { Byline } from './design-system/Byline.js'; +import { Dialog } from './design-system/Dialog.js'; +import { KeyboardShortcutHint } from './design-system/KeyboardShortcutHint.js'; +import TextInput from './TextInput.js'; + +// This value was determined experimentally by testing the URL length limit +const GITHUB_URL_LIMIT = 7250; +const GITHUB_ISSUES_REPO_URL = "external" === 'ant' ? 'https://github.com/anthropics/claude-cli-internal/issues' : 'https://github.com/anthropics/claude-code/issues'; +type Props = { + abortSignal: AbortSignal; + messages: Message[]; + initialDescription?: string; + onDone(result: string, options?: { + display?: CommandResultDisplay; + }): void; + backgroundTasks?: { + [taskId: string]: { + type: string; + identity?: { + agentId: string; + }; + messages?: Message[]; + }; + }; +}; +type Step = 'userInput' | 'consent' | 'submitting' | 'done'; +type FeedbackData = { + // latestAssistantMessageId is the message ID from the latest main model call + latestAssistantMessageId: string | null; + message_count: number; + datetime: string; + description: string; + platform: string; + gitRepo: boolean; + version: string | null; + transcript: Message[]; + subagentTranscripts?: { + [agentId: string]: Message[]; + }; + rawTranscriptJsonl?: string; +}; + +// Utility function to redact sensitive information from strings +export function redactSensitiveInfo(text: string): string { + let redacted = text; + + // Anthropic API keys (sk-ant...) with or without quotes + // First handle the case with quotes + redacted = redacted.replace(/"(sk-ant[^\s"']{24,})"/g, '"[REDACTED_API_KEY]"'); + // Then handle the cases without quotes - more general pattern + redacted = redacted.replace( + // eslint-disable-next-line custom-rules/no-lookbehind-regex -- .replace(re, string) on /bug path: no-match returns same string (Object.is) + /(? { + // Sanitize error logs to remove any API keys + return getInMemoryErrors().map(errorInfo => { + // Create a copy of the error info to avoid modifying the original + const errorCopy = { + ...errorInfo + } as { + error?: string; + timestamp?: string; + }; + + // Sanitize error if present and is a string + if (errorCopy && typeof errorCopy.error === 'string') { + errorCopy.error = redactSensitiveInfo(errorCopy.error); + } + return errorCopy; + }); +} +async function loadRawTranscriptJsonl(): Promise { + try { + const transcriptPath = getTranscriptPath(); + const { + size + } = await stat(transcriptPath); + if (size > MAX_TRANSCRIPT_READ_BYTES) { + logForDebugging(`Skipping raw transcript read: file too large (${size} bytes)`, { + level: 'warn' + }); + return null; + } + return await readFile(transcriptPath, 'utf-8'); + } catch { + return null; + } +} +export function Feedback({ + abortSignal, + messages, + initialDescription, + onDone, + backgroundTasks = {} +}: Props): React.ReactNode { + const [step, setStep] = useState('userInput'); + const [cursorOffset, setCursorOffset] = useState(0); + const [description, setDescription] = useState(initialDescription ?? ''); + const [feedbackId, setFeedbackId] = useState(null); + const [error, setError] = useState(null); + const [envInfo, setEnvInfo] = useState<{ + isGit: boolean; + gitState: GitRepoState | null; + }>({ + isGit: false, + gitState: null + }); + const [title, setTitle] = useState(null); + const textInputColumns = useTerminalSize().columns - 4; + useEffect(() => { + async function loadEnvInfo() { + const isGit = await getIsGit(); + let gitState: GitRepoState | null = null; + if (isGit) { + gitState = await getGitState(); + } + setEnvInfo({ + isGit, + gitState + }); + } + void loadEnvInfo(); + }, []); + const submitReport = useCallback(async () => { + setStep('submitting'); + setError(null); + setFeedbackId(null); + + // Get sanitized errors for the report + const sanitizedErrors = getSanitizedErrorLogs(); + + // Extract last assistant message ID from messages array + const lastAssistantMessage = getLastAssistantMessage(messages); + const lastAssistantMessageId = lastAssistantMessage?.requestId ?? null; + const [diskTranscripts, rawTranscriptJsonl] = await Promise.all([loadAllSubagentTranscriptsFromDisk(), loadRawTranscriptJsonl()]); + const teammateTranscripts = extractTeammateTranscriptsFromTasks(backgroundTasks); + const subagentTranscripts = { + ...diskTranscripts, + ...teammateTranscripts + }; + const reportData = { + latestAssistantMessageId: lastAssistantMessageId, + message_count: messages.length, + datetime: new Date().toISOString(), + description, + platform: env.platform, + gitRepo: envInfo.isGit, + terminal: env.terminal, + version: MACRO.VERSION, + transcript: normalizeMessagesForAPI(messages), + errors: sanitizedErrors, + lastApiRequest: getLastAPIRequest(), + ...(Object.keys(subagentTranscripts).length > 0 && { + subagentTranscripts + }), + ...(rawTranscriptJsonl && { + rawTranscriptJsonl + }) + }; + const [result, t] = await Promise.all([submitFeedback(reportData, abortSignal), generateTitle(description, abortSignal)]); + setTitle(t); + if (result.success) { + if (result.feedbackId) { + setFeedbackId(result.feedbackId); + logEvent('tengu_bug_report_submitted', { + feedback_id: result.feedbackId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + last_assistant_message_id: lastAssistantMessageId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + // 1P-only: freeform text approved for BQ. Join on feedback_id. + logEventTo1P('tengu_bug_report_description', { + feedback_id: result.feedbackId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + description: redactSensitiveInfo(description) as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + } + setStep('done'); + } else { + if (result.isZdrOrg) { + setError('Feedback collection is not available for organizations with custom data retention policies.'); + } else { + setError('Could not submit feedback. Please try again later.'); + } + // Stay on userInput step so user can retry with their content preserved + setStep('userInput'); + } + }, [description, envInfo.isGit, messages]); + + // Handle cancel - this will be called by Dialog's automatic Esc handling + const handleCancel = useCallback(() => { + // Don't cancel when done - let other keys close the dialog + if (step === 'done') { + if (error) { + onDone('Error submitting feedback / bug report', { + display: 'system' + }); + } else { + onDone('Feedback / bug report submitted', { + display: 'system' + }); + } + return; + } + onDone('Feedback / bug report cancelled', { + display: 'system' + }); + }, [step, error, onDone]); + + // During text input, use Settings context where only Escape (not 'n') triggers confirm:no. + // This allows typing 'n' in the text field while still supporting Escape to cancel. + useKeybinding('confirm:no', handleCancel, { + context: 'Settings', + isActive: step === 'userInput' + }); + useInput((input, key) => { + // Allow any key press to close the dialog when done or when there's an error + if (step === 'done') { + if (key.return && title) { + // Open GitHub issue URL when Enter is pressed + const issueUrl = createGitHubIssueUrl(feedbackId ?? '', title, description, getSanitizedErrorLogs()); + void openBrowser(issueUrl); + } + if (error) { + onDone('Error submitting feedback / bug report', { + display: 'system' + }); + } else { + onDone('Feedback / bug report submitted', { + display: 'system' + }); + } + return; + } + + // When in userInput step with error, allow user to edit and retry + // (don't close on any keypress - they can still press Esc to cancel) + if (error && step !== 'userInput') { + onDone('Error submitting feedback / bug report', { + display: 'system' + }); + return; + } + if (step === 'consent' && (key.return || input === ' ')) { + void submitReport(); + } + }); + return exitState.pending ? Press {exitState.keyName} again to exit : step === 'userInput' ? + + + : step === 'consent' ? + + + : null}> + {step === 'userInput' && + Describe the issue below: + { + setDescription(value); + // Clear error when user starts editing to allow retry + if (error) { + setError(null); + } + }} columns={textInputColumns} onSubmit={() => setStep('consent')} onExitMessage={() => onDone('Feedback cancelled', { + display: 'system' + })} cursorOffset={cursorOffset} onChangeCursorOffset={setCursorOffset} showCursor /> + {error && + {error} + + Edit and press Enter to retry, or Esc to cancel + + } + } + + {step === 'consent' && + This report will include: + + + - Your feedback / bug description:{' '} + {description} + + + - Environment info:{' '} + + {env.platform}, {env.terminal}, v{MACRO.VERSION} + + + {envInfo.gitState && + - Git repo metadata:{' '} + + {envInfo.gitState.branchName} + {envInfo.gitState.commitHash ? `, ${envInfo.gitState.commitHash.slice(0, 7)}` : ''} + {envInfo.gitState.remoteUrl ? ` @ ${envInfo.gitState.remoteUrl}` : ''} + {!envInfo.gitState.isHeadOnRemote && ', not synced'} + {!envInfo.gitState.isClean && ', has local changes'} + + } + - Current session transcript + + + + We will use your feedback to debug related issues or to improve{' '} + Claude Code's functionality (eg. to reduce the risk of bugs + occurring in the future). + + + + + Press Enter to confirm and submit. + + + } + + {step === 'submitting' && + Submitting report… + } + + {step === 'done' && + {error ? {error} : Thank you for your report!} + {feedbackId && Feedback ID: {feedbackId}} + + Press + Enter + + to open your browser and draft a GitHub issue, or any other key to + close. + + + } + ; +} +export function createGitHubIssueUrl(feedbackId: string, title: string, description: string, errors: Array<{ + error?: string; + timestamp?: string; +}>): string { + const sanitizedTitle = redactSensitiveInfo(title); + const sanitizedDescription = redactSensitiveInfo(description); + const bodyPrefix = `**Bug Description**\n${sanitizedDescription}\n\n` + `**Environment Info**\n` + `- Platform: ${env.platform}\n` + `- Terminal: ${env.terminal}\n` + `- Version: ${MACRO.VERSION || 'unknown'}\n` + `- Feedback ID: ${feedbackId}\n` + `\n**Errors**\n\`\`\`json\n`; + const errorSuffix = `\n\`\`\`\n`; + const errorsJson = jsonStringify(errors); + const baseUrl = `${GITHUB_ISSUES_REPO_URL}/new?title=${encodeURIComponent(sanitizedTitle)}&labels=user-reported,bug&body=`; + const truncationNote = `\n**Note:** Content was truncated.\n`; + const encodedPrefix = encodeURIComponent(bodyPrefix); + const encodedSuffix = encodeURIComponent(errorSuffix); + const encodedNote = encodeURIComponent(truncationNote); + const encodedErrors = encodeURIComponent(errorsJson); + + // Calculate space available for errors + const spaceForErrors = GITHUB_URL_LIMIT - baseUrl.length - encodedPrefix.length - encodedSuffix.length - encodedNote.length; + + // If description alone exceeds limit, truncate everything + if (spaceForErrors <= 0) { + const ellipsis = encodeURIComponent('…'); + const buffer = 50; // Extra safety margin + const maxEncodedLength = GITHUB_URL_LIMIT - baseUrl.length - ellipsis.length - encodedNote.length - buffer; + const fullBody = bodyPrefix + errorsJson + errorSuffix; + let encodedFullBody = encodeURIComponent(fullBody); + if (encodedFullBody.length > maxEncodedLength) { + encodedFullBody = encodedFullBody.slice(0, maxEncodedLength); + // Don't cut in middle of %XX sequence + const lastPercent = encodedFullBody.lastIndexOf('%'); + if (lastPercent >= encodedFullBody.length - 2) { + encodedFullBody = encodedFullBody.slice(0, lastPercent); + } + } + return baseUrl + encodedFullBody + ellipsis + encodedNote; + } + + // If errors fit, no truncation needed + if (encodedErrors.length <= spaceForErrors) { + return baseUrl + encodedPrefix + encodedErrors + encodedSuffix; + } + + // Truncate errors to fit (prioritize keeping description) + // Slice encoded errors directly, then trim to avoid cutting %XX sequences + const ellipsis = encodeURIComponent('…'); + const buffer = 50; // Extra safety margin + let truncatedEncodedErrors = encodedErrors.slice(0, spaceForErrors - ellipsis.length - buffer); + // If we cut in middle of %XX, back up to before the % + const lastPercent = truncatedEncodedErrors.lastIndexOf('%'); + if (lastPercent >= truncatedEncodedErrors.length - 2) { + truncatedEncodedErrors = truncatedEncodedErrors.slice(0, lastPercent); + } + return baseUrl + encodedPrefix + truncatedEncodedErrors + ellipsis + encodedSuffix + encodedNote; +} +async function generateTitle(description: string, abortSignal: AbortSignal): Promise { + try { + const response = await queryHaiku({ + systemPrompt: asSystemPrompt(['Generate a concise, technical issue title (max 80 chars) for a public GitHub issue based on this bug report for Claude Code.', 'Claude Code is an agentic coding CLI based on the Anthropic API.', 'The title should:', '- Include the type of issue [Bug] or [Feature Request] as the first thing in the title', '- Be concise, specific and descriptive of the actual problem', '- Use technical terminology appropriate for a software issue', '- For error messages, extract the key error (e.g., "Missing Tool Result Block" rather than the full message)', '- Be direct and clear for developers to understand the problem', '- If you cannot determine a clear issue, use "Bug Report: [brief description]"', '- Any LLM API errors are from the Anthropic API, not from any other model provider', 'Your response will be directly used as the title of the Github issue, and as such should not contain any other commentary or explaination', 'Examples of good titles include: "[Bug] Auto-Compact triggers to soon", "[Bug] Anthropic API Error: Missing Tool Result Block", "[Bug] Error: Invalid Model Name for Opus"']), + userPrompt: description, + signal: abortSignal, + options: { + hasAppendSystemPrompt: false, + toolChoice: undefined, + isNonInteractiveSession: false, + agents: [], + querySource: 'feedback', + mcpTools: [] + } + }); + const title = response.message.content[0]?.type === 'text' ? response.message.content[0].text : 'Bug Report'; + + // Check if the title contains an API error message + if (startsWithApiErrorPrefix(title)) { + return createFallbackTitle(description); + } + return title; + } catch (error) { + // If there's any error in title generation, use a fallback title + logError(error); + return createFallbackTitle(description); + } +} +function createFallbackTitle(description: string): string { + // Create a safe fallback title based on the bug description + + // Try to extract a meaningful title from the first line + const firstLine = description.split('\n')[0] || ''; + + // If the first line is very short, use it directly + if (firstLine.length <= 60 && firstLine.length > 5) { + return firstLine; + } + + // For longer descriptions, create a truncated version + // Truncate at word boundaries when possible + let truncated = firstLine.slice(0, 60); + if (firstLine.length > 60) { + // Find the last space before the 60 char limit + const lastSpace = truncated.lastIndexOf(' '); + if (lastSpace > 30) { + // Only trim at word if we're not cutting too much + truncated = truncated.slice(0, lastSpace); + } + truncated += '...'; + } + return truncated.length < 10 ? 'Bug Report' : truncated; +} + +// Helper function to sanitize and log errors without exposing API keys +function sanitizeAndLogError(err: unknown): void { + if (err instanceof Error) { + // Create a copy with potentially sensitive info redacted + const safeError = new Error(redactSensitiveInfo(err.message)); + + // Also redact the stack trace if present + if (err.stack) { + safeError.stack = redactSensitiveInfo(err.stack); + } + logError(safeError); + } else { + // For non-Error objects, convert to string and redact sensitive info + const errorString = redactSensitiveInfo(String(err)); + logError(new Error(errorString)); + } +} +async function submitFeedback(data: FeedbackData, signal?: AbortSignal): Promise<{ + success: boolean; + feedbackId?: string; + isZdrOrg?: boolean; +}> { + if (isEssentialTrafficOnly()) { + return { + success: false + }; + } + try { + // Ensure OAuth token is fresh before getting auth headers + // This prevents 401 errors from stale cached tokens + await checkAndRefreshOAuthTokenIfNeeded(); + const authResult = getAuthHeaders(); + if (authResult.error) { + return { + success: false + }; + } + const headers: Record = { + 'Content-Type': 'application/json', + 'User-Agent': getUserAgent(), + ...authResult.headers + }; + const response = await axios.post('https://api.anthropic.com/api/claude_cli_feedback', { + content: jsonStringify(data) + }, { + headers, + timeout: 30000, + // 30 second timeout to prevent hanging + signal + }); + if (response.status === 200) { + const result = response.data; + if (result?.feedback_id) { + return { + success: true, + feedbackId: result.feedback_id + }; + } + sanitizeAndLogError(new Error('Failed to submit feedback: request did not return feedback_id')); + return { + success: false + }; + } + sanitizeAndLogError(new Error('Failed to submit feedback:' + response.status)); + return { + success: false + }; + } catch (err) { + // Handle cancellation/abort - don't log as error + if (axios.isCancel(err)) { + return { + success: false + }; + } + if (axios.isAxiosError(err) && err.response?.status === 403) { + const errorData = err.response.data; + if (errorData?.error?.type === 'permission_error' && errorData?.error?.message?.includes('Custom data retention settings')) { + sanitizeAndLogError(new Error('Cannot submit feedback because custom data retention settings are enabled')); + return { + success: false, + isZdrOrg: true + }; + } + } + // Use our safe error logging function to avoid leaking API keys + sanitizeAndLogError(err); + return { + success: false + }; + } +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJheGlvcyIsInJlYWRGaWxlIiwic3RhdCIsIlJlYWN0IiwidXNlQ2FsbGJhY2siLCJ1c2VFZmZlY3QiLCJ1c2VTdGF0ZSIsImdldExhc3RBUElSZXF1ZXN0IiwibG9nRXZlbnRUbzFQIiwiQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyIsImxvZ0V2ZW50IiwiZ2V0TGFzdEFzc2lzdGFudE1lc3NhZ2UiLCJub3JtYWxpemVNZXNzYWdlc0ZvckFQSSIsIkNvbW1hbmRSZXN1bHREaXNwbGF5IiwidXNlVGVybWluYWxTaXplIiwiQm94IiwiVGV4dCIsInVzZUlucHV0IiwidXNlS2V5YmluZGluZyIsInF1ZXJ5SGFpa3UiLCJzdGFydHNXaXRoQXBpRXJyb3JQcmVmaXgiLCJNZXNzYWdlIiwiY2hlY2tBbmRSZWZyZXNoT0F1dGhUb2tlbklmTmVlZGVkIiwib3BlbkJyb3dzZXIiLCJsb2dGb3JEZWJ1Z2dpbmciLCJlbnYiLCJHaXRSZXBvU3RhdGUiLCJnZXRHaXRTdGF0ZSIsImdldElzR2l0IiwiZ2V0QXV0aEhlYWRlcnMiLCJnZXRVc2VyQWdlbnQiLCJnZXRJbk1lbW9yeUVycm9ycyIsImxvZ0Vycm9yIiwiaXNFc3NlbnRpYWxUcmFmZmljT25seSIsImV4dHJhY3RUZWFtbWF0ZVRyYW5zY3JpcHRzRnJvbVRhc2tzIiwiZ2V0VHJhbnNjcmlwdFBhdGgiLCJsb2FkQWxsU3ViYWdlbnRUcmFuc2NyaXB0c0Zyb21EaXNrIiwiTUFYX1RSQU5TQ1JJUFRfUkVBRF9CWVRFUyIsImpzb25TdHJpbmdpZnkiLCJhc1N5c3RlbVByb21wdCIsIkNvbmZpZ3VyYWJsZVNob3J0Y3V0SGludCIsIkJ5bGluZSIsIkRpYWxvZyIsIktleWJvYXJkU2hvcnRjdXRIaW50IiwiVGV4dElucHV0IiwiR0lUSFVCX1VSTF9MSU1JVCIsIkdJVEhVQl9JU1NVRVNfUkVQT19VUkwiLCJQcm9wcyIsImFib3J0U2lnbmFsIiwiQWJvcnRTaWduYWwiLCJtZXNzYWdlcyIsImluaXRpYWxEZXNjcmlwdGlvbiIsIm9uRG9uZSIsInJlc3VsdCIsIm9wdGlvbnMiLCJkaXNwbGF5IiwiYmFja2dyb3VuZFRhc2tzIiwidGFza0lkIiwidHlwZSIsImlkZW50aXR5IiwiYWdlbnRJZCIsIlN0ZXAiLCJGZWVkYmFja0RhdGEiLCJsYXRlc3RBc3Npc3RhbnRNZXNzYWdlSWQiLCJtZXNzYWdlX2NvdW50IiwiZGF0ZXRpbWUiLCJkZXNjcmlwdGlvbiIsInBsYXRmb3JtIiwiZ2l0UmVwbyIsInZlcnNpb24iLCJ0cmFuc2NyaXB0Iiwic3ViYWdlbnRUcmFuc2NyaXB0cyIsInJhd1RyYW5zY3JpcHRKc29ubCIsInJlZGFjdFNlbnNpdGl2ZUluZm8iLCJ0ZXh0IiwicmVkYWN0ZWQiLCJyZXBsYWNlIiwiZ2V0U2FuaXRpemVkRXJyb3JMb2dzIiwiQXJyYXkiLCJlcnJvciIsInRpbWVzdGFtcCIsIm1hcCIsImVycm9ySW5mbyIsImVycm9yQ29weSIsImxvYWRSYXdUcmFuc2NyaXB0SnNvbmwiLCJQcm9taXNlIiwidHJhbnNjcmlwdFBhdGgiLCJzaXplIiwibGV2ZWwiLCJGZWVkYmFjayIsIlJlYWN0Tm9kZSIsInN0ZXAiLCJzZXRTdGVwIiwiY3Vyc29yT2Zmc2V0Iiwic2V0Q3Vyc29yT2Zmc2V0Iiwic2V0RGVzY3JpcHRpb24iLCJmZWVkYmFja0lkIiwic2V0RmVlZGJhY2tJZCIsInNldEVycm9yIiwiZW52SW5mbyIsInNldEVudkluZm8iLCJpc0dpdCIsImdpdFN0YXRlIiwidGl0bGUiLCJzZXRUaXRsZSIsInRleHRJbnB1dENvbHVtbnMiLCJjb2x1bW5zIiwibG9hZEVudkluZm8iLCJzdWJtaXRSZXBvcnQiLCJzYW5pdGl6ZWRFcnJvcnMiLCJsYXN0QXNzaXN0YW50TWVzc2FnZSIsImxhc3RBc3Npc3RhbnRNZXNzYWdlSWQiLCJyZXF1ZXN0SWQiLCJkaXNrVHJhbnNjcmlwdHMiLCJhbGwiLCJ0ZWFtbWF0ZVRyYW5zY3JpcHRzIiwicmVwb3J0RGF0YSIsImxlbmd0aCIsIkRhdGUiLCJ0b0lTT1N0cmluZyIsInRlcm1pbmFsIiwiTUFDUk8iLCJWRVJTSU9OIiwiZXJyb3JzIiwibGFzdEFwaVJlcXVlc3QiLCJPYmplY3QiLCJrZXlzIiwidCIsInN1Ym1pdEZlZWRiYWNrIiwiZ2VuZXJhdGVUaXRsZSIsInN1Y2Nlc3MiLCJmZWVkYmFja19pZCIsImxhc3RfYXNzaXN0YW50X21lc3NhZ2VfaWQiLCJpc1pkck9yZyIsImhhbmRsZUNhbmNlbCIsImNvbnRleHQiLCJpc0FjdGl2ZSIsImlucHV0Iiwia2V5IiwicmV0dXJuIiwiaXNzdWVVcmwiLCJjcmVhdGVHaXRIdWJJc3N1ZVVybCIsImV4aXRTdGF0ZSIsInBlbmRpbmciLCJrZXlOYW1lIiwidmFsdWUiLCJicmFuY2hOYW1lIiwiY29tbWl0SGFzaCIsInNsaWNlIiwicmVtb3RlVXJsIiwiaXNIZWFkT25SZW1vdGUiLCJpc0NsZWFuIiwic2FuaXRpemVkVGl0bGUiLCJzYW5pdGl6ZWREZXNjcmlwdGlvbiIsImJvZHlQcmVmaXgiLCJlcnJvclN1ZmZpeCIsImVycm9yc0pzb24iLCJiYXNlVXJsIiwiZW5jb2RlVVJJQ29tcG9uZW50IiwidHJ1bmNhdGlvbk5vdGUiLCJlbmNvZGVkUHJlZml4IiwiZW5jb2RlZFN1ZmZpeCIsImVuY29kZWROb3RlIiwiZW5jb2RlZEVycm9ycyIsInNwYWNlRm9yRXJyb3JzIiwiZWxsaXBzaXMiLCJidWZmZXIiLCJtYXhFbmNvZGVkTGVuZ3RoIiwiZnVsbEJvZHkiLCJlbmNvZGVkRnVsbEJvZHkiLCJsYXN0UGVyY2VudCIsImxhc3RJbmRleE9mIiwidHJ1bmNhdGVkRW5jb2RlZEVycm9ycyIsInJlc3BvbnNlIiwic3lzdGVtUHJvbXB0IiwidXNlclByb21wdCIsInNpZ25hbCIsImhhc0FwcGVuZFN5c3RlbVByb21wdCIsInRvb2xDaG9pY2UiLCJ1bmRlZmluZWQiLCJpc05vbkludGVyYWN0aXZlU2Vzc2lvbiIsImFnZW50cyIsInF1ZXJ5U291cmNlIiwibWNwVG9vbHMiLCJtZXNzYWdlIiwiY29udGVudCIsImNyZWF0ZUZhbGxiYWNrVGl0bGUiLCJmaXJzdExpbmUiLCJzcGxpdCIsInRydW5jYXRlZCIsImxhc3RTcGFjZSIsInNhbml0aXplQW5kTG9nRXJyb3IiLCJlcnIiLCJFcnJvciIsInNhZmVFcnJvciIsInN0YWNrIiwiZXJyb3JTdHJpbmciLCJTdHJpbmciLCJkYXRhIiwiYXV0aFJlc3VsdCIsImhlYWRlcnMiLCJSZWNvcmQiLCJwb3N0IiwidGltZW91dCIsInN0YXR1cyIsImlzQ2FuY2VsIiwiaXNBeGlvc0Vycm9yIiwiZXJyb3JEYXRhIiwiaW5jbHVkZXMiXSwic291cmNlcyI6WyJGZWVkYmFjay50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGF4aW9zIGZyb20gJ2F4aW9zJ1xuaW1wb3J0IHsgcmVhZEZpbGUsIHN0YXQgfSBmcm9tICdmcy9wcm9taXNlcydcbmltcG9ydCAqIGFzIFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgdXNlQ2FsbGJhY2ssIHVzZUVmZmVjdCwgdXNlU3RhdGUgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IGdldExhc3RBUElSZXF1ZXN0IH0gZnJvbSAnc3JjL2Jvb3RzdHJhcC9zdGF0ZS5qcydcbmltcG9ydCB7IGxvZ0V2ZW50VG8xUCB9IGZyb20gJ3NyYy9zZXJ2aWNlcy9hbmFseXRpY3MvZmlyc3RQYXJ0eUV2ZW50TG9nZ2VyLmpzJ1xuaW1wb3J0IHtcbiAgdHlwZSBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICBsb2dFdmVudCxcbn0gZnJvbSAnc3JjL3NlcnZpY2VzL2FuYWx5dGljcy9pbmRleC5qcydcbmltcG9ydCB7XG4gIGdldExhc3RBc3Npc3RhbnRNZXNzYWdlLFxuICBub3JtYWxpemVNZXNzYWdlc0ZvckFQSSxcbn0gZnJvbSAnc3JjL3V0aWxzL21lc3NhZ2VzLmpzJ1xuaW1wb3J0IHR5cGUgeyBDb21tYW5kUmVzdWx0RGlzcGxheSB9IGZyb20gJy4uL2NvbW1hbmRzLmpzJ1xuaW1wb3J0IHsgdXNlVGVybWluYWxTaXplIH0gZnJvbSAnLi4vaG9va3MvdXNlVGVybWluYWxTaXplLmpzJ1xuaW1wb3J0IHsgQm94LCBUZXh0LCB1c2VJbnB1dCB9IGZyb20gJy4uL2luay5qcydcbmltcG9ydCB7IHVzZUtleWJpbmRpbmcgfSBmcm9tICcuLi9rZXliaW5kaW5ncy91c2VLZXliaW5kaW5nLmpzJ1xuaW1wb3J0IHsgcXVlcnlIYWlrdSB9IGZyb20gJy4uL3NlcnZpY2VzL2FwaS9jbGF1ZGUuanMnXG5pbXBvcnQgeyBzdGFydHNXaXRoQXBpRXJyb3JQcmVmaXggfSBmcm9tICcuLi9zZXJ2aWNlcy9hcGkvZXJyb3JzLmpzJ1xuaW1wb3J0IHR5cGUgeyBNZXNzYWdlIH0gZnJvbSAnLi4vdHlwZXMvbWVzc2FnZS5qcydcbmltcG9ydCB7IGNoZWNrQW5kUmVmcmVzaE9BdXRoVG9rZW5JZk5lZWRlZCB9IGZyb20gJy4uL3V0aWxzL2F1dGguanMnXG5pbXBvcnQgeyBvcGVuQnJvd3NlciB9IGZyb20gJy4uL3V0aWxzL2Jyb3dzZXIuanMnXG5pbXBvcnQgeyBsb2dGb3JEZWJ1Z2dpbmcgfSBmcm9tICcuLi91dGlscy9kZWJ1Zy5qcydcbmltcG9ydCB7IGVudiB9IGZyb20gJy4uL3V0aWxzL2Vudi5qcydcbmltcG9ydCB7IHR5cGUgR2l0UmVwb1N0YXRlLCBnZXRHaXRTdGF0ZSwgZ2V0SXNHaXQgfSBmcm9tICcuLi91dGlscy9naXQuanMnXG5pbXBvcnQgeyBnZXRBdXRoSGVhZGVycywgZ2V0VXNlckFnZW50IH0gZnJvbSAnLi4vdXRpbHMvaHR0cC5qcydcbmltcG9ydCB7IGdldEluTWVtb3J5RXJyb3JzLCBsb2dFcnJvciB9IGZyb20gJy4uL3V0aWxzL2xvZy5qcydcbmltcG9ydCB7IGlzRXNzZW50aWFsVHJhZmZpY09ubHkgfSBmcm9tICcuLi91dGlscy9wcml2YWN5TGV2ZWwuanMnXG5pbXBvcnQge1xuICBleHRyYWN0VGVhbW1hdGVUcmFuc2NyaXB0c0Zyb21UYXNrcyxcbiAgZ2V0VHJhbnNjcmlwdFBhdGgsXG4gIGxvYWRBbGxTdWJhZ2VudFRyYW5zY3JpcHRzRnJvbURpc2ssXG4gIE1BWF9UUkFOU0NSSVBUX1JFQURfQllURVMsXG59IGZyb20gJy4uL3V0aWxzL3Nlc3Npb25TdG9yYWdlLmpzJ1xuaW1wb3J0IHsganNvblN0cmluZ2lmeSB9IGZyb20gJy4uL3V0aWxzL3Nsb3dPcGVyYXRpb25zLmpzJ1xuaW1wb3J0IHsgYXNTeXN0ZW1Qcm9tcHQgfSBmcm9tICcuLi91dGlscy9zeXN0ZW1Qcm9tcHRUeXBlLmpzJ1xuaW1wb3J0IHsgQ29uZmlndXJhYmxlU2hvcnRjdXRIaW50IH0gZnJvbSAnLi9Db25maWd1cmFibGVTaG9ydGN1dEhpbnQuanMnXG5pbXBvcnQgeyBCeWxpbmUgfSBmcm9tICcuL2Rlc2lnbi1zeXN0ZW0vQnlsaW5lLmpzJ1xuaW1wb3J0IHsgRGlhbG9nIH0gZnJvbSAnLi9kZXNpZ24tc3lzdGVtL0RpYWxvZy5qcydcbmltcG9ydCB7IEtleWJvYXJkU2hvcnRjdXRIaW50IH0gZnJvbSAnLi9kZXNpZ24tc3lzdGVtL0tleWJvYXJkU2hvcnRjdXRIaW50LmpzJ1xuaW1wb3J0IFRleHRJbnB1dCBmcm9tICcuL1RleHRJbnB1dC5qcydcblxuLy8gVGhpcyB2YWx1ZSB3YXMgZGV0ZXJtaW5lZCBleHBlcmltZW50YWxseSBieSB0ZXN0aW5nIHRoZSBVUkwgbGVuZ3RoIGxpbWl0XG5jb25zdCBHSVRIVUJfVVJMX0xJTUlUID0gNzI1MFxuY29uc3QgR0lUSFVCX0lTU1VFU19SRVBPX1VSTCA9XG4gIFwiZXh0ZXJuYWxcIiA9PT0gJ2FudCdcbiAgICA/ICdodHRwczovL2dpdGh1Yi5jb20vYW50aHJvcGljcy9jbGF1ZGUtY2xpLWludGVybmFsL2lzc3VlcydcbiAgICA6ICdodHRwczovL2dpdGh1Yi5jb20vYW50aHJvcGljcy9jbGF1ZGUtY29kZS9pc3N1ZXMnXG5cbnR5cGUgUHJvcHMgPSB7XG4gIGFib3J0U2lnbmFsOiBBYm9ydFNpZ25hbFxuICBtZXNzYWdlczogTWVzc2FnZVtdXG4gIGluaXRpYWxEZXNjcmlwdGlvbj86IHN0cmluZ1xuICBvbkRvbmUocmVzdWx0OiBzdHJpbmcsIG9wdGlvbnM/OiB7IGRpc3BsYXk/OiBDb21tYW5kUmVzdWx0RGlzcGxheSB9KTogdm9pZFxuICBiYWNrZ3JvdW5kVGFza3M/OiB7XG4gICAgW3Rhc2tJZDogc3RyaW5nXToge1xuICAgICAgdHlwZTogc3RyaW5nXG4gICAgICBpZGVudGl0eT86IHsgYWdlbnRJZDogc3RyaW5nIH1cbiAgICAgIG1lc3NhZ2VzPzogTWVzc2FnZVtdXG4gICAgfVxuICB9XG59XG5cbnR5cGUgU3RlcCA9ICd1c2VySW5wdXQnIHwgJ2NvbnNlbnQnIHwgJ3N1Ym1pdHRpbmcnIHwgJ2RvbmUnXG5cbnR5cGUgRmVlZGJhY2tEYXRhID0ge1xuICAvLyBsYXRlc3RBc3Npc3RhbnRNZXNzYWdlSWQgaXMgdGhlIG1lc3NhZ2UgSUQgZnJvbSB0aGUgbGF0ZXN0IG1haW4gbW9kZWwgY2FsbFxuICBsYXRlc3RBc3Npc3RhbnRNZXNzYWdlSWQ6IHN0cmluZyB8IG51bGxcbiAgbWVzc2FnZV9jb3VudDogbnVtYmVyXG4gIGRhdGV0aW1lOiBzdHJpbmdcbiAgZGVzY3JpcHRpb246IHN0cmluZ1xuICBwbGF0Zm9ybTogc3RyaW5nXG4gIGdpdFJlcG86IGJvb2xlYW5cbiAgdmVyc2lvbjogc3RyaW5nIHwgbnVsbFxuICB0cmFuc2NyaXB0OiBNZXNzYWdlW11cbiAgc3ViYWdlbnRUcmFuc2NyaXB0cz86IHsgW2FnZW50SWQ6IHN0cmluZ106IE1lc3NhZ2VbXSB9XG4gIHJhd1RyYW5zY3JpcHRKc29ubD86IHN0cmluZ1xufVxuXG4vLyBVdGlsaXR5IGZ1bmN0aW9uIHRvIHJlZGFjdCBzZW5zaXRpdmUgaW5mb3JtYXRpb24gZnJvbSBzdHJpbmdzXG5leHBvcnQgZnVuY3Rpb24gcmVkYWN0U2Vuc2l0aXZlSW5mbyh0ZXh0OiBzdHJpbmcpOiBzdHJpbmcge1xuICBsZXQgcmVkYWN0ZWQgPSB0ZXh0XG5cbiAgLy8gQW50aHJvcGljIEFQSSBrZXlzIChzay1hbnQuLi4pIHdpdGggb3Igd2l0aG91dCBxdW90ZXNcbiAgLy8gRmlyc3QgaGFuZGxlIHRoZSBjYXNlIHdpdGggcXVvdGVzXG4gIHJlZGFjdGVkID0gcmVkYWN0ZWQucmVwbGFjZSgvXCIoc2stYW50W15cXHNcIiddezI0LH0pXCIvZywgJ1wiW1JFREFDVEVEX0FQSV9LRVldXCInKVxuICAvLyBUaGVuIGhhbmRsZSB0aGUgY2FzZXMgd2l0aG91dCBxdW90ZXMgLSBtb3JlIGdlbmVyYWwgcGF0dGVyblxuICByZWRhY3RlZCA9IHJlZGFjdGVkLnJlcGxhY2UoXG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIGN1c3RvbS1ydWxlcy9uby1sb29rYmVoaW5kLXJlZ2V4IC0tIC5yZXBsYWNlKHJlLCBzdHJpbmcpIG9uIC9idWcgcGF0aDogbm8tbWF0Y2ggcmV0dXJucyBzYW1lIHN0cmluZyAoT2JqZWN0LmlzKVxuICAgIC8oPzwhW0EtWmEtejAtOVwiJ10pKHNrLWFudC0/W0EtWmEtejAtOV8tXXsxMCx9KSg/IVtBLVphLXowLTlcIiddKS9nLFxuICAgICdbUkVEQUNURURfQVBJX0tFWV0nLFxuICApXG5cbiAgLy8gQVdTIGtleXMgLSBBV1NYWFhYIGZvcm1hdCAtIGFkZCB0aGUgcGF0dGVybiB3ZSBuZWVkIGZvciB0aGUgdGVzdFxuICByZWRhY3RlZCA9IHJlZGFjdGVkLnJlcGxhY2UoXG4gICAgL0FXUyBrZXk6IFwiKEFXU1tBLVowLTldezIwLH0pXCIvZyxcbiAgICAnQVdTIGtleTogXCJbUkVEQUNURURfQVdTX0tFWV1cIicsXG4gIClcblxuICAvLyBBV1MgQUtJQVhYWCBrZXlzXG4gIHJlZGFjdGVkID0gcmVkYWN0ZWQucmVwbGFjZSgvKEFLSUFbQS1aMC05XXsxNn0pL2csICdbUkVEQUNURURfQVdTX0tFWV0nKVxuXG4gIC8vIEdvb2dsZSBDbG91ZCBrZXlzXG4gIHJlZGFjdGVkID0gcmVkYWN0ZWQucmVwbGFjZShcbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgY3VzdG9tLXJ1bGVzL25vLWxvb2tiZWhpbmQtcmVnZXggLS0gc2FtZSBhcyBhYm92ZVxuICAgIC8oPzwhW0EtWmEtejAtOV0pKEFJemFbQS1aYS16MC05Xy1dezM1fSkoPyFbQS1aYS16MC05XSkvZyxcbiAgICAnW1JFREFDVEVEX0dDUF9LRVldJyxcbiAgKVxuXG4gIC8vIFZlcnRleCBBSSBzZXJ2aWNlIGFjY291bnQga2V5c1xuICByZWRhY3RlZCA9IHJlZGFjdGVkLnJlcGxhY2UoXG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIGN1c3RvbS1ydWxlcy9uby1sb29rYmVoaW5kLXJlZ2V4IC0tIHNhbWUgYXMgYWJvdmVcbiAgICAvKD88IVtBLVphLXowLTldKShbYS16MC05LV0rQFthLXowLTktXStcXC5pYW1cXC5nc2VydmljZWFjY291bnRcXC5jb20pKD8hW0EtWmEtejAtOV0pL2csXG4gICAgJ1tSRURBQ1RFRF9HQ1BfU0VSVklDRV9BQ0NPVU5UXScsXG4gIClcblxuICAvLyBHZW5lcmljIEFQSSBrZXlzIGluIGhlYWRlcnNcbiAgcmVkYWN0ZWQgPSByZWRhY3RlZC5yZXBsYWNlKFxuICAgIC8oW1wiJ10/eC1hcGkta2V5W1wiJ10/XFxzKls6PV1cXHMqW1wiJ10/KVteXCInLFxccyl9XFxdXSsvZ2ksXG4gICAgJyQxW1JFREFDVEVEX0FQSV9LRVldJyxcbiAgKVxuXG4gIC8vIEF1dGhvcml6YXRpb24gaGVhZGVycyBhbmQgQmVhcmVyIHRva2Vuc1xuICByZWRhY3RlZCA9IHJlZGFjdGVkLnJlcGxhY2UoXG4gICAgLyhbXCInXT9hdXRob3JpemF0aW9uW1wiJ10/XFxzKls6PV1cXHMqW1wiJ10/KGJlYXJlclxccyspPylbXlwiJyxcXHMpfVxcXV0rL2dpLFxuICAgICckMVtSRURBQ1RFRF9UT0tFTl0nLFxuICApXG5cbiAgLy8gQVdTIGVudmlyb25tZW50IHZhcmlhYmxlc1xuICByZWRhY3RlZCA9IHJlZGFjdGVkLnJlcGxhY2UoXG4gICAgLyhBV1NbXy1dW0EtWmEtejAtOV9dK1xccypbPTpdXFxzKilbXCInXT9bXlwiJyxcXHMpfVxcXV0rW1wiJ10/L2dpLFxuICAgICckMVtSRURBQ1RFRF9BV1NfVkFMVUVdJyxcbiAgKVxuXG4gIC8vIEdDUCBlbnZpcm9ubWVudCB2YXJpYWJsZXNcbiAgcmVkYWN0ZWQgPSByZWRhY3RlZC5yZXBsYWNlKFxuICAgIC8oR09PR0xFW18tXVtBLVphLXowLTlfXStcXHMqWz06XVxccyopW1wiJ10/W15cIicsXFxzKX1cXF1dK1tcIiddPy9naSxcbiAgICAnJDFbUkVEQUNURURfR0NQX1ZBTFVFXScsXG4gIClcblxuICAvLyBFbnZpcm9ubWVudCB2YXJpYWJsZXMgd2l0aCBrZXlzXG4gIHJlZGFjdGVkID0gcmVkYWN0ZWQucmVwbGFjZShcbiAgICAvKChBUElbLV9dP0tFWXxUT0tFTnxTRUNSRVR8UEFTU1dPUkQpXFxzKls9Ol1cXHMqKVtcIiddP1teXCInLFxccyl9XFxdXStbXCInXT8vZ2ksXG4gICAgJyQxW1JFREFDVEVEXScsXG4gIClcblxuICByZXR1cm4gcmVkYWN0ZWRcbn1cblxuLy8gR2V0IHNhbml0aXplZCBlcnJvciBsb2dzIHdpdGggc2Vuc2l0aXZlIGluZm9ybWF0aW9uIHJlZGFjdGVkXG5mdW5jdGlvbiBnZXRTYW5pdGl6ZWRFcnJvckxvZ3MoKTogQXJyYXk8e1xuICBlcnJvcj86IHN0cmluZ1xuICB0aW1lc3RhbXA/OiBzdHJpbmdcbn0+IHtcbiAgLy8gU2FuaXRpemUgZXJyb3IgbG9ncyB0byByZW1vdmUgYW55IEFQSSBrZXlzXG4gIHJldHVybiBnZXRJbk1lbW9yeUVycm9ycygpLm1hcChlcnJvckluZm8gPT4ge1xuICAgIC8vIENyZWF0ZSBhIGNvcHkgb2YgdGhlIGVycm9yIGluZm8gdG8gYXZvaWQgbW9kaWZ5aW5nIHRoZSBvcmlnaW5hbFxuICAgIGNvbnN0IGVycm9yQ29weSA9IHsgLi4uZXJyb3JJbmZvIH0gYXMgeyBlcnJvcj86IHN0cmluZzsgdGltZXN0YW1wPzogc3RyaW5nIH1cblxuICAgIC8vIFNhbml0aXplIGVycm9yIGlmIHByZXNlbnQgYW5kIGlzIGEgc3RyaW5nXG4gICAgaWYgKGVycm9yQ29weSAmJiB0eXBlb2YgZXJyb3JDb3B5LmVycm9yID09PSAnc3RyaW5nJykge1xuICAgICAgZXJyb3JDb3B5LmVycm9yID0gcmVkYWN0U2Vuc2l0aXZlSW5mbyhlcnJvckNvcHkuZXJyb3IpXG4gICAgfVxuXG4gICAgcmV0dXJuIGVycm9yQ29weVxuICB9KVxufVxuXG5hc3luYyBmdW5jdGlvbiBsb2FkUmF3VHJhbnNjcmlwdEpzb25sKCk6IFByb21pc2U8c3RyaW5nIHwgbnVsbD4ge1xuICB0cnkge1xuICAgIGNvbnN0IHRyYW5zY3JpcHRQYXRoID0gZ2V0VHJhbnNjcmlwdFBhdGgoKVxuICAgIGNvbnN0IHsgc2l6ZSB9ID0gYXdhaXQgc3RhdCh0cmFuc2NyaXB0UGF0aClcbiAgICBpZiAoc2l6ZSA+IE1BWF9UUkFOU0NSSVBUX1JFQURfQllURVMpIHtcbiAgICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgICAgYFNraXBwaW5nIHJhdyB0cmFuc2NyaXB0IHJlYWQ6IGZpbGUgdG9vIGxhcmdlICgke3NpemV9IGJ5dGVzKWAsXG4gICAgICAgIHsgbGV2ZWw6ICd3YXJuJyB9LFxuICAgICAgKVxuICAgICAgcmV0dXJuIG51bGxcbiAgICB9XG4gICAgcmV0dXJuIGF3YWl0IHJlYWRGaWxlKHRyYW5zY3JpcHRQYXRoLCAndXRmLTgnKVxuICB9IGNhdGNoIHtcbiAgICByZXR1cm4gbnVsbFxuICB9XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBGZWVkYmFjayh7XG4gIGFib3J0U2lnbmFsLFxuICBtZXNzYWdlcyxcbiAgaW5pdGlhbERlc2NyaXB0aW9uLFxuICBvbkRvbmUsXG4gIGJhY2tncm91bmRUYXNrcyA9IHt9LFxufTogUHJvcHMpOiBSZWFjdC5SZWFjdE5vZGUge1xuICBjb25zdCBbc3RlcCwgc2V0U3RlcF0gPSB1c2VTdGF0ZTxTdGVwPigndXNlcklucHV0JylcbiAgY29uc3QgW2N1cnNvck9mZnNldCwgc2V0Q3Vyc29yT2Zmc2V0XSA9IHVzZVN0YXRlKDApXG4gIGNvbnN0IFtkZXNjcmlwdGlvbiwgc2V0RGVzY3JpcHRpb25dID0gdXNlU3RhdGUoaW5pdGlhbERlc2NyaXB0aW9uID8/ICcnKVxuICBjb25zdCBbZmVlZGJhY2tJZCwgc2V0RmVlZGJhY2tJZF0gPSB1c2VTdGF0ZTxzdHJpbmcgfCBudWxsPihudWxsKVxuICBjb25zdCBbZXJyb3IsIHNldEVycm9yXSA9IHVzZVN0YXRlPHN0cmluZyB8IG51bGw+KG51bGwpXG4gIGNvbnN0IFtlbnZJbmZvLCBzZXRFbnZJbmZvXSA9IHVzZVN0YXRlPHtcbiAgICBpc0dpdDogYm9vbGVhblxuICAgIGdpdFN0YXRlOiBHaXRSZXBvU3RhdGUgfCBudWxsXG4gIH0+KHsgaXNHaXQ6IGZhbHNlLCBnaXRTdGF0ZTogbnVsbCB9KVxuICBjb25zdCBbdGl0bGUsIHNldFRpdGxlXSA9IHVzZVN0YXRlPHN0cmluZyB8IG51bGw+KG51bGwpXG4gIGNvbnN0IHRleHRJbnB1dENvbHVtbnMgPSB1c2VUZXJtaW5hbFNpemUoKS5jb2x1bW5zIC0gNFxuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgYXN5bmMgZnVuY3Rpb24gbG9hZEVudkluZm8oKSB7XG4gICAgICBjb25zdCBpc0dpdCA9IGF3YWl0IGdldElzR2l0KClcbiAgICAgIGxldCBnaXRTdGF0ZTogR2l0UmVwb1N0YXRlIHwgbnVsbCA9IG51bGxcbiAgICAgIGlmIChpc0dpdCkge1xuICAgICAgICBnaXRTdGF0ZSA9IGF3YWl0IGdldEdpdFN0YXRlKClcbiAgICAgIH1cbiAgICAgIHNldEVudkluZm8oeyBpc0dpdCwgZ2l0U3RhdGUgfSlcbiAgICB9XG4gICAgdm9pZCBsb2FkRW52SW5mbygpXG4gIH0sIFtdKVxuXG4gIGNvbnN0IHN1Ym1pdFJlcG9ydCA9IHVzZUNhbGxiYWNrKGFzeW5jICgpID0+IHtcbiAgICBzZXRTdGVwKCdzdWJtaXR0aW5nJylcbiAgICBzZXRFcnJvcihudWxsKVxuICAgIHNldEZlZWRiYWNrSWQobnVsbClcblxuICAgIC8vIEdldCBzYW5pdGl6ZWQgZXJyb3JzIGZvciB0aGUgcmVwb3J0XG4gICAgY29uc3Qgc2FuaXRpemVkRXJyb3JzID0gZ2V0U2FuaXRpemVkRXJyb3JMb2dzKClcblxuICAgIC8vIEV4dHJhY3QgbGFzdCBhc3Npc3RhbnQgbWVzc2FnZSBJRCBmcm9tIG1lc3NhZ2VzIGFycmF5XG4gICAgY29uc3QgbGFzdEFzc2lzdGFudE1lc3NhZ2UgPSBnZXRMYXN0QXNzaXN0YW50TWVzc2FnZShtZXNzYWdlcylcbiAgICBjb25zdCBsYXN0QXNzaXN0YW50TWVzc2FnZUlkID0gbGFzdEFzc2lzdGFudE1lc3NhZ2U/LnJlcXVlc3RJZCA/PyBudWxsXG5cbiAgICBjb25zdCBbZGlza1RyYW5zY3JpcHRzLCByYXdUcmFuc2NyaXB0SnNvbmxdID0gYXdhaXQgUHJvbWlzZS5hbGwoW1xuICAgICAgbG9hZEFsbFN1YmFnZW50VHJhbnNjcmlwdHNGcm9tRGlzaygpLFxuICAgICAgbG9hZFJhd1RyYW5zY3JpcHRKc29ubCgpLFxuICAgIF0pXG4gICAgY29uc3QgdGVhbW1hdGVUcmFuc2NyaXB0cyA9XG4gICAgICBleHRyYWN0VGVhbW1hdGVUcmFuc2NyaXB0c0Zyb21UYXNrcyhiYWNrZ3JvdW5kVGFza3MpXG4gICAgY29uc3Qgc3ViYWdlbnRUcmFuc2NyaXB0cyA9IHsgLi4uZGlza1RyYW5zY3JpcHRzLCAuLi50ZWFtbWF0ZVRyYW5zY3JpcHRzIH1cblxuICAgIGNvbnN0IHJlcG9ydERhdGEgPSB7XG4gICAgICBsYXRlc3RBc3Npc3RhbnRNZXNzYWdlSWQ6IGxhc3RBc3Npc3RhbnRNZXNzYWdlSWQsXG4gICAgICBtZXNzYWdlX2NvdW50OiBtZXNzYWdlcy5sZW5ndGgsXG4gICAgICBkYXRldGltZTogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxuICAgICAgZGVzY3JpcHRpb24sXG4gICAgICBwbGF0Zm9ybTogZW52LnBsYXRmb3JtLFxuICAgICAgZ2l0UmVwbzogZW52SW5mby5pc0dpdCxcbiAgICAgIHRlcm1pbmFsOiBlbnYudGVybWluYWwsXG4gICAgICB2ZXJzaW9uOiBNQUNSTy5WRVJTSU9OLFxuICAgICAgdHJhbnNjcmlwdDogbm9ybWFsaXplTWVzc2FnZXNGb3JBUEkobWVzc2FnZXMpLFxuICAgICAgZXJyb3JzOiBzYW5pdGl6ZWRFcnJvcnMsXG4gICAgICBsYXN0QXBpUmVxdWVzdDogZ2V0TGFzdEFQSVJlcXVlc3QoKSxcbiAgICAgIC4uLihPYmplY3Qua2V5cyhzdWJhZ2VudFRyYW5zY3JpcHRzKS5sZW5ndGggPiAwICYmIHtcbiAgICAgICAgc3ViYWdlbnRUcmFuc2NyaXB0cyxcbiAgICAgIH0pLFxuICAgICAgLi4uKHJhd1RyYW5zY3JpcHRKc29ubCAmJiB7IHJhd1RyYW5zY3JpcHRKc29ubCB9KSxcbiAgICB9XG5cbiAgICBjb25zdCBbcmVzdWx0LCB0XSA9IGF3YWl0IFByb21pc2UuYWxsKFtcbiAgICAgIHN1Ym1pdEZlZWRiYWNrKHJlcG9ydERhdGEsIGFib3J0U2lnbmFsKSxcbiAgICAgIGdlbmVyYXRlVGl0bGUoZGVzY3JpcHRpb24sIGFib3J0U2lnbmFsKSxcbiAgICBdKVxuXG4gICAgc2V0VGl0bGUodClcblxuICAgIGlmIChyZXN1bHQuc3VjY2Vzcykge1xuICAgICAgaWYgKHJlc3VsdC5mZWVkYmFja0lkKSB7XG4gICAgICAgIHNldEZlZWRiYWNrSWQocmVzdWx0LmZlZWRiYWNrSWQpXG4gICAgICAgIGxvZ0V2ZW50KCd0ZW5ndV9idWdfcmVwb3J0X3N1Ym1pdHRlZCcsIHtcbiAgICAgICAgICBmZWVkYmFja19pZDpcbiAgICAgICAgICAgIHJlc3VsdC5mZWVkYmFja0lkIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgICAgbGFzdF9hc3Npc3RhbnRfbWVzc2FnZV9pZDpcbiAgICAgICAgICAgIGxhc3RBc3Npc3RhbnRNZXNzYWdlSWQgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICAgICAgfSlcbiAgICAgICAgLy8gMVAtb25seTogZnJlZWZvcm0gdGV4dCBhcHByb3ZlZCBmb3IgQlEuIEpvaW4gb24gZmVlZGJhY2tfaWQuXG4gICAgICAgIGxvZ0V2ZW50VG8xUCgndGVuZ3VfYnVnX3JlcG9ydF9kZXNjcmlwdGlvbicsIHtcbiAgICAgICAgICBmZWVkYmFja19pZDpcbiAgICAgICAgICAgIHJlc3VsdC5mZWVkYmFja0lkIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgICAgZGVzY3JpcHRpb246IHJlZGFjdFNlbnNpdGl2ZUluZm8oXG4gICAgICAgICAgICBkZXNjcmlwdGlvbixcbiAgICAgICAgICApIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgIH0pXG4gICAgICB9XG4gICAgICBzZXRTdGVwKCdkb25lJylcbiAgICB9IGVsc2Uge1xuICAgICAgaWYgKHJlc3VsdC5pc1pkck9yZykge1xuICAgICAgICBzZXRFcnJvcihcbiAgICAgICAgICAnRmVlZGJhY2sgY29sbGVjdGlvbiBpcyBub3QgYXZhaWxhYmxlIGZvciBvcmdhbml6YXRpb25zIHdpdGggY3VzdG9tIGRhdGEgcmV0ZW50aW9uIHBvbGljaWVzLicsXG4gICAgICAgIClcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHNldEVycm9yKCdDb3VsZCBub3Qgc3VibWl0IGZlZWRiYWNrLiBQbGVhc2UgdHJ5IGFnYWluIGxhdGVyLicpXG4gICAgICB9XG4gICAgICAvLyBTdGF5IG9uIHVzZXJJbnB1dCBzdGVwIHNvIHVzZXIgY2FuIHJldHJ5IHdpdGggdGhlaXIgY29udGVudCBwcmVzZXJ2ZWRcbiAgICAgIHNldFN0ZXAoJ3VzZXJJbnB1dCcpXG4gICAgfVxuICB9LCBbZGVzY3JpcHRpb24sIGVudkluZm8uaXNHaXQsIG1lc3NhZ2VzXSlcblxuICAvLyBIYW5kbGUgY2FuY2VsIC0gdGhpcyB3aWxsIGJlIGNhbGxlZCBieSBEaWFsb2cncyBhdXRvbWF0aWMgRXNjIGhhbmRsaW5nXG4gIGNvbnN0IGhhbmRsZUNhbmNlbCA9IHVzZUNhbGxiYWNrKCgpID0+IHtcbiAgICAvLyBEb24ndCBjYW5jZWwgd2hlbiBkb25lIC0gbGV0IG90aGVyIGtleXMgY2xvc2UgdGhlIGRpYWxvZ1xuICAgIGlmIChzdGVwID09PSAnZG9uZScpIHtcbiAgICAgIGlmIChlcnJvcikge1xuICAgICAgICBvbkRvbmUoJ0Vycm9yIHN1Ym1pdHRpbmcgZmVlZGJhY2sgLyBidWcgcmVwb3J0Jywge1xuICAgICAgICAgIGRpc3BsYXk6ICdzeXN0ZW0nLFxuICAgICAgICB9KVxuICAgICAgfSBlbHNlIHtcbiAgICAgICAgb25Eb25lKCdGZWVkYmFjayAvIGJ1ZyByZXBvcnQgc3VibWl0dGVkJywgeyBkaXNwbGF5OiAnc3lzdGVtJyB9KVxuICAgICAgfVxuICAgICAgcmV0dXJuXG4gICAgfVxuICAgIG9uRG9uZSgnRmVlZGJhY2sgLyBidWcgcmVwb3J0IGNhbmNlbGxlZCcsIHsgZGlzcGxheTogJ3N5c3RlbScgfSlcbiAgfSwgW3N0ZXAsIGVycm9yLCBvbkRvbmVdKVxuXG4gIC8vIER1cmluZyB0ZXh0IGlucHV0LCB1c2UgU2V0dGluZ3MgY29udGV4dCB3aGVyZSBvbmx5IEVzY2FwZSAobm90ICduJykgdHJpZ2dlcnMgY29uZmlybTpuby5cbiAgLy8gVGhpcyBhbGxvd3MgdHlwaW5nICduJyBpbiB0aGUgdGV4dCBmaWVsZCB3aGlsZSBzdGlsbCBzdXBwb3J0aW5nIEVzY2FwZSB0byBjYW5jZWwuXG4gIHVzZUtleWJpbmRpbmcoJ2NvbmZpcm06bm8nLCBoYW5kbGVDYW5jZWwsIHtcbiAgICBjb250ZXh0OiAnU2V0dGluZ3MnLFxuICAgIGlzQWN0aXZlOiBzdGVwID09PSAndXNlcklucHV0JyxcbiAgfSlcblxuICB1c2VJbnB1dCgoaW5wdXQsIGtleSkgPT4ge1xuICAgIC8vIEFsbG93IGFueSBrZXkgcHJlc3MgdG8gY2xvc2UgdGhlIGRpYWxvZyB3aGVuIGRvbmUgb3Igd2hlbiB0aGVyZSdzIGFuIGVycm9yXG4gICAgaWYgKHN0ZXAgPT09ICdkb25lJykge1xuICAgICAgaWYgKGtleS5yZXR1cm4gJiYgdGl0bGUpIHtcbiAgICAgICAgLy8gT3BlbiBHaXRIdWIgaXNzdWUgVVJMIHdoZW4gRW50ZXIgaXMgcHJlc3NlZFxuICAgICAgICBjb25zdCBpc3N1ZVVybCA9IGNyZWF0ZUdpdEh1Yklzc3VlVXJsKFxuICAgICAgICAgIGZlZWRiYWNrSWQgPz8gJycsXG4gICAgICAgICAgdGl0bGUsXG4gICAgICAgICAgZGVzY3JpcHRpb24sXG4gICAgICAgICAgZ2V0U2FuaXRpemVkRXJyb3JMb2dzKCksXG4gICAgICAgIClcbiAgICAgICAgdm9pZCBvcGVuQnJvd3Nlcihpc3N1ZVVybClcbiAgICAgIH1cbiAgICAgIGlmIChlcnJvcikge1xuICAgICAgICBvbkRvbmUoJ0Vycm9yIHN1Ym1pdHRpbmcgZmVlZGJhY2sgLyBidWcgcmVwb3J0Jywge1xuICAgICAgICAgIGRpc3BsYXk6ICdzeXN0ZW0nLFxuICAgICAgICB9KVxuICAgICAgfSBlbHNlIHtcbiAgICAgICAgb25Eb25lKCdGZWVkYmFjayAvIGJ1ZyByZXBvcnQgc3VibWl0dGVkJywgeyBkaXNwbGF5OiAnc3lzdGVtJyB9KVxuICAgICAgfVxuICAgICAgcmV0dXJuXG4gICAgfVxuXG4gICAgLy8gV2hlbiBpbiB1c2VySW5wdXQgc3RlcCB3aXRoIGVycm9yLCBhbGxvdyB1c2VyIHRvIGVkaXQgYW5kIHJldHJ5XG4gICAgLy8gKGRvbid0IGNsb3NlIG9uIGFueSBrZXlwcmVzcyAtIHRoZXkgY2FuIHN0aWxsIHByZXNzIEVzYyB0byBjYW5jZWwpXG4gICAgaWYgKGVycm9yICYmIHN0ZXAgIT09ICd1c2VySW5wdXQnKSB7XG4gICAgICBvbkRvbmUoJ0Vycm9yIHN1Ym1pdHRpbmcgZmVlZGJhY2sgLyBidWcgcmVwb3J0Jywge1xuICAgICAgICBkaXNwbGF5OiAnc3lzdGVtJyxcbiAgICAgIH0pXG4gICAgICByZXR1cm5cbiAgICB9XG5cbiAgICBpZiAoc3RlcCA9PT0gJ2NvbnNlbnQnICYmIChrZXkucmV0dXJuIHx8IGlucHV0ID09PSAnICcpKSB7XG4gICAgICB2b2lkIHN1Ym1pdFJlcG9ydCgpXG4gICAgfVxuICB9KVxuXG4gIHJldHVybiAoXG4gICAgPERpYWxvZ1xuICAgICAgdGl0bGU9XCJTdWJtaXQgRmVlZGJhY2sgLyBCdWcgUmVwb3J0XCJcbiAgICAgIG9uQ2FuY2VsPXtoYW5kbGVDYW5jZWx9XG4gICAgICBpc0NhbmNlbEFjdGl2ZT17c3RlcCAhPT0gJ3VzZXJJbnB1dCd9XG4gICAgICBpbnB1dEd1aWRlPXtleGl0U3RhdGUgPT5cbiAgICAgICAgZXhpdFN0YXRlLnBlbmRpbmcgPyAoXG4gICAgICAgICAgPFRleHQ+UHJlc3Mge2V4aXRTdGF0ZS5rZXlOYW1lfSBhZ2FpbiB0byBleGl0PC9UZXh0PlxuICAgICAgICApIDogc3RlcCA9PT0gJ3VzZXJJbnB1dCcgPyAoXG4gICAgICAgICAgPEJ5bGluZT5cbiAgICAgICAgICAgIDxLZXlib2FyZFNob3J0Y3V0SGludCBzaG9ydGN1dD1cIkVudGVyXCIgYWN0aW9uPVwiY29udGludWVcIiAvPlxuICAgICAgICAgICAgPENvbmZpZ3VyYWJsZVNob3J0Y3V0SGludFxuICAgICAgICAgICAgICBhY3Rpb249XCJjb25maXJtOm5vXCJcbiAgICAgICAgICAgICAgY29udGV4dD1cIkNvbmZpcm1hdGlvblwiXG4gICAgICAgICAgICAgIGZhbGxiYWNrPVwiRXNjXCJcbiAgICAgICAgICAgICAgZGVzY3JpcHRpb249XCJjYW5jZWxcIlxuICAgICAgICAgICAgLz5cbiAgICAgICAgICA8L0J5bGluZT5cbiAgICAgICAgKSA6IHN0ZXAgPT09ICdjb25zZW50JyA/IChcbiAgICAgICAgICA8QnlsaW5lPlxuICAgICAgICAgICAgPEtleWJvYXJkU2hvcnRjdXRIaW50IHNob3J0Y3V0PVwiRW50ZXJcIiBhY3Rpb249XCJzdWJtaXRcIiAvPlxuICAgICAgICAgICAgPENvbmZpZ3VyYWJsZVNob3J0Y3V0SGludFxuICAgICAgICAgICAgICBhY3Rpb249XCJjb25maXJtOm5vXCJcbiAgICAgICAgICAgICAgY29udGV4dD1cIkNvbmZpcm1hdGlvblwiXG4gICAgICAgICAgICAgIGZhbGxiYWNrPVwiRXNjXCJcbiAgICAgICAgICAgICAgZGVzY3JpcHRpb249XCJjYW5jZWxcIlxuICAgICAgICAgICAgLz5cbiAgICAgICAgICA8L0J5bGluZT5cbiAgICAgICAgKSA6IG51bGxcbiAgICAgIH1cbiAgICA+XG4gICAgICB7c3RlcCA9PT0gJ3VzZXJJbnB1dCcgJiYgKFxuICAgICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIiBnYXA9ezF9PlxuICAgICAgICAgIDxUZXh0PkRlc2NyaWJlIHRoZSBpc3N1ZSBiZWxvdzo8L1RleHQ+XG4gICAgICAgICAgPFRleHRJbnB1dFxuICAgICAgICAgICAgdmFsdWU9e2Rlc2NyaXB0aW9ufVxuICAgICAgICAgICAgb25DaGFuZ2U9e3ZhbHVlID0+IHtcbiAgICAgICAgICAgICAgc2V0RGVzY3JpcHRpb24odmFsdWUpXG4gICAgICAgICAgICAgIC8vIENsZWFyIGVycm9yIHdoZW4gdXNlciBzdGFydHMgZWRpdGluZyB0byBhbGxvdyByZXRyeVxuICAgICAgICAgICAgICBpZiAoZXJyb3IpIHtcbiAgICAgICAgICAgICAgICBzZXRFcnJvcihudWxsKVxuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9fVxuICAgICAgICAgICAgY29sdW1ucz17dGV4dElucHV0Q29sdW1uc31cbiAgICAgICAgICAgIG9uU3VibWl0PXsoKSA9PiBzZXRTdGVwKCdjb25zZW50Jyl9XG4gICAgICAgICAgICBvbkV4aXRNZXNzYWdlPXsoKSA9PlxuICAgICAgICAgICAgICBvbkRvbmUoJ0ZlZWRiYWNrIGNhbmNlbGxlZCcsIHsgZGlzcGxheTogJ3N5c3RlbScgfSlcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGN1cnNvck9mZnNldD17Y3Vyc29yT2Zmc2V0fVxuICAgICAgICAgICAgb25DaGFuZ2VDdXJzb3JPZmZzZXQ9e3NldEN1cnNvck9mZnNldH1cbiAgICAgICAgICAgIHNob3dDdXJzb3JcbiAgICAgICAgICAvPlxuICAgICAgICAgIHtlcnJvciAmJiAoXG4gICAgICAgICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIiBnYXA9ezF9PlxuICAgICAgICAgICAgICA8VGV4dCBjb2xvcj1cImVycm9yXCI+e2Vycm9yfTwvVGV4dD5cbiAgICAgICAgICAgICAgPFRleHQgZGltQ29sb3I+XG4gICAgICAgICAgICAgICAgRWRpdCBhbmQgcHJlc3MgRW50ZXIgdG8gcmV0cnksIG9yIEVzYyB0byBjYW5jZWxcbiAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgPC9Cb3g+XG4gICAgICAgICAgKX1cbiAgICAgICAgPC9Cb3g+XG4gICAgICApfVxuXG4gICAgICB7c3RlcCA9PT0gJ2NvbnNlbnQnICYmIChcbiAgICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCI+XG4gICAgICAgICAgPFRleHQ+VGhpcyByZXBvcnQgd2lsbCBpbmNsdWRlOjwvVGV4dD5cbiAgICAgICAgICA8Qm94IG1hcmdpbkxlZnQ9ezJ9IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIj5cbiAgICAgICAgICAgIDxUZXh0PlxuICAgICAgICAgICAgICAtIFlvdXIgZmVlZGJhY2sgLyBidWcgZGVzY3JpcHRpb246eycgJ31cbiAgICAgICAgICAgICAgPFRleHQgZGltQ29sb3I+e2Rlc2NyaXB0aW9ufTwvVGV4dD5cbiAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgIDxUZXh0PlxuICAgICAgICAgICAgICAtIEVudmlyb25tZW50IGluZm86eycgJ31cbiAgICAgICAgICAgICAgPFRleHQgZGltQ29sb3I+XG4gICAgICAgICAgICAgICAge2Vudi5wbGF0Zm9ybX0sIHtlbnYudGVybWluYWx9LCB2e01BQ1JPLlZFUlNJT059XG4gICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgIHtlbnZJbmZvLmdpdFN0YXRlICYmIChcbiAgICAgICAgICAgICAgPFRleHQ+XG4gICAgICAgICAgICAgICAgLSBHaXQgcmVwbyBtZXRhZGF0YTp7JyAnfVxuICAgICAgICAgICAgICAgIDxUZXh0IGRpbUNvbG9yPlxuICAgICAgICAgICAgICAgICAge2VudkluZm8uZ2l0U3RhdGUuYnJhbmNoTmFtZX1cbiAgICAgICAgICAgICAgICAgIHtlbnZJbmZvLmdpdFN0YXRlLmNvbW1pdEhhc2hcbiAgICAgICAgICAgICAgICAgICAgPyBgLCAke2VudkluZm8uZ2l0U3RhdGUuY29tbWl0SGFzaC5zbGljZSgwLCA3KX1gXG4gICAgICAgICAgICAgICAgICAgIDogJyd9XG4gICAgICAgICAgICAgICAgICB7ZW52SW5mby5naXRTdGF0ZS5yZW1vdGVVcmxcbiAgICAgICAgICAgICAgICAgICAgPyBgIEAgJHtlbnZJbmZvLmdpdFN0YXRlLnJlbW90ZVVybH1gXG4gICAgICAgICAgICAgICAgICAgIDogJyd9XG4gICAgICAgICAgICAgICAgICB7IWVudkluZm8uZ2l0U3RhdGUuaXNIZWFkT25SZW1vdGUgJiYgJywgbm90IHN5bmNlZCd9XG4gICAgICAgICAgICAgICAgICB7IWVudkluZm8uZ2l0U3RhdGUuaXNDbGVhbiAmJiAnLCBoYXMgbG9jYWwgY2hhbmdlcyd9XG4gICAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICApfVxuICAgICAgICAgICAgPFRleHQ+LSBDdXJyZW50IHNlc3Npb24gdHJhbnNjcmlwdDwvVGV4dD5cbiAgICAgICAgICA8L0JveD5cbiAgICAgICAgICA8Qm94IG1hcmdpblRvcD17MX0+XG4gICAgICAgICAgICA8VGV4dCB3cmFwPVwid3JhcFwiIGRpbUNvbG9yPlxuICAgICAgICAgICAgICBXZSB3aWxsIHVzZSB5b3VyIGZlZWRiYWNrIHRvIGRlYnVnIHJlbGF0ZWQgaXNzdWVzIG9yIHRvIGltcHJvdmV7JyAnfVxuICAgICAgICAgICAgICBDbGF1ZGUgQ29kZSZhcG9zO3MgZnVuY3Rpb25hbGl0eSAoZWcuIHRvIHJlZHVjZSB0aGUgcmlzayBvZiBidWdzXG4gICAgICAgICAgICAgIG9jY3VycmluZyBpbiB0aGUgZnV0dXJlKS5cbiAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICA8L0JveD5cbiAgICAgICAgICA8Qm94IG1hcmdpblRvcD17MX0+XG4gICAgICAgICAgICA8VGV4dD5cbiAgICAgICAgICAgICAgUHJlc3MgPFRleHQgYm9sZD5FbnRlcjwvVGV4dD4gdG8gY29uZmlybSBhbmQgc3VibWl0LlxuICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgIDwvQm94PlxuICAgICAgICA8L0JveD5cbiAgICAgICl9XG5cbiAgICAgIHtzdGVwID09PSAnc3VibWl0dGluZycgJiYgKFxuICAgICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJyb3dcIiBnYXA9ezF9PlxuICAgICAgICAgIDxUZXh0PlN1Ym1pdHRpbmcgcmVwb3J04oCmPC9UZXh0PlxuICAgICAgICA8L0JveD5cbiAgICAgICl9XG5cbiAgICAgIHtzdGVwID09PSAnZG9uZScgJiYgKFxuICAgICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIj5cbiAgICAgICAgICB7ZXJyb3IgPyAoXG4gICAgICAgICAgICA8VGV4dCBjb2xvcj1cImVycm9yXCI+e2Vycm9yfTwvVGV4dD5cbiAgICAgICAgICApIDogKFxuICAgICAgICAgICAgPFRleHQgY29sb3I9XCJzdWNjZXNzXCI+VGhhbmsgeW91IGZvciB5b3VyIHJlcG9ydCE8L1RleHQ+XG4gICAgICAgICAgKX1cbiAgICAgICAgICB7ZmVlZGJhY2tJZCAmJiA8VGV4dCBkaW1Db2xvcj5GZWVkYmFjayBJRDoge2ZlZWRiYWNrSWR9PC9UZXh0Pn1cbiAgICAgICAgICA8Qm94IG1hcmdpblRvcD17MX0+XG4gICAgICAgICAgICA8VGV4dD5QcmVzcyA8L1RleHQ+XG4gICAgICAgICAgICA8VGV4dCBib2xkPkVudGVyIDwvVGV4dD5cbiAgICAgICAgICAgIDxUZXh0PlxuICAgICAgICAgICAgICB0byBvcGVuIHlvdXIgYnJvd3NlciBhbmQgZHJhZnQgYSBHaXRIdWIgaXNzdWUsIG9yIGFueSBvdGhlciBrZXkgdG9cbiAgICAgICAgICAgICAgY2xvc2UuXG4gICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgPC9Cb3g+XG4gICAgICAgIDwvQm94PlxuICAgICAgKX1cbiAgICA8L0RpYWxvZz5cbiAgKVxufVxuXG5leHBvcnQgZnVuY3Rpb24gY3JlYXRlR2l0SHViSXNzdWVVcmwoXG4gIGZlZWRiYWNrSWQ6IHN0cmluZyxcbiAgdGl0bGU6IHN0cmluZyxcbiAgZGVzY3JpcHRpb246IHN0cmluZyxcbiAgZXJyb3JzOiBBcnJheTx7XG4gICAgZXJyb3I/OiBzdHJpbmdcbiAgICB0aW1lc3RhbXA/OiBzdHJpbmdcbiAgfT4sXG4pOiBzdHJpbmcge1xuICBjb25zdCBzYW5pdGl6ZWRUaXRsZSA9IHJlZGFjdFNlbnNpdGl2ZUluZm8odGl0bGUpXG4gIGNvbnN0IHNhbml0aXplZERlc2NyaXB0aW9uID0gcmVkYWN0U2Vuc2l0aXZlSW5mbyhkZXNjcmlwdGlvbilcblxuICBjb25zdCBib2R5UHJlZml4ID1cbiAgICBgKipCdWcgRGVzY3JpcHRpb24qKlxcbiR7c2FuaXRpemVkRGVzY3JpcHRpb259XFxuXFxuYCArXG4gICAgYCoqRW52aXJvbm1lbnQgSW5mbyoqXFxuYCArXG4gICAgYC0gUGxhdGZvcm06ICR7ZW52LnBsYXRmb3JtfVxcbmAgK1xuICAgIGAtIFRlcm1pbmFsOiAke2Vudi50ZXJtaW5hbH1cXG5gICtcbiAgICBgLSBWZXJzaW9uOiAke01BQ1JPLlZFUlNJT04gfHwgJ3Vua25vd24nfVxcbmAgK1xuICAgIGAtIEZlZWRiYWNrIElEOiAke2ZlZWRiYWNrSWR9XFxuYCArXG4gICAgYFxcbioqRXJyb3JzKipcXG5cXGBcXGBcXGBqc29uXFxuYFxuICBjb25zdCBlcnJvclN1ZmZpeCA9IGBcXG5cXGBcXGBcXGBcXG5gXG4gIGNvbnN0IGVycm9yc0pzb24gPSBqc29uU3RyaW5naWZ5KGVycm9ycylcblxuICBjb25zdCBiYXNlVXJsID0gYCR7R0lUSFVCX0lTU1VFU19SRVBPX1VSTH0vbmV3P3RpdGxlPSR7ZW5jb2RlVVJJQ29tcG9uZW50KHNhbml0aXplZFRpdGxlKX0mbGFiZWxzPXVzZXItcmVwb3J0ZWQsYnVnJmJvZHk9YFxuICBjb25zdCB0cnVuY2F0aW9uTm90ZSA9IGBcXG4qKk5vdGU6KiogQ29udGVudCB3YXMgdHJ1bmNhdGVkLlxcbmBcblxuICBjb25zdCBlbmNvZGVkUHJlZml4ID0gZW5jb2RlVVJJQ29tcG9uZW50KGJvZHlQcmVmaXgpXG4gIGNvbnN0IGVuY29kZWRTdWZmaXggPSBlbmNvZGVVUklDb21wb25lbnQoZXJyb3JTdWZmaXgpXG4gIGNvbnN0IGVuY29kZWROb3RlID0gZW5jb2RlVVJJQ29tcG9uZW50KHRydW5jYXRpb25Ob3RlKVxuICBjb25zdCBlbmNvZGVkRXJyb3JzID0gZW5jb2RlVVJJQ29tcG9uZW50KGVycm9yc0pzb24pXG5cbiAgLy8gQ2FsY3VsYXRlIHNwYWNlIGF2YWlsYWJsZSBmb3IgZXJyb3JzXG4gIGNvbnN0IHNwYWNlRm9yRXJyb3JzID1cbiAgICBHSVRIVUJfVVJMX0xJTUlUIC1cbiAgICBiYXNlVXJsLmxlbmd0aCAtXG4gICAgZW5jb2RlZFByZWZpeC5sZW5ndGggLVxuICAgIGVuY29kZWRTdWZmaXgubGVuZ3RoIC1cbiAgICBlbmNvZGVkTm90ZS5sZW5ndGhcblxuICAvLyBJZiBkZXNjcmlwdGlvbiBhbG9uZSBleGNlZWRzIGxpbWl0LCB0cnVuY2F0ZSBldmVyeXRoaW5nXG4gIGlmIChzcGFjZUZvckVycm9ycyA8PSAwKSB7XG4gICAgY29uc3QgZWxsaXBzaXMgPSBlbmNvZGVVUklDb21wb25lbnQoJ+KApicpXG4gICAgY29uc3QgYnVmZmVyID0gNTAgLy8gRXh0cmEgc2FmZXR5IG1hcmdpblxuICAgIGNvbnN0IG1heEVuY29kZWRMZW5ndGggPVxuICAgICAgR0lUSFVCX1VSTF9MSU1JVCAtXG4gICAgICBiYXNlVXJsLmxlbmd0aCAtXG4gICAgICBlbGxpcHNpcy5sZW5ndGggLVxuICAgICAgZW5jb2RlZE5vdGUubGVuZ3RoIC1cbiAgICAgIGJ1ZmZlclxuICAgIGNvbnN0IGZ1bGxCb2R5ID0gYm9keVByZWZpeCArIGVycm9yc0pzb24gKyBlcnJvclN1ZmZpeFxuICAgIGxldCBlbmNvZGVkRnVsbEJvZHkgPSBlbmNvZGVVUklDb21wb25lbnQoZnVsbEJvZHkpXG5cbiAgICBpZiAoZW5jb2RlZEZ1bGxCb2R5Lmxlbmd0aCA+IG1heEVuY29kZWRMZW5ndGgpIHtcbiAgICAgIGVuY29kZWRGdWxsQm9keSA9IGVuY29kZWRGdWxsQm9keS5zbGljZSgwLCBtYXhFbmNvZGVkTGVuZ3RoKVxuICAgICAgLy8gRG9uJ3QgY3V0IGluIG1pZGRsZSBvZiAlWFggc2VxdWVuY2VcbiAgICAgIGNvbnN0IGxhc3RQZXJjZW50ID0gZW5jb2RlZEZ1bGxCb2R5Lmxhc3RJbmRleE9mKCclJylcbiAgICAgIGlmIChsYXN0UGVyY2VudCA+PSBlbmNvZGVkRnVsbEJvZHkubGVuZ3RoIC0gMikge1xuICAgICAgICBlbmNvZGVkRnVsbEJvZHkgPSBlbmNvZGVkRnVsbEJvZHkuc2xpY2UoMCwgbGFzdFBlcmNlbnQpXG4gICAgICB9XG4gICAgfVxuXG4gICAgcmV0dXJuIGJhc2VVcmwgKyBlbmNvZGVkRnVsbEJvZHkgKyBlbGxpcHNpcyArIGVuY29kZWROb3RlXG4gIH1cblxuICAvLyBJZiBlcnJvcnMgZml0LCBubyB0cnVuY2F0aW9uIG5lZWRlZFxuICBpZiAoZW5jb2RlZEVycm9ycy5sZW5ndGggPD0gc3BhY2VGb3JFcnJvcnMpIHtcbiAgICByZXR1cm4gYmFzZVVybCArIGVuY29kZWRQcmVmaXggKyBlbmNvZGVkRXJyb3JzICsgZW5jb2RlZFN1ZmZpeFxuICB9XG5cbiAgLy8gVHJ1bmNhdGUgZXJyb3JzIHRvIGZpdCAocHJpb3JpdGl6ZSBrZWVwaW5nIGRlc2NyaXB0aW9uKVxuICAvLyBTbGljZSBlbmNvZGVkIGVycm9ycyBkaXJlY3RseSwgdGhlbiB0cmltIHRvIGF2b2lkIGN1dHRpbmcgJVhYIHNlcXVlbmNlc1xuICBjb25zdCBlbGxpcHNpcyA9IGVuY29kZVVSSUNvbXBvbmVudCgn4oCmJylcbiAgY29uc3QgYnVmZmVyID0gNTAgLy8gRXh0cmEgc2FmZXR5IG1hcmdpblxuICBsZXQgdHJ1bmNhdGVkRW5jb2RlZEVycm9ycyA9IGVuY29kZWRFcnJvcnMuc2xpY2UoXG4gICAgMCxcbiAgICBzcGFjZUZvckVycm9ycyAtIGVsbGlwc2lzLmxlbmd0aCAtIGJ1ZmZlcixcbiAgKVxuICAvLyBJZiB3ZSBjdXQgaW4gbWlkZGxlIG9mICVYWCwgYmFjayB1cCB0byBiZWZvcmUgdGhlICVcbiAgY29uc3QgbGFzdFBlcmNlbnQgPSB0cnVuY2F0ZWRFbmNvZGVkRXJyb3JzLmxhc3RJbmRleE9mKCclJylcbiAgaWYgKGxhc3RQZXJjZW50ID49IHRydW5jYXRlZEVuY29kZWRFcnJvcnMubGVuZ3RoIC0gMikge1xuICAgIHRydW5jYXRlZEVuY29kZWRFcnJvcnMgPSB0cnVuY2F0ZWRFbmNvZGVkRXJyb3JzLnNsaWNlKDAsIGxhc3RQZXJjZW50KVxuICB9XG5cbiAgcmV0dXJuIChcbiAgICBiYXNlVXJsICtcbiAgICBlbmNvZGVkUHJlZml4ICtcbiAgICB0cnVuY2F0ZWRFbmNvZGVkRXJyb3JzICtcbiAgICBlbGxpcHNpcyArXG4gICAgZW5jb2RlZFN1ZmZpeCArXG4gICAgZW5jb2RlZE5vdGVcbiAgKVxufVxuXG5hc3luYyBmdW5jdGlvbiBnZW5lcmF0ZVRpdGxlKFxuICBkZXNjcmlwdGlvbjogc3RyaW5nLFxuICBhYm9ydFNpZ25hbDogQWJvcnRTaWduYWwsXG4pOiBQcm9taXNlPHN0cmluZz4ge1xuICB0cnkge1xuICAgIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgcXVlcnlIYWlrdSh7XG4gICAgICBzeXN0ZW1Qcm9tcHQ6IGFzU3lzdGVtUHJvbXB0KFtcbiAgICAgICAgJ0dlbmVyYXRlIGEgY29uY2lzZSwgdGVjaG5pY2FsIGlzc3VlIHRpdGxlIChtYXggODAgY2hhcnMpIGZvciBhIHB1YmxpYyBHaXRIdWIgaXNzdWUgYmFzZWQgb24gdGhpcyBidWcgcmVwb3J0IGZvciBDbGF1ZGUgQ29kZS4nLFxuICAgICAgICAnQ2xhdWRlIENvZGUgaXMgYW4gYWdlbnRpYyBjb2RpbmcgQ0xJIGJhc2VkIG9uIHRoZSBBbnRocm9waWMgQVBJLicsXG4gICAgICAgICdUaGUgdGl0bGUgc2hvdWxkOicsXG4gICAgICAgICctIEluY2x1ZGUgdGhlIHR5cGUgb2YgaXNzdWUgW0J1Z10gb3IgW0ZlYXR1cmUgUmVxdWVzdF0gYXMgdGhlIGZpcnN0IHRoaW5nIGluIHRoZSB0aXRsZScsXG4gICAgICAgICctIEJlIGNvbmNpc2UsIHNwZWNpZmljIGFuZCBkZXNjcmlwdGl2ZSBvZiB0aGUgYWN0dWFsIHByb2JsZW0nLFxuICAgICAgICAnLSBVc2UgdGVjaG5pY2FsIHRlcm1pbm9sb2d5IGFwcHJvcHJpYXRlIGZvciBhIHNvZnR3YXJlIGlzc3VlJyxcbiAgICAgICAgJy0gRm9yIGVycm9yIG1lc3NhZ2VzLCBleHRyYWN0IHRoZSBrZXkgZXJyb3IgKGUuZy4sIFwiTWlzc2luZyBUb29sIFJlc3VsdCBCbG9ja1wiIHJhdGhlciB0aGFuIHRoZSBmdWxsIG1lc3NhZ2UpJyxcbiAgICAgICAgJy0gQmUgZGlyZWN0IGFuZCBjbGVhciBmb3IgZGV2ZWxvcGVycyB0byB1bmRlcnN0YW5kIHRoZSBwcm9ibGVtJyxcbiAgICAgICAgJy0gSWYgeW91IGNhbm5vdCBkZXRlcm1pbmUgYSBjbGVhciBpc3N1ZSwgdXNlIFwiQnVnIFJlcG9ydDogW2JyaWVmIGRlc2NyaXB0aW9uXVwiJyxcbiAgICAgICAgJy0gQW55IExMTSBBUEkgZXJyb3JzIGFyZSBmcm9tIHRoZSBBbnRocm9waWMgQVBJLCBub3QgZnJvbSBhbnkgb3RoZXIgbW9kZWwgcHJvdmlkZXInLFxuICAgICAgICAnWW91ciByZXNwb25zZSB3aWxsIGJlIGRpcmVjdGx5IHVzZWQgYXMgdGhlIHRpdGxlIG9mIHRoZSBHaXRodWIgaXNzdWUsIGFuZCBhcyBzdWNoIHNob3VsZCBub3QgY29udGFpbiBhbnkgb3RoZXIgY29tbWVudGFyeSBvciBleHBsYWluYXRpb24nLFxuICAgICAgICAnRXhhbXBsZXMgb2YgZ29vZCB0aXRsZXMgaW5jbHVkZTogXCJbQnVnXSBBdXRvLUNvbXBhY3QgdHJpZ2dlcnMgdG8gc29vblwiLCBcIltCdWddIEFudGhyb3BpYyBBUEkgRXJyb3I6IE1pc3NpbmcgVG9vbCBSZXN1bHQgQmxvY2tcIiwgXCJbQnVnXSBFcnJvcjogSW52YWxpZCBNb2RlbCBOYW1lIGZvciBPcHVzXCInLFxuICAgICAgXSksXG4gICAgICB1c2VyUHJvbXB0OiBkZXNjcmlwdGlvbixcbiAgICAgIHNpZ25hbDogYWJvcnRTaWduYWwsXG4gICAgICBvcHRpb25zOiB7XG4gICAgICAgIGhhc0FwcGVuZFN5c3RlbVByb21wdDogZmFsc2UsXG4gICAgICAgIHRvb2xDaG9pY2U6IHVuZGVmaW5lZCxcbiAgICAgICAgaXNOb25JbnRlcmFjdGl2ZVNlc3Npb246IGZhbHNlLFxuICAgICAgICBhZ2VudHM6IFtdLFxuICAgICAgICBxdWVyeVNvdXJjZTogJ2ZlZWRiYWNrJyxcbiAgICAgICAgbWNwVG9vbHM6IFtdLFxuICAgICAgfSxcbiAgICB9KVxuXG4gICAgY29uc3QgdGl0bGUgPVxuICAgICAgcmVzcG9uc2UubWVzc2FnZS5jb250ZW50WzBdPy50eXBlID09PSAndGV4dCdcbiAgICAgICAgPyByZXNwb25zZS5tZXNzYWdlLmNvbnRlbnRbMF0udGV4dFxuICAgICAgICA6ICdCdWcgUmVwb3J0J1xuXG4gICAgLy8gQ2hlY2sgaWYgdGhlIHRpdGxlIGNvbnRhaW5zIGFuIEFQSSBlcnJvciBtZXNzYWdlXG4gICAgaWYgKHN0YXJ0c1dpdGhBcGlFcnJvclByZWZpeCh0aXRsZSkpIHtcbiAgICAgIHJldHVybiBjcmVhdGVGYWxsYmFja1RpdGxlKGRlc2NyaXB0aW9uKVxuICAgIH1cblxuICAgIHJldHVybiB0aXRsZVxuICB9IGNhdGNoIChlcnJvcikge1xuICAgIC8vIElmIHRoZXJlJ3MgYW55IGVycm9yIGluIHRpdGxlIGdlbmVyYXRpb24sIHVzZSBhIGZhbGxiYWNrIHRpdGxlXG4gICAgbG9nRXJyb3IoZXJyb3IpXG4gICAgcmV0dXJuIGNyZWF0ZUZhbGxiYWNrVGl0bGUoZGVzY3JpcHRpb24pXG4gIH1cbn1cblxuZnVuY3Rpb24gY3JlYXRlRmFsbGJhY2tUaXRsZShkZXNjcmlwdGlvbjogc3RyaW5nKTogc3RyaW5nIHtcbiAgLy8gQ3JlYXRlIGEgc2FmZSBmYWxsYmFjayB0aXRsZSBiYXNlZCBvbiB0aGUgYnVnIGRlc2NyaXB0aW9uXG5cbiAgLy8gVHJ5IHRvIGV4dHJhY3QgYSBtZWFuaW5nZnVsIHRpdGxlIGZyb20gdGhlIGZpcnN0IGxpbmVcbiAgY29uc3QgZmlyc3RMaW5lID0gZGVzY3JpcHRpb24uc3BsaXQoJ1xcbicpWzBdIHx8ICcnXG5cbiAgLy8gSWYgdGhlIGZpcnN0IGxpbmUgaXMgdmVyeSBzaG9ydCwgdXNlIGl0IGRpcmVjdGx5XG4gIGlmIChmaXJzdExpbmUubGVuZ3RoIDw9IDYwICYmIGZpcnN0TGluZS5sZW5ndGggPiA1KSB7XG4gICAgcmV0dXJuIGZpcnN0TGluZVxuICB9XG5cbiAgLy8gRm9yIGxvbmdlciBkZXNjcmlwdGlvbnMsIGNyZWF0ZSBhIHRydW5jYXRlZCB2ZXJzaW9uXG4gIC8vIFRydW5jYXRlIGF0IHdvcmQgYm91bmRhcmllcyB3aGVuIHBvc3NpYmxlXG4gIGxldCB0cnVuY2F0ZWQgPSBmaXJzdExpbmUuc2xpY2UoMCwgNjApXG4gIGlmIChmaXJzdExpbmUubGVuZ3RoID4gNjApIHtcbiAgICAvLyBGaW5kIHRoZSBsYXN0IHNwYWNlIGJlZm9yZSB0aGUgNjAgY2hhciBsaW1pdFxuICAgIGNvbnN0IGxhc3RTcGFjZSA9IHRydW5jYXRlZC5sYXN0SW5kZXhPZignICcpXG4gICAgaWYgKGxhc3RTcGFjZSA+IDMwKSB7XG4gICAgICAvLyBPbmx5IHRyaW0gYXQgd29yZCBpZiB3ZSdyZSBub3QgY3V0dGluZyB0b28gbXVjaFxuICAgICAgdHJ1bmNhdGVkID0gdHJ1bmNhdGVkLnNsaWNlKDAsIGxhc3RTcGFjZSlcbiAgICB9XG4gICAgdHJ1bmNhdGVkICs9ICcuLi4nXG4gIH1cblxuICByZXR1cm4gdHJ1bmNhdGVkLmxlbmd0aCA8IDEwID8gJ0J1ZyBSZXBvcnQnIDogdHJ1bmNhdGVkXG59XG5cbi8vIEhlbHBlciBmdW5jdGlvbiB0byBzYW5pdGl6ZSBhbmQgbG9nIGVycm9ycyB3aXRob3V0IGV4cG9zaW5nIEFQSSBrZXlzXG5mdW5jdGlvbiBzYW5pdGl6ZUFuZExvZ0Vycm9yKGVycjogdW5rbm93bik6IHZvaWQge1xuICBpZiAoZXJyIGluc3RhbmNlb2YgRXJyb3IpIHtcbiAgICAvLyBDcmVhdGUgYSBjb3B5IHdpdGggcG90ZW50aWFsbHkgc2Vuc2l0aXZlIGluZm8gcmVkYWN0ZWRcbiAgICBjb25zdCBzYWZlRXJyb3IgPSBuZXcgRXJyb3IocmVkYWN0U2Vuc2l0aXZlSW5mbyhlcnIubWVzc2FnZSkpXG5cbiAgICAvLyBBbHNvIHJlZGFjdCB0aGUgc3RhY2sgdHJhY2UgaWYgcHJlc2VudFxuICAgIGlmIChlcnIuc3RhY2spIHtcbiAgICAgIHNhZmVFcnJvci5zdGFjayA9IHJlZGFjdFNlbnNpdGl2ZUluZm8oZXJyLnN0YWNrKVxuICAgIH1cblxuICAgIGxvZ0Vycm9yKHNhZmVFcnJvcilcbiAgfSBlbHNlIHtcbiAgICAvLyBGb3Igbm9uLUVycm9yIG9iamVjdHMsIGNvbnZlcnQgdG8gc3RyaW5nIGFuZCByZWRhY3Qgc2Vuc2l0aXZlIGluZm9cbiAgICBjb25zdCBlcnJvclN0cmluZyA9IHJlZGFjdFNlbnNpdGl2ZUluZm8oU3RyaW5nKGVycikpXG4gICAgbG9nRXJyb3IobmV3IEVycm9yKGVycm9yU3RyaW5nKSlcbiAgfVxufVxuXG5hc3luYyBmdW5jdGlvbiBzdWJtaXRGZWVkYmFjayhcbiAgZGF0YTogRmVlZGJhY2tEYXRhLFxuICBzaWduYWw/OiBBYm9ydFNpZ25hbCxcbik6IFByb21pc2U8eyBzdWNjZXNzOiBib29sZWFuOyBmZWVkYmFja0lkPzogc3RyaW5nOyBpc1pkck9yZz86IGJvb2xlYW4gfT4ge1xuICBpZiAoaXNFc3NlbnRpYWxUcmFmZmljT25seSgpKSB7XG4gICAgcmV0dXJuIHsgc3VjY2VzczogZmFsc2UgfVxuICB9XG5cbiAgdHJ5IHtcbiAgICAvLyBFbnN1cmUgT0F1dGggdG9rZW4gaXMgZnJlc2ggYmVmb3JlIGdldHRpbmcgYXV0aCBoZWFkZXJzXG4gICAgLy8gVGhpcyBwcmV2ZW50cyA0MDEgZXJyb3JzIGZyb20gc3RhbGUgY2FjaGVkIHRva2Vuc1xuICAgIGF3YWl0IGNoZWNrQW5kUmVmcmVzaE9BdXRoVG9rZW5JZk5lZWRlZCgpXG5cbiAgICBjb25zdCBhdXRoUmVzdWx0ID0gZ2V0QXV0aEhlYWRlcnMoKVxuICAgIGlmIChhdXRoUmVzdWx0LmVycm9yKSB7XG4gICAgICByZXR1cm4geyBzdWNjZXNzOiBmYWxzZSB9XG4gICAgfVxuXG4gICAgY29uc3QgaGVhZGVyczogUmVjb3JkPHN0cmluZywgc3RyaW5nPiA9IHtcbiAgICAgICdDb250ZW50LVR5cGUnOiAnYXBwbGljYXRpb24vanNvbicsXG4gICAgICAnVXNlci1BZ2VudCc6IGdldFVzZXJBZ2VudCgpLFxuICAgICAgLi4uYXV0aFJlc3VsdC5oZWFkZXJzLFxuICAgIH1cblxuICAgIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgYXhpb3MucG9zdChcbiAgICAgICdodHRwczovL2FwaS5hbnRocm9waWMuY29tL2FwaS9jbGF1ZGVfY2xpX2ZlZWRiYWNrJyxcbiAgICAgIHtcbiAgICAgICAgY29udGVudDoganNvblN0cmluZ2lmeShkYXRhKSxcbiAgICAgIH0sXG4gICAgICB7XG4gICAgICAgIGhlYWRlcnMsXG4gICAgICAgIHRpbWVvdXQ6IDMwMDAwLCAvLyAzMCBzZWNvbmQgdGltZW91dCB0byBwcmV2ZW50IGhhbmdpbmdcbiAgICAgICAgc2lnbmFsLFxuICAgICAgfSxcbiAgICApXG5cbiAgICBpZiAocmVzcG9uc2Uuc3RhdHVzID09PSAyMDApIHtcbiAgICAgIGNvbnN0IHJlc3VsdCA9IHJlc3BvbnNlLmRhdGFcbiAgICAgIGlmIChyZXN1bHQ/LmZlZWRiYWNrX2lkKSB7XG4gICAgICAgIHJldHVybiB7IHN1Y2Nlc3M6IHRydWUsIGZlZWRiYWNrSWQ6IHJlc3VsdC5mZWVkYmFja19pZCB9XG4gICAgICB9XG4gICAgICBzYW5pdGl6ZUFuZExvZ0Vycm9yKFxuICAgICAgICBuZXcgRXJyb3IoXG4gICAgICAgICAgJ0ZhaWxlZCB0byBzdWJtaXQgZmVlZGJhY2s6IHJlcXVlc3QgZGlkIG5vdCByZXR1cm4gZmVlZGJhY2tfaWQnLFxuICAgICAgICApLFxuICAgICAgKVxuICAgICAgcmV0dXJuIHsgc3VjY2VzczogZmFsc2UgfVxuICAgIH1cblxuICAgIHNhbml0aXplQW5kTG9nRXJyb3IoXG4gICAgICBuZXcgRXJyb3IoJ0ZhaWxlZCB0byBzdWJtaXQgZmVlZGJhY2s6JyArIHJlc3BvbnNlLnN0YXR1cyksXG4gICAgKVxuICAgIHJldHVybiB7IHN1Y2Nlc3M6IGZhbHNlIH1cbiAgfSBjYXRjaCAoZXJyKSB7XG4gICAgLy8gSGFuZGxlIGNhbmNlbGxhdGlvbi9hYm9ydCAtIGRvbid0IGxvZyBhcyBlcnJvclxuICAgIGlmIChheGlvcy5pc0NhbmNlbChlcnIpKSB7XG4gICAgICByZXR1cm4geyBzdWNjZXNzOiBmYWxzZSB9XG4gICAgfVxuXG4gICAgaWYgKGF4aW9zLmlzQXhpb3NFcnJvcihlcnIpICYmIGVyci5yZXNwb25zZT8uc3RhdHVzID09PSA0MDMpIHtcbiAgICAgIGNvbnN0IGVycm9yRGF0YSA9IGVyci5yZXNwb25zZS5kYXRhXG4gICAgICBpZiAoXG4gICAgICAgIGVycm9yRGF0YT8uZXJyb3I/LnR5cGUgPT09ICdwZXJtaXNzaW9uX2Vycm9yJyAmJlxuICAgICAgICBlcnJvckRhdGE/LmVycm9yPy5tZXNzYWdlPy5pbmNsdWRlcygnQ3VzdG9tIGRhdGEgcmV0ZW50aW9uIHNldHRpbmdzJylcbiAgICAgICkge1xuICAgICAgICBzYW5pdGl6ZUFuZExvZ0Vycm9yKFxuICAgICAgICAgIG5ldyBFcnJvcihcbiAgICAgICAgICAgICdDYW5ub3Qgc3VibWl0IGZlZWRiYWNrIGJlY2F1c2UgY3VzdG9tIGRhdGEgcmV0ZW50aW9uIHNldHRpbmdzIGFyZSBlbmFibGVkJyxcbiAgICAgICAgICApLFxuICAgICAgICApXG4gICAgICAgIHJldHVybiB7IHN1Y2Nlc3M6IGZhbHNlLCBpc1pkck9yZzogdHJ1ZSB9XG4gICAgICB9XG4gICAgfVxuICAgIC8vIFVzZSBvdXIgc2FmZSBlcnJvciBsb2dnaW5nIGZ1bmN0aW9uIHRvIGF2b2lkIGxlYWtpbmcgQVBJIGtleXNcbiAgICBzYW5pdGl6ZUFuZExvZ0Vycm9yKGVycilcbiAgICByZXR1cm4geyBzdWNjZXNzOiBmYWxzZSB9XG4gIH1cbn1cbiJdLCJtYXBwaW5ncyI6IkFBQUEsT0FBT0EsS0FBSyxNQUFNLE9BQU87QUFDekIsU0FBU0MsUUFBUSxFQUFFQyxJQUFJLFFBQVEsYUFBYTtBQUM1QyxPQUFPLEtBQUtDLEtBQUssTUFBTSxPQUFPO0FBQzlCLFNBQVNDLFdBQVcsRUFBRUMsU0FBUyxFQUFFQyxRQUFRLFFBQVEsT0FBTztBQUN4RCxTQUFTQyxpQkFBaUIsUUFBUSx3QkFBd0I7QUFDMUQsU0FBU0MsWUFBWSxRQUFRLGlEQUFpRDtBQUM5RSxTQUNFLEtBQUtDLDBEQUEwRCxFQUMvREMsUUFBUSxRQUNILGlDQUFpQztBQUN4QyxTQUNFQyx1QkFBdUIsRUFDdkJDLHVCQUF1QixRQUNsQix1QkFBdUI7QUFDOUIsY0FBY0Msb0JBQW9CLFFBQVEsZ0JBQWdCO0FBQzFELFNBQVNDLGVBQWUsUUFBUSw2QkFBNkI7QUFDN0QsU0FBU0MsR0FBRyxFQUFFQyxJQUFJLEVBQUVDLFFBQVEsUUFBUSxXQUFXO0FBQy9DLFNBQVNDLGFBQWEsUUFBUSxp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 \ No newline at end of file diff --git a/src/components/FeedbackSurvey/FeedbackSurvey.tsx b/src/components/FeedbackSurvey/FeedbackSurvey.tsx new file mode 100644 index 0000000..dc3e712 --- /dev/null +++ b/src/components/FeedbackSurvey/FeedbackSurvey.tsx @@ -0,0 +1,174 @@ +import { c as _c } from "react/compiler-runtime"; +import React from 'react'; +import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from 'src/services/analytics/index.js'; +import { Box, Text } from '../../ink.js'; +import { FeedbackSurveyView, isValidResponseInput } from './FeedbackSurveyView.js'; +import type { TranscriptShareResponse } from './TranscriptSharePrompt.js'; +import { TranscriptSharePrompt } from './TranscriptSharePrompt.js'; +import { useDebouncedDigitInput } from './useDebouncedDigitInput.js'; +import type { FeedbackSurveyResponse } from './utils.js'; +type Props = { + state: 'closed' | 'open' | 'thanks' | 'transcript_prompt' | 'submitting' | 'submitted'; + lastResponse: FeedbackSurveyResponse | null; + handleSelect: (selected: FeedbackSurveyResponse) => void; + handleTranscriptSelect?: (selected: TranscriptShareResponse) => void; + inputValue: string; + setInputValue: (value: string) => void; + onRequestFeedback?: () => void; + message?: string; +}; +export function FeedbackSurvey(t0) { + const $ = _c(16); + const { + state, + lastResponse, + handleSelect, + handleTranscriptSelect, + inputValue, + setInputValue, + onRequestFeedback, + message + } = t0; + if (state === "closed") { + return null; + } + if (state === "thanks") { + let t1; + if ($[0] !== inputValue || $[1] !== lastResponse || $[2] !== onRequestFeedback || $[3] !== setInputValue) { + t1 = ; + $[0] = inputValue; + $[1] = lastResponse; + $[2] = onRequestFeedback; + $[3] = setInputValue; + $[4] = t1; + } else { + t1 = $[4]; + } + return t1; + } + if (state === "submitted") { + let t1; + if ($[5] === Symbol.for("react.memo_cache_sentinel")) { + t1 = {"\u2713"} Thanks for sharing your transcript!; + $[5] = t1; + } else { + t1 = $[5]; + } + return t1; + } + if (state === "submitting") { + let t1; + if ($[6] === Symbol.for("react.memo_cache_sentinel")) { + t1 = Sharing transcript{"\u2026"}; + $[6] = t1; + } else { + t1 = $[6]; + } + return t1; + } + if (state === "transcript_prompt") { + if (!handleTranscriptSelect) { + return null; + } + if (inputValue && !["1", "2", "3"].includes(inputValue)) { + return null; + } + let t1; + if ($[7] !== handleTranscriptSelect || $[8] !== inputValue || $[9] !== setInputValue) { + t1 = ; + $[7] = handleTranscriptSelect; + $[8] = inputValue; + $[9] = setInputValue; + $[10] = t1; + } else { + t1 = $[10]; + } + return t1; + } + if (inputValue && !isValidResponseInput(inputValue)) { + return null; + } + let t1; + if ($[11] !== handleSelect || $[12] !== inputValue || $[13] !== message || $[14] !== setInputValue) { + t1 = ; + $[11] = handleSelect; + $[12] = inputValue; + $[13] = message; + $[14] = setInputValue; + $[15] = t1; + } else { + t1 = $[15]; + } + return t1; +} +type ThanksProps = { + lastResponse: FeedbackSurveyResponse | null; + inputValue: string; + setInputValue: (value: string) => void; + onRequestFeedback?: () => void; +}; +const isFollowUpDigit = (char: string): char is '1' => char === '1'; +function FeedbackSurveyThanks(t0) { + const $ = _c(12); + const { + lastResponse, + inputValue, + setInputValue, + onRequestFeedback + } = t0; + const showFollowUp = onRequestFeedback && lastResponse === "good"; + const t1 = Boolean(showFollowUp); + let t2; + if ($[0] !== lastResponse || $[1] !== onRequestFeedback) { + t2 = () => { + logEvent("tengu_feedback_survey_event", { + event_type: "followup_accepted" as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + response: lastResponse as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + onRequestFeedback?.(); + }; + $[0] = lastResponse; + $[1] = onRequestFeedback; + $[2] = t2; + } else { + t2 = $[2]; + } + let t3; + if ($[3] !== inputValue || $[4] !== setInputValue || $[5] !== t1 || $[6] !== t2) { + t3 = { + inputValue, + setInputValue, + isValidDigit: isFollowUpDigit, + enabled: t1, + once: true, + onDigit: t2 + }; + $[3] = inputValue; + $[4] = setInputValue; + $[5] = t1; + $[6] = t2; + $[7] = t3; + } else { + t3 = $[7]; + } + useDebouncedDigitInput(t3); + const feedbackCommand = false ? "/issue" : "/feedback"; + let t4; + if ($[8] === Symbol.for("react.memo_cache_sentinel")) { + t4 = Thanks for the feedback!; + $[8] = t4; + } else { + t4 = $[8]; + } + let t5; + if ($[9] !== lastResponse || $[10] !== showFollowUp) { + t5 = {t4}{showFollowUp ? (Optional) Press [1] to tell us what went well {" \xB7 "}{feedbackCommand} : lastResponse === "bad" ? Use /issue to report model behavior issues. : Use {feedbackCommand} to share detailed feedback anytime.}; + $[9] = lastResponse; + $[10] = showFollowUp; + $[11] = t5; + } else { + t5 = $[11]; + } + return t5; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMiLCJsb2dFdmVudCIsIkJveCIsIlRleHQiLCJGZWVkYmFja1N1cnZleVZpZXciLCJpc1ZhbGlkUmVzcG9uc2VJbnB1dCIsIlRyYW5zY3JpcHRTaGFyZVJlc3BvbnNlIiwiVHJhbnNjcmlwdFNoYXJlUHJvbXB0IiwidXNlRGVib3VuY2VkRGlnaXRJbnB1dCIsIkZlZWRiYWNrU3VydmV5UmVzcG9uc2UiLCJQcm9wcyIsInN0YXRlIiwibGFzdFJlc3BvbnNlIiwiaGFuZGxlU2VsZWN0Iiwic2VsZWN0ZWQiLCJoYW5kbGVUcmFuc2NyaXB0U2VsZWN0IiwiaW5wdXRWYWx1ZSIsInNldElucHV0VmFsdWUiLCJ2YWx1ZSIsIm9uUmVxdWVzdEZlZWRiYWNrIiwibWVzc2FnZSIsIkZlZWRiYWNrU3VydmV5IiwidDAiLCIkIiwiX2MiLCJ0MSIsIlN5bWJvbCIsImZvciIsImluY2x1ZGVzIiwiVGhhbmtzUHJvcHMiLCJpc0ZvbGxvd1VwRGlnaXQiLCJjaGFyIiwiRmVlZGJhY2tTdXJ2ZXlUaGFua3MiLCJzaG93Rm9sbG93VXAiLCJCb29sZWFuIiwidDIiLCJldmVudF90eXBlIiwicmVzcG9uc2UiLCJ0MyIsImlzVmFsaWREaWdpdCIsImVuYWJsZWQiLCJvbmNlIiwib25EaWdpdCIsImZlZWRiYWNrQ29tbWFuZCIsInQ0IiwidDUiXSwic291cmNlcyI6WyJGZWVkYmFja1N1cnZleS50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHtcbiAgdHlwZSBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICBsb2dFdmVudCxcbn0gZnJvbSAnc3JjL3NlcnZpY2VzL2FuYWx5dGljcy9pbmRleC5qcydcbmltcG9ydCB7IEJveCwgVGV4dCB9IGZyb20gJy4uLy4uL2luay5qcydcbmltcG9ydCB7XG4gIEZlZWRiYWNrU3VydmV5VmlldyxcbiAgaXNWYWxpZFJlc3BvbnNlSW5wdXQsXG59IGZyb20gJy4vRmVlZGJhY2tTdXJ2ZXlWaWV3LmpzJ1xuaW1wb3J0IHR5cGUgeyBUcmFuc2NyaXB0U2hhcmVSZXNwb25zZSB9IGZyb20gJy4vVHJhbnNjcmlwdFNoYXJlUHJvbXB0LmpzJ1xuaW1wb3J0IHsgVHJhbnNjcmlwdFNoYXJlUHJvbXB0IH0gZnJvbSAnLi9UcmFuc2NyaXB0U2hhcmVQcm9tcHQuanMnXG5pbXBvcnQgeyB1c2VEZWJvdW5jZWREaWdpdElucHV0IH0gZnJvbSAnLi91c2VEZWJvdW5jZWREaWdpdElucHV0LmpzJ1xuaW1wb3J0IHR5cGUgeyBGZWVkYmFja1N1cnZleVJlc3BvbnNlIH0gZnJvbSAnLi91dGlscy5qcydcblxudHlwZSBQcm9wcyA9IHtcbiAgc3RhdGU6XG4gICAgfCAnY2xvc2VkJ1xuICAgIHwgJ29wZW4nXG4gICAgfCAndGhhbmtzJ1xuICAgIHwgJ3RyYW5zY3JpcHRfcHJvbXB0J1xuICAgIHwgJ3N1Ym1pdHRpbmcnXG4gICAgfCAnc3VibWl0dGVkJ1xuICBsYXN0UmVzcG9uc2U6IEZlZWRiYWNrU3VydmV5UmVzcG9uc2UgfCBudWxsXG4gIGhhbmRsZVNlbGVjdDogKHNlbGVjdGVkOiBGZWVkYmFja1N1cnZleVJlc3BvbnNlKSA9PiB2b2lkXG4gIGhhbmRsZVRyYW5zY3JpcHRTZWxlY3Q/OiAoc2VsZWN0ZWQ6IFRyYW5zY3JpcHRTaGFyZVJlc3BvbnNlKSA9PiB2b2lkXG4gIGlucHV0VmFsdWU6IHN0cmluZ1xuICBzZXRJbnB1dFZhbHVlOiAodmFsdWU6IHN0cmluZykgPT4gdm9pZFxuICBvblJlcXVlc3RGZWVkYmFjaz86ICgpID0+IHZvaWRcbiAgbWVzc2FnZT86IHN0cmluZ1xufVxuXG5leHBvcnQgZnVuY3Rpb24gRmVlZGJhY2tTdXJ2ZXkoe1xuICBzdGF0ZSxcbiAgbGFzdFJlc3BvbnNlLFxuICBoYW5kbGVTZWxlY3QsXG4gIGhhbmRsZVRyYW5zY3JpcHRTZWxlY3QsXG4gIGlucHV0VmFsdWUsXG4gIHNldElucHV0VmFsdWUsXG4gIG9uUmVxdWVzdEZlZWRiYWNrLFxuICBtZXNzYWdlLFxufTogUHJvcHMpOiBSZWFjdC5SZWFjdE5vZGUge1xuICBpZiAoc3RhdGUgPT09ICdjbG9zZWQnKSB7XG4gICAgcmV0dXJuIG51bGxcbiAgfVxuXG4gIGlmIChzdGF0ZSA9PT0gJ3RoYW5rcycpIHtcbiAgICByZXR1cm4gKFxuICAgICAgPEZlZWRiYWNrU3VydmV5VGhhbmtzXG4gICAgICAgIGxhc3RSZXNwb25zZT17bGFzdFJlc3BvbnNlfVxuICAgICAgICBpbnB1dFZhbHVlPXtpbnB1dFZhbHVlfVxuICAgICAgICBzZXRJbnB1dFZhbHVlPXtzZXRJbnB1dFZhbHVlfVxuICAgICAgICBvblJlcXVlc3RGZWVkYmFjaz17b25SZXF1ZXN0RmVlZGJhY2t9XG4gICAgICAvPlxuICAgIClcbiAgfVxuXG4gIGlmIChzdGF0ZSA9PT0gJ3N1Ym1pdHRlZCcpIHtcbiAgICByZXR1cm4gKFxuICAgICAgPEJveCBtYXJnaW5Ub3A9ezF9PlxuICAgICAgICA8VGV4dCBjb2xvcj1cInN1Y2Nlc3NcIj5cbiAgICAgICAgICB7J1xcdTI3MTMnfSBUaGFua3MgZm9yIHNoYXJpbmcgeW91ciB0cmFuc2NyaXB0IVxuICAgICAgICA8L1RleHQ+XG4gICAgICA8L0JveD5cbiAgICApXG4gIH1cblxuICBpZiAoc3RhdGUgPT09ICdzdWJtaXR0aW5nJykge1xuICAgIHJldHVybiAoXG4gICAgICA8Qm94IG1hcmdpblRvcD17MX0+XG4gICAgICAgIDxUZXh0IGRpbUNvbG9yPlNoYXJpbmcgdHJhbnNjcmlwdHsnXFx1MjAyNid9PC9UZXh0PlxuICAgICAgPC9Cb3g+XG4gICAgKVxuICB9XG5cbiAgaWYgKHN0YXRlID09PSAndHJhbnNjcmlwdF9wcm9tcHQnKSB7XG4gICAgaWYgKCFoYW5kbGVUcmFuc2NyaXB0U2VsZWN0KSB7XG4gICAgICByZXR1cm4gbnVsbFxuICAgIH1cbiAgICAvLyBIaWRlIHByb21wdCBpZiB1c2VyIGlzIHR5cGluZyBub24tcmVzcG9uc2UgY2hhcmFjdGVyc1xuICAgIGlmIChpbnB1dFZhbHVlICYmICFbJzEnLCAnMicsICczJ10uaW5jbHVkZXMoaW5wdXRWYWx1ZSkpIHtcbiAgICAgIHJldHVybiBudWxsXG4gICAgfVxuICAgIHJldHVybiAoXG4gICAgICA8VHJhbnNjcmlwdFNoYXJlUHJvbXB0XG4gICAgICAgIG9uU2VsZWN0PXtoYW5kbGVUcmFuc2NyaXB0U2VsZWN0fVxuICAgICAgICBpbnB1dFZhbHVlPXtpbnB1dFZhbHVlfVxuICAgICAgICBzZXRJbnB1dFZhbHVlPXtzZXRJbnB1dFZhbHVlfVxuICAgICAgLz5cbiAgICApXG4gIH1cblxuICAvLyBzdGF0ZSA9PT0gJ29wZW4nXG4gIC8vIEhpZGUgdGhlIHN1cnZleSBpZiB0aGUgdXNlciBpcyB0eXBpbmcgYW55dGhpbmcgb3RoZXIgdGhhbiBhIHN1cnZleSByZXNwb25zZS5cbiAgLy8gVGhpcyBwcmV2ZW50cyB0aGUgc3VydmV5IGZyb20gc2hvd2luZyB1cCB3aGVuIHRoZSB1c2VyIGlzIHR5cGluZyBhIG1lc3NhZ2UsXG4gIC8vIHdoaWNoIGNhbiByZXN1bHQgaW4gYWNjaWRlbnRhbCBzdXJ2ZXkgc3VibWlzc2lvbnMgKGUuZy4gXCJzM2NtZFwiKS5cbiAgaWYgKGlucHV0VmFsdWUgJiYgIWlzVmFsaWRSZXNwb25zZUlucHV0KGlucHV0VmFsdWUpKSB7XG4gICAgcmV0dXJuIG51bGxcbiAgfVxuXG4gIHJldHVybiAoXG4gICAgPEZlZWRiYWNrU3VydmV5Vmlld1xuICAgICAgb25TZWxlY3Q9e2hhbmRsZVNlbGVjdH1cbiAgICAgIGlucHV0VmFsdWU9e2lucHV0VmFsdWV9XG4gICAgICBzZXRJbnB1dFZhbHVlPXtzZXRJbnB1dFZhbHVlfVxuICAgICAgbWVzc2FnZT17bWVzc2FnZX1cbiAgICAvPlxuICApXG59XG5cbnR5cGUgVGhhbmtzUHJvcHMgPSB7XG4gIGxhc3RSZXNwb25zZTogRmVlZGJhY2tTdXJ2ZXlSZXNwb25zZSB8IG51bGxcbiAgaW5wdXRWYWx1ZTogc3RyaW5nXG4gIHNldElucHV0VmFsdWU6ICh2YWx1ZTogc3RyaW5nKSA9PiB2b2lkXG4gIG9uUmVxdWVzdEZlZWRiYWNrPzogKCkgPT4gdm9pZFxufVxuXG5jb25zdCBpc0ZvbGxvd1VwRGlnaXQgPSAoY2hhcjogc3RyaW5nKTogY2hhciBpcyAnMScgPT4gY2hhciA9PT0gJzEnXG5cbmZ1bmN0aW9uIEZlZWRiYWNrU3VydmV5VGhhbmtzKHtcbiAgbGFzdFJlc3BvbnNlLFxuICBpbnB1dFZhbHVlLFxuICBzZXRJbnB1dFZhbHVlLFxuICBvblJlcXVlc3RGZWVkYmFjayxcbn06IFRoYW5rc1Byb3BzKTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgY29uc3Qgc2hvd0ZvbGxvd1VwID0gb25SZXF1ZXN0RmVlZGJhY2sgJiYgbGFzdFJlc3BvbnNlID09PSAnZ29vZCdcblxuICAvLyBMaXN0ZW4gZm9yIFwiMVwiIGtleXByZXNzIHRvIGxhdW5jaCAvZmVlZGJhY2tcbiAgdXNlRGVib3VuY2VkRGlnaXRJbnB1dCh7XG4gICAgaW5wdXRWYWx1ZSxcbiAgICBzZXRJbnB1dFZhbHVlLFxuICAgIGlzVmFsaWREaWdpdDogaXNGb2xsb3dVcERpZ2l0LFxuICAgIGVuYWJsZWQ6IEJvb2xlYW4oc2hvd0ZvbGxvd1VwKSxcbiAgICBvbmNlOiB0cnVlLFxuICAgIG9uRGlnaXQ6ICgpID0+IHtcbiAgICAgIGxvZ0V2ZW50KCd0ZW5ndV9mZWVkYmFja19zdXJ2ZXlfZXZlbnQnLCB7XG4gICAgICAgIGV2ZW50X3R5cGU6XG4gICAgICAgICAgJ2ZvbGxvd3VwX2FjY2VwdGVkJyBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICByZXNwb25zZTpcbiAgICAgICAgICBsYXN0UmVzcG9uc2UgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICAgIH0pXG4gICAgICBvblJlcXVlc3RGZWVkYmFjaz8uKClcbiAgICB9LFxuICB9KVxuXG4gIGNvbnN0IGZlZWRiYWNrQ29tbWFuZCA9XG4gICAgXCJleHRlcm5hbFwiID09PSAnYW50JyA/ICcvaXNzdWUnIDogJy9mZWVkYmFjaydcblxuICByZXR1cm4gKFxuICAgIDxCb3ggbWFyZ2luVG9wPXsxfSBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCI+XG4gICAgICA8VGV4dCBjb2xvcj1cInN1Y2Nlc3NcIj5UaGFua3MgZm9yIHRoZSBmZWVkYmFjayE8L1RleHQ+XG4gICAgICB7c2hvd0ZvbGxvd1VwID8gKFxuICAgICAgICA8VGV4dCBkaW1Db2xvcj5cbiAgICAgICAgICAoT3B0aW9uYWwpIFByZXNzIFs8VGV4dCBjb2xvcj1cImFuc2k6Y3lhblwiPjE8L1RleHQ+XSB0byB0ZWxsIHVzIHdoYXRcbiAgICAgICAgICB3ZW50IHdlbGwgeycgXFx1MDBiNyAnfVxuICAgICAgICAgIHtmZWVkYmFja0NvbW1hbmR9XG4gICAgICAgIDwvVGV4dD5cbiAgICAgICkgOiBsYXN0UmVzcG9uc2UgPT09ICdiYWQnID8gKFxuICAgICAgICA8VGV4dCBkaW1Db2xvcj5Vc2UgL2lzc3VlIHRvIHJlcG9ydCBtb2RlbCBiZWhhdmlvciBpc3N1ZXMuPC9UZXh0PlxuICAgICAgKSA6IChcbiAgICAgICAgPFRleHQgZGltQ29sb3I+XG4gICAgICAgICAgVXNlIHtmZWVkYmFja0NvbW1hbmR9IHRvIHNoYXJlIGRldGFpbGVkIGZlZWRiYWNrIGFueXRpbWUuXG4gICAgICAgIDwvVGV4dD5cbiAgICAgICl9XG4gICAgPC9Cb3g+XG4gIClcbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU9BLEtBQUssTUFBTSxPQUFPO0FBQ3pCLFNBQ0UsS0FBS0MsMERBQTBELEVBQy9EQyxRQUFRLFFBQ0gsaUNBQWlDO0FBQ3hDLFNBQVNDLEdBQUcsRUFBRUMsSUFBSSxRQUFRLGNBQWM7QUFDeEMsU0FDRUMsa0JB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 \ No newline at end of file diff --git a/src/components/FeedbackSurvey/FeedbackSurveyView.tsx b/src/components/FeedbackSurvey/FeedbackSurveyView.tsx new file mode 100644 index 0000000..a2f3757 --- /dev/null +++ b/src/components/FeedbackSurvey/FeedbackSurveyView.tsx @@ -0,0 +1,108 @@ +import { c as _c } from "react/compiler-runtime"; +import React from 'react'; +import { Box, Text } from '../../ink.js'; +import { useDebouncedDigitInput } from './useDebouncedDigitInput.js'; +import type { FeedbackSurveyResponse } from './utils.js'; +type Props = { + onSelect: (option: FeedbackSurveyResponse) => void; + inputValue: string; + setInputValue: (value: string) => void; + message?: string; +}; +const RESPONSE_INPUTS = ['0', '1', '2', '3'] as const; +type ResponseInput = (typeof RESPONSE_INPUTS)[number]; +const inputToResponse: Record = { + '0': 'dismissed', + '1': 'bad', + '2': 'fine', + '3': 'good' +} as const; +export const isValidResponseInput = (input: string): input is ResponseInput => (RESPONSE_INPUTS as readonly string[]).includes(input); +const DEFAULT_MESSAGE = 'How is Claude doing this session? (optional)'; +export function FeedbackSurveyView(t0) { + const $ = _c(15); + const { + onSelect, + inputValue, + setInputValue, + message: t1 + } = t0; + const message = t1 === undefined ? DEFAULT_MESSAGE : t1; + let t2; + if ($[0] !== onSelect) { + t2 = digit => onSelect(inputToResponse[digit]); + $[0] = onSelect; + $[1] = t2; + } else { + t2 = $[1]; + } + let t3; + if ($[2] !== inputValue || $[3] !== setInputValue || $[4] !== t2) { + t3 = { + inputValue, + setInputValue, + isValidDigit: isValidResponseInput, + onDigit: t2 + }; + $[2] = inputValue; + $[3] = setInputValue; + $[4] = t2; + $[5] = t3; + } else { + t3 = $[5]; + } + useDebouncedDigitInput(t3); + let t4; + if ($[6] === Symbol.for("react.memo_cache_sentinel")) { + t4 = ; + $[6] = t4; + } else { + t4 = $[6]; + } + let t5; + if ($[7] !== message) { + t5 = {t4}{message}; + $[7] = message; + $[8] = t5; + } else { + t5 = $[8]; + } + let t6; + if ($[9] === Symbol.for("react.memo_cache_sentinel")) { + t6 = 1: Bad; + $[9] = t6; + } else { + t6 = $[9]; + } + let t7; + if ($[10] === Symbol.for("react.memo_cache_sentinel")) { + t7 = 2: Fine; + $[10] = t7; + } else { + t7 = $[10]; + } + let t8; + if ($[11] === Symbol.for("react.memo_cache_sentinel")) { + t8 = 3: Good; + $[11] = t8; + } else { + t8 = $[11]; + } + let t9; + if ($[12] === Symbol.for("react.memo_cache_sentinel")) { + t9 = {t6}{t7}{t8}0: Dismiss; + $[12] = t9; + } else { + t9 = $[12]; + } + let t10; + if ($[13] !== t5) { + t10 = {t5}{t9}; + $[13] = t5; + $[14] = t10; + } else { + t10 = $[14]; + } + return t10; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkJveCIsIlRleHQiLCJ1c2VEZWJvdW5jZWREaWdpdElucHV0IiwiRmVlZGJhY2tTdXJ2ZXlSZXNwb25zZSIsIlByb3BzIiwib25TZWxlY3QiLCJvcHRpb24iLCJpbnB1dFZhbHVlIiwic2V0SW5wdXRWYWx1ZSIsInZhbHVlIiwibWVzc2FnZSIsIlJFU1BPTlNFX0lOUFVUUyIsImNvbnN0IiwiUmVzcG9uc2VJbnB1dCIsImlucHV0VG9SZXNwb25zZSIsIlJlY29yZCIsImlzVmFsaWRSZXNwb25zZUlucHV0IiwiaW5wdXQiLCJpbmNsdWRlcyIsIkRFRkFVTFRfTUVTU0FHRSIsIkZlZWRiYWNrU3VydmV5VmlldyIsInQwIiwiJCIsIl9jIiwidDEiLCJ1bmRlZmluZWQiLCJ0MiIsImRpZ2l0IiwidDMiLCJpc1ZhbGlkRGlnaXQiLCJvbkRpZ2l0IiwidDQiLCJTeW1ib2wiLCJmb3IiLCJ0NSIsInQ2IiwidDciLCJ0OCIsInQ5IiwidDEwIl0sInNvdXJjZXMiOlsiRmVlZGJhY2tTdXJ2ZXlWaWV3LnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBCb3gsIFRleHQgfSBmcm9tICcuLi8uLi9pbmsuanMnXG5pbXBvcnQgeyB1c2VEZWJvdW5jZWREaWdpdElucHV0IH0gZnJvbSAnLi91c2VEZWJvdW5jZWREaWdpdElucHV0LmpzJ1xuaW1wb3J0IHR5cGUgeyBGZWVkYmFja1N1cnZleVJlc3BvbnNlIH0gZnJvbSAnLi91dGlscy5qcydcblxudHlwZSBQcm9wcyA9IHtcbiAgb25TZWxlY3Q6IChvcHRpb246IEZlZWRiYWNrU3VydmV5UmVzcG9uc2UpID0+IHZvaWRcbiAgaW5wdXRWYWx1ZTogc3RyaW5nXG4gIHNldElucHV0VmFsdWU6ICh2YWx1ZTogc3RyaW5nKSA9PiB2b2lkXG4gIG1lc3NhZ2U/OiBzdHJpbmdcbn1cblxuY29uc3QgUkVTUE9OU0VfSU5QVVRTID0gWycwJywgJzEnLCAnMicsICczJ10gYXMgY29uc3RcbnR5cGUgUmVzcG9uc2VJbnB1dCA9ICh0eXBlb2YgUkVTUE9OU0VfSU5QVVRTKVtudW1iZXJdXG5cbmNvbnN0IGlucHV0VG9SZXNwb25zZTogUmVjb3JkPFJlc3BvbnNlSW5wdXQsIEZlZWRiYWNrU3VydmV5UmVzcG9uc2U+ID0ge1xuICAnMCc6ICdkaXNtaXNzZWQnLFxuICAnMSc6ICdiYWQnLFxuICAnMic6ICdmaW5lJyxcbiAgJzMnOiAnZ29vZCcsXG59IGFzIGNvbnN0XG5cbmV4cG9ydCBjb25zdCBpc1ZhbGlkUmVzcG9uc2VJbnB1dCA9IChpbnB1dDogc3RyaW5nKTogaW5wdXQgaXMgUmVzcG9uc2VJbnB1dCA9PlxuICAoUkVTUE9OU0VfSU5QVVRTIGFzIHJlYWRvbmx5IHN0cmluZ1tdKS5pbmNsdWRlcyhpbnB1dClcblxuY29uc3QgREVGQVVMVF9NRVNTQUdFID0gJ0hvdyBpcyBDbGF1ZGUgZG9pbmcgdGhpcyBzZXNzaW9uPyAob3B0aW9uYWwpJ1xuXG5leHBvcnQgZnVuY3Rpb24gRmVlZGJhY2tTdXJ2ZXlWaWV3KHtcbiAgb25TZWxlY3QsXG4gIGlucHV0VmFsdWUsXG4gIHNldElucHV0VmFsdWUsXG4gIG1lc3NhZ2UgPSBERUZBVUxUX01FU1NBR0UsXG59OiBQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIHVzZURlYm91bmNlZERpZ2l0SW5wdXQoe1xuICAgIGlucHV0VmFsdWUsXG4gICAgc2V0SW5wdXRWYWx1ZSxcbiAgICBpc1ZhbGlkRGlnaXQ6IGlzVmFsaWRSZXNwb25zZUlucHV0LFxuICAgIG9uRGlnaXQ6IGRpZ2l0ID0+IG9uU2VsZWN0KGlucHV0VG9SZXNwb25zZVtkaWdpdF0pLFxuICB9KVxuXG4gIHJldHVybiAoXG4gICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCIgbWFyZ2luVG9wPXsxfT5cbiAgICAgIDxCb3g+XG4gICAgICAgIDxUZXh0IGNvbG9yPVwiYW5zaTpjeWFuXCI+4pePIDwvVGV4dD5cbiAgICAgICAgPFRleHQgYm9sZD57bWVzc2FnZX08L1RleHQ+XG4gICAgICA8L0JveD5cblxuICAgICAgPEJveCBtYXJnaW5MZWZ0PXsyfT5cbiAgICAgICAgPEJveCB3aWR0aD17MTB9PlxuICAgICAgICAgIDxUZXh0PlxuICAgICAgICAgICAgPFRleHQgY29sb3I9XCJhbnNpOmN5YW5cIj4xPC9UZXh0PjogQmFkXG4gICAgICAgICAgPC9UZXh0PlxuICAgICAgICA8L0JveD5cbiAgICAgICAgPEJveCB3aWR0aD17MTB9PlxuICAgICAgICAgIDxUZXh0PlxuICAgICAgICAgICAgPFRleHQgY29sb3I9XCJhbnNpOmN5YW5cIj4yPC9UZXh0PjogRmluZVxuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgPC9Cb3g+XG4gICAgICAgIDxCb3ggd2lkdGg9ezEwfT5cbiAgICAgICAgICA8VGV4dD5cbiAgICAgICAgICAgIDxUZXh0IGNvbG9yPVwiYW5zaTpjeWFuXCI+MzwvVGV4dD46IEdvb2RcbiAgICAgICAgICA8L1RleHQ+XG4gICAgICAgIDwvQm94PlxuICAgICAgICA8Qm94PlxuICAgICAgICAgIDxUZXh0PlxuICAgICAgICAgICAgPFRleHQgY29sb3I9XCJhbnNpOmN5YW5cIj4wPC9UZXh0PjogRGlzbWlzc1xuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgPC9Cb3g+XG4gICAgICA8L0JveD5cbiAgICA8L0JveD5cbiAgKVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsT0FBT0EsS0FBSyxNQUFNLE9BQU87QUFDekIsU0FBU0MsR0FBRyxFQUFFQyxJQUFJLFFBQVEsY0FBYztBQUN4QyxTQUFTQyxzQkFBc0IsUUFBUSw2QkFBNkI7QUFDcEUsY0FBY0Msc0JBQXNCLFFBQVEsWUFBWTtBQUV4RCxLQUFLQyxLQUFLLEdBQUc7RUFDWEMsUUFBUSxFQUFFLENBQUNDLE1BQU0sRUFBRUgsc0JBQXNCLEVBQUUsR0FBRyxJQUFJO0VBQ2xESSxVQUFVLEVBQUUsTUFBTTtFQUNsQkMsYUFBYSxFQUFFLENBQUNDLEtBQUssRUFBRSxNQUFNLEVBQUUsR0FB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 \ No newline at end of file diff --git a/src/components/FeedbackSurvey/TranscriptSharePrompt.tsx b/src/components/FeedbackSurvey/TranscriptSharePrompt.tsx new file mode 100644 index 0000000..b9556c8 --- /dev/null +++ b/src/components/FeedbackSurvey/TranscriptSharePrompt.tsx @@ -0,0 +1,88 @@ +import { c as _c } from "react/compiler-runtime"; +import React from 'react'; +import { BLACK_CIRCLE } from '../../constants/figures.js'; +import { Box, Text } from '../../ink.js'; +import { useDebouncedDigitInput } from './useDebouncedDigitInput.js'; +export type TranscriptShareResponse = 'yes' | 'no' | 'dont_ask_again'; +type Props = { + onSelect: (option: TranscriptShareResponse) => void; + inputValue: string; + setInputValue: (value: string) => void; +}; +const RESPONSE_INPUTS = ['1', '2', '3'] as const; +type ResponseInput = (typeof RESPONSE_INPUTS)[number]; +const inputToResponse: Record = { + '1': 'yes', + '2': 'no', + '3': 'dont_ask_again' +} as const; +const isValidResponseInput = (input: string): input is ResponseInput => (RESPONSE_INPUTS as readonly string[]).includes(input); +export function TranscriptSharePrompt(t0) { + const $ = _c(11); + const { + onSelect, + inputValue, + setInputValue + } = t0; + let t1; + if ($[0] !== onSelect) { + t1 = digit => onSelect(inputToResponse[digit]); + $[0] = onSelect; + $[1] = t1; + } else { + t1 = $[1]; + } + let t2; + if ($[2] !== inputValue || $[3] !== setInputValue || $[4] !== t1) { + t2 = { + inputValue, + setInputValue, + isValidDigit: isValidResponseInput, + onDigit: t1 + }; + $[2] = inputValue; + $[3] = setInputValue; + $[4] = t1; + $[5] = t2; + } else { + t2 = $[5]; + } + useDebouncedDigitInput(t2); + let t3; + if ($[6] === Symbol.for("react.memo_cache_sentinel")) { + t3 = {BLACK_CIRCLE} Can Anthropic look at your session transcript to help us improve Claude Code?; + $[6] = t3; + } else { + t3 = $[6]; + } + let t4; + if ($[7] === Symbol.for("react.memo_cache_sentinel")) { + t4 = Learn more: https://code.claude.com/docs/en/data-usage#session-quality-surveys; + $[7] = t4; + } else { + t4 = $[7]; + } + let t5; + if ($[8] === Symbol.for("react.memo_cache_sentinel")) { + t5 = 1: Yes; + $[8] = t5; + } else { + t5 = $[8]; + } + let t6; + if ($[9] === Symbol.for("react.memo_cache_sentinel")) { + t6 = 2: No; + $[9] = t6; + } else { + t6 = $[9]; + } + let t7; + if ($[10] === Symbol.for("react.memo_cache_sentinel")) { + t7 = {t3}{t4}{t5}{t6}3: Don't ask again; + $[10] = t7; + } else { + t7 = $[10]; + } + return t7; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkJMQUNLX0NJUkNMRSIsIkJveCIsIlRleHQiLCJ1c2VEZWJvdW5jZWREaWdpdElucHV0IiwiVHJhbnNjcmlwdFNoYXJlUmVzcG9uc2UiLCJQcm9wcyIsIm9uU2VsZWN0Iiwib3B0aW9uIiwiaW5wdXRWYWx1ZSIsInNldElucHV0VmFsdWUiLCJ2YWx1ZSIsIlJFU1BPTlNFX0lOUFVUUyIsImNvbnN0IiwiUmVzcG9uc2VJbnB1dCIsImlucHV0VG9SZXNwb25zZSIsIlJlY29yZCIsImlzVmFsaWRSZXNwb25zZUlucHV0IiwiaW5wdXQiLCJpbmNsdWRlcyIsIlRyYW5zY3JpcHRTaGFyZVByb21wdCIsInQwIiwiJCIsIl9jIiwidDEiLCJkaWdpdCIsInQyIiwiaXNWYWxpZERpZ2l0Iiwib25EaWdpdCIsInQzIiwiU3ltYm9sIiwiZm9yIiwidDQiLCJ0NSIsInQ2IiwidDciXSwic291cmNlcyI6WyJUcmFuc2NyaXB0U2hhcmVQcm9tcHQudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB7IEJMQUNLX0NJUkNMRSB9IGZyb20gJy4uLy4uL2NvbnN0YW50cy9maWd1cmVzLmpzJ1xuaW1wb3J0IHsgQm94LCBUZXh0IH0gZnJvbSAnLi4vLi4vaW5rLmpzJ1xuaW1wb3J0IHsgdXNlRGVib3VuY2VkRGlnaXRJbnB1dCB9IGZyb20gJy4vdXNlRGVib3VuY2VkRGlnaXRJbnB1dC5qcydcblxuZXhwb3J0IHR5cGUgVHJhbnNjcmlwdFNoYXJlUmVzcG9uc2UgPSAneWVzJyB8ICdubycgfCAnZG9udF9hc2tfYWdhaW4nXG5cbnR5cGUgUHJvcHMgPSB7XG4gIG9uU2VsZWN0OiAob3B0aW9uOiBUcmFuc2NyaXB0U2hhcmVSZXNwb25zZSkgPT4gdm9pZFxuICBpbnB1dFZhbHVlOiBzdHJpbmdcbiAgc2V0SW5wdXRWYWx1ZTogKHZhbHVlOiBzdHJpbmcpID0+IHZvaWRcbn1cblxuY29uc3QgUkVTUE9OU0VfSU5QVVRTID0gWycxJywgJzInLCAnMyddIGFzIGNvbnN0XG50eXBlIFJlc3BvbnNlSW5wdXQgPSAodHlwZW9mIFJFU1BPTlNFX0lOUFVUUylbbnVtYmVyXVxuXG5jb25zdCBpbnB1dFRvUmVzcG9uc2U6IFJlY29yZDxSZXNwb25zZUlucHV0LCBUcmFuc2NyaXB0U2hhcmVSZXNwb25zZT4gPSB7XG4gICcxJzogJ3llcycsXG4gICcyJzogJ25vJyxcbiAgJzMnOiAnZG9udF9hc2tfYWdhaW4nLFxufSBhcyBjb25zdFxuXG5jb25zdCBpc1ZhbGlkUmVzcG9uc2VJbnB1dCA9IChpbnB1dDogc3RyaW5nKTogaW5wdXQgaXMgUmVzcG9uc2VJbnB1dCA9PlxuICAoUkVTUE9OU0VfSU5QVVRTIGFzIHJlYWRvbmx5IHN0cmluZ1tdKS5pbmNsdWRlcyhpbnB1dClcblxuZXhwb3J0IGZ1bmN0aW9uIFRyYW5zY3JpcHRTaGFyZVByb21wdCh7XG4gIG9uU2VsZWN0LFxuICBpbnB1dFZhbHVlLFxuICBzZXRJbnB1dFZhbHVlLFxufTogUHJvcHMpOiBSZWFjdC5SZWFjdE5vZGUge1xuICB1c2VEZWJvdW5jZWREaWdpdElucHV0KHtcbiAgICBpbnB1dFZhbHVlLFxuICAgIHNldElucHV0VmFsdWUsXG4gICAgaXNWYWxpZERpZ2l0OiBpc1ZhbGlkUmVzcG9uc2VJbnB1dCxcbiAgICBvbkRpZ2l0OiBkaWdpdCA9PiBvblNlbGVjdChpbnB1dFRvUmVzcG9uc2VbZGlnaXRdKSxcbiAgfSlcblxuICByZXR1cm4gKFxuICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiIG1hcmdpblRvcD17MX0+XG4gICAgICA8Qm94PlxuICAgICAgICA8VGV4dCBjb2xvcj1cImFuc2k6Y3lhblwiPntCTEFDS19DSVJDTEV9IDwvVGV4dD5cbiAgICAgICAgPFRleHQgYm9sZD5cbiAgICAgICAgICBDYW4gQW50aHJvcGljIGxvb2sgYXQgeW91ciBzZXNzaW9uIHRyYW5zY3JpcHQgdG8gaGVscCB1cyBpbXByb3ZlXG4gICAgICAgICAgQ2xhdWRlIENvZGU/XG4gICAgICAgIDwvVGV4dD5cbiAgICAgIDwvQm94PlxuXG4gICAgICA8Qm94IG1hcmdpbkxlZnQ9ezJ9PlxuICAgICAgICA8VGV4dCBkaW1Db2xvcj5cbiAgICAgICAgICBMZWFybiBtb3JlOlxuICAgICAgICAgIGh0dHBzOi8vY29kZS5jbGF1ZGUuY29tL2RvY3MvZW4vZGF0YS11c2FnZSNzZXNzaW9uLXF1YWxpdHktc3VydmV5c1xuICAgICAgICA8L1RleHQ+XG4gICAgICA8L0JveD5cblxuICAgICAgPEJveCBtYXJnaW5MZWZ0PXsyfT5cbiAgICAgICAgPEJveCB3aWR0aD17MTB9PlxuICAgICAgICAgIDxUZXh0PlxuICAgICAgICAgICAgPFRleHQgY29sb3I9XCJhbnNpOmN5YW5cIj4xPC9UZXh0PjogWWVzXG4gICAgICAgICAgPC9UZXh0PlxuICAgICAgICA8L0JveD5cbiAgICAgICAgPEJveCB3aWR0aD17MTB9PlxuICAgICAgICAgIDxUZXh0PlxuICAgICAgICAgICAgPFRleHQgY29sb3I9XCJhbnNpOmN5YW5cIj4yPC9UZXh0PjogTm9cbiAgICAgICAgICA8L1RleHQ+XG4gICAgICAgIDwvQm94PlxuICAgICAgICA8Qm94PlxuICAgICAgICAgIDxUZXh0PlxuICAgICAgICAgICAgPFRleHQgY29sb3I9XCJhbnNpOmN5YW5cIj4zPC9UZXh0PjogRG9uJmFwb3M7dCBhc2sgYWdhaW5cbiAgICAgICAgICA8L1RleHQ+XG4gICAgICAgIDwvQm94PlxuICAgICAgPC9Cb3g+XG4gICAgPC9Cb3g+XG4gIClcbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU9BLEtBQUssTUFBTSxPQUFPO0FBQ3pCLFNBQVNDLFlBQVksUUFBUSw0QkFBNEI7QUFDekQsU0FBU0MsR0FBRyxFQUFFQyxJQUFJLFFBQVEsY0FBYztBQUN4QyxTQUFTQyxzQkFBc0IsUUFBUSw2QkFBNkI7QUFFcEUsT0FBTyxLQUFLQyx1QkFBdUIsR0FBRyxLQUFLLEdBQUcsSUFBSSxHQUFHLGdCQUFnQjtBQUVyRSxLQUFLQyxLQUFL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 \ No newline at end of file diff --git a/src/components/FeedbackSurvey/submitTranscriptShare.ts b/src/components/FeedbackSurvey/submitTranscriptShare.ts new file mode 100644 index 0000000..52e1425 --- /dev/null +++ b/src/components/FeedbackSurvey/submitTranscriptShare.ts @@ -0,0 +1,112 @@ +import axios from 'axios' +import { readFile, stat } from 'fs/promises' +import type { Message } from '../../types/message.js' +import { checkAndRefreshOAuthTokenIfNeeded } from '../../utils/auth.js' +import { logForDebugging } from '../../utils/debug.js' +import { errorMessage } from '../../utils/errors.js' +import { getAuthHeaders, getUserAgent } from '../../utils/http.js' +import { normalizeMessagesForAPI } from '../../utils/messages.js' +import { + extractAgentIdsFromMessages, + getTranscriptPath, + loadSubagentTranscripts, + MAX_TRANSCRIPT_READ_BYTES, +} from '../../utils/sessionStorage.js' +import { jsonStringify } from '../../utils/slowOperations.js' +import { redactSensitiveInfo } from '../Feedback.js' + +type TranscriptShareResult = { + success: boolean + transcriptId?: string +} + +export type TranscriptShareTrigger = + | 'bad_feedback_survey' + | 'good_feedback_survey' + | 'frustration' + | 'memory_survey' + +export async function submitTranscriptShare( + messages: Message[], + trigger: TranscriptShareTrigger, + appearanceId: string, +): Promise { + try { + logForDebugging('Collecting transcript for sharing', { level: 'info' }) + + const transcript = normalizeMessagesForAPI(messages) + + // Collect subagent transcripts + const agentIds = extractAgentIdsFromMessages(messages) + const subagentTranscripts = await loadSubagentTranscripts(agentIds) + + // Read raw JSONL transcript (with size guard to prevent OOM) + let rawTranscriptJsonl: string | undefined + try { + const transcriptPath = getTranscriptPath() + const { size } = await stat(transcriptPath) + if (size <= MAX_TRANSCRIPT_READ_BYTES) { + rawTranscriptJsonl = await readFile(transcriptPath, 'utf-8') + } else { + logForDebugging( + `Skipping raw transcript read: file too large (${size} bytes)`, + { level: 'warn' }, + ) + } + } catch { + // File may not exist + } + + const data = { + trigger, + version: MACRO.VERSION, + platform: process.platform, + transcript, + subagentTranscripts: + Object.keys(subagentTranscripts).length > 0 + ? subagentTranscripts + : undefined, + rawTranscriptJsonl, + } + + const content = redactSensitiveInfo(jsonStringify(data)) + + await checkAndRefreshOAuthTokenIfNeeded() + + const authResult = getAuthHeaders() + if (authResult.error) { + return { success: false } + } + + const headers: Record = { + 'Content-Type': 'application/json', + 'User-Agent': getUserAgent(), + ...authResult.headers, + } + + const response = await axios.post( + 'https://api.anthropic.com/api/claude_code_shared_session_transcripts', + { content, appearance_id: appearanceId }, + { + headers, + timeout: 30000, + }, + ) + + if (response.status === 200 || response.status === 201) { + const result = response.data + logForDebugging('Transcript shared successfully', { level: 'info' }) + return { + success: true, + transcriptId: result?.transcript_id, + } + } + + return { success: false } + } catch (err) { + logForDebugging(errorMessage(err), { + level: 'error', + }) + return { success: false } + } +} diff --git a/src/components/FeedbackSurvey/useDebouncedDigitInput.ts b/src/components/FeedbackSurvey/useDebouncedDigitInput.ts new file mode 100644 index 0000000..072eaeb --- /dev/null +++ b/src/components/FeedbackSurvey/useDebouncedDigitInput.ts @@ -0,0 +1,82 @@ +import { useEffect, useRef } from 'react' +import { normalizeFullWidthDigits } from '../../utils/stringUtils.js' + +// Delay before accepting a digit as a response, to prevent accidental +// submissions when users start messages with numbers (e.g., numbered lists). +// Short enough to feel instant for intentional presses, long enough to +// cancel when the user types more characters. +const DEFAULT_DEBOUNCE_MS = 400 + +/** + * Detects when the user types a single valid digit into the prompt input, + * debounces to avoid accidental submissions (e.g., "1. First item"), + * trims the digit from the input, and fires a callback. + * + * Used by survey components that accept numeric responses typed directly + * into the main prompt input. + */ +export function useDebouncedDigitInput({ + inputValue, + setInputValue, + isValidDigit, + onDigit, + enabled = true, + once = false, + debounceMs = DEFAULT_DEBOUNCE_MS, +}: { + inputValue: string + setInputValue: (value: string) => void + isValidDigit: (char: string) => char is T + onDigit: (digit: T) => void + enabled?: boolean + once?: boolean + debounceMs?: number +}): void { + const initialInputValue = useRef(inputValue) + const hasTriggeredRef = useRef(false) + const debounceRef = useRef | null>(null) + + // Latest-ref pattern so callers can pass inline callbacks without causing + // the effect to re-run (which would reset the debounce timer every render). + const callbacksRef = useRef({ setInputValue, isValidDigit, onDigit }) + callbacksRef.current = { setInputValue, isValidDigit, onDigit } + + useEffect(() => { + if (!enabled || (once && hasTriggeredRef.current)) { + return + } + + if (debounceRef.current !== null) { + clearTimeout(debounceRef.current) + debounceRef.current = null + } + + if (inputValue !== initialInputValue.current) { + const lastChar = normalizeFullWidthDigits(inputValue.slice(-1)) + if (callbacksRef.current.isValidDigit(lastChar)) { + const trimmed = inputValue.slice(0, -1) + debounceRef.current = setTimeout( + (debounceRef, hasTriggeredRef, callbacksRef, trimmed, lastChar) => { + debounceRef.current = null + hasTriggeredRef.current = true + callbacksRef.current.setInputValue(trimmed) + callbacksRef.current.onDigit(lastChar) + }, + debounceMs, + debounceRef, + hasTriggeredRef, + callbacksRef, + trimmed, + lastChar, + ) + } + } + + return () => { + if (debounceRef.current !== null) { + clearTimeout(debounceRef.current) + debounceRef.current = null + } + } + }, [inputValue, enabled, once, debounceMs]) +} diff --git a/src/components/FeedbackSurvey/useFeedbackSurvey.tsx b/src/components/FeedbackSurvey/useFeedbackSurvey.tsx new file mode 100644 index 0000000..20bab3d --- /dev/null +++ b/src/components/FeedbackSurvey/useFeedbackSurvey.tsx @@ -0,0 +1,296 @@ +import { useCallback, useEffect, useMemo, useRef, useState } from 'react'; +import { useDynamicConfig } from 'src/hooks/useDynamicConfig.js'; +import { isFeedbackSurveyDisabled } from 'src/services/analytics/config.js'; +import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from 'src/services/analytics/index.js'; +import { isPolicyAllowed } from '../../services/policyLimits/index.js'; +import type { Message } from '../../types/message.js'; +import { getGlobalConfig, saveGlobalConfig } from '../../utils/config.js'; +import { isEnvTruthy } from '../../utils/envUtils.js'; +import { getLastAssistantMessage } from '../../utils/messages.js'; +import { getMainLoopModel } from '../../utils/model/model.js'; +import { getInitialSettings } from '../../utils/settings/settings.js'; +import { logOTelEvent } from '../../utils/telemetry/events.js'; +import { submitTranscriptShare, type TranscriptShareTrigger } from './submitTranscriptShare.js'; +import type { TranscriptShareResponse } from './TranscriptSharePrompt.js'; +import { useSurveyState } from './useSurveyState.js'; +import type { FeedbackSurveyResponse, FeedbackSurveyType } from './utils.js'; +type FeedbackSurveyConfig = { + minTimeBeforeFeedbackMs: number; + minTimeBetweenFeedbackMs: number; + minTimeBetweenGlobalFeedbackMs: number; + minUserTurnsBeforeFeedback: number; + minUserTurnsBetweenFeedback: number; + hideThanksAfterMs: number; + onForModels: string[]; + probability: number; +}; +type TranscriptAskConfig = { + probability: number; +}; +const DEFAULT_FEEDBACK_SURVEY_CONFIG: FeedbackSurveyConfig = { + minTimeBeforeFeedbackMs: 600000, + minTimeBetweenFeedbackMs: 3600000, + minTimeBetweenGlobalFeedbackMs: 100000000, + minUserTurnsBeforeFeedback: 5, + minUserTurnsBetweenFeedback: 10, + hideThanksAfterMs: 3000, + onForModels: ['*'], + probability: 0.005 +}; +const DEFAULT_TRANSCRIPT_ASK_CONFIG: TranscriptAskConfig = { + probability: 0 +}; +export function useFeedbackSurvey(messages: Message[], isLoading: boolean, submitCount: number, surveyType: FeedbackSurveyType = 'session', hasActivePrompt: boolean = false): { + state: 'closed' | 'open' | 'thanks' | 'transcript_prompt' | 'submitting' | 'submitted'; + lastResponse: FeedbackSurveyResponse | null; + handleSelect: (selected: FeedbackSurveyResponse) => boolean; + handleTranscriptSelect: (selected: TranscriptShareResponse) => void; +} { + const lastAssistantMessageIdRef = useRef('unknown'); + lastAssistantMessageIdRef.current = getLastAssistantMessage(messages)?.message?.id || 'unknown'; + const [feedbackSurvey, setFeedbackSurvey] = useState<{ + timeLastShown: number | null; + submitCountAtLastAppearance: number | null; + }>(() => ({ + timeLastShown: null, + submitCountAtLastAppearance: null + })); + const config = useDynamicConfig('tengu_feedback_survey_config', DEFAULT_FEEDBACK_SURVEY_CONFIG); + const badTranscriptAskConfig = useDynamicConfig('tengu_bad_survey_transcript_ask_config', DEFAULT_TRANSCRIPT_ASK_CONFIG); + const goodTranscriptAskConfig = useDynamicConfig('tengu_good_survey_transcript_ask_config', DEFAULT_TRANSCRIPT_ASK_CONFIG); + const settingsRate = getInitialSettings().feedbackSurveyRate; + const sessionStartTime = useRef(Date.now()); + const submitCountAtSessionStart = useRef(submitCount); + const submitCountRef = useRef(submitCount); + submitCountRef.current = submitCount; + const messagesRef = useRef(messages); + messagesRef.current = messages; + // Probability gate: roll once when eligibility conditions are met, not on every + // useMemo re-evaluation. Without this, each dependency change (submitCount, + // isLoading toggle, etc.) re-rolls Math.random(), making the survey almost + // certain to appear after enough renders. + const probabilityPassedRef = useRef(false); + const lastEligibleSubmitCountRef = useRef(null); + const updateLastShownTime = useCallback((timestamp: number, submitCountValue: number) => { + setFeedbackSurvey(prev => { + if (prev.timeLastShown === timestamp && prev.submitCountAtLastAppearance === submitCountValue) { + return prev; + } + return { + timeLastShown: timestamp, + submitCountAtLastAppearance: submitCountValue + }; + }); + // Persist cross-session pacing state (previously done by onChangeAppState observer) + if (getGlobalConfig().feedbackSurveyState?.lastShownTime !== timestamp) { + saveGlobalConfig(current => ({ + ...current, + feedbackSurveyState: { + lastShownTime: timestamp + } + })); + } + }, []); + const onOpen = useCallback((appearanceId: string) => { + updateLastShownTime(Date.now(), submitCountRef.current); + logEvent('tengu_feedback_survey_event', { + event_type: 'appeared' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + appearance_id: appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + last_assistant_message_id: lastAssistantMessageIdRef.current as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + survey_type: surveyType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + void logOTelEvent('feedback_survey', { + event_type: 'appeared', + appearance_id: appearanceId, + survey_type: surveyType + }); + }, [updateLastShownTime, surveyType]); + const onSelect = useCallback((appearanceId_0: string, selected: FeedbackSurveyResponse) => { + updateLastShownTime(Date.now(), submitCountRef.current); + logEvent('tengu_feedback_survey_event', { + event_type: 'responded' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + appearance_id: appearanceId_0 as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + response: selected as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + last_assistant_message_id: lastAssistantMessageIdRef.current as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + survey_type: surveyType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + void logOTelEvent('feedback_survey', { + event_type: 'responded', + appearance_id: appearanceId_0, + response: selected, + survey_type: surveyType + }); + }, [updateLastShownTime, surveyType]); + const shouldShowTranscriptPrompt = useCallback((selected_0: FeedbackSurveyResponse) => { + // Only bad and good ratings trigger the transcript ask + if (selected_0 !== 'bad' && selected_0 !== 'good') { + return false; + } + + // Don't show if user previously chose "Don't ask again" + if (getGlobalConfig().transcriptShareDismissed) { + return false; + } + + // Don't show if product feedback is blocked by org policy (ZDR) + if (!isPolicyAllowed('allow_product_feedback')) { + return false; + } + + // Probability gate from GrowthBook config (separate per rating) + const probability = selected_0 === 'bad' ? badTranscriptAskConfig.probability : goodTranscriptAskConfig.probability; + return Math.random() <= probability; + }, [badTranscriptAskConfig.probability, goodTranscriptAskConfig.probability]); + const onTranscriptPromptShown = useCallback((appearanceId_1: string, surveyResponse: FeedbackSurveyResponse) => { + const trigger: TranscriptShareTrigger = surveyResponse === 'good' ? 'good_feedback_survey' : 'bad_feedback_survey'; + logEvent('tengu_feedback_survey_event', { + event_type: 'transcript_prompt_appeared' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + appearance_id: appearanceId_1 as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + last_assistant_message_id: lastAssistantMessageIdRef.current as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + survey_type: surveyType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + trigger: trigger as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + void logOTelEvent('feedback_survey', { + event_type: 'transcript_prompt_appeared', + appearance_id: appearanceId_1, + survey_type: surveyType + }); + }, [surveyType]); + const onTranscriptSelect = useCallback(async (appearanceId_2: string, selected_1: TranscriptShareResponse, surveyResponse_0: FeedbackSurveyResponse | null): Promise => { + const trigger_0: TranscriptShareTrigger = surveyResponse_0 === 'good' ? 'good_feedback_survey' : 'bad_feedback_survey'; + logEvent('tengu_feedback_survey_event', { + event_type: `transcript_share_${selected_1}` as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + appearance_id: appearanceId_2 as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + last_assistant_message_id: lastAssistantMessageIdRef.current as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + survey_type: surveyType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + trigger: trigger_0 as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + if (selected_1 === 'dont_ask_again') { + saveGlobalConfig(current_0 => ({ + ...current_0, + transcriptShareDismissed: true + })); + } + if (selected_1 === 'yes') { + const result = await submitTranscriptShare(messagesRef.current, trigger_0, appearanceId_2); + logEvent('tengu_feedback_survey_event', { + event_type: (result.success ? 'transcript_share_submitted' : 'transcript_share_failed') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + appearance_id: appearanceId_2 as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + trigger: trigger_0 as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + return result.success; + } + return false; + }, [surveyType]); + const { + state, + lastResponse, + open, + handleSelect, + handleTranscriptSelect + } = useSurveyState({ + hideThanksAfterMs: config.hideThanksAfterMs, + onOpen, + onSelect, + shouldShowTranscriptPrompt, + onTranscriptPromptShown, + onTranscriptSelect + }); + const currentModel = getMainLoopModel(); + const isModelAllowed = useMemo(() => { + if (config.onForModels.length === 0) { + return false; + } + if (config.onForModels.includes('*')) { + return true; + } + return config.onForModels.includes(currentModel); + }, [config.onForModels, currentModel]); + const shouldOpen = useMemo(() => { + if (state !== 'closed') { + return false; + } + if (isLoading) { + return false; + } + + // Don't show survey when permission or ask question prompts are visible + if (hasActivePrompt) { + return false; + } + + // Force display for testing + if (process.env.CLAUDE_FORCE_DISPLAY_SURVEY && !feedbackSurvey.timeLastShown) { + return true; + } + if (!isModelAllowed) { + return false; + } + if (isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_FEEDBACK_SURVEY)) { + return false; + } + if (isFeedbackSurveyDisabled()) { + return false; + } + + // Check if product feedback is allowed by org policy + if (!isPolicyAllowed('allow_product_feedback')) { + return false; + } + + // Check session-local pacing + if (feedbackSurvey.timeLastShown) { + // Check time elapsed since last appearance in this session + const timeSinceLastShown = Date.now() - feedbackSurvey.timeLastShown; + if (timeSinceLastShown < config.minTimeBetweenFeedbackMs) { + return false; + } + // Check user turn requirement for subsequent appearances + if (feedbackSurvey.submitCountAtLastAppearance !== null && submitCount < feedbackSurvey.submitCountAtLastAppearance + config.minUserTurnsBetweenFeedback) { + return false; + } + } else { + // First appearance in this session + const timeSinceSessionStart = Date.now() - sessionStartTime.current; + if (timeSinceSessionStart < config.minTimeBeforeFeedbackMs) { + return false; + } + if (submitCount < submitCountAtSessionStart.current + config.minUserTurnsBeforeFeedback) { + return false; + } + } + + // Probability check: roll once per eligibility window to avoid re-rolling + // on every useMemo re-evaluation (which would make triggering near-certain). + if (lastEligibleSubmitCountRef.current !== submitCount) { + lastEligibleSubmitCountRef.current = submitCount; + probabilityPassedRef.current = Math.random() <= (settingsRate ?? config.probability); + } + if (!probabilityPassedRef.current) { + return false; + } + + // Check global pacing (across all sessions) + // Leave this till last because it reads from the filesystem which is expensive. + const globalFeedbackState = getGlobalConfig().feedbackSurveyState; + if (globalFeedbackState?.lastShownTime) { + const timeSinceGlobalLastShown = Date.now() - globalFeedbackState.lastShownTime; + if (timeSinceGlobalLastShown < config.minTimeBetweenGlobalFeedbackMs) { + return false; + } + } + return true; + }, [state, isLoading, hasActivePrompt, isModelAllowed, feedbackSurvey.timeLastShown, feedbackSurvey.submitCountAtLastAppearance, submitCount, config.minTimeBetweenFeedbackMs, config.minTimeBetweenGlobalFeedbackMs, config.minUserTurnsBetweenFeedback, config.minTimeBeforeFeedbackMs, config.minUserTurnsBeforeFeedback, config.probability, settingsRate]); + useEffect(() => { + if (shouldOpen) { + open(); + } + }, [shouldOpen, open]); + return { + state, + lastResponse, + handleSelect, + handleTranscriptSelect + }; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJ1c2VDYWxsYmFjayIsInVzZUVmZmVjdCIsInVzZU1lbW8iLCJ1c2VSZWYiLCJ1c2VTdGF0ZSIsInVzZUR5bmFtaWNDb25maWciLCJpc0ZlZWRiYWNrU3VydmV5RGlzYWJsZWQiLCJBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTIiwibG9nRXZlbnQiLCJpc1BvbGljeUFsbG93ZWQiLCJNZXNzYWdlIiwiZ2V0R2xvYmFsQ29uZmlnIiwic2F2ZUdsb2JhbENvbmZpZyIsImlzRW52VHJ1dGh5IiwiZ2V0TGFzdEFzc2lzdGFudE1lc3NhZ2UiLCJnZXRNYWluTG9vcE1vZGVsIiwiZ2V0SW5pdGlhbFNldHRpbmdzIiwibG9nT1RlbEV2ZW50Iiwic3VibWl0VHJhbnNjcmlwdFNoYXJlIiwiVHJhbnNjcmlwdFNoYXJlVHJpZ2dlciIsIlRyYW5zY3JpcHRTaGFyZVJlc3BvbnNlIiwidXNlU3VydmV5U3RhdGUiLCJGZWVkYmFja1N1cnZleVJlc3BvbnNlIiwiRmVlZGJhY2tTdXJ2ZXlUeXBlIiwiRmVlZGJhY2tTdXJ2ZXlDb25maWciLCJtaW5UaW1lQmVmb3JlRmVlZGJhY2tNcyIsIm1pblRpbWVCZXR3ZWVuRmVlZGJhY2tNcyIsIm1pblRpbWVCZXR3ZWVuR2xvYmFsRmVlZGJhY2tNcyIsIm1pblVzZXJUdXJuc0JlZm9yZUZlZWRiYWNrIiwibWluVXNlclR1cm5zQmV0d2VlbkZlZWRiYWNrIiwiaGlkZVRoYW5rc0FmdGVyTXMiLCJvbkZvck1vZGVscyIsInByb2JhYmlsaXR5IiwiVHJhbnNjcmlwdEFza0NvbmZpZyIsIkRFRkFVTFRfRkVFREJBQ0tfU1VSVkVZX0NPTkZJRyIsIkRFRkFVTFRfVFJBTlNDUklQVF9BU0tfQ09ORklHIiwidXNlRmVlZGJhY2tTdXJ2ZXkiLCJtZXNzYWdlcyIsImlzTG9hZGluZyIsInN1Ym1pdENvdW50Iiwic3VydmV5VHlwZSIsImhhc0FjdGl2ZVByb21wdCIsInN0YXRlIiwibGFzdFJlc3BvbnNlIiwiaGFuZGxlU2VsZWN0Iiwic2VsZWN0ZWQiLCJoYW5kbGVUcmFuc2NyaXB0U2VsZWN0IiwibGFzdEFzc2lzdGFudE1lc3NhZ2VJZFJlZiIsImN1cnJlbnQiLCJtZXNzYWdlIiwiaWQiLCJmZWVkYmFja1N1cnZleSIsInNldEZlZWRiYWNrU3VydmV5IiwidGltZUxhc3RTaG93biIsInN1Ym1pdENvdW50QXRMYXN0QXBwZWFyYW5jZSIsImNvbmZpZyIsImJhZFRyYW5zY3JpcHRBc2tDb25maWciLCJnb29kVHJhbnNjcmlwdEFza0NvbmZpZyIsInNldHRpbmdzUmF0ZSIsImZlZWRiYWNrU3VydmV5UmF0ZSIsInNlc3Npb25TdGFydFRpbWUiLCJEYXRlIiwibm93Iiwic3VibWl0Q291bnRBdFNlc3Npb25TdGFydCIsInN1Ym1pdENvdW50UmVmIiwibWVzc2FnZXNSZWYiLCJwcm9iYWJpbGl0eVBhc3NlZFJlZiIsImxhc3RFbGlnaWJsZVN1Ym1pdENvdW50UmVmIiwidXBkYXRlTGFzdFNob3duVGltZSIsInRpbWVzdGFtcCIsInN1Ym1pdENvdW50VmFsdWUiLCJwcmV2IiwiZmVlZGJhY2tTdXJ2ZXlTdGF0ZSIsImxhc3RTaG93blRpbWUiLCJvbk9wZW4iLCJhcHBlYXJhbmNlSWQiLCJldmVudF90eXBlIiwiYXBwZWFyYW5jZV9pZCIsImxhc3RfYXNzaXN0YW50X21lc3NhZ2VfaWQiLCJzdXJ2ZXlfdHlwZSIsIm9uU2VsZWN0IiwicmVzcG9uc2UiLCJzaG91bGRTaG93VHJhbnNjcmlwdFByb21wdCIsInRyYW5zY3JpcHRTaGFyZURpc21pc3NlZCIsIk1hdGgiLCJyYW5kb20iLCJvblRyYW5zY3JpcHRQcm9tcHRTaG93biIsInN1cnZleVJlc3BvbnNlIiwidHJpZ2dlciIsIm9uVHJhbnNjcmlwdFNlbGVjdCIsIlByb21pc2UiLCJyZXN1bHQiLCJzdWNjZXNzIiwib3BlbiIsImN1cnJlbnRNb2RlbCIsImlzTW9kZWxBbGxvd2VkIiwibGVuZ3RoIiwiaW5jbHVkZXMiLCJzaG91bGRPcGVuIiwicHJvY2VzcyIsImVudiIsIkNMQVVERV9GT1JDRV9ESVNQTEFZX1NVUlZFWSIsIkNMQVVERV9DT0RFX0RJU0FCTEVfRkVFREJBQ0tfU1VSVkVZIiwidGltZVNpbmNlTGFzdFNob3duIiwidGltZVNpbmNlU2Vzc2lvblN0YXJ0IiwiZ2xvYmFsRmVlZGJhY2tTdGF0ZSIsInRpbWVTaW5jZUdsb2JhbExhc3RTaG93biJdLCJzb3VyY2VzIjpbInVzZUZlZWRiYWNrU3VydmV5LnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyB1c2VDYWxsYmFjaywgdXNlRWZmZWN0LCB1c2VNZW1vLCB1c2VSZWYsIHVzZVN0YXRlIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyB1c2VEeW5hbWljQ29uZmlnIH0gZnJvbSAnc3JjL2hvb2tzL3VzZUR5bmFtaWNDb25maWcuanMnXG5pbXBvcnQgeyBpc0ZlZWRiYWNrU3VydmV5RGlzYWJsZWQgfSBmcm9tICdzcmMvc2VydmljZXMvYW5hbHl0aWNzL2NvbmZpZy5qcydcbmltcG9ydCB7XG4gIHR5cGUgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgbG9nRXZlbnQsXG59IGZyb20gJ3NyYy9zZXJ2aWNlcy9hbmFseXRpY3MvaW5kZXguanMnXG5pbXBvcnQgeyBpc1BvbGljeUFsbG93ZWQgfSBmcm9tICcuLi8uLi9zZXJ2aWNlcy9wb2xpY3lMaW1pdHMvaW5kZXguanMnXG5pbXBvcnQgdHlwZSB7IE1lc3NhZ2UgfSBmcm9tICcuLi8uLi90eXBlcy9tZXNzYWdlLmpzJ1xuaW1wb3J0IHsgZ2V0R2xvYmFsQ29uZmlnLCBzYXZlR2xvYmFsQ29uZmlnIH0gZnJvbSAnLi4vLi4vdXRpbHMvY29uZmlnLmpzJ1xuaW1wb3J0IHsgaXNFbnZUcnV0aHkgfSBmcm9tICcuLi8uLi91dGlscy9lbnZVdGlscy5qcydcbmltcG9ydCB7IGdldExhc3RBc3Npc3RhbnRNZXNzYWdlIH0gZnJvbSAnLi4vLi4vdXRpbHMvbWVzc2FnZXMuanMnXG5pbXBvcnQgeyBnZXRNYWluTG9vcE1vZGVsIH0gZnJvbSAnLi4vLi4vdXRpbHMvbW9kZWwvbW9kZWwuanMnXG5pbXBvcnQgeyBnZXRJbml0aWFsU2V0dGluZ3MgfSBmcm9tICcuLi8uLi91dGlscy9zZXR0aW5ncy9zZXR0aW5ncy5qcydcbmltcG9ydCB7IGxvZ09UZWxFdmVudCB9IGZyb20gJy4uLy4uL3V0aWxzL3RlbGVtZXRyeS9ldmVudHMuanMnXG5pbXBvcnQge1xuICBzdWJtaXRUcmFuc2NyaXB0U2hhcmUsXG4gIHR5cGUgVHJhbnNjcmlwdFNoYXJlVHJpZ2dlcixcbn0gZnJvbSAnLi9zdWJtaXRUcmFuc2NyaXB0U2hhcmUuanMnXG5pbXBvcnQgdHlwZSB7IFRyYW5zY3JpcHRTaGFyZVJlc3BvbnNlIH0gZnJvbSAnLi9UcmFuc2NyaXB0U2hhcmVQcm9tcHQuanMnXG5pbXBvcnQgeyB1c2VTdXJ2ZXlTdGF0ZSB9IGZyb20gJy4vdXNlU3VydmV5U3RhdGUuanMnXG5pbXBvcnQgdHlwZSB7IEZlZWRiYWNrU3VydmV5UmVzcG9uc2UsIEZlZWRiYWNrU3VydmV5VHlwZSB9IGZyb20gJy4vdXRpbHMuanMnXG5cbnR5cGUgRmVlZGJhY2tTdXJ2ZXlDb25maWcgPSB7XG4gIG1pblRpbWVCZWZvcmVGZWVkYmFja01zOiBudW1iZXJcbiAgbWluVGltZUJldHdlZW5GZWVkYmFja01zOiBudW1iZXJcbiAgbWluVGltZUJldHdlZW5HbG9iYWxGZWVkYmFja01zOiBudW1iZXJcbiAgbWluVXNlclR1cm5zQmVmb3JlRmVlZGJhY2s6IG51bWJlclxuICBtaW5Vc2VyVHVybnNCZXR3ZWVuRmVlZGJhY2s6IG51bWJlclxuICBoaWRlVGhhbmtzQWZ0ZXJNczogbnVtYmVyXG4gIG9uRm9yTW9kZWxzOiBzdHJpbmdbXVxuICBwcm9iYWJpbGl0eTogbnVtYmVyXG59XG5cbnR5cGUgVHJhbnNjcmlwdEFza0NvbmZpZyA9IHtcbiAgcHJvYmFiaWxpdHk6IG51bWJlclxufVxuXG5jb25zdCBERUZBVUxUX0ZFRURCQUNLX1NVUlZFWV9DT05GSUc6IEZlZWRiYWNrU3VydmV5Q29uZmlnID0ge1xuICBtaW5UaW1lQmVmb3JlRmVlZGJhY2tNczogNjAwMDAwLFxuICBtaW5UaW1lQmV0d2VlbkZlZWRiYWNrTXM6IDM2MDAwMDAsXG4gIG1pblRpbWVCZXR3ZWVuR2xvYmFsRmVlZGJhY2tNczogMTAwMDAwMDAwLFxuICBtaW5Vc2VyVHVybnNCZWZvcmVGZWVkYmFjazogNSxcbiAgbWluVXNlclR1cm5zQmV0d2VlbkZlZWRiYWNrOiAxMCxcbiAgaGlkZVRoYW5rc0FmdGVyTXM6IDMwMDAsXG4gIG9uRm9yTW9kZWxzOiBbJyonXSxcbiAgcHJvYmFiaWxpdHk6IDAuMDA1LFxufVxuXG5jb25zdCBERUZBVUxUX1RSQU5TQ1JJUFRfQVNLX0NPTkZJRzogVHJhbnNjcmlwdEFza0NvbmZpZyA9IHtcbiAgcHJvYmFiaWxpdHk6IDAsXG59XG5cbmV4cG9ydCBmdW5jdGlvbiB1c2VGZWVkYmFja1N1cnZleShcbiAgbWVzc2FnZXM6IE1lc3NhZ2VbXSxcbiAgaXNMb2FkaW5nOiBib29sZWFuLFxuICBzdWJtaXRDb3VudDogbnVtYmVyLFxuICBzdXJ2ZXlUeXBlOiBGZWVkYmFja1N1cnZleVR5cGUgPSAnc2Vzc2lvbicsXG4gIGhhc0FjdGl2ZVByb21wdDogYm9vbGVhbiA9IGZhbHNlLFxuKToge1xuICBzdGF0ZTpcbiAgICB8ICdjbG9zZWQnXG4gICAgfCAnb3BlbidcbiAgICB8ICd0aGFua3MnXG4gICAgfCAndHJhbnNjcmlwdF9wcm9tcHQnXG4gICAgfCAnc3VibWl0dGluZydcbiAgICB8ICdzdWJtaXR0ZWQnXG4gIGxhc3RSZXNwb25zZTogRmVlZGJhY2tTdXJ2ZXlSZXNwb25zZSB8IG51bGxcbiAgaGFuZGxlU2VsZWN0OiAoc2VsZWN0ZWQ6IEZlZWRiYWNrU3VydmV5UmVzcG9uc2UpID0+IGJvb2xlYW5cbiAgaGFuZGxlVHJhbnNjcmlwdFNlbGVjdDogKHNlbGVjdGVkOiBUcmFuc2NyaXB0U2hhcmVSZXNwb25zZSkgPT4gdm9pZFxufSB7XG4gIGNvbnN0IGxhc3RBc3Npc3RhbnRNZXNzYWdlSWRSZWYgPSB1c2VSZWYoJ3Vua25vd24nKVxuICBsYXN0QXNzaXN0YW50TWVzc2FnZUlkUmVmLmN1cnJlbnQgPVxuICAgIGdldExhc3RBc3Npc3RhbnRNZXNzYWdlKG1lc3NhZ2VzKT8ubWVzc2FnZT8uaWQgfHwgJ3Vua25vd24nXG4gIGNvbnN0IFtmZWVkYmFja1N1cnZleSwgc2V0RmVlZGJhY2tTdXJ2ZXldID0gdXNlU3RhdGU8e1xuICAgIHRpbWVMYXN0U2hvd246IG51bWJlciB8IG51bGxcbiAgICBzdWJtaXRDb3VudEF0TGFzdEFwcGVhcmFuY2U6IG51bWJlciB8IG51bGxcbiAgfT4oKCkgPT4gKHsgdGltZUxhc3RTaG93bjogbnVsbCwgc3VibWl0Q291bnRBdExhc3RBcHBlYXJhbmNlOiBudWxsIH0pKVxuICBjb25zdCBjb25maWcgPSB1c2VEeW5hbWljQ29uZmlnPEZlZWRiYWNrU3VydmV5Q29uZmlnPihcbiAgICAndGVuZ3VfZmVlZGJhY2tfc3VydmV5X2NvbmZpZycsXG4gICAgREVGQVVMVF9GRUVEQkFDS19TVVJWRVlfQ09ORklHLFxuICApXG4gIGNvbnN0IGJhZFRyYW5zY3JpcHRBc2tDb25maWcgPSB1c2VEeW5hbWljQ29uZmlnPFRyYW5zY3JpcHRBc2tDb25maWc+KFxuICAgICd0ZW5ndV9iYWRfc3VydmV5X3RyYW5zY3JpcHRfYXNrX2NvbmZpZycsXG4gICAgREVGQVVMVF9UUkFOU0NSSVBUX0FTS19DT05GSUcsXG4gIClcbiAgY29uc3QgZ29vZFRyYW5zY3JpcHRBc2tDb25maWcgPSB1c2VEeW5hbWljQ29uZmlnPFRyYW5zY3JpcHRBc2tDb25maWc+KFxuICAgICd0ZW5ndV9nb29kX3N1cnZleV90cmFuc2NyaXB0X2Fza19jb25maWcnLFxuICAgIERFRkFVTFRfVFJBTlNDUklQVF9BU0tfQ09ORklHLFxuICApXG4gIGNvbnN0IHNldHRpbmdzUmF0ZSA9IGdldEluaXRpYWxTZXR0aW5ncygpLmZlZWRiYWNrU3VydmV5UmF0ZVxuICBjb25zdCBzZXNzaW9uU3RhcnRUaW1lID0gdXNlUmVmKERhdGUubm93KCkpXG4gIGNvbnN0IHN1Ym1pdENvdW50QXRTZXNzaW9uU3RhcnQgPSB1c2VSZWYoc3VibWl0Q291bnQpXG4gIGNvbnN0IHN1Ym1pdENvdW50UmVmID0gdXNlUmVmKHN1Ym1pdENvdW50KVxuICBzdWJtaXRDb3VudFJlZi5jdXJyZW50ID0gc3VibWl0Q291bnRcbiAgY29uc3QgbWVzc2FnZXNSZWYgPSB1c2VSZWYobWVzc2FnZXMpXG4gIG1lc3NhZ2VzUmVmLmN1cnJlbnQgPSBtZXNzYWdlc1xuICAvLyBQcm9iYWJpbGl0eSBnYXRlOiByb2xsIG9uY2Ugd2hlbiBlbGlnaWJpbGl0eSBjb25kaXRpb25zIGFyZSBtZXQsIG5vdCBvbiBldmVyeVxuICAvLyB1c2VNZW1vIHJlLWV2YWx1YXRpb24uIFdpdGhvdXQgdGhpcywgZWFjaCBkZXBlbmRlbmN5IGNoYW5nZSAoc3VibWl0Q291bnQsXG4gIC8vIGlzTG9hZGluZyB0b2dnbGUsIGV0Yy4pIHJlLXJvbGxzIE1hdGgucmFuZG9tKCksIG1ha2luZyB0aGUgc3VydmV5IGFsbW9zdFxuICAvLyBjZXJ0YWluIHRvIGFwcGVhciBhZnRlciBlbm91Z2ggcmVuZGVycy5cbiAgY29uc3QgcHJvYmFiaWxpdHlQYXNzZWRSZWYgPSB1c2VSZWYoZmFsc2UpXG4gIGNvbnN0IGxhc3RFbGlnaWJsZVN1Ym1pdENvdW50UmVmID0gdXNlUmVmPG51bWJlciB8IG51bGw+KG51bGwpXG5cbiAgY29uc3QgdXBkYXRlTGFzdFNob3duVGltZSA9IHVzZUNhbGxiYWNrKFxuICAgICh0aW1lc3RhbXA6IG51bWJlciwgc3VibWl0Q291bnRWYWx1ZTogbnVtYmVyKSA9PiB7XG4gICAgICBzZXRGZWVkYmFja1N1cnZleShwcmV2ID0+IHtcbiAgICAgICAgaWYgKFxuICAgICAgICAgIHByZXYudGltZUxhc3RTaG93biA9PT0gdGltZXN0YW1wICYmXG4gICAgICAgICAgcHJldi5zdWJtaXRDb3VudEF0TGFzdEFwcGVhcmFuY2UgPT09IHN1Ym1pdENvdW50VmFsdWVcbiAgICAgICAgKSB7XG4gICAgICAgICAgcmV0dXJuIHByZXZcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgIHRpbWVMYXN0U2hvd246IHRpbWVzdGFtcCxcbiAgICAgICAgICBzdWJtaXRDb3VudEF0TGFzdEFwcGVhcmFuY2U6IHN1Ym1pdENvdW50VmFsdWUsXG4gICAgICAgIH1cbiAgICAgIH0pXG4gICAgICAvLyBQZXJzaXN0IGNyb3NzLXNlc3Npb24gcGFjaW5nIHN0YXRlIChwcmV2aW91c2x5IGRvbmUgYnkgb25DaGFuZ2VBcHBTdGF0ZSBvYnNlcnZlcilcbiAgICAgIGlmIChnZXRHbG9iYWxDb25maWcoKS5mZWVkYmFja1N1cnZleVN0YXRlPy5sYXN0U2hvd25UaW1lICE9PSB0aW1lc3RhbXApIHtcbiAgICAgICAgc2F2ZUdsb2JhbENvbmZpZyhjdXJyZW50ID0+ICh7XG4gICAgICAgICAgLi4uY3VycmVudCxcbiAgICAgICAgICBmZWVkYmFja1N1cnZleVN0YXRlOiB7XG4gICAgICAgICAgICBsYXN0U2hvd25UaW1lOiB0aW1lc3RhbXAsXG4gICAgICAgICAgfSxcbiAgICAgICAgfSkpXG4gICAgICB9XG4gICAgfSxcbiAgICBbXSxcbiAgKVxuXG4gIGNvbnN0IG9uT3BlbiA9IHVzZUNhbGxiYWNrKFxuICAgIChhcHBlYXJhbmNlSWQ6IHN0cmluZykgPT4ge1xuICAgICAgdXBkYXRlTGFzdFNob3duVGltZShEYXRlLm5vdygpLCBzdWJtaXRDb3VudFJlZi5jdXJyZW50KVxuICAgICAgbG9nRXZlbnQoJ3Rlbmd1X2ZlZWRiYWNrX3N1cnZleV9ldmVudCcsIHtcbiAgICAgICAgZXZlbnRfdHlwZTpcbiAgICAgICAgICAnYXBwZWFyZWQnIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgIGFwcGVhcmFuY2VfaWQ6XG4gICAgICAgICAgYXBwZWFyYW5jZUlkIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgIGxhc3RfYXNzaXN0YW50X21lc3NhZ2VfaWQ6XG4gICAgICAgICAgbGFzdEFzc2lzdGFudE1lc3NhZ2VJZFJlZi5jdXJyZW50IGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgIHN1cnZleV90eXBlOlxuICAgICAgICAgIHN1cnZleVR5cGUgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICAgIH0pXG4gICAgICB2b2lkIGxvZ09UZWxFdmVudCgnZmVlZGJhY2tfc3VydmV5Jywge1xuICAgICAgICBldmVudF90eXBlOiAnYXBwZWFyZWQnLFxuICAgICAgICBhcHBlYXJhbmNlX2lkOiBhcHBlYXJhbmNlSWQsXG4gICAgICAgIHN1cnZleV90eXBlOiBzdXJ2ZXlUeXBlLFxuICAgICAgfSlcbiAgICB9LFxuICAgIFt1cGRhdGVMYXN0U2hvd25UaW1lLCBzdXJ2ZXlUeXBlXSxcbiAgKVxuXG4gIGNvbnN0IG9uU2VsZWN0ID0gdXNlQ2FsbGJhY2soXG4gICAgKGFwcGVhcmFuY2VJZDogc3RyaW5nLCBzZWxlY3RlZDogRmVlZGJhY2tTdXJ2ZXlSZXNwb25zZSkgPT4ge1xuICAgICAgdXBkYXRlTGFzdFNob3duVGltZShEYXRlLm5vdygpLCBzdWJtaXRDb3VudFJlZi5jdXJyZW50KVxuICAgICAgbG9nRXZlbnQoJ3Rlbmd1X2ZlZWRiYWNrX3N1cnZleV9ldmVudCcsIHtcbiAgICAgICAgZXZlbnRfdHlwZTpcbiAgICAgICAgICAncmVzcG9uZGVkJyBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICBhcHBlYXJhbmNlX2lkOlxuICAgICAgICAgIGFwcGVhcmFuY2VJZCBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICByZXNwb25zZTpcbiAgICAgICAgICBzZWxlY3RlZCBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICBsYXN0X2Fzc2lzdGFudF9tZXNzYWdlX2lkOlxuICAgICAgICAgIGxhc3RBc3Npc3RhbnRNZXNzYWdlSWRSZWYuY3VycmVudCBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICBzdXJ2ZXlfdHlwZTpcbiAgICAgICAgICBzdXJ2ZXlUeXBlIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICB9KVxuICAgICAgdm9pZCBsb2dPVGVsRXZlbnQoJ2ZlZWRiYWNrX3N1cnZleScsIHtcbiAgICAgICAgZXZlbnRfdHlwZTogJ3Jlc3BvbmRlZCcsXG4gICAgICAgIGFwcGVhcmFuY2VfaWQ6IGFwcGVhcmFuY2VJZCxcbiAgICAgICAgcmVzcG9uc2U6IHNlbGVjdGVkLFxuICAgICAgICBzdXJ2ZXlfdHlwZTogc3VydmV5VHlwZSxcbiAgICAgIH0pXG4gICAgfSxcbiAgICBbdXBkYXRlTGFzdFNob3duVGltZSwgc3VydmV5VHlwZV0sXG4gIClcblxuICBjb25zdCBzaG91bGRTaG93VHJhbnNjcmlwdFByb21wdCA9IHVzZUNhbGxiYWNrKFxuICAgIChzZWxlY3RlZDogRmVlZGJhY2tTdXJ2ZXlSZXNwb25zZSkgPT4ge1xuICAgICAgLy8gT25seSBiYWQgYW5kIGdvb2QgcmF0aW5ncyB0cmlnZ2VyIHRoZSB0cmFuc2NyaXB0IGFza1xuICAgICAgaWYgKHNlbGVjdGVkICE9PSAnYmFkJyAmJiBzZWxlY3RlZCAhPT0gJ2dvb2QnKSB7XG4gICAgICAgIHJldHVybiBmYWxzZVxuICAgICAgfVxuXG4gICAgICAvLyBEb24ndCBzaG93IGlmIHVzZXIgcHJldmlvdXNseSBjaG9zZSBcIkRvbid0IGFzayBhZ2FpblwiXG4gICAgICBpZiAoZ2V0R2xvYmFsQ29uZmlnKCkudHJhbnNjcmlwdFNoYXJlRGlzbWlzc2VkKSB7XG4gICAgICAgIHJldHVybiBmYWxzZVxuICAgICAgfVxuXG4gICAgICAvLyBEb24ndCBzaG93IGlmIHByb2R1Y3QgZmVlZGJhY2sgaXMgYmxvY2tlZCBieSBvcmcgcG9saWN5IChaRFIpXG4gICAgICBpZiAoIWlzUG9saWN5QWxsb3dlZCgnYWxsb3dfcHJvZHVjdF9mZWVkYmFjaycpKSB7XG4gICAgICAgIHJldHVybiBmYWxzZVxuICAgICAgfVxuXG4gICAgICAvLyBQcm9iYWJpbGl0eSBnYXRlIGZyb20gR3Jvd3RoQm9vayBjb25maWcgKHNlcGFyYXRlIHBlciByYXRpbmcpXG4gICAgICBjb25zdCBwcm9iYWJpbGl0eSA9XG4gICAgICAgIHNlbGVjdGVkID09PSAnYmFkJ1xuICAgICAgICAgID8gYmFkVHJhbnNjcmlwdEFza0NvbmZpZy5wcm9iYWJpbGl0eVxuICAgICAgICAgIDogZ29vZFRyYW5zY3JpcHRBc2tDb25maWcucHJvYmFiaWxpdHlcbiAgICAgIHJldHVybiBNYXRoLnJhbmRvbSgpIDw9IHByb2JhYmlsaXR5XG4gICAgfSxcbiAgICBbYmFkVHJhbnNjcmlwdEFza0NvbmZpZy5wcm9iYWJpbGl0eSwgZ29vZFRyYW5zY3JpcHRBc2tDb25maWcucHJvYmFiaWxpdHldLFxuICApXG5cbiAgY29uc3Qgb25UcmFuc2NyaXB0UHJvbXB0U2hvd24gPSB1c2VDYWxsYmFjayhcbiAgICAoYXBwZWFyYW5jZUlkOiBzdHJpbmcsIHN1cnZleVJlc3BvbnNlOiBGZWVkYmFja1N1cnZleVJlc3BvbnNlKSA9PiB7XG4gICAgICBjb25zdCB0cmlnZ2VyOiBUcmFuc2NyaXB0U2hhcmVUcmlnZ2VyID1cbiAgICAgICAgc3VydmV5UmVzcG9uc2UgPT09ICdnb29kJ1xuICAgICAgICAgID8gJ2dvb2RfZmVlZGJhY2tfc3VydmV5J1xuICAgICAgICAgIDogJ2JhZF9mZWVkYmFja19zdXJ2ZXknXG4gICAgICBsb2dFdmVudCgndGVuZ3VfZmVlZGJhY2tfc3VydmV5X2V2ZW50Jywge1xuICAgICAgICBldmVudF90eXBlOlxuICAgICAgICAgICd0cmFuc2NyaXB0X3Byb21wdF9hcHBlYXJlZCcgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICAgICAgYXBwZWFyYW5jZV9pZDpcbiAgICAgICAgICBhcHBlYXJhbmNlSWQgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICAgICAgbGFzdF9hc3Npc3RhbnRfbWVzc2FnZV9pZDpcbiAgICAgICAgICBsYXN0QXNzaXN0YW50TWVzc2FnZUlkUmVmLmN1cnJlbnQgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICAgICAgc3VydmV5X3R5cGU6XG4gICAgICAgICAgc3VydmV5VHlwZSBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICB0cmlnZ2VyOlxuICAgICAgICAgIHRyaWdnZXIgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICAgIH0pXG4gICAgICB2b2lkIGxvZ09UZWxFdmVudCgnZmVlZGJhY2tfc3VydmV5Jywge1xuICAgICAgICBldmVudF90eXBlOiAndHJhbnNjcmlwdF9wcm9tcHRfYXBwZWFyZWQnLFxuICAgICAgICBhcHBlYXJhbmNlX2lkOiBhcHBlYXJhbmNlSWQsXG4gICAgICAgIHN1cnZleV90eXBlOiBzdXJ2ZXlUeXBlLFxuICAgICAgfSlcbiAgICB9LFxuICAgIFtzdXJ2ZXlUeXBlXSxcbiAgKVxuXG4gIGNvbnN0IG9uVHJhbnNjcmlwdFNlbGVjdCA9IHVzZUNhbGxiYWNrKFxuICAgIGFzeW5jIChcbiAgICAgIGFwcGVhcmFuY2VJZDogc3RyaW5nLFxuICAgICAgc2VsZWN0ZWQ6IFRyYW5zY3JpcHRTaGFyZVJlc3BvbnNlLFxuICAgICAgc3VydmV5UmVzcG9uc2U6IEZlZWRiYWNrU3VydmV5UmVzcG9uc2UgfCBudWxsLFxuICAgICk6IFByb21pc2U8Ym9vbGVhbj4gPT4ge1xuICAgICAgY29uc3QgdHJpZ2dlcjogVHJhbnNjcmlwdFNoYXJlVHJpZ2dlciA9XG4gICAgICAgIHN1cnZleVJlc3BvbnNlID09PSAnZ29vZCdcbiAgICAgICAgICA/ICdnb29kX2ZlZWRiYWNrX3N1cnZleSdcbiAgICAgICAgICA6ICdiYWRfZmVlZGJhY2tfc3VydmV5J1xuXG4gICAgICBsb2dFdmVudCgndGVuZ3VfZmVlZGJhY2tfc3VydmV5X2V2ZW50Jywge1xuICAgICAgICBldmVudF90eXBlOlxuICAgICAgICAgIGB0cmFuc2NyaXB0X3NoYXJlXyR7c2VsZWN0ZWR9YCBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICBhcHBlYXJhbmNlX2lkOlxuICAgICAgICAgIGFwcGVhcmFuY2VJZCBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICBsYXN0X2Fzc2lzdGFudF9tZXNzYWdlX2lkOlxuICAgICAgICAgIGxhc3RBc3Npc3RhbnRNZXNzYWdlSWRSZWYuY3VycmVudCBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICBzdXJ2ZXlfdHlwZTpcbiAgICAgICAgICBzdXJ2ZXlUeXBlIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgIHRyaWdnZXI6XG4gICAgICAgICAgdHJpZ2dlciBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgfSlcblxuICAgICAgaWYgKHNlbGVjdGVkID09PSAnZG9udF9hc2tfYWdhaW4nKSB7XG4gICAgICAgIHNhdmVHbG9iYWxDb25maWcoY3VycmVudCA9PiAoe1xuICAgICAgICAgIC4uLmN1cnJlbnQsXG4gICAgICAgICAgdHJhbnNjcmlwdFNoYXJlRGlzbWlzc2VkOiB0cnVlLFxuICAgICAgICB9KSlcbiAgICAgIH1cblxuICAgICAgaWYgKHNlbGVjdGVkID09PSAneWVzJykge1xuICAgICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCBzdWJtaXRUcmFuc2NyaXB0U2hhcmUoXG4gICAgICAgICAgbWVzc2FnZXNSZWYuY3VycmVudCxcbiAgICAgICAgICB0cmlnZ2VyLFxuICAgICAgICAgIGFwcGVhcmFuY2VJZCxcbiAgICAgICAgKVxuICAgICAgICBsb2dFdmVudCgndGVuZ3VfZmVlZGJhY2tfc3VydmV5X2V2ZW50Jywge1xuICAgICAgICAgIGV2ZW50X3R5cGU6IChyZXN1bHQuc3VjY2Vzc1xuICAgICAgICAgICAgPyAndHJhbnNjcmlwdF9zaGFyZV9zdWJtaXR0ZWQnXG4gICAgICAgICAgICA6ICd0cmFuc2NyaXB0X3NoYXJlX2ZhaWxlZCcpIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgICAgYXBwZWFyYW5jZV9pZDpcbiAgICAgICAgICAgIGFwcGVhcmFuY2VJZCBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICAgIHRyaWdnZXI6XG4gICAgICAgICAgICB0cmlnZ2VyIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgIH0pXG4gICAgICAgIHJldHVybiByZXN1bHQuc3VjY2Vzc1xuICAgICAgfVxuXG4gICAgICByZXR1cm4gZmFsc2VcbiAgICB9LFxuICAgIFtzdXJ2ZXlUeXBlXSxcbiAgKVxuXG4gIGNvbnN0IHsgc3RhdGUsIGxhc3RSZXNwb25zZSwgb3BlbiwgaGFuZGxlU2VsZWN0LCBoYW5kbGVUcmFuc2NyaXB0U2VsZWN0IH0gPVxuICAgIHVzZVN1cnZleVN0YXRlKHtcbiAgICAgIGhpZGVUaGFua3NBZnRlck1zOiBjb25maWcuaGlkZVRoYW5rc0FmdGVyTXMsXG4gICAgICBvbk9wZW4sXG4gICAgICBvblNlbGVjdCxcbiAgICAgIHNob3VsZFNob3dUcmFuc2NyaXB0UHJvbXB0LFxuICAgICAgb25UcmFuc2NyaXB0UHJvbXB0U2hvd24sXG4gICAgICBvblRyYW5zY3JpcHRTZWxlY3QsXG4gICAgfSlcblxuICBjb25zdCBjdXJyZW50TW9kZWwgPSBnZXRNYWluTG9vcE1vZGVsKClcbiAgY29uc3QgaXNNb2RlbEFsbG93ZWQgPSB1c2VNZW1vKCgpID0+IHtcbiAgICBpZiAoY29uZmlnLm9uRm9yTW9kZWxzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgcmV0dXJuIGZhbHNlXG4gICAgfVxuICAgIGlmIChjb25maWcub25Gb3JNb2RlbHMuaW5jbHVkZXMoJyonKSkge1xuICAgICAgcmV0dXJuIHRydWVcbiAgICB9XG4gICAgcmV0dXJuIGNvbmZpZy5vbkZvck1vZGVscy5pbmNsdWRlcyhjdXJyZW50TW9kZWwpXG4gIH0sIFtjb25maWcub25Gb3JNb2RlbHMsIGN1cnJlbnRNb2RlbF0pXG5cbiAgY29uc3Qgc2hvdWxkT3BlbiA9IHVzZU1lbW8oKCkgPT4ge1xuICAgIGlmIChzdGF0ZSAhPT0gJ2Nsb3NlZCcpIHtcbiAgICAgIHJldHVybiBmYWxzZVxuICAgIH1cblxuICAgIGlmIChpc0xvYWRpbmcpIHtcbiAgICAgIHJldHVybiBmYWxzZVxuICAgIH1cblxuICAgIC8vIERvbid0IHNob3cgc3VydmV5IHdoZW4gcGVybWlzc2lvbiBvciBhc2sgcXVlc3Rpb24gcHJvbXB0cyBhcmUgdmlzaWJsZVxuICAgIGlmIChoYXNBY3RpdmVQcm9tcHQpIHtcbiAgICAgIHJldHVybiBmYWxzZVxuICAgIH1cblxuICAgIC8vIEZvcmNlIGRpc3BsYXkgZm9yIHRlc3RpbmdcbiAgICBpZiAoXG4gICAgICBwcm9jZXNzLmVudi5DTEFVREVfRk9SQ0VfRElTUExBWV9TVVJWRVkgJiZcbiAgICAgICFmZWVkYmFja1N1cnZleS50aW1lTGFzdFNob3duXG4gICAgKSB7XG4gICAgICByZXR1cm4gdHJ1ZVxuICAgIH1cblxuICAgIGlmICghaXNNb2RlbEFsbG93ZWQpIHtcbiAgICAgIHJldHVybiBmYWxzZVxuICAgIH1cblxuICAgIGlmIChpc0VudlRydXRoeShwcm9jZXNzLmVudi5DTEFVREVfQ09ERV9ESVNBQkxFX0ZFRURCQUNLX1NVUlZFWSkpIHtcbiAgICAgIHJldHVybiBmYWxzZVxuICAgIH1cblxuICAgIGlmIChpc0ZlZWRiYWNrU3VydmV5RGlzYWJsZWQoKSkge1xuICAgICAgcmV0dXJuIGZhbHNlXG4gICAgfVxuXG4gICAgLy8gQ2hlY2sgaWYgcHJvZHVjdCBmZWVkYmFjayBpcyBhbGxvd2VkIGJ5IG9yZyBwb2xpY3lcbiAgICBpZiAoIWlzUG9saWN5QWxsb3dlZCgnYWxsb3dfcHJvZHVjdF9mZWVkYmFjaycpKSB7XG4gICAgICByZXR1cm4gZmFsc2VcbiAgICB9XG5cbiAgICAvLyBDaGVjayBzZXNzaW9uLWxvY2FsIHBhY2luZ1xuICAgIGlmIChmZWVkYmFja1N1cnZleS50aW1lTGFzdFNob3duKSB7XG4gICAgICAvLyBDaGVjayB0aW1lIGVsYXBzZWQgc2luY2UgbGFzdCBhcHBlYXJhbmNlIGluIHRoaXMgc2Vzc2lvblxuICAgICAgY29uc3QgdGltZVNpbmNlTGFzdFNob3duID0gRGF0ZS5ub3coKSAtIGZlZWRiYWNrU3VydmV5LnRpbWVMYXN0U2hvd25cbiAgICAgIGlmICh0aW1lU2luY2VMYXN0U2hvd24gPCBjb25maWcubWluVGltZUJldHdlZW5GZWVkYmFja01zKSB7XG4gICAgICAgIHJldHVybiBmYWxzZVxuICAgICAgfVxuICAgICAgLy8gQ2hlY2sgdXNlciB0dXJuIHJlcXVpcmVtZW50IGZvciBzdWJzZXF1ZW50IGFwcGVhcmFuY2VzXG4gICAgICBpZiAoXG4gICAgICAgIGZlZWRiYWNrU3VydmV5LnN1Ym1pdENvdW50QXRMYXN0QXBwZWFyYW5jZSAhPT0gbnVsbCAmJlxuICAgICAgICBzdWJtaXRDb3VudCA8XG4gICAgICAgICAgZmVlZGJhY2tTdXJ2ZXkuc3VibWl0Q291bnRBdExhc3RBcHBlYXJhbmNlICtcbiAgICAgICAgICAgIGNvbmZpZy5taW5Vc2VyVHVybnNCZXR3ZWVuRmVlZGJhY2tcbiAgICAgICkge1xuICAgICAgICByZXR1cm4gZmFsc2VcbiAgICAgIH1cbiAgICB9IGVsc2Uge1xuICAgICAgLy8gRmlyc3QgYXBwZWFyYW5jZSBpbiB0aGlzIHNlc3Npb25cbiAgICAgIGNvbnN0IHRpbWVTaW5jZVNlc3Npb25TdGFydCA9IERhdGUubm93KCkgLSBzZXNzaW9uU3RhcnRUaW1lLmN1cnJlbnRcbiAgICAgIGlmICh0aW1lU2luY2VTZXNzaW9uU3RhcnQgPCBjb25maWcubWluVGltZUJlZm9yZUZlZWRiYWNrTXMpIHtcbiAgICAgICAgcmV0dXJuIGZhbHNlXG4gICAgICB9XG4gICAgICBpZiAoXG4gICAgICAgIHN1Ym1pdENvdW50IDxcbiAgICAgICAgc3VibWl0Q291bnRBdFNlc3Npb25TdGFydC5jdXJyZW50ICsgY29uZmlnLm1pblVzZXJUdXJuc0JlZm9yZUZlZWRiYWNrXG4gICAgICApIHtcbiAgICAgICAgcmV0dXJuIGZhbHNlXG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8gUHJvYmFiaWxpdHkgY2hlY2s6IHJvbGwgb25jZSBwZXIgZWxpZ2liaWxpdHkgd2luZG93IHRvIGF2b2lkIHJlLXJvbGxpbmdcbiAgICAvLyBvbiBldmVyeSB1c2VNZW1vIHJlLWV2YWx1YXRpb24gKHdoaWNoIHdvdWxkIG1ha2UgdHJpZ2dlcmluZyBuZWFyLWNlcnRhaW4pLlxuICAgIGlmIChsYXN0RWxpZ2libGVTdWJtaXRDb3VudFJlZi5jdXJyZW50ICE9PSBzdWJtaXRDb3VudCkge1xuICAgICAgbGFzdEVsaWdpYmxlU3VibWl0Q291bnRSZWYuY3VycmVudCA9IHN1Ym1pdENvdW50XG4gICAgICBwcm9iYWJpbGl0eVBhc3NlZFJlZi5jdXJyZW50ID1cbiAgICAgICAgTWF0aC5yYW5kb20oKSA8PSAoc2V0dGluZ3NSYXRlID8/IGNvbmZpZy5wcm9iYWJpbGl0eSlcbiAgICB9XG4gICAgaWYgKCFwcm9iYWJpbGl0eVBhc3NlZFJlZi5jdXJyZW50KSB7XG4gICAgICByZXR1cm4gZmFsc2VcbiAgICB9XG5cbiAgICAvLyBDaGVjayBnbG9iYWwgcGFjaW5nIChhY3Jvc3MgYWxsIHNlc3Npb25zKVxuICAgIC8vIExlYXZlIHRoaXMgdGlsbCBsYXN0IGJlY2F1c2UgaXQgcmVhZHMgZnJvbSB0aGUgZmlsZXN5c3RlbSB3aGljaCBpcyBleHBlbnNpdmUuXG4gICAgY29uc3QgZ2xvYmFsRmVlZGJhY2tTdGF0ZSA9IGdldEdsb2JhbENvbmZpZygpLmZlZWRiYWNrU3VydmV5U3RhdGVcbiAgICBpZiAoZ2xvYmFsRmVlZGJhY2tTdGF0ZT8ubGFzdFNob3duVGltZSkge1xuICAgICAgY29uc3QgdGltZVNpbmNlR2xvYmFsTGFzdFNob3duID1cbiAgICAgICAgRGF0ZS5ub3coKSAtIGdsb2JhbEZlZWRiYWNrU3RhdGUubGFzdFNob3duVGltZVxuICAgICAgaWYgKHRpbWVTaW5jZUdsb2JhbExhc3RTaG93biA8IGNvbmZpZy5taW5UaW1lQmV0d2Vlbkdsb2JhbEZlZWRiYWNrTXMpIHtcbiAgICAgICAgcmV0dXJuIGZhbHNlXG4gICAgICB9XG4gICAgfVxuXG4gICAgcmV0dXJuIHRydWVcbiAgfSwgW1xuICAgIHN0YXRlLFxuICAgIGlzTG9hZGluZyxcbiAgICBoYXNBY3RpdmVQcm9tcHQsXG4gICAgaXNNb2RlbEFsbG93ZWQsXG4gICAgZmVlZGJhY2tTdXJ2ZXkudGltZUxhc3RTaG93bixcbiAgICBmZWVkYmFja1N1cnZleS5zdWJtaXRDb3VudEF0TGFzdEFwcGVhcmFuY2UsXG4gICAgc3VibWl0Q291bnQsXG4gICAgY29uZmlnLm1pblRpbWVCZXR3ZWVuRmVlZGJhY2tNcyxcbiAgICBjb25maWcubWluVGltZUJldHdlZW5HbG9iYWxGZWVkYmFja01zLFxuICAgIGNvbmZpZy5taW5Vc2VyVHVybnNCZXR3ZWVuRmVlZGJhY2ssXG4gICAgY29uZmlnLm1pblRpbWVCZWZvcmVGZWVkYmFja01zLFxuICAgIGNvbmZpZy5taW5Vc2VyVHVybnNCZWZvcmVGZWVkYmFjayxcbiAgICBjb25maWcucHJvYmFiaWxpdHksXG4gICAgc2V0dGluZ3NSYXRlLFxuICBdKVxuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKHNob3VsZE9wZW4pIHtcbiAgICAgIG9wZW4oKVxuICAgIH1cbiAgfSwgW3Nob3VsZE9wZW4sIG9wZW5dKVxuXG4gIHJldHVybiB7IHN0YXRlLCBsYXN0UmVzcG9uc2UsIGhhbmRsZVNlbGVjdCwgaGFuZGxlVHJhbnNjcmlwdFNlbGVjdCB9XG59XG4iXSwibWFwcGluZ3MiOiJBQUFBLFNBQVNBLFdBQVcsRUFBRUMsU0FBUyxFQUFFQyxPQUFPLEVBQUVDLE1BQU0sRUFBRUMsUUFBUSxRQUFRLE9BQU87QUFDekUsU0FBU0MsZ0JBQWdCLFFBQVEsK0JBQStCO0FBQ2hFLFNBQVNDLHdCQUF3QixRQUFRLGtDQUFrQztBQUMzRSxTQUNFLEtBQUtDLDBEQUEwRCxFQUMvREMsUUFBUSxRQUNILGlDQUFpQztBQUN4QyxTQUFTQyxlQUFlLFFBQVEsc0NBQXNDO0FBQ3RFLGNBQWNDLE9BQU8sUUFBUSx3QkFBd0I7QUFDckQsU0FBU0MsZUFBZSxFQUFFQyxnQkFBZ0IsUUFBUSx1QkFBdUI7QUFDekUsU0FBU0MsV0FBVyxRQUFR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 \ No newline at end of file diff --git a/src/components/FeedbackSurvey/useMemorySurvey.tsx b/src/components/FeedbackSurvey/useMemorySurvey.tsx new file mode 100644 index 0000000..e7d9b18 --- /dev/null +++ b/src/components/FeedbackSurvey/useMemorySurvey.tsx @@ -0,0 +1,213 @@ +import { useCallback, useEffect, useMemo, useRef } from 'react'; +import { isFeedbackSurveyDisabled } from 'src/services/analytics/config.js'; +import { getFeatureValue_CACHED_MAY_BE_STALE } from 'src/services/analytics/growthbook.js'; +import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from 'src/services/analytics/index.js'; +import { isAutoMemoryEnabled } from '../../memdir/paths.js'; +import { isPolicyAllowed } from '../../services/policyLimits/index.js'; +import { FILE_READ_TOOL_NAME } from '../../tools/FileReadTool/prompt.js'; +import type { Message } from '../../types/message.js'; +import { getGlobalConfig, saveGlobalConfig } from '../../utils/config.js'; +import { isEnvTruthy } from '../../utils/envUtils.js'; +import { isAutoManagedMemoryFile } from '../../utils/memoryFileDetection.js'; +import { extractTextContent, getLastAssistantMessage } from '../../utils/messages.js'; +import { logOTelEvent } from '../../utils/telemetry/events.js'; +import { submitTranscriptShare } from './submitTranscriptShare.js'; +import type { TranscriptShareResponse } from './TranscriptSharePrompt.js'; +import { useSurveyState } from './useSurveyState.js'; +import type { FeedbackSurveyResponse } from './utils.js'; +const HIDE_THANKS_AFTER_MS = 3000; +const MEMORY_SURVEY_GATE = 'tengu_dunwich_bell'; +const MEMORY_SURVEY_EVENT = 'tengu_memory_survey_event'; +const SURVEY_PROBABILITY = 0.2; +const TRANSCRIPT_SHARE_TRIGGER = 'memory_survey'; +const MEMORY_WORD_RE = /\bmemor(?:y|ies)\b/i; +function hasMemoryFileRead(messages: Message[]): boolean { + for (const message of messages) { + if (message.type !== 'assistant') { + continue; + } + const content = message.message.content; + if (!Array.isArray(content)) { + continue; + } + for (const block of content) { + if (block.type !== 'tool_use' || block.name !== FILE_READ_TOOL_NAME) { + continue; + } + const input = block.input as { + file_path?: unknown; + }; + if (typeof input.file_path === 'string' && isAutoManagedMemoryFile(input.file_path)) { + return true; + } + } + } + return false; +} +export function useMemorySurvey(messages: Message[], isLoading: boolean, hasActivePrompt = false, { + enabled = true +}: { + enabled?: boolean; +} = {}): { + state: 'closed' | 'open' | 'thanks' | 'transcript_prompt' | 'submitting' | 'submitted'; + lastResponse: FeedbackSurveyResponse | null; + handleSelect: (selected: FeedbackSurveyResponse) => void; + handleTranscriptSelect: (selected: TranscriptShareResponse) => void; +} { + // Track assistant message UUIDs that were already evaluated so we don't + // re-roll probability on re-renders or re-scan messages for the same turn. + const seenAssistantUuids = useRef>(new Set()); + // Once a memory file read is observed it stays true for the session — + // skip the O(n) scan on subsequent turns. + const memoryReadSeen = useRef(false); + const messagesRef = useRef(messages); + messagesRef.current = messages; + const onOpen = useCallback((appearanceId: string) => { + logEvent(MEMORY_SURVEY_EVENT, { + event_type: 'appeared' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + appearance_id: appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + void logOTelEvent('feedback_survey', { + event_type: 'appeared', + appearance_id: appearanceId, + survey_type: 'memory' + }); + }, []); + const onSelect = useCallback((appearanceId_0: string, selected: FeedbackSurveyResponse) => { + logEvent(MEMORY_SURVEY_EVENT, { + event_type: 'responded' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + appearance_id: appearanceId_0 as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + response: selected as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + void logOTelEvent('feedback_survey', { + event_type: 'responded', + appearance_id: appearanceId_0, + response: selected, + survey_type: 'memory' + }); + }, []); + const shouldShowTranscriptPrompt = useCallback((selected_0: FeedbackSurveyResponse) => { + if ("external" !== 'ant') { + return false; + } + if (selected_0 !== 'bad' && selected_0 !== 'good') { + return false; + } + if (getGlobalConfig().transcriptShareDismissed) { + return false; + } + if (!isPolicyAllowed('allow_product_feedback')) { + return false; + } + return true; + }, []); + const onTranscriptPromptShown = useCallback((appearanceId_1: string) => { + logEvent(MEMORY_SURVEY_EVENT, { + event_type: 'transcript_prompt_appeared' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + appearance_id: appearanceId_1 as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + trigger: TRANSCRIPT_SHARE_TRIGGER as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + void logOTelEvent('feedback_survey', { + event_type: 'transcript_prompt_appeared', + appearance_id: appearanceId_1, + survey_type: 'memory' + }); + }, []); + const onTranscriptSelect = useCallback(async (appearanceId_2: string, selected_1: TranscriptShareResponse): Promise => { + logEvent(MEMORY_SURVEY_EVENT, { + event_type: `transcript_share_${selected_1}` as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + appearance_id: appearanceId_2 as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + trigger: TRANSCRIPT_SHARE_TRIGGER as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + if (selected_1 === 'dont_ask_again') { + saveGlobalConfig(current => ({ + ...current, + transcriptShareDismissed: true + })); + } + if (selected_1 === 'yes') { + const result = await submitTranscriptShare(messagesRef.current, TRANSCRIPT_SHARE_TRIGGER, appearanceId_2); + logEvent(MEMORY_SURVEY_EVENT, { + event_type: (result.success ? 'transcript_share_submitted' : 'transcript_share_failed') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + appearance_id: appearanceId_2 as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + trigger: TRANSCRIPT_SHARE_TRIGGER as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + return result.success; + } + return false; + }, []); + const { + state, + lastResponse, + open, + handleSelect, + handleTranscriptSelect + } = useSurveyState({ + hideThanksAfterMs: HIDE_THANKS_AFTER_MS, + onOpen, + onSelect, + shouldShowTranscriptPrompt, + onTranscriptPromptShown, + onTranscriptSelect + }); + const lastAssistant = useMemo(() => getLastAssistantMessage(messages), [messages]); + useEffect(() => { + if (!enabled) return; + + // /clear resets messages but REPL stays mounted — reset refs so a memory + // read from the previous conversation doesn't leak into the new one. + if (messages.length === 0) { + memoryReadSeen.current = false; + seenAssistantUuids.current.clear(); + return; + } + if (state !== 'closed' || isLoading || hasActivePrompt) { + return; + } + + // 3P default: survey off (no GrowthBook on Bedrock/Vertex/Foundry). + if (!getFeatureValue_CACHED_MAY_BE_STALE(MEMORY_SURVEY_GATE, false)) { + return; + } + if (!isAutoMemoryEnabled()) { + return; + } + if (isFeedbackSurveyDisabled()) { + return; + } + if (!isPolicyAllowed('allow_product_feedback')) { + return; + } + if (isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_FEEDBACK_SURVEY)) { + return; + } + if (!lastAssistant || seenAssistantUuids.current.has(lastAssistant.uuid)) { + return; + } + const text = extractTextContent(lastAssistant.message.content, ' '); + if (!MEMORY_WORD_RE.test(text)) { + return; + } + + // Mark as evaluated before the memory-read scan so a turn that mentions + // "memory" but has no memory read doesn't trigger repeated O(n) scans + // on subsequent renders with the same last assistant message. + seenAssistantUuids.current.add(lastAssistant.uuid); + if (!memoryReadSeen.current) { + memoryReadSeen.current = hasMemoryFileRead(messages); + } + if (!memoryReadSeen.current) { + return; + } + if (Math.random() < SURVEY_PROBABILITY) { + open(); + } + }, [enabled, state, isLoading, hasActivePrompt, lastAssistant, messages, open]); + return { + state, + lastResponse, + handleSelect, + handleTranscriptSelect + }; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJ1c2VDYWxsYmFjayIsInVzZUVmZmVjdCIsInVzZU1lbW8iLCJ1c2VSZWYiLCJpc0ZlZWRiYWNrU3VydmV5RGlzYWJsZWQiLCJnZXRGZWF0dXJlVmFsdWVfQ0FDSEVEX01BWV9CRV9TVEFMRSIsIkFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMiLCJsb2dFdmVudCIsImlzQXV0b01lbW9yeUVuYWJsZWQiLCJpc1BvbGljeUFsbG93ZWQiLCJGSUxFX1JFQURfVE9PTF9OQU1FIiwiTWVzc2FnZSIsImdldEdsb2JhbENvbmZpZyIsInNhdmVHbG9iYWxDb25maWciLCJpc0VudlRydXRoeSIsImlzQXV0b01hbmFnZWRNZW1vcnlGaWxlIiwiZXh0cmFjdFRleHRDb250ZW50IiwiZ2V0TGFzdEFzc2lzdGFudE1lc3NhZ2UiLCJsb2dPVGVsRXZlbnQiLCJzdWJtaXRUcmFuc2NyaXB0U2hhcmUiLCJUcmFuc2NyaXB0U2hhcmVSZXNwb25zZSIsInVzZVN1cnZleVN0YXRlIiwiRmVlZGJhY2tTdXJ2ZXlSZXNwb25zZSIsIkhJREVfVEhBTktTX0FGVEVSX01TIiwiTUVNT1JZX1NVUlZFWV9HQVRFIiwiTUVNT1JZX1NVUlZFWV9FVkVOVCIsIlNVUlZFWV9QUk9CQUJJTElUWSIsIlRSQU5TQ1JJUFRfU0hBUkVfVFJJR0dFUiIsIk1FTU9SWV9XT1JEX1JFIiwiaGFzTWVtb3J5RmlsZVJlYWQiLCJtZXNzYWdlcyIsIm1lc3NhZ2UiLCJ0eXBlIiwiY29udGVudCIsIkFycmF5IiwiaXNBcnJheSIsImJsb2NrIiwibmFtZSIsImlucHV0IiwiZmlsZV9wYXRoIiwidXNlTWVtb3J5U3VydmV5IiwiaXNMb2FkaW5nIiwiaGFzQWN0aXZlUHJvbXB0IiwiZW5hYmxlZCIsInN0YXRlIiwibGFzdFJlc3BvbnNlIiwiaGFuZGxlU2VsZWN0Iiwic2VsZWN0ZWQiLCJoYW5kbGVUcmFuc2NyaXB0U2VsZWN0Iiwic2VlbkFzc2lzdGFudFV1aWRzIiwiU2V0IiwibWVtb3J5UmVhZFNlZW4iLCJtZXNzYWdlc1JlZiIsImN1cnJlbnQiLCJvbk9wZW4iLCJhcHBlYXJhbmNlSWQiLCJldmVudF90eXBlIiwiYXBwZWFyYW5jZV9pZCIsInN1cnZleV90eXBlIiwib25TZWxlY3QiLCJyZXNwb25zZSIsInNob3VsZFNob3dUcmFuc2NyaXB0UHJvbXB0IiwidHJhbnNjcmlwdFNoYXJlRGlzbWlzc2VkIiwib25UcmFuc2NyaXB0UHJvbXB0U2hvd24iLCJ0cmlnZ2VyIiwib25UcmFuc2NyaXB0U2VsZWN0IiwiUHJvbWlzZSIsInJlc3VsdCIsInN1Y2Nlc3MiLCJvcGVuIiwiaGlkZVRoYW5rc0FmdGVyTXMiLCJsYXN0QXNzaXN0YW50IiwibGVuZ3RoIiwiY2xlYXIiLCJwcm9jZXNzIiwiZW52IiwiQ0xBVURFX0NPREVfRElTQUJMRV9GRUVEQkFDS19TVVJWRVkiLCJoYXMiLCJ1dWlkIiwidGV4dCIsInRlc3QiLCJhZGQiLCJNYXRoIiwicmFuZG9tIl0sInNvdXJjZXMiOlsidXNlTWVtb3J5U3VydmV5LnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyB1c2VDYWxsYmFjaywgdXNlRWZmZWN0LCB1c2VNZW1vLCB1c2VSZWYgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IGlzRmVlZGJhY2tTdXJ2ZXlEaXNhYmxlZCB9IGZyb20gJ3NyYy9zZXJ2aWNlcy9hbmFseXRpY3MvY29uZmlnLmpzJ1xuaW1wb3J0IHsgZ2V0RmVhdHVyZVZhbHVlX0NBQ0hFRF9NQVlfQkVfU1RBTEUgfSBmcm9tICdzcmMvc2VydmljZXMvYW5hbHl0aWNzL2dyb3d0aGJvb2suanMnXG5pbXBvcnQge1xuICB0eXBlIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gIGxvZ0V2ZW50LFxufSBmcm9tICdzcmMvc2VydmljZXMvYW5hbHl0aWNzL2luZGV4LmpzJ1xuaW1wb3J0IHsgaXNBdXRvTWVtb3J5RW5hYmxlZCB9IGZyb20gJy4uLy4uL21lbWRpci9wYXRocy5qcydcbmltcG9ydCB7IGlzUG9saWN5QWxsb3dlZCB9IGZyb20gJy4uLy4uL3NlcnZpY2VzL3BvbGljeUxpbWl0cy9pbmRleC5qcydcbmltcG9ydCB7IEZJTEVfUkVBRF9UT09MX05BTUUgfSBmcm9tICcuLi8uLi90b29scy9GaWxlUmVhZFRvb2wvcHJvbXB0LmpzJ1xuaW1wb3J0IHR5cGUgeyBNZXNzYWdlIH0gZnJvbSAnLi4vLi4vdHlwZXMvbWVzc2FnZS5qcydcbmltcG9ydCB7IGdldEdsb2JhbENvbmZpZywgc2F2ZUdsb2JhbENvbmZpZyB9IGZyb20gJy4uLy4uL3V0aWxzL2NvbmZpZy5qcydcbmltcG9ydCB7IGlzRW52VHJ1dGh5IH0gZnJvbSAnLi4vLi4vdXRpbHMvZW52VXRpbHMuanMnXG5pbXBvcnQgeyBpc0F1dG9NYW5hZ2VkTWVtb3J5RmlsZSB9IGZyb20gJy4uLy4uL3V0aWxzL21lbW9yeUZpbGVEZXRlY3Rpb24uanMnXG5pbXBvcnQge1xuICBleHRyYWN0VGV4dENvbnRlbnQsXG4gIGdldExhc3RBc3Npc3RhbnRNZXNzYWdlLFxufSBmcm9tICcuLi8uLi91dGlscy9tZXNzYWdlcy5qcydcbmltcG9ydCB7IGxvZ09UZWxFdmVudCB9IGZyb20gJy4uLy4uL3V0aWxzL3RlbGVtZXRyeS9ldmVudHMuanMnXG5pbXBvcnQgeyBzdWJtaXRUcmFuc2NyaXB0U2hhcmUgfSBmcm9tICcuL3N1Ym1pdFRyYW5zY3JpcHRTaGFyZS5qcydcbmltcG9ydCB0eXBlIHsgVHJhbnNjcmlwdFNoYXJlUmVzcG9uc2UgfSBmcm9tICcuL1RyYW5zY3JpcHRTaGFyZVByb21wdC5qcydcbmltcG9ydCB7IHVzZVN1cnZleVN0YXRlIH0gZnJvbSAnLi91c2VTdXJ2ZXlTdGF0ZS5qcydcbmltcG9ydCB0eXBlIHsgRmVlZGJhY2tTdXJ2ZXlSZXNwb25zZSB9IGZyb20gJy4vdXRpbHMuanMnXG5cbmNvbnN0IEhJREVfVEhBTktTX0FGVEVSX01TID0gMzAwMFxuY29uc3QgTUVNT1JZX1NVUlZFWV9HQVRFID0gJ3Rlbmd1X2R1bndpY2hfYmVsbCdcbmNvbnN0IE1FTU9SWV9TVVJWRVlfRVZFTlQgPSAndGVuZ3VfbWVtb3J5X3N1cnZleV9ldmVudCdcbmNvbnN0IFNVUlZFWV9QUk9CQUJJTElUWSA9IDAuMlxuY29uc3QgVFJBTlNDUklQVF9TSEFSRV9UUklHR0VSID0gJ21lbW9yeV9zdXJ2ZXknXG5cbmNvbnN0IE1FTU9SWV9XT1JEX1JFID0gL1xcYm1lbW9yKD86eXxpZXMpXFxiL2lcblxuZnVuY3Rpb24gaGFzTWVtb3J5RmlsZVJlYWQobWVzc2FnZXM6IE1lc3NhZ2VbXSk6IGJvb2xlYW4ge1xuICBmb3IgKGNvbnN0IG1lc3NhZ2Ugb2YgbWVzc2FnZXMpIHtcbiAgICBpZiAobWVzc2FnZS50eXBlICE9PSAnYXNzaXN0YW50Jykge1xuICAgICAgY29udGludWVcbiAgICB9XG4gICAgY29uc3QgY29udGVudCA9IG1lc3NhZ2UubWVzc2FnZS5jb250ZW50XG4gICAgaWYgKCFBcnJheS5pc0FycmF5KGNvbnRlbnQpKSB7XG4gICAgICBjb250aW51ZVxuICAgIH1cbiAgICBmb3IgKGNvbnN0IGJsb2NrIG9mIGNvbnRlbnQpIHtcbiAgICAgIGlmIChibG9jay50eXBlICE9PSAndG9vbF91c2UnIHx8IGJsb2NrLm5hbWUgIT09IEZJTEVfUkVBRF9UT09MX05BTUUpIHtcbiAgICAgICAgY29udGludWVcbiAgICAgIH1cbiAgICAgIGNvbnN0IGlucHV0ID0gYmxvY2suaW5wdXQgYXMgeyBmaWxlX3BhdGg/OiB1bmtub3duIH1cbiAgICAgIGlmIChcbiAgICAgICAgdHlwZW9mIGlucHV0LmZpbGVfcGF0aCA9PT0gJ3N0cmluZycgJiZcbiAgICAgICAgaXNBdXRvTWFuYWdlZE1lbW9yeUZpbGUoaW5wdXQuZmlsZV9wYXRoKVxuICAgICAgKSB7XG4gICAgICAgIHJldHVybiB0cnVlXG4gICAgICB9XG4gICAgfVxuICB9XG4gIHJldHVybiBmYWxzZVxufVxuXG5leHBvcnQgZnVuY3Rpb24gdXNlTWVtb3J5U3VydmV5KFxuICBtZXNzYWdlczogTWVzc2FnZVtdLFxuICBpc0xvYWRpbmc6IGJvb2xlYW4sXG4gIGhhc0FjdGl2ZVByb21wdCA9IGZhbHNlLFxuICB7IGVuYWJsZWQgPSB0cnVlIH06IHsgZW5hYmxlZD86IGJvb2xlYW4gfSA9IHt9LFxuKToge1xuICBzdGF0ZTpcbiAgICB8ICdjbG9zZWQnXG4gICAgfCAnb3BlbidcbiAgICB8ICd0aGFua3MnXG4gICAgfCAndHJhbnNjcmlwdF9wcm9tcHQnXG4gICAgfCAnc3VibWl0dGluZydcbiAgICB8ICdzdWJtaXR0ZWQnXG4gIGxhc3RSZXNwb25zZTogRmVlZGJhY2tTdXJ2ZXlSZXNwb25zZSB8IG51bGxcbiAgaGFuZGxlU2VsZWN0OiAoc2VsZWN0ZWQ6IEZlZWRiYWNrU3VydmV5UmVzcG9uc2UpID0+IHZvaWRcbiAgaGFuZGxlVHJhbnNjcmlwdFNlbGVjdDogKHNlbGVjdGVkOiBUcmFuc2NyaXB0U2hhcmVSZXNwb25zZSkgPT4gdm9pZFxufSB7XG4gIC8vIFRyYWNrIGFzc2lzdGFudCBtZXNzYWdlIFVVSURzIHRoYXQgd2VyZSBhbHJlYWR5IGV2YWx1YXRlZCBzbyB3ZSBkb24ndFxuICAvLyByZS1yb2xsIHByb2JhYmlsaXR5IG9uIHJlLXJlbmRlcnMgb3IgcmUtc2NhbiBtZXNzYWdlcyBmb3IgdGhlIHNhbWUgdHVybi5cbiAgY29uc3Qgc2VlbkFzc2lzdGFudFV1aWRzID0gdXNlUmVmPFNldDxzdHJpbmc+PihuZXcgU2V0KCkpXG4gIC8vIE9uY2UgYSBtZW1vcnkgZmlsZSByZWFkIGlzIG9ic2VydmVkIGl0IHN0YXlzIHRydWUgZm9yIHRoZSBzZXNzaW9uIOKAlFxuICAvLyBza2lwIHRoZSBPKG4pIHNjYW4gb24gc3Vic2VxdWVudCB0dXJucy5cbiAgY29uc3QgbWVtb3J5UmVhZFNlZW4gPSB1c2VSZWYoZmFsc2UpXG4gIGNvbnN0IG1lc3NhZ2VzUmVmID0gdXNlUmVmKG1lc3NhZ2VzKVxuICBtZXNzYWdlc1JlZi5jdXJyZW50ID0gbWVzc2FnZXNcblxuICBjb25zdCBvbk9wZW4gPSB1c2VDYWxsYmFjaygoYXBwZWFyYW5jZUlkOiBzdHJpbmcpID0+IHtcbiAgICBsb2dFdmVudChNRU1PUllfU1VSVkVZX0VWRU5ULCB7XG4gICAgICBldmVudF90eXBlOlxuICAgICAgICAnYXBwZWFyZWQnIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICBhcHBlYXJhbmNlX2lkOlxuICAgICAgICBhcHBlYXJhbmNlSWQgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICB9KVxuICAgIHZvaWQgbG9nT1RlbEV2ZW50KCdmZWVkYmFja19zdXJ2ZXknLCB7XG4gICAgICBldmVudF90eXBlOiAnYXBwZWFyZWQnLFxuICAgICAgYXBwZWFyYW5jZV9pZDogYXBwZWFyYW5jZUlkLFxuICAgICAgc3VydmV5X3R5cGU6ICdtZW1vcnknLFxuICAgIH0pXG4gIH0sIFtdKVxuXG4gIGNvbnN0IG9uU2VsZWN0ID0gdXNlQ2FsbGJhY2soXG4gICAgKGFwcGVhcmFuY2VJZDogc3RyaW5nLCBzZWxlY3RlZDogRmVlZGJhY2tTdXJ2ZXlSZXNwb25zZSkgPT4ge1xuICAgICAgbG9nRXZlbnQoTUVNT1JZX1NVUlZFWV9FVkVOVCwge1xuICAgICAgICBldmVudF90eXBlOlxuICAgICAgICAgICdyZXNwb25kZWQnIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgIGFwcGVhcmFuY2VfaWQ6XG4gICAgICAgICAgYXBwZWFyYW5jZUlkIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgIHJlc3BvbnNlOlxuICAgICAgICAgIHNlbGVjdGVkIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICB9KVxuICAgICAgdm9pZCBsb2dPVGVsRXZlbnQoJ2ZlZWRiYWNrX3N1cnZleScsIHtcbiAgICAgICAgZXZlbnRfdHlwZTogJ3Jlc3BvbmRlZCcsXG4gICAgICAgIGFwcGVhcmFuY2VfaWQ6IGFwcGVhcmFuY2VJZCxcbiAgICAgICAgcmVzcG9uc2U6IHNlbGVjdGVkLFxuICAgICAgICBzdXJ2ZXlfdHlwZTogJ21lbW9yeScsXG4gICAgICB9KVxuICAgIH0sXG4gICAgW10sXG4gIClcblxuICBjb25zdCBzaG91bGRTaG93VHJhbnNjcmlwdFByb21wdCA9IHVzZUNhbGxiYWNrKFxuICAgIChzZWxlY3RlZDogRmVlZGJhY2tTdXJ2ZXlSZXNwb25zZSkgPT4ge1xuICAgICAgaWYgKFwiZXh0ZXJuYWxcIiAhPT0gJ2FudCcpIHtcbiAgICAgICAgcmV0dXJuIGZhbHNlXG4gICAgICB9XG4gICAgICBpZiAoc2VsZWN0ZWQgIT09ICdiYWQnICYmIHNlbGVjdGVkICE9PSAnZ29vZCcpIHtcbiAgICAgICAgcmV0dXJuIGZhbHNlXG4gICAgICB9XG4gICAgICBpZiAoZ2V0R2xvYmFsQ29uZmlnKCkudHJhbnNjcmlwdFNoYXJlRGlzbWlzc2VkKSB7XG4gICAgICAgIHJldHVybiBmYWxzZVxuICAgICAgfVxuICAgICAgaWYgKCFpc1BvbGljeUFsbG93ZWQoJ2FsbG93X3Byb2R1Y3RfZmVlZGJhY2snKSkge1xuICAgICAgICByZXR1cm4gZmFsc2VcbiAgICAgIH1cbiAgICAgIHJldHVybiB0cnVlXG4gICAgfSxcbiAgICBbXSxcbiAgKVxuXG4gIGNvbnN0IG9uVHJhbnNjcmlwdFByb21wdFNob3duID0gdXNlQ2FsbGJhY2soKGFwcGVhcmFuY2VJZDogc3RyaW5nKSA9PiB7XG4gICAgbG9nRXZlbnQoTUVNT1JZX1NVUlZFWV9FVkVOVCwge1xuICAgICAgZXZlbnRfdHlwZTpcbiAgICAgICAgJ3RyYW5zY3JpcHRfcHJvbXB0X2FwcGVhcmVkJyBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgYXBwZWFyYW5jZV9pZDpcbiAgICAgICAgYXBwZWFyYW5jZUlkIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICB0cmlnZ2VyOlxuICAgICAgICBUUkFOU0NSSVBUX1NIQVJFX1RSSUdHRVIgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICB9KVxuICAgIHZvaWQgbG9nT1RlbEV2ZW50KCdmZWVkYmFja19zdXJ2ZXknLCB7XG4gICAgICBldmVudF90eXBlOiAndHJhbnNjcmlwdF9wcm9tcHRfYXBwZWFyZWQnLFxuICAgICAgYXBwZWFyYW5jZV9pZDogYXBwZWFyYW5jZUlkLFxuICAgICAgc3VydmV5X3R5cGU6ICdtZW1vcnknLFxuICAgIH0pXG4gIH0sIFtdKVxuXG4gIGNvbnN0IG9uVHJhbnNjcmlwdFNlbGVjdCA9IHVzZUNhbGxiYWNrKFxuICAgIGFzeW5jIChcbiAgICAgIGFwcGVhcmFuY2VJZDogc3RyaW5nLFxuICAgICAgc2VsZWN0ZWQ6IFRyYW5zY3JpcHRTaGFyZVJlc3BvbnNlLFxuICAgICk6IFByb21pc2U8Ym9vbGVhbj4gPT4ge1xuICAgICAgbG9nRXZlbnQoTUVNT1JZX1NVUlZFWV9FVkVOVCwge1xuICAgICAgICBldmVudF90eXBlOlxuICAgICAgICAgIGB0cmFuc2NyaXB0X3NoYXJlXyR7c2VsZWN0ZWR9YCBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICBhcHBlYXJhbmNlX2lkOlxuICAgICAgICAgIGFwcGVhcmFuY2VJZCBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICB0cmlnZ2VyOlxuICAgICAgICAgIFRSQU5TQ1JJUFRfU0hBUkVfVFJJR0dFUiBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgfSlcblxuICAgICAgaWYgKHNlbGVjdGVkID09PSAnZG9udF9hc2tfYWdhaW4nKSB7XG4gICAgICAgIHNhdmVHbG9iYWxDb25maWcoY3VycmVudCA9PiAoe1xuICAgICAgICAgIC4uLmN1cnJlbnQsXG4gICAgICAgICAgdHJhbnNjcmlwdFNoYXJlRGlzbWlzc2VkOiB0cnVlLFxuICAgICAgICB9KSlcbiAgICAgIH1cblxuICAgICAgaWYgKHNlbGVjdGVkID09PSAneWVzJykge1xuICAgICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCBzdWJtaXRUcmFuc2NyaXB0U2hhcmUoXG4gICAgICAgICAgbWVzc2FnZXNSZWYuY3VycmVudCxcbiAgICAgICAgICBUUkFOU0NSSVBUX1NIQVJFX1RSSUdHRVIsXG4gICAgICAgICAgYXBwZWFyYW5jZUlkLFxuICAgICAgICApXG4gICAgICAgIGxvZ0V2ZW50KE1FTU9SWV9TVVJWRVlfRVZFTlQsIHtcbiAgICAgICAgICBldmVudF90eXBlOiAocmVzdWx0LnN1Y2Nlc3NcbiAgICAgICAgICAgID8gJ3RyYW5zY3JpcHRfc2hhcmVfc3VibWl0dGVkJ1xuICAgICAgICAgICAgOiAndHJhbnNjcmlwdF9zaGFyZV9mYWlsZWQnKSBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICAgIGFwcGVhcmFuY2VfaWQ6XG4gICAgICAgICAgICBhcHBlYXJhbmNlSWQgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICAgICAgICB0cmlnZ2VyOlxuICAgICAgICAgICAgVFJBTlNDUklQVF9TSEFSRV9UUklHR0VSIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgIH0pXG4gICAgICAgIHJldHVybiByZXN1bHQuc3VjY2Vzc1xuICAgICAgfVxuXG4gICAgICByZXR1cm4gZmFsc2VcbiAgICB9LFxuICAgIFtdLFxuICApXG5cbiAgY29uc3QgeyBzdGF0ZSwgbGFzdFJlc3BvbnNlLCBvcGVuLCBoYW5kbGVTZWxlY3QsIGhhbmRsZVRyYW5zY3JpcHRTZWxlY3QgfSA9XG4gICAgdXNlU3VydmV5U3RhdGUoe1xuICAgICAgaGlkZVRoYW5rc0FmdGVyTXM6IEhJREVfVEhBTktTX0FGVEVSX01TLFxuICAgICAgb25PcGVuLFxuICAgICAgb25TZWxlY3QsXG4gICAgICBzaG91bGRTaG93VHJhbnNjcmlwdFByb21wdCxcbiAgICAgIG9uVHJhbnNjcmlwdFByb21wdFNob3duLFxuICAgICAgb25UcmFuc2NyaXB0U2VsZWN0LFxuICAgIH0pXG5cbiAgY29uc3QgbGFzdEFzc2lzdGFudCA9IHVzZU1lbW8oXG4gICAgKCkgPT4gZ2V0TGFzdEFzc2lzdGFudE1lc3NhZ2UobWVzc2FnZXMpLFxuICAgIFttZXNzYWdlc10sXG4gIClcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmICghZW5hYmxlZCkgcmV0dXJuXG5cbiAgICAvLyAvY2xlYXIgcmVzZXRzIG1lc3NhZ2VzIGJ1dCBSRVBMIHN0YXlzIG1vdW50ZWQg4oCUIHJlc2V0IHJlZnMgc28gYSBtZW1vcnlcbiAgICAvLyByZWFkIGZyb20gdGhlIHByZXZpb3VzIGNvbnZlcnNhdGlvbiBkb2Vzbid0IGxlYWsgaW50byB0aGUgbmV3IG9uZS5cbiAgICBpZiAobWVzc2FnZXMubGVuZ3RoID09PSAwKSB7XG4gICAgICBtZW1vcnlSZWFkU2Vlbi5jdXJyZW50ID0gZmFsc2VcbiAgICAgIHNlZW5Bc3Npc3RhbnRVdWlkcy5jdXJyZW50LmNsZWFyKClcbiAgICAgIHJldHVyblxuICAgIH1cblxuICAgIGlmIChzdGF0ZSAhPT0gJ2Nsb3NlZCcgfHwgaXNMb2FkaW5nIHx8IGhhc0FjdGl2ZVByb21wdCkge1xuICAgICAgcmV0dXJuXG4gICAgfVxuXG4gICAgLy8gM1AgZGVmYXVsdDogc3VydmV5IG9mZiAobm8gR3Jvd3RoQm9vayBvbiBCZWRyb2NrL1ZlcnRleC9Gb3VuZHJ5KS5cbiAgICBpZiAoIWdldEZlYXR1cmVWYWx1ZV9DQUNIRURfTUFZX0JFX1NUQUxFKE1FTU9SWV9TVVJWRVlfR0FURSwgZmFsc2UpKSB7XG4gICAgICByZXR1cm5cbiAgICB9XG5cbiAgICBpZiAoIWlzQXV0b01lbW9yeUVuYWJsZWQoKSkge1xuICAgICAgcmV0dXJuXG4gICAgfVxuXG4gICAgaWYgKGlzRmVlZGJhY2tTdXJ2ZXlEaXNhYmxlZCgpKSB7XG4gICAgICByZXR1cm5cbiAgICB9XG5cbiAgICBpZiAoIWlzUG9saWN5QWxsb3dlZCgnYWxsb3dfcHJvZHVjdF9mZWVkYmFjaycpKSB7XG4gICAgICByZXR1cm5cbiAgICB9XG5cbiAgICBpZiAoaXNFbnZUcnV0aHkocHJvY2Vzcy5lbnYuQ0xBVURFX0NPREVfRElTQUJMRV9GRUVEQkFDS19TVVJWRVkpKSB7XG4gICAgICByZXR1cm5cbiAgICB9XG5cbiAgICBpZiAoIWxhc3RBc3Npc3RhbnQgfHwgc2VlbkFzc2lzdGFudFV1aWRzLmN1cnJlbnQuaGFzKGxhc3RBc3Npc3RhbnQudXVpZCkpIHtcbiAgICAgIHJldHVyblxuICAgIH1cblxuICAgIGNvbnN0IHRleHQgPSBleHRyYWN0VGV4dENvbnRlbnQobGFzdEFzc2lzdGFudC5tZXNzYWdlLmNvbnRlbnQsICcgJylcbiAgICBpZiAoIU1FTU9SWV9XT1JEX1JFLnRlc3QodGV4dCkpIHtcbiAgICAgIHJldHVyblxuICAgIH1cblxuICAgIC8vIE1hcmsgYXMgZXZhbHVhdGVkIGJlZm9yZSB0aGUgbWVtb3J5LXJlYWQgc2NhbiBzbyBhIHR1cm4gdGhhdCBtZW50aW9uc1xuICAgIC8vIFwibWVtb3J5XCIgYnV0IGhhcyBubyBtZW1vcnkgcmVhZCBkb2Vzbid0IHRyaWdnZXIgcmVwZWF0ZWQgTyhuKSBzY2Fuc1xuICAgIC8vIG9uIHN1YnNlcXVlbnQgcmVuZGVycyB3aXRoIHRoZSBzYW1lIGxhc3QgYXNzaXN0YW50IG1lc3NhZ2UuXG4gICAgc2VlbkFzc2lzdGFudFV1aWRzLmN1cnJlbnQuYWRkKGxhc3RBc3Npc3RhbnQudXVpZClcblxuICAgIGlmICghbWVtb3J5UmVhZFNlZW4uY3VycmVudCkge1xuICAgICAgbWVtb3J5UmVhZFNlZW4uY3VycmVudCA9IGhhc01lbW9yeUZpbGVSZWFkKG1lc3NhZ2VzKVxuICAgIH1cbiAgICBpZiAoIW1lbW9yeVJlYWRTZWVuLmN1cnJlbnQpIHtcbiAgICAgIHJldHVyblxuICAgIH1cblxuICAgIGlmIChNYXRoLnJhbmRvbSgpIDwgU1VSVkVZX1BST0JBQklMSVRZKSB7XG4gICAgICBvcGVuKClcbiAgICB9XG4gIH0sIFtcbiAgICBlbmFibGVkLFxuICAgIHN0YXRlLFxuICAgIGlzTG9hZGluZyxcbiAgICBoYXNBY3RpdmVQcm9tcHQsXG4gICAgbGFzdEFzc2lzdGFudCxcbiAgICBtZXNzYWdlcyxcbiAgICBvcGVuLFxuICBdKVxuXG4gIHJldHVybiB7IHN0YXRlLCBsYXN0UmVzcG9uc2UsIGhhbmRsZVNlbGVjdCwgaGFuZGxlVHJhbnNjcmlwdFNlbGVjdCB9XG59XG4iXSwibWFwcGluZ3MiOiJBQUFBLFNBQVNBLFdBQVcsRUFBRUMsU0FBUyxFQUFFQyxPQUFPLEVBQUVDLE1BQU0sUUFBUSxPQUFPO0FBQy9ELFNBQVNDLHdCQUF3QixRQUFRLGtDQUFrQztBQUMzRSxTQUFTQyxtQ0FBbUMsUUFBUSxzQ0FBc0M7QUFDMUYsU0FDRSxLQUFLQywwREFBMEQsRUFDL0RDLFFBQVEsUUFDSCxpQ0FBaUM7QUFDeEMsU0FBU0MsbUJBQW1CLFFBQVEsdUJBQXVCO0FBQzNELFNBQVNDLGVBQWUsUUFBUSxzQ0FBc0M7QUFDdEUsU0FBU0MsbUJBQW1CLFFBQVEsb0NBQW9DO0FBQ3hFLGNBQWNDLE9BQU8sUUFBUSx3QkFBd0I7QUFDckQs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O0lBQ0Y7SUFFQSxJQUFJN0IsS0FBSyxLQUFLLFFBQVEsSUFBSUgsU0FBUyxJQUFJQyxlQUFlLEVBQUU7TUFDdEQ7SUFDRjs7SUFFQTtJQUNBLElBQUksQ0FBQ3JDLG1DQUFtQyxDQUFDbUIsa0JBQWtCLEVBQUUsS0FBSyxDQUFDLEVBQUU7TUFDbkU7SUFDRjtJQUVBLElBQUksQ0FBQ2hCLG1CQUFtQixDQUFDLENBQUMsRUFBRTtNQUMxQjtJQUNGO0lBRUEsSUFBSUosd0JBQXdCLENBQUMsQ0FBQyxFQUFFO01BQzlCO0lBQ0Y7SUFFQSxJQUFJLENBQUNLLGVBQWUsQ0FBQyx3QkFBd0IsQ0FBQyxFQUFFO01BQzlDO0lBQ0Y7SUFFQSxJQUFJSyxXQUFXLENBQUM0RCxPQUFPLENBQUNDLEdBQUcsQ0FBQ0MsbUNBQW1DLENBQUMsRUFBRTtNQUNoRTtJQUNGO0lBRUEsSUFBSSxDQUFDTCxhQUFhLElBQUl0QixrQkFBa0IsQ0FBQ0ksT0FBTyxDQUFDd0IsR0FBRyxDQUFDTixhQUFhLENBQUNPLElBQUksQ0FBQyxFQUFFO01BQ3hFO0lBQ0Y7SUFFQSxNQUFNQyxJQUFJLEdBQUcvRCxrQkFBa0IsQ0FBQ3VELGFBQWEsQ0FBQ3hDLE9BQU8sQ0FBQ0UsT0FBTyxFQUFFLEdBQUcsQ0FBQztJQUNuRSxJQUFJLENBQUNMLGNBQWMsQ0FBQ29ELElBQUksQ0FBQ0QsSUFBSSxDQUFDLEVBQUU7TUFDOUI7SUFDRjs7SUFFQTtJQUNBO0lBQ0E7SUFDQTlCLGtCQUFrQixDQUFDSSxPQUFPLENBQUM0QixHQUFHLENBQUNWLGFBQWEsQ0FBQ08sSUFBSSxDQUFDO0lBRWxELElBQUksQ0FBQzNCLGNBQWMsQ0FBQ0UsT0FBTyxFQUFFO01BQzNCRixjQUFjLENBQUNFLE9BQU8sR0FBR3hCLGlCQUFpQixDQUFDQyxRQUFRLENBQUM7SUFDdEQ7SUFDQSxJQUFJLENBQUNxQixjQUFjLENBQUNFLE9BQU8sRUFBRTtNQUMzQjtJQUNGO0lBRUEsSUFBSTZCLElBQUksQ0FBQ0MsTUFBTSxDQUFDLENBQUMsR0FBR3pELGtCQUFrQixFQUFFO01BQ3RDMkMsSUFBSSxDQUFDLENBQUM7SUFDUjtFQUNGLENBQUMsRUFBRSxDQUNEMUIsT0FBTyxFQUNQQyxLQUFLLEVBQ0xILFNBQVMsRUFDVEMsZUFBZSxFQUNmNkIsYUFBYSxFQUNiekMsUUFBUSxFQUNSdUMsSUFBSSxDQUNMLENBQUM7RUFFRixPQUFPO0lBQUV6QixLQUFLO0lBQUVDLFlBQVk7SUFBRUMsWUFBWTtJQUFFRTtFQUF1QixDQUFDO0FBQ3RFIiwiaWdub3JlTGlzdCI6W119 \ No newline at end of file diff --git a/src/components/FeedbackSurvey/usePostCompactSurvey.tsx b/src/components/FeedbackSurvey/usePostCompactSurvey.tsx new file mode 100644 index 0000000..b33281a --- /dev/null +++ b/src/components/FeedbackSurvey/usePostCompactSurvey.tsx @@ -0,0 +1,206 @@ +import { c as _c } from "react/compiler-runtime"; +import { useCallback, useEffect, useMemo, useRef, useState } from 'react'; +import { isFeedbackSurveyDisabled } from 'src/services/analytics/config.js'; +import { checkStatsigFeatureGate_CACHED_MAY_BE_STALE } from 'src/services/analytics/growthbook.js'; +import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from 'src/services/analytics/index.js'; +import { shouldUseSessionMemoryCompaction } from '../../services/compact/sessionMemoryCompact.js'; +import type { Message } from '../../types/message.js'; +import { isEnvTruthy } from '../../utils/envUtils.js'; +import { isCompactBoundaryMessage } from '../../utils/messages.js'; +import { logOTelEvent } from '../../utils/telemetry/events.js'; +import { useSurveyState } from './useSurveyState.js'; +import type { FeedbackSurveyResponse } from './utils.js'; +const HIDE_THANKS_AFTER_MS = 3000; +const POST_COMPACT_SURVEY_GATE = 'tengu_post_compact_survey'; +const SURVEY_PROBABILITY = 0.2; // Show survey 20% of the time after compaction + +function hasMessageAfterBoundary(messages: Message[], boundaryUuid: string): boolean { + const boundaryIndex = messages.findIndex(msg => msg.uuid === boundaryUuid); + if (boundaryIndex === -1) { + return false; + } + + // Check if there's a user or assistant message after the boundary + for (let i = boundaryIndex + 1; i < messages.length; i++) { + const msg = messages[i]; + if (msg && (msg.type === 'user' || msg.type === 'assistant')) { + return true; + } + } + return false; +} +export function usePostCompactSurvey(messages, isLoading, t0, t1) { + const $ = _c(23); + const hasActivePrompt = t0 === undefined ? false : t0; + let t2; + if ($[0] !== t1) { + t2 = t1 === undefined ? {} : t1; + $[0] = t1; + $[1] = t2; + } else { + t2 = $[1]; + } + const { + enabled: t3 + } = t2; + const enabled = t3 === undefined ? true : t3; + const [gateEnabled, setGateEnabled] = useState(null); + let t4; + if ($[2] === Symbol.for("react.memo_cache_sentinel")) { + t4 = new Set(); + $[2] = t4; + } else { + t4 = $[2]; + } + const seenCompactBoundaries = useRef(t4); + const pendingCompactBoundaryUuid = useRef(null); + const onOpen = _temp; + const onSelect = _temp2; + let t5; + if ($[3] === Symbol.for("react.memo_cache_sentinel")) { + t5 = { + hideThanksAfterMs: HIDE_THANKS_AFTER_MS, + onOpen, + onSelect + }; + $[3] = t5; + } else { + t5 = $[3]; + } + const { + state, + lastResponse, + open, + handleSelect + } = useSurveyState(t5); + let t6; + let t7; + if ($[4] !== enabled) { + t6 = () => { + if (!enabled) { + return; + } + setGateEnabled(checkStatsigFeatureGate_CACHED_MAY_BE_STALE(POST_COMPACT_SURVEY_GATE)); + }; + t7 = [enabled]; + $[4] = enabled; + $[5] = t6; + $[6] = t7; + } else { + t6 = $[5]; + t7 = $[6]; + } + useEffect(t6, t7); + let t8; + if ($[7] !== messages) { + t8 = new Set(messages.filter(_temp3).map(_temp4)); + $[7] = messages; + $[8] = t8; + } else { + t8 = $[8]; + } + const currentCompactBoundaries = t8; + let t10; + let t9; + if ($[9] !== currentCompactBoundaries || $[10] !== enabled || $[11] !== gateEnabled || $[12] !== hasActivePrompt || $[13] !== isLoading || $[14] !== messages || $[15] !== open || $[16] !== state) { + t9 = () => { + if (!enabled) { + return; + } + if (state !== "closed" || isLoading) { + return; + } + if (hasActivePrompt) { + return; + } + if (gateEnabled !== true) { + return; + } + if (isFeedbackSurveyDisabled()) { + return; + } + if (isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_FEEDBACK_SURVEY)) { + return; + } + if (pendingCompactBoundaryUuid.current !== null) { + if (hasMessageAfterBoundary(messages, pendingCompactBoundaryUuid.current)) { + pendingCompactBoundaryUuid.current = null; + if (Math.random() < SURVEY_PROBABILITY) { + open(); + } + return; + } + } + const newBoundaries = Array.from(currentCompactBoundaries).filter(uuid => !seenCompactBoundaries.current.has(uuid)); + if (newBoundaries.length > 0) { + seenCompactBoundaries.current = new Set(currentCompactBoundaries); + pendingCompactBoundaryUuid.current = newBoundaries[newBoundaries.length - 1]; + } + }; + t10 = [enabled, currentCompactBoundaries, state, isLoading, hasActivePrompt, gateEnabled, messages, open]; + $[9] = currentCompactBoundaries; + $[10] = enabled; + $[11] = gateEnabled; + $[12] = hasActivePrompt; + $[13] = isLoading; + $[14] = messages; + $[15] = open; + $[16] = state; + $[17] = t10; + $[18] = t9; + } else { + t10 = $[17]; + t9 = $[18]; + } + useEffect(t9, t10); + let t11; + if ($[19] !== handleSelect || $[20] !== lastResponse || $[21] !== state) { + t11 = { + state, + lastResponse, + handleSelect + }; + $[19] = handleSelect; + $[20] = lastResponse; + $[21] = state; + $[22] = t11; + } else { + t11 = $[22]; + } + return t11; +} +function _temp4(msg_0) { + return msg_0.uuid; +} +function _temp3(msg) { + return isCompactBoundaryMessage(msg); +} +function _temp2(appearanceId_0, selected) { + const smCompactionEnabled_0 = shouldUseSessionMemoryCompaction(); + logEvent("tengu_post_compact_survey_event", { + event_type: "responded" as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + appearance_id: appearanceId_0 as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + response: selected as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + session_memory_compaction_enabled: smCompactionEnabled_0 as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + logOTelEvent("feedback_survey", { + event_type: "responded", + appearance_id: appearanceId_0, + response: selected, + survey_type: "post_compact" + }); +} +function _temp(appearanceId) { + const smCompactionEnabled = shouldUseSessionMemoryCompaction(); + logEvent("tengu_post_compact_survey_event", { + event_type: "appeared" as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + appearance_id: appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, + session_memory_compaction_enabled: smCompactionEnabled as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS + }); + logOTelEvent("feedback_survey", { + event_type: "appeared", + appearance_id: appearanceId, + survey_type: "post_compact" + }); +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJ1c2VDYWxsYmFjayIsInVzZUVmZmVjdCIsInVzZU1lbW8iLCJ1c2VSZWYiLCJ1c2VTdGF0ZSIsImlzRmVlZGJhY2tTdXJ2ZXlEaXNhYmxlZCIsImNoZWNrU3RhdHNpZ0ZlYXR1cmVHYXRlX0NBQ0hFRF9NQVlfQkVfU1RBTEUiLCJBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTIiwibG9nRXZlbnQiLCJzaG91bGRVc2VTZXNzaW9uTWVtb3J5Q29tcGFjdGlvbiIsIk1lc3NhZ2UiLCJpc0VudlRydXRoeSIsImlzQ29tcGFjdEJvdW5kYXJ5TWVzc2FnZSIsImxvZ09UZWxFdmVudCIsInVzZVN1cnZleVN0YXRlIiwiRmVlZGJhY2tTdXJ2ZXlSZXNwb25zZSIsIkhJREVfVEhBTktTX0FGVEVSX01TIiwiUE9TVF9DT01QQUNUX1NVUlZFWV9HQVRFIiwiU1VSVkVZX1BST0JBQklMSVRZIiwiaGFzTWVzc2FnZUFmdGVyQm91bmRhcnkiLCJtZXNzYWdlcyIsImJvdW5kYXJ5VXVpZCIsImJvdW5kYXJ5SW5kZXgiLCJmaW5kSW5kZXgiLCJtc2ciLCJ1dWlkIiwiaSIsImxlbmd0aCIsInR5cGUiLCJ1c2VQb3N0Q29tcGFjdFN1cnZleSIsImlzTG9hZGluZyIsInQwIiwidDEiLCIkIiwiX2MiLCJoYXNBY3RpdmVQcm9tcHQiLCJ1bmRlZmluZWQiLCJ0MiIsImVuYWJsZWQiLCJ0MyIsImdhdGVFbmFibGVkIiwic2V0R2F0ZUVuYWJsZWQiLCJ0NCIsIlN5bWJvbCIsImZvciIsIlNldCIsInNlZW5Db21wYWN0Qm91bmRhcmllcyIsInBlbmRpbmdDb21wYWN0Qm91bmRhcnlVdWlkIiwib25PcGVuIiwiX3RlbXAiLCJvblNlbGVjdCIsIl90ZW1wMiIsInQ1IiwiaGlkZVRoYW5rc0FmdGVyTXMiLCJzdGF0ZSIsImxhc3RSZXNwb25zZSIsIm9wZW4iLCJoYW5kbGVTZWxlY3QiLCJ0NiIsInQ3IiwidDgiLCJmaWx0ZXIiLCJfdGVtcDMiLCJtYXAiLCJfdGVtcDQiLCJjdXJyZW50Q29tcGFjdEJvdW5kYXJpZXMiLCJ0MTAiLCJ0OSIsInByb2Nlc3MiLCJlbnYiLCJDTEFVREVfQ09ERV9ESVNBQkxFX0ZFRURCQUNLX1NVUlZFWSIsImN1cnJlbnQiLCJNYXRoIiwicmFuZG9tIiwibmV3Qm91bmRhcmllcyIsIkFycmF5IiwiZnJvbSIsImhhcyIsInQxMSIsIm1zZ18wIiwiYXBwZWFyYW5jZUlkXzAiLCJzZWxlY3RlZCIsInNtQ29tcGFjdGlvbkVuYWJsZWRfMCIsImV2ZW50X3R5cGUiLCJhcHBlYXJhbmNlX2lkIiwiYXBwZWFyYW5jZUlkIiwicmVzcG9uc2UiLCJzZXNzaW9uX21lbW9yeV9jb21wYWN0aW9uX2VuYWJsZWQiLCJzbUNvbXBhY3Rpb25FbmFibGVkIiwic3VydmV5X3R5cGUiXSwic291cmNlcyI6WyJ1c2VQb3N0Q29tcGFjdFN1cnZleS50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgdXNlQ2FsbGJhY2ssIHVzZUVmZmVjdCwgdXNlTWVtbywgdXNlUmVmLCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgaXNGZWVkYmFja1N1cnZleURpc2FibGVkIH0gZnJvbSAnc3JjL3NlcnZpY2VzL2FuYWx5dGljcy9jb25maWcuanMnXG5pbXBvcnQgeyBjaGVja1N0YXRzaWdGZWF0dXJlR2F0ZV9DQUNIRURfTUFZX0JFX1NUQUxFIH0gZnJvbSAnc3JjL3NlcnZpY2VzL2FuYWx5dGljcy9ncm93dGhib29rLmpzJ1xuaW1wb3J0IHtcbiAgdHlwZSBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICBsb2dFdmVudCxcbn0gZnJvbSAnc3JjL3NlcnZpY2VzL2FuYWx5dGljcy9pbmRleC5qcydcbmltcG9ydCB7IHNob3VsZFVzZVNlc3Npb25NZW1vcnlDb21wYWN0aW9uIH0gZnJvbSAnLi4vLi4vc2VydmljZXMvY29tcGFjdC9zZXNzaW9uTWVtb3J5Q29tcGFjdC5qcydcbmltcG9ydCB0eXBlIHsgTWVzc2FnZSB9IGZyb20gJy4uLy4uL3R5cGVzL21lc3NhZ2UuanMnXG5pbXBvcnQgeyBpc0VudlRydXRoeSB9IGZyb20gJy4uLy4uL3V0aWxzL2VudlV0aWxzLmpzJ1xuaW1wb3J0IHsgaXNDb21wYWN0Qm91bmRhcnlNZXNzYWdlIH0gZnJvbSAnLi4vLi4vdXRpbHMvbWVzc2FnZXMuanMnXG5pbXBvcnQgeyBsb2dPVGVsRXZlbnQgfSBmcm9tICcuLi8uLi91dGlscy90ZWxlbWV0cnkvZXZlbnRzLmpzJ1xuaW1wb3J0IHsgdXNlU3VydmV5U3RhdGUgfSBmcm9tICcuL3VzZVN1cnZleVN0YXRlLmpzJ1xuaW1wb3J0IHR5cGUgeyBGZWVkYmFja1N1cnZleVJlc3BvbnNlIH0gZnJvbSAnLi91dGlscy5qcydcblxuY29uc3QgSElERV9USEFOS1NfQUZURVJfTVMgPSAzMDAwXG5jb25zdCBQT1NUX0NPTVBBQ1RfU1VSVkVZX0dBVEUgPSAndGVuZ3VfcG9zdF9jb21wYWN0X3N1cnZleSdcbmNvbnN0IFNVUlZFWV9QUk9CQUJJTElUWSA9IDAuMiAvLyBTaG93IHN1cnZleSAyMCUgb2YgdGhlIHRpbWUgYWZ0ZXIgY29tcGFjdGlvblxuXG5mdW5jdGlvbiBoYXNNZXNzYWdlQWZ0ZXJCb3VuZGFyeShcbiAgbWVzc2FnZXM6IE1lc3NhZ2VbXSxcbiAgYm91bmRhcnlVdWlkOiBzdHJpbmcsXG4pOiBib29sZWFuIHtcbiAgY29uc3QgYm91bmRhcnlJbmRleCA9IG1lc3NhZ2VzLmZpbmRJbmRleChtc2cgPT4gbXNnLnV1aWQgPT09IGJvdW5kYXJ5VXVpZClcbiAgaWYgKGJvdW5kYXJ5SW5kZXggPT09IC0xKSB7XG4gICAgcmV0dXJuIGZhbHNlXG4gIH1cblxuICAvLyBDaGVjayBpZiB0aGVyZSdzIGEgdXNlciBvciBhc3Npc3RhbnQgbWVzc2FnZSBhZnRlciB0aGUgYm91bmRhcnlcbiAgZm9yIChsZXQgaSA9IGJvdW5kYXJ5SW5kZXggKyAxOyBpIDwgbWVzc2FnZXMubGVuZ3RoOyBpKyspIHtcbiAgICBjb25zdCBtc2cgPSBtZXNzYWdlc1tpXVxuICAgIGlmIChtc2cgJiYgKG1zZy50eXBlID09PSAndXNlcicgfHwgbXNnLnR5cGUgPT09ICdhc3Npc3RhbnQnKSkge1xuICAgICAgcmV0dXJuIHRydWVcbiAgICB9XG4gIH1cbiAgcmV0dXJuIGZhbHNlXG59XG5cbmV4cG9ydCBmdW5jdGlvbiB1c2VQb3N0Q29tcGFjdFN1cnZleShcbiAgbWVzc2FnZXM6IE1lc3NhZ2VbXSxcbiAgaXNMb2FkaW5nOiBib29sZWFuLFxuICBoYXNBY3RpdmVQcm9tcHQgPSBmYWxzZSxcbiAgeyBlbmFibGVkID0gdHJ1ZSB9OiB7IGVuYWJsZWQ/OiBib29sZWFuIH0gPSB7fSxcbik6IHtcbiAgc3RhdGU6XG4gICAgfCAnY2xvc2VkJ1xuICAgIHwgJ29wZW4nXG4gICAgfCAndGhhbmtzJ1xuICAgIHwgJ3RyYW5zY3JpcHRfcHJvbXB0J1xuICAgIHwgJ3N1Ym1pdHRpbmcnXG4gICAgfCAnc3VibWl0dGVkJ1xuICBsYXN0UmVzcG9uc2U6IEZlZWRiYWNrU3VydmV5UmVzcG9uc2UgfCBudWxsXG4gIGhhbmRsZVNlbGVjdDogKHNlbGVjdGVkOiBGZWVkYmFja1N1cnZleVJlc3BvbnNlKSA9PiB2b2lkXG59IHtcbiAgY29uc3QgW2dhdGVFbmFibGVkLCBzZXRHYXRlRW5hYmxlZF0gPSB1c2VTdGF0ZTxib29sZWFuIHwgbnVsbD4obnVsbClcbiAgY29uc3Qgc2VlbkNvbXBhY3RCb3VuZGFyaWVzID0gdXNlUmVmPFNldDxzdHJpbmc+PihuZXcgU2V0KCkpXG4gIC8vIFRyYWNrIHRoZSBjb21wYWN0IGJvdW5kYXJ5IHdlJ3JlIHdhaXRpbmcgb24gKHRvIHNob3cgc3VydmV5IGFmdGVyIG5leHQgbWVzc2FnZSlcbiAgY29uc3QgcGVuZGluZ0NvbXBhY3RCb3VuZGFyeVV1aWQgPSB1c2VSZWY8c3RyaW5nIHwgbnVsbD4obnVsbClcblxuICBjb25zdCBvbk9wZW4gPSB1c2VDYWxsYmFjaygoYXBwZWFyYW5jZUlkOiBzdHJpbmcpID0+IHtcbiAgICBjb25zdCBzbUNvbXBhY3Rpb25FbmFibGVkID0gc2hvdWxkVXNlU2Vzc2lvbk1lbW9yeUNvbXBhY3Rpb24oKVxuICAgIGxvZ0V2ZW50KCd0ZW5ndV9wb3N0X2NvbXBhY3Rfc3VydmV5X2V2ZW50Jywge1xuICAgICAgZXZlbnRfdHlwZTpcbiAgICAgICAgJ2FwcGVhcmVkJyBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgYXBwZWFyYW5jZV9pZDpcbiAgICAgICAgYXBwZWFyYW5jZUlkIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICBzZXNzaW9uX21lbW9yeV9jb21wYWN0aW9uX2VuYWJsZWQ6XG4gICAgICAgIHNtQ29tcGFjdGlvbkVuYWJsZWQgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICB9KVxuICAgIHZvaWQgbG9nT1RlbEV2ZW50KCdmZWVkYmFja19zdXJ2ZXknLCB7XG4gICAgICBldmVudF90eXBlOiAnYXBwZWFyZWQnLFxuICAgICAgYXBwZWFyYW5jZV9pZDogYXBwZWFyYW5jZUlkLFxuICAgICAgc3VydmV5X3R5cGU6ICdwb3N0X2NvbXBhY3QnLFxuICAgIH0pXG4gIH0sIFtdKVxuXG4gIGNvbnN0IG9uU2VsZWN0ID0gdXNlQ2FsbGJhY2soXG4gICAgKGFwcGVhcmFuY2VJZDogc3RyaW5nLCBzZWxlY3RlZDogRmVlZGJhY2tTdXJ2ZXlSZXNwb25zZSkgPT4ge1xuICAgICAgY29uc3Qgc21Db21wYWN0aW9uRW5hYmxlZCA9IHNob3VsZFVzZVNlc3Npb25NZW1vcnlDb21wYWN0aW9uKClcbiAgICAgIGxvZ0V2ZW50KCd0ZW5ndV9wb3N0X2NvbXBhY3Rfc3VydmV5X2V2ZW50Jywge1xuICAgICAgICBldmVudF90eXBlOlxuICAgICAgICAgICdyZXNwb25kZWQnIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgIGFwcGVhcmFuY2VfaWQ6XG4gICAgICAgICAgYXBwZWFyYW5jZUlkIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgIHJlc3BvbnNlOlxuICAgICAgICAgIHNlbGVjdGVkIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgIHNlc3Npb25fbWVtb3J5X2NvbXBhY3Rpb25fZW5hYmxlZDpcbiAgICAgICAgICBzbUNvbXBhY3Rpb25FbmFibGVkIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICB9KVxuICAgICAgdm9pZCBsb2dPVGVsRXZlbnQoJ2ZlZWRiYWNrX3N1cnZleScsIHtcbiAgICAgICAgZXZlbnRfdHlwZTogJ3Jlc3BvbmRlZCcsXG4gICAgICAgIGFwcGVhcmFuY2VfaWQ6IGFwcGVhcmFuY2VJZCxcbiAgICAgICAgcmVzcG9uc2U6IHNlbGVjdGVkLFxuICAgICAgICBzdXJ2ZXlfdHlwZTogJ3Bvc3RfY29tcGFjdCcsXG4gICAgICB9KVxuICAgIH0sXG4gICAgW10sXG4gIClcblxuICBjb25zdCB7IHN0YXRlLCBsYXN0UmVzcG9uc2UsIG9wZW4sIGhhbmRsZVNlbGVjdCB9ID0gdXNlU3VydmV5U3RhdGUoe1xuICAgIGhpZGVUaGFua3NBZnRlck1zOiBISURFX1RIQU5LU19BRlRFUl9NUyxcbiAgICBvbk9wZW4sXG4gICAgb25TZWxlY3QsXG4gIH0pXG5cbiAgLy8gQ2hlY2sgdGhlIGZlYXR1cmUgZ2F0ZSBvbiBtb3VudFxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmICghZW5hYmxlZCkgcmV0dXJuXG4gICAgc2V0R2F0ZUVuYWJsZWQoXG4gICAgICBjaGVja1N0YXRzaWdGZWF0dXJlR2F0ZV9DQUNIRURfTUFZX0JFX1NUQUxFKFBPU1RfQ09NUEFDVF9TVVJWRVlfR0FURSksXG4gICAgKVxuICB9LCBbZW5hYmxlZF0pXG5cbiAgLy8gRmluZCBjb21wYWN0IGJvdW5kYXJ5IG1lc3NhZ2VzXG4gIGNvbnN0IGN1cnJlbnRDb21wYWN0Qm91bmRhcmllcyA9IHVzZU1lbW8oXG4gICAgKCkgPT5cbiAgICAgIG5ldyBTZXQoXG4gICAgICAgIG1lc3NhZ2VzXG4gICAgICAgICAgLmZpbHRlcihtc2cgPT4gaXNDb21wYWN0Qm91bmRhcnlNZXNzYWdlKG1zZykpXG4gICAgICAgICAgLm1hcChtc2cgPT4gbXNnLnV1aWQpLFxuICAgICAgKSxcbiAgICBbbWVzc2FnZXNdLFxuICApXG5cbiAgLy8gRGV0ZWN0IG5ldyBjb21wYWN0IGJvdW5kYXJpZXMgYW5kIGRlZmVyIHNob3dpbmcgc3VydmV5IHVudGlsIG5leHQgbWVzc2FnZVxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmICghZW5hYmxlZCkgcmV0dXJuXG5cbiAgICAvLyBEb24ndCBwcm9jZXNzIGlmIGFscmVhZHkgc2hvd2luZ1xuICAgIGlmIChzdGF0ZSAhPT0gJ2Nsb3NlZCcgfHwgaXNMb2FkaW5nKSB7XG4gICAgICByZXR1cm5cbiAgICB9XG5cbiAgICAvLyBEb24ndCBzaG93IHN1cnZleSB3aGVuIHBlcm1pc3Npb24gb3IgYXNrIHF1ZXN0aW9uIHByb21wdHMgYXJlIHZpc2libGVcbiAgICBpZiAoaGFzQWN0aXZlUHJvbXB0KSB7XG4gICAgICByZXR1cm5cbiAgICB9XG5cbiAgICAvLyBDaGVjayBpZiB0aGUgZ2F0ZSBpcyBlbmFibGVkXG4gICAgaWYgKGdhdGVFbmFibGVkICE9PSB0cnVlKSB7XG4gICAgICByZXR1cm5cbiAgICB9XG5cbiAgICBpZiAoaXNGZWVkYmFja1N1cnZleURpc2FibGVkKCkpIHtcbiAgICAgIHJldHVyblxuICAgIH1cblxuICAgIC8vIENoZWNrIGlmIHN1cnZleSBpcyBleHBsaWNpdGx5IGRpc2FibGVkXG4gICAgaWYgKGlzRW52VHJ1dGh5KHByb2Nlc3MuZW52LkNMQVVERV9DT0RFX0RJU0FCTEVfRkVFREJBQ0tfU1VSVkVZKSkge1xuICAgICAgcmV0dXJuXG4gICAgfVxuXG4gICAgLy8gRmlyc3QsIGNoZWNrIGlmIHdlIGhhdmUgYSBwZW5kaW5nIGNvbXBhY3QgYW5kIGEgbmV3IG1lc3NhZ2UgaGFzIGFycml2ZWRcbiAgICBpZiAocGVuZGluZ0NvbXBhY3RCb3VuZGFyeVV1aWQuY3VycmVudCAhPT0gbnVsbCkge1xuICAgICAgaWYgKFxuICAgICAgICBoYXNNZXNzYWdlQWZ0ZXJCb3VuZGFyeShtZXNzYWdlcywgcGVuZGluZ0NvbXBhY3RCb3VuZGFyeVV1aWQuY3VycmVudClcbiAgICAgICkge1xuICAgICAgICAvLyBBIG5ldyBtZXNzYWdlIGFycml2ZWQgYWZ0ZXIgdGhlIGNvbXBhY3QgLSBkZWNpZGUgd2hldGhlciB0byBzaG93IHN1cnZleVxuICAgICAgICBwZW5kaW5nQ29tcGFjdEJvdW5kYXJ5VXVpZC5jdXJyZW50ID0gbnVsbFxuXG4gICAgICAgIC8vIE9ubHkgc2hvdyBzdXJ2ZXkgMjAlIG9mIHRoZSB0aW1lXG4gICAgICAgIGlmIChNYXRoLnJhbmRvbSgpIDwgU1VSVkVZX1BST0JBQklMSVRZKSB7XG4gICAgICAgICAgb3BlbigpXG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuXG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8gRmluZCBuZXcgY29tcGFjdCBib3VuZGFyaWVzIHRoYXQgd2UgaGF2ZW4ndCBzZWVuIHlldFxuICAgIGNvbnN0IG5ld0JvdW5kYXJpZXMgPSBBcnJheS5mcm9tKGN1cnJlbnRDb21wYWN0Qm91bmRhcmllcykuZmlsdGVyKFxuICAgICAgdXVpZCA9PiAhc2VlbkNvbXBhY3RCb3VuZGFyaWVzLmN1cnJlbnQuaGFzKHV1aWQpLFxuICAgIClcblxuICAgIGlmIChuZXdCb3VuZGFyaWVzLmxlbmd0aCA+IDApIHtcbiAgICAgIC8vIE1hcmsgdGhlc2UgYm91bmRhcmllcyBhcyBzZWVuXG4gICAgICBzZWVuQ29tcGFjdEJvdW5kYXJpZXMuY3VycmVudCA9IG5ldyBTZXQoY3VycmVudENvbXBhY3RCb3VuZGFyaWVzKVxuXG4gICAgICAvLyBEb24ndCBzaG93IHN1cnZleSBpbW1lZGlhdGVseSAtIHdhaXQgZm9yIG5leHQgbWVzc2FnZVxuICAgICAgLy8gU3RvcmUgdGhlIG1vc3QgcmVjZW50IG5ldyBib3VuZGFyeSBVVUlEXG4gICAgICBwZW5kaW5nQ29tcGFjdEJvdW5kYXJ5VXVpZC5jdXJyZW50ID1cbiAgICAgICAgbmV3Qm91bmRhcmllc1tuZXdCb3VuZGFyaWVzLmxlbmd0aCAtIDFdIVxuICAgIH1cbiAgfSwgW1xuICAgIGVuYWJsZWQsXG4gICAgY3VycmVudENvbXBhY3RCb3VuZGFyaWVzLFxuICAgIHN0YXRlLFxuICAgIGlzTG9hZGluZyxcbiAgICBoYXNBY3RpdmVQcm9tcHQsXG4gICAgZ2F0ZUVuYWJsZWQsXG4gICAgbWVzc2FnZXMsXG4gICAgb3BlbixcbiAgXSlcblxuICByZXR1cm4geyBzdGF0ZSwgbGFzdFJlc3BvbnNlLCBoYW5kbGVTZWxlY3QgfVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsU0FBU0EsV0FBVyxFQUFFQyxTQUFTLEVBQUVDLE9BQU8sRUFBRUMsTUFBTSxFQUFFQyxRQUFRLFFBQVEsT0FBTztBQUN6RSxTQUFTQyx3QkFBd0IsUUFBUSxrQ0FBa0M7QUFDM0UsU0FBU0MsMkNBQTJDLFFBQVEsc0NBQXNDO0FBQ2xHLFNBQ0UsS0FBS0MsMERBQTBELEVBQy9EQyxRQUFRLFFBQ0gsaUNBQWlDO0FBQ3hDLFNBQVNDLGdDQUFnQyxRQUFRLGdEQUFnRDtBQUNqRyxjQUFjQyxPQUFPLFFBQVEsd0JBQXdCO0FBQ3JELFNBQVNDLFdBQVcsUUFBUSx5QkFBeUI7QUFDckQsU0FBU0Msd0JBQXdCLFFBQVEseUJBQXlCO0FBQ2xFLFNBQVNDLFlBQVksUUFBUSxpQ0FBaUM7QUFDOUQsU0FB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 \ No newline at end of file diff --git a/src/components/FeedbackSurvey/useSurveyState.tsx b/src/components/FeedbackSurvey/useSurveyState.tsx new file mode 100644 index 0000000..a2758ed --- /dev/null +++ b/src/components/FeedbackSurvey/useSurveyState.tsx @@ -0,0 +1,100 @@ +import { randomUUID } from 'crypto'; +import { useCallback, useRef, useState } from 'react'; +import type { TranscriptShareResponse } from './TranscriptSharePrompt.js'; +import type { FeedbackSurveyResponse } from './utils.js'; +type SurveyState = 'closed' | 'open' | 'thanks' | 'transcript_prompt' | 'submitting' | 'submitted'; +type UseSurveyStateOptions = { + hideThanksAfterMs: number; + onOpen: (appearanceId: string) => void | Promise; + onSelect: (appearanceId: string, selected: FeedbackSurveyResponse) => void | Promise; + shouldShowTranscriptPrompt?: (selected: FeedbackSurveyResponse) => boolean; + onTranscriptPromptShown?: (appearanceId: string, surveyResponse: FeedbackSurveyResponse) => void; + onTranscriptSelect?: (appearanceId: string, selected: TranscriptShareResponse, surveyResponse: FeedbackSurveyResponse | null) => boolean | Promise; +}; +export function useSurveyState({ + hideThanksAfterMs, + onOpen, + onSelect, + shouldShowTranscriptPrompt, + onTranscriptPromptShown, + onTranscriptSelect +}: UseSurveyStateOptions): { + state: SurveyState; + lastResponse: FeedbackSurveyResponse | null; + open: () => void; + handleSelect: (selected: FeedbackSurveyResponse) => boolean; + handleTranscriptSelect: (selected: TranscriptShareResponse) => void; +} { + const [state, setState] = useState('closed'); + const [lastResponse, setLastResponse] = useState(null); + const appearanceId = useRef(randomUUID()); + const lastResponseRef = useRef(null); + const showThanksThenClose = useCallback(() => { + setState('thanks'); + setTimeout((setState_0, setLastResponse_0) => { + setState_0('closed'); + setLastResponse_0(null); + }, hideThanksAfterMs, setState, setLastResponse); + }, [hideThanksAfterMs]); + const showSubmittedThenClose = useCallback(() => { + setState('submitted'); + setTimeout(setState, hideThanksAfterMs, 'closed'); + }, [hideThanksAfterMs]); + const open = useCallback(() => { + if (state !== 'closed') { + return; + } + setState('open'); + appearanceId.current = randomUUID(); + void onOpen(appearanceId.current); + }, [state, onOpen]); + const handleSelect = useCallback((selected: FeedbackSurveyResponse): boolean => { + setLastResponse(selected); + lastResponseRef.current = selected; + // Always fire the survey response event first + void onSelect(appearanceId.current, selected); + if (selected === 'dismissed') { + setState('closed'); + setLastResponse(null); + } else if (shouldShowTranscriptPrompt?.(selected)) { + setState('transcript_prompt'); + onTranscriptPromptShown?.(appearanceId.current, selected); + return true; + } else { + showThanksThenClose(); + } + return false; + }, [showThanksThenClose, onSelect, shouldShowTranscriptPrompt, onTranscriptPromptShown]); + const handleTranscriptSelect = useCallback((selected_0: TranscriptShareResponse) => { + switch (selected_0) { + case 'yes': + setState('submitting'); + void (async () => { + try { + const success = await onTranscriptSelect?.(appearanceId.current, selected_0, lastResponseRef.current); + if (success) { + showSubmittedThenClose(); + } else { + showThanksThenClose(); + } + } catch { + showThanksThenClose(); + } + })(); + break; + case 'no': + case 'dont_ask_again': + void onTranscriptSelect?.(appearanceId.current, selected_0, lastResponseRef.current); + showThanksThenClose(); + break; + } + }, [showThanksThenClose, showSubmittedThenClose, onTranscriptSelect]); + return { + state, + lastResponse, + open, + handleSelect, + handleTranscriptSelect + }; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJyYW5kb21VVUlEIiwidXNlQ2FsbGJhY2siLCJ1c2VSZWYiLCJ1c2VTdGF0ZSIsIlRyYW5zY3JpcHRTaGFyZVJlc3BvbnNlIiwiRmVlZGJhY2tTdXJ2ZXlSZXNwb25zZSIsIlN1cnZleVN0YXRlIiwiVXNlU3VydmV5U3RhdGVPcHRpb25zIiwiaGlkZVRoYW5rc0FmdGVyTXMiLCJvbk9wZW4iLCJhcHBlYXJhbmNlSWQiLCJQcm9taXNlIiwib25TZWxlY3QiLCJzZWxlY3RlZCIsInNob3VsZFNob3dUcmFuc2NyaXB0UHJvbXB0Iiwib25UcmFuc2NyaXB0UHJvbXB0U2hvd24iLCJzdXJ2ZXlSZXNwb25zZSIsIm9uVHJhbnNjcmlwdFNlbGVjdCIsInVzZVN1cnZleVN0YXRlIiwic3RhdGUiLCJsYXN0UmVzcG9uc2UiLCJvcGVuIiwiaGFuZGxlU2VsZWN0IiwiaGFuZGxlVHJhbnNjcmlwdFNlbGVjdCIsInNldFN0YXRlIiwic2V0TGFzdFJlc3BvbnNlIiwibGFzdFJlc3BvbnNlUmVmIiwic2hvd1RoYW5rc1RoZW5DbG9zZSIsInNldFRpbWVvdXQiLCJzaG93U3VibWl0dGVkVGhlbkNsb3NlIiwiY3VycmVudCIsInN1Y2Nlc3MiXSwic291cmNlcyI6WyJ1c2VTdXJ2ZXlTdGF0ZS50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgcmFuZG9tVVVJRCB9IGZyb20gJ2NyeXB0bydcbmltcG9ydCB7IHVzZUNhbGxiYWNrLCB1c2VSZWYsIHVzZVN0YXRlIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgdHlwZSB7IFRyYW5zY3JpcHRTaGFyZVJlc3BvbnNlIH0gZnJvbSAnLi9UcmFuc2NyaXB0U2hhcmVQcm9tcHQuanMnXG5pbXBvcnQgdHlwZSB7IEZlZWRiYWNrU3VydmV5UmVzcG9uc2UgfSBmcm9tICcuL3V0aWxzLmpzJ1xuXG50eXBlIFN1cnZleVN0YXRlID1cbiAgfCAnY2xvc2VkJ1xuICB8ICdvcGVuJ1xuICB8ICd0aGFua3MnXG4gIHwgJ3RyYW5zY3JpcHRfcHJvbXB0J1xuICB8ICdzdWJtaXR0aW5nJ1xuICB8ICdzdWJtaXR0ZWQnXG5cbnR5cGUgVXNlU3VydmV5U3RhdGVPcHRpb25zID0ge1xuICBoaWRlVGhhbmtzQWZ0ZXJNczogbnVtYmVyXG4gIG9uT3BlbjogKGFwcGVhcmFuY2VJZDogc3RyaW5nKSA9PiB2b2lkIHwgUHJvbWlzZTx2b2lkPlxuICBvblNlbGVjdDogKFxuICAgIGFwcGVhcmFuY2VJZDogc3RyaW5nLFxuICAgIHNlbGVjdGVkOiBGZWVkYmFja1N1cnZleVJlc3BvbnNlLFxuICApID0+IHZvaWQgfCBQcm9taXNlPHZvaWQ+XG4gIHNob3VsZFNob3dUcmFuc2NyaXB0UHJvbXB0PzogKHNlbGVjdGVkOiBGZWVkYmFja1N1cnZleVJlc3BvbnNlKSA9PiBib29sZWFuXG4gIG9uVHJhbnNjcmlwdFByb21wdFNob3duPzogKFxuICAgIGFwcGVhcmFuY2VJZDogc3RyaW5nLFxuICAgIHN1cnZleVJlc3BvbnNlOiBGZWVkYmFja1N1cnZleVJlc3BvbnNlLFxuICApID0+IHZvaWRcbiAgb25UcmFuc2NyaXB0U2VsZWN0PzogKFxuICAgIGFwcGVhcmFuY2VJZDogc3RyaW5nLFxuICAgIHNlbGVjdGVkOiBUcmFuc2NyaXB0U2hhcmVSZXNwb25zZSxcbiAgICBzdXJ2ZXlSZXNwb25zZTogRmVlZGJhY2tTdXJ2ZXlSZXNwb25zZSB8IG51bGwsXG4gICkgPT4gYm9vbGVhbiB8IFByb21pc2U8Ym9vbGVhbj5cbn1cblxuZXhwb3J0IGZ1bmN0aW9uIHVzZVN1cnZleVN0YXRlKHtcbiAgaGlkZVRoYW5rc0FmdGVyTXMsXG4gIG9uT3BlbixcbiAgb25TZWxlY3QsXG4gIHNob3VsZFNob3dUcmFuc2NyaXB0UHJvbXB0LFxuICBvblRyYW5zY3JpcHRQcm9tcHRTaG93bixcbiAgb25UcmFuc2NyaXB0U2VsZWN0LFxufTogVXNlU3VydmV5U3RhdGVPcHRpb25zKToge1xuICBzdGF0ZTogU3VydmV5U3RhdGVcbiAgbGFzdFJlc3BvbnNlOiBGZWVkYmFja1N1cnZleVJlc3BvbnNlIHwgbnVsbFxuICBvcGVuOiAoKSA9PiB2b2lkXG4gIGhhbmRsZVNlbGVjdDogKHNlbGVjdGVkOiBGZWVkYmFja1N1cnZleVJlc3BvbnNlKSA9PiBib29sZWFuXG4gIGhhbmRsZVRyYW5zY3JpcHRTZWxlY3Q6IChzZWxlY3RlZDogVHJhbnNjcmlwdFNoYXJlUmVzcG9uc2UpID0+IHZvaWRcbn0ge1xuICBjb25zdCBbc3RhdGUsIHNldFN0YXRlXSA9IHVzZVN0YXRlPFN1cnZleVN0YXRlPignY2xvc2VkJylcbiAgY29uc3QgW2xhc3RSZXNwb25zZSwgc2V0TGFzdFJlc3BvbnNlXSA9XG4gICAgdXNlU3RhdGU8RmVlZGJhY2tTdXJ2ZXlSZXNwb25zZSB8IG51bGw+KG51bGwpXG4gIGNvbnN0IGFwcGVhcmFuY2VJZCA9IHVzZVJlZihyYW5kb21VVUlEKCkpXG4gIGNvbnN0IGxhc3RSZXNwb25zZVJlZiA9IHVzZVJlZjxGZWVkYmFja1N1cnZleVJlc3BvbnNlIHwgbnVsbD4obnVsbClcblxuICBjb25zdCBzaG93VGhhbmtzVGhlbkNsb3NlID0gdXNlQ2FsbGJhY2soKCkgPT4ge1xuICAgIHNldFN0YXRlKCd0aGFua3MnKVxuICAgIHNldFRpbWVvdXQoXG4gICAgICAoc2V0U3RhdGUsIHNldExhc3RSZXNwb25zZSkgPT4ge1xuICAgICAgICBzZXRTdGF0ZSgnY2xvc2VkJylcbiAgICAgICAgc2V0TGFzdFJlc3BvbnNlKG51bGwpXG4gICAgICB9LFxuICAgICAgaGlkZVRoYW5rc0FmdGVyTXMsXG4gICAgICBzZXRTdGF0ZSxcbiAgICAgIHNldExhc3RSZXNwb25zZSxcbiAgICApXG4gIH0sIFtoaWRlVGhhbmtzQWZ0ZXJNc10pXG5cbiAgY29uc3Qgc2hvd1N1Ym1pdHRlZFRoZW5DbG9zZSA9IHVzZUNhbGxiYWNrKCgpID0+IHtcbiAgICBzZXRTdGF0ZSgnc3VibWl0dGVkJylcbiAgICBzZXRUaW1lb3V0KHNldFN0YXRlLCBoaWRlVGhhbmtzQWZ0ZXJNcywgJ2Nsb3NlZCcpXG4gIH0sIFtoaWRlVGhhbmtzQWZ0ZXJNc10pXG5cbiAgY29uc3Qgb3BlbiA9IHVzZUNhbGxiYWNrKCgpID0+IHtcbiAgICBpZiAoc3RhdGUgIT09ICdjbG9zZWQnKSB7XG4gICAgICByZXR1cm5cbiAgICB9XG4gICAgc2V0U3RhdGUoJ29wZW4nKVxuICAgIGFwcGVhcmFuY2VJZC5jdXJyZW50ID0gcmFuZG9tVVVJRCgpXG4gICAgdm9pZCBvbk9wZW4oYXBwZWFyYW5jZUlkLmN1cnJlbnQpXG4gIH0sIFtzdGF0ZSwgb25PcGVuXSlcblxuICBjb25zdCBoYW5kbGVTZWxlY3QgPSB1c2VDYWxsYmFjayhcbiAgICAoc2VsZWN0ZWQ6IEZlZWRiYWNrU3VydmV5UmVzcG9uc2UpOiBib29sZWFuID0+IHtcbiAgICAgIHNldExhc3RSZXNwb25zZShzZWxlY3RlZClcbiAgICAgIGxhc3RSZXNwb25zZVJlZi5jdXJyZW50ID0gc2VsZWN0ZWRcbiAgICAgIC8vIEFsd2F5cyBmaXJlIHRoZSBzdXJ2ZXkgcmVzcG9uc2UgZXZlbnQgZmlyc3RcbiAgICAgIHZvaWQgb25TZWxlY3QoYXBwZWFyYW5jZUlkLmN1cnJlbnQsIHNlbGVjdGVkKVxuXG4gICAgICBpZiAoc2VsZWN0ZWQgPT09ICdkaXNtaXNzZWQnKSB7XG4gICAgICAgIHNldFN0YXRlKCdjbG9zZWQnKVxuICAgICAgICBzZXRMYXN0UmVzcG9uc2UobnVsbClcbiAgICAgIH0gZWxzZSBpZiAoc2hvdWxkU2hvd1RyYW5zY3JpcHRQcm9tcHQ/LihzZWxlY3RlZCkpIHtcbiAgICAgICAgc2V0U3RhdGUoJ3RyYW5zY3JpcHRfcHJvbXB0JylcbiAgICAgICAgb25UcmFuc2NyaXB0UHJvbXB0U2hvd24/LihhcHBlYXJhbmNlSWQuY3VycmVudCwgc2VsZWN0ZWQpXG4gICAgICAgIHJldHVybiB0cnVlXG4gICAgICB9IGVsc2Uge1xuICAgICAgICBzaG93VGhhbmtzVGhlbkNsb3NlKClcbiAgICAgIH1cbiAgICAgIHJldHVybiBmYWxzZVxuICAgIH0sXG4gICAgW1xuICAgICAgc2hvd1RoYW5rc1RoZW5DbG9zZSxcbiAgICAgIG9uU2VsZWN0LFxuICAgICAgc2hvdWxkU2hvd1RyYW5zY3JpcHRQcm9tcHQsXG4gICAgICBvblRyYW5zY3JpcHRQcm9tcHRTaG93bixcbiAgICBdLFxuICApXG5cbiAgY29uc3QgaGFuZGxlVHJhbnNjcmlwdFNlbGVjdCA9IHVzZUNhbGxiYWNrKFxuICAgIChzZWxlY3RlZDogVHJhbnNjcmlwdFNoYXJlUmVzcG9uc2UpID0+IHtcbiAgICAgIHN3aXRjaCAoc2VsZWN0ZWQpIHtcbiAgICAgICAgY2FzZSAneWVzJzpcbiAgICAgICAgICBzZXRTdGF0ZSgnc3VibWl0dGluZycpXG4gICAgICAgICAgdm9pZCAoYXN5bmMgKCkgPT4ge1xuICAgICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgICAgY29uc3Qgc3VjY2VzcyA9IGF3YWl0IG9uVHJhbnNjcmlwdFNlbGVjdD8uKFxuICAgICAgICAgICAgICAgIGFwcGVhcmFuY2VJZC5jdXJyZW50LFxuICAgICAgICAgICAgICAgIHNlbGVjdGVkLFxuICAgICAgICAgICAgICAgIGxhc3RSZXNwb25zZVJlZi5jdXJyZW50LFxuICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgIGlmIChzdWNjZXNzKSB7XG4gICAgICAgICAgICAgICAgc2hvd1N1Ym1pdHRlZFRoZW5DbG9zZSgpXG4gICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgc2hvd1RoYW5rc1RoZW5DbG9zZSgpXG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH0gY2F0Y2gge1xuICAgICAgICAgICAgICBzaG93VGhhbmtzVGhlbkNsb3NlKClcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9KSgpXG4gICAgICAgICAgYnJlYWtcbiAgICAgICAgY2FzZSAnbm8nOlxuICAgICAgICBjYXNlICdkb250X2Fza19hZ2Fpbic6XG4gICAgICAgICAgdm9pZCBvblRyYW5zY3JpcHRTZWxlY3Q/LihcbiAgICAgICAgICAgIGFwcGVhcmFuY2VJZC5jdXJyZW50LFxuICAgICAgICAgICAgc2VsZWN0ZWQsXG4gICAgICAgICAgICBsYXN0UmVzcG9uc2VSZWYuY3VycmVudCxcbiAgICAgICAgICApXG4gICAgICAgICAgc2hvd1RoYW5rc1RoZW5DbG9zZSgpXG4gICAgICAgICAgYnJlYWtcbiAgICAgIH1cbiAgICB9LFxuICAgIFtzaG93VGhhbmtzVGhlbkNsb3NlLCBzaG93U3VibWl0dGVkVGhlbkNsb3NlLCBvblRyYW5zY3JpcHRTZWxlY3RdLFxuICApXG5cbiAgcmV0dXJuIHsgc3RhdGUsIGxhc3RSZXNwb25zZSwgb3BlbiwgaGFuZGxlU2VsZWN0LCBoYW5kbGVUcmFuc2NyaXB0U2VsZWN0IH1cbn1cbiJdLCJtYXBwaW5ncyI6IkFBQUEsU0FBU0EsVUFBVSxRQUFRLFFBQVE7QUFDbkMsU0FBU0MsV0FBVyxFQUFFQyxNQUFNLEVBQUVDLFFBQVEsUUFBUSxPQUFPO0FBQ3JELGNBQWNDLHVCQUF1QixRQUFRLDRCQUE0QjtBQUN6RSxjQUFjQyxzQkFBc0IsUUFBUSxZQUFZO0FBRXhELEtBQUtDLFdBQVcsR0FDWixRQUFRLEdBQ1IsTUFBTSxHQUNOLFFBQVEsR0FDUixtQkFBbUIsR0FDbkIsWUFBWSxHQUNaLFdBQVc7QUFFZixLQUFLQyxxQkFBcUIsR0FBRztFQUMzQkMsaUJBQWlCLEVBQUUsTUFBTTtFQUN6QkMsTUFBTSxFQUFFLENBQUNDLFlBQVksRUFBRSxNQUFNLEVBQUUsR0FBRyxJQUFJLEdBQUdDLE9BQU8sQ0FBQyxJQUFJLENBQUM7RUFDdERDLFFBQVEsRUFBRSxDQUNSRixZQUFZLEVBQUUsTUFBTSxFQUNwQkcsUUFBUSxFQUFFUixzQkFBc0IsRUFDaEMsR0FBRyxJQUFJLEdBQUdNLE9BQU8sQ0FBQyxJQUFJLENBQUM7RUFDekJHLDBCQUEwQixDQUFDLEVBQUUsQ0FBQ0QsUUFBUSxFQUFFUixzQkFBc0IsRUFBRSxHQUFHLE9BQU87RUFDMUVVLHVCQUF1QixDQUFDLEVBQUUsQ0FDeEJMLFlBQVksRUFBRSxNQUFNLEVBQ3BCTSxjQUFjLEVBQUVYLHNCQUFzQixFQUN0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 \ No newline at end of file diff --git a/src/components/FileEditToolDiff.tsx b/src/components/FileEditToolDiff.tsx new file mode 100644 index 0000000..6b4896d --- /dev/null +++ b/src/components/FileEditToolDiff.tsx @@ -0,0 +1,181 @@ +import { c as _c } from "react/compiler-runtime"; +import type { StructuredPatchHunk } from 'diff'; +import * as React from 'react'; +import { Suspense, use, useState } from 'react'; +import { useTerminalSize } from '../hooks/useTerminalSize.js'; +import { Box, Text } from '../ink.js'; +import type { FileEdit } from '../tools/FileEditTool/types.js'; +import { findActualString, preserveQuoteStyle } from '../tools/FileEditTool/utils.js'; +import { adjustHunkLineNumbers, CONTEXT_LINES, getPatchForDisplay } from '../utils/diff.js'; +import { logError } from '../utils/log.js'; +import { CHUNK_SIZE, openForScan, readCapped, scanForContext } from '../utils/readEditContext.js'; +import { firstLineOf } from '../utils/stringUtils.js'; +import { StructuredDiffList } from './StructuredDiffList.js'; +type Props = { + file_path: string; + edits: FileEdit[]; +}; +type DiffData = { + patch: StructuredPatchHunk[]; + firstLine: string | null; + fileContent: string | undefined; +}; +export function FileEditToolDiff(props) { + const $ = _c(7); + let t0; + if ($[0] !== props.edits || $[1] !== props.file_path) { + t0 = () => loadDiffData(props.file_path, props.edits); + $[0] = props.edits; + $[1] = props.file_path; + $[2] = t0; + } else { + t0 = $[2]; + } + const [dataPromise] = useState(t0); + let t1; + if ($[3] === Symbol.for("react.memo_cache_sentinel")) { + t1 = ; + $[3] = t1; + } else { + t1 = $[3]; + } + let t2; + if ($[4] !== dataPromise || $[5] !== props.file_path) { + t2 = ; + $[4] = dataPromise; + $[5] = props.file_path; + $[6] = t2; + } else { + t2 = $[6]; + } + return t2; +} +function DiffBody(t0) { + const $ = _c(6); + const { + promise, + file_path + } = t0; + const { + patch, + firstLine, + fileContent + } = use(promise); + const { + columns + } = useTerminalSize(); + let t1; + if ($[0] !== columns || $[1] !== fileContent || $[2] !== file_path || $[3] !== firstLine || $[4] !== patch) { + t1 = ; + $[0] = columns; + $[1] = fileContent; + $[2] = file_path; + $[3] = firstLine; + $[4] = patch; + $[5] = t1; + } else { + t1 = $[5]; + } + return t1; +} +function DiffFrame(t0) { + const $ = _c(5); + const { + children, + placeholder + } = t0; + let t1; + if ($[0] !== children || $[1] !== placeholder) { + t1 = placeholder ? : children; + $[0] = children; + $[1] = placeholder; + $[2] = t1; + } else { + t1 = $[2]; + } + let t2; + if ($[3] !== t1) { + t2 = {t1}; + $[3] = t1; + $[4] = t2; + } else { + t2 = $[4]; + } + return t2; +} +async function loadDiffData(file_path: string, edits: FileEdit[]): Promise { + const valid = edits.filter(e => e.old_string != null && e.new_string != null); + const single = valid.length === 1 ? valid[0]! : undefined; + + // SedEditPermissionRequest passes the entire file as old_string. Scanning for + // a needle ≥ CHUNK_SIZE allocates O(needle) for the overlap buffer — skip the + // file read entirely and diff the inputs we already have. + if (single && single.old_string.length >= CHUNK_SIZE) { + return diffToolInputsOnly(file_path, [single]); + } + try { + const handle = await openForScan(file_path); + if (handle === null) return diffToolInputsOnly(file_path, valid); + try { + // Multi-edit and empty old_string genuinely need full-file for sequential + // replacements — structuredPatch needs before/after strings. replace_all + // routes through the chunked path below (shows first-occurrence window; + // matches within the slice still replace via edit.replace_all). + if (!single || single.old_string === '') { + const file = await readCapped(handle); + if (file === null) return diffToolInputsOnly(file_path, valid); + const normalized = valid.map(e => normalizeEdit(file, e)); + return { + patch: getPatchForDisplay({ + filePath: file_path, + fileContents: file, + edits: normalized + }), + firstLine: firstLineOf(file), + fileContent: file + }; + } + const ctx = await scanForContext(handle, single.old_string, CONTEXT_LINES); + if (ctx.truncated || ctx.content === '') { + return diffToolInputsOnly(file_path, [single]); + } + const normalized = normalizeEdit(ctx.content, single); + const hunks = getPatchForDisplay({ + filePath: file_path, + fileContents: ctx.content, + edits: [normalized] + }); + return { + patch: adjustHunkLineNumbers(hunks, ctx.lineOffset - 1), + firstLine: ctx.lineOffset === 1 ? firstLineOf(ctx.content) : null, + fileContent: ctx.content + }; + } finally { + await handle.close(); + } + } catch (e) { + logError(e as Error); + return diffToolInputsOnly(file_path, valid); + } +} +function diffToolInputsOnly(filePath: string, edits: FileEdit[]): DiffData { + return { + patch: edits.flatMap(e => getPatchForDisplay({ + filePath, + fileContents: e.old_string, + edits: [e] + })), + firstLine: null, + fileContent: undefined + }; +} +function normalizeEdit(fileContent: string, edit: FileEdit): FileEdit { + const actualOld = findActualString(fileContent, edit.old_string) || edit.old_string; + const actualNew = preserveQuoteStyle(edit.old_string, actualOld, edit.new_string); + return { + ...edit, + old_string: actualOld, + new_string: actualNew + }; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJTdHJ1Y3R1cmVkUGF0Y2hIdW5rIiwiUmVhY3QiLCJTdXNwZW5zZSIsInVzZSIsInVzZVN0YXRlIiwidXNlVGVybWluYWxTaXplIiwiQm94IiwiVGV4dCIsIkZpbGVFZGl0IiwiZmluZEFjdHVhbFN0cmluZyIsInByZXNlcnZlUXVvdGVTdHlsZSIsImFkanVzdEh1bmtMaW5lTnVtYmVycyIsIkNPTlRFWFRfTElORVMiLCJnZXRQYXRjaEZvckRpc3BsYXkiLCJsb2dFcnJvciIsIkNIVU5LX1NJWkUiLCJvcGVuRm9yU2NhbiIsInJlYWRDYXBwZWQiLCJzY2FuRm9yQ29udGV4dCIsImZpcnN0TGluZU9mIiwiU3RydWN0dXJlZERpZmZMaXN0IiwiUHJvcHMiLCJmaWxlX3BhdGgiLCJlZGl0cyIsIkRpZmZEYXRhIiwicGF0Y2giLCJmaXJzdExpbmUiLCJmaWxlQ29udGVudCIsIkZpbGVFZGl0VG9vbERpZmYiLCJwcm9wcyIsIiQiLCJfYyIsInQwIiwibG9hZERpZmZEYXRhIiwiZGF0YVByb21pc2UiLCJ0MSIsIlN5bWJvbCIsImZvciIsInQyIiwiRGlmZkJvZHkiLCJwcm9taXNlIiwiY29sdW1ucyIsIkRpZmZGcmFtZSIsImNoaWxkcmVuIiwicGxhY2Vob2xkZXIiLCJQcm9taXNlIiwidmFsaWQiLCJmaWx0ZXIiLCJlIiwib2xkX3N0cmluZyIsIm5ld19zdHJpbmciLCJzaW5nbGUiLCJsZW5ndGgiLCJ1bmRlZmluZWQiLCJkaWZmVG9vbElucHV0c09ubHkiLCJoYW5kbGUiLCJmaWxlIiwibm9ybWFsaXplZCIsIm1hcCIsIm5vcm1hbGl6ZUVkaXQiLCJmaWxlUGF0aCIsImZpbGVDb250ZW50cyIsImN0eCIsInRydW5jYXRlZCIsImNvbnRlbnQiLCJodW5rcyIsImxpbmVPZmZzZXQiLCJjbG9zZSIsIkVycm9yIiwiZmxhdE1hcCIsImVkaXQiLCJhY3R1YWxPbGQiLCJhY3R1YWxOZXciXSwic291cmNlcyI6WyJGaWxlRWRpdFRvb2xEaWZmLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IFN0cnVjdHVyZWRQYXRjaEh1bmsgfSBmcm9tICdkaWZmJ1xuaW1wb3J0ICogYXMgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBTdXNwZW5zZSwgdXNlLCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgdXNlVGVybWluYWxTaXplIH0gZnJvbSAnLi4vaG9va3MvdXNlVGVybWluYWxTaXplLmpzJ1xuaW1wb3J0IHsgQm94LCBUZXh0IH0gZnJvbSAnLi4vaW5rLmpzJ1xuaW1wb3J0IHR5cGUgeyBGaWxlRWRpdCB9IGZyb20gJy4uL3Rvb2xzL0ZpbGVFZGl0VG9vbC90eXBlcy5qcydcbmltcG9ydCB7XG4gIGZpbmRBY3R1YWxTdHJpbmcsXG4gIHByZXNlcnZlUXVvdGVTdHlsZSxcbn0gZnJvbSAnLi4vdG9vbHMvRmlsZUVkaXRUb29sL3V0aWxzLmpzJ1xuaW1wb3J0IHtcbiAgYWRqdXN0SHVua0xpbmVOdW1iZXJzLFxuICBDT05URVhUX0xJTkVTLFxuICBnZXRQYXRjaEZvckRpc3BsYXksXG59IGZyb20gJy4uL3V0aWxzL2RpZmYuanMnXG5pbXBvcnQgeyBsb2dFcnJvciB9IGZyb20gJy4uL3V0aWxzL2xvZy5qcydcbmltcG9ydCB7XG4gIENIVU5LX1NJWkUsXG4gIG9wZW5Gb3JTY2FuLFxuICByZWFkQ2FwcGVkLFxuICBzY2FuRm9yQ29udGV4dCxcbn0gZnJvbSAnLi4vdXRpbHMvcmVhZEVkaXRDb250ZXh0LmpzJ1xuaW1wb3J0IHsgZmlyc3RMaW5lT2YgfSBmcm9tICcuLi91dGlscy9zdHJpbmdVdGlscy5qcydcbmltcG9ydCB7IFN0cnVjdHVyZWREaWZmTGlzdCB9IGZyb20gJy4vU3RydWN0dXJlZERpZmZMaXN0LmpzJ1xuXG50eXBlIFByb3BzID0ge1xuICBmaWxlX3BhdGg6IHN0cmluZ1xuICBlZGl0czogRmlsZUVkaXRbXVxufVxuXG50eXBlIERpZmZEYXRhID0ge1xuICBwYXRjaDogU3RydWN0dXJlZFBhdGNoSHVua1tdXG4gIGZpcnN0TGluZTogc3RyaW5nIHwgbnVsbFxuICBmaWxlQ29udGVudDogc3RyaW5nIHwgdW5kZWZpbmVkXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBGaWxlRWRpdFRvb2xEaWZmKHByb3BzOiBQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIC8vIFNuYXBzaG90IG9uIG1vdW50IOKAlCB0aGUgZGlmZiBtdXN0IHN0YXkgY29uc2lzdGVudCBldmVuIGlmIHRoZSBmaWxlIGNoYW5nZXNcbiAgLy8gd2hpbGUgdGhlIGRpYWxvZyBpcyBvcGVuLiB1c2VNZW1vIG9uIHByb3BzLmVkaXRzIHdvdWxkIHJlLXJlYWQgdGhlIGZpbGUgb25cbiAgLy8gZXZlcnkgcmVuZGVyIGJlY2F1c2UgY2FsbGVycyBwYXNzIGZyZXNoIGFycmF5IGxpdGVyYWxzLlxuICBjb25zdCBbZGF0YVByb21pc2VdID0gdXNlU3RhdGUoKCkgPT5cbiAgICBsb2FkRGlmZkRhdGEocHJvcHMuZmlsZV9wYXRoLCBwcm9wcy5lZGl0cyksXG4gIClcbiAgcmV0dXJuIChcbiAgICA8U3VzcGVuc2UgZmFsbGJhY2s9ezxEaWZmRnJhbWUgcGxhY2Vob2xkZXIgLz59PlxuICAgICAgPERpZmZCb2R5IHByb21pc2U9e2RhdGFQcm9taXNlfSBmaWxlX3BhdGg9e3Byb3BzLmZpbGVfcGF0aH0gLz5cbiAgICA8L1N1c3BlbnNlPlxuICApXG59XG5cbmZ1bmN0aW9uIERpZmZCb2R5KHtcbiAgcHJvbWlzZSxcbiAgZmlsZV9wYXRoLFxufToge1xuICBwcm9taXNlOiBQcm9taXNlPERpZmZEYXRhPlxuICBmaWxlX3BhdGg6IHN0cmluZ1xufSk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IHsgcGF0Y2gsIGZpcnN0TGluZSwgZmlsZUNvbnRlbnQgfSA9IHVzZShwcm9taXNlKVxuICBjb25zdCB7IGNvbHVtbnMgfSA9IHVzZVRlcm1pbmFsU2l6ZSgpXG4gIHJldHVybiAoXG4gICAgPERpZmZGcmFtZT5cbiAgICAgIDxTdHJ1Y3R1cmVkRGlmZkxpc3RcbiAgICAgICAgaHVua3M9e3BhdGNofVxuICAgICAgICBkaW09e2ZhbHNlfVxuICAgICAgICB3aWR0aD17Y29sdW1uc31cbiAgICAgICAgZmlsZVBhdGg9e2ZpbGVfcGF0aH1cbiAgICAgICAgZmlyc3RMaW5lPXtmaXJzdExpbmV9XG4gICAgICAgIGZpbGVDb250ZW50PXtmaWxlQ29udGVudH1cbiAgICAgIC8+XG4gICAgPC9EaWZmRnJhbWU+XG4gIClcbn1cblxuZnVuY3Rpb24gRGlmZkZyYW1lKHtcbiAgY2hpbGRyZW4sXG4gIHBsYWNlaG9sZGVyLFxufToge1xuICBjaGlsZHJlbj86IFJlYWN0LlJlYWN0Tm9kZVxuICBwbGFjZWhvbGRlcj86IGJvb2xlYW5cbn0pOiBSZWFjdC5SZWFjdE5vZGUge1xuICByZXR1cm4gKFxuICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiPlxuICAgICAgPEJveFxuICAgICAgICBib3JkZXJDb2xvcj1cInN1YnRsZVwiXG4gICAgICAgIGJvcmRlclN0eWxlPVwiZGFzaGVkXCJcbiAgICAgICAgZmxleERpcmVjdGlvbj1cImNvbHVtblwiXG4gICAgICAgIGJvcmRlckxlZnQ9e2ZhbHNlfVxuICAgICAgICBib3JkZXJSaWdodD17ZmFsc2V9XG4gICAgICA+XG4gICAgICAgIHtwbGFjZWhvbGRlciA/IDxUZXh0IGRpbUNvbG9yPuKApjwvVGV4dD4gOiBjaGlsZHJlbn1cbiAgICAgIDwvQm94PlxuICAgIDwvQm94PlxuICApXG59XG5cbmFzeW5jIGZ1bmN0aW9uIGxvYWREaWZmRGF0YShcbiAgZmlsZV9wYXRoOiBzdHJpbmcsXG4gIGVkaXRzOiBGaWxlRWRpdFtdLFxuKTogUHJvbWlzZTxEaWZmRGF0YT4ge1xuICBjb25zdCB2YWxpZCA9IGVkaXRzLmZpbHRlcihlID0+IGUub2xkX3N0cmluZyAhPSBudWxsICYmIGUubmV3X3N0cmluZyAhPSBudWxsKVxuICBjb25zdCBzaW5nbGUgPSB2YWxpZC5sZW5ndGggPT09IDEgPyB2YWxpZFswXSEgOiB1bmRlZmluZWRcblxuICAvLyBTZWRFZGl0UGVybWlzc2lvblJlcXVlc3QgcGFzc2VzIHRoZSBlbnRpcmUgZmlsZSBhcyBvbGRfc3RyaW5nLiBTY2FubmluZyBmb3JcbiAgLy8gYSBuZWVkbGUg4omlIENIVU5LX1NJWkUgYWxsb2NhdGVzIE8obmVlZGxlKSBmb3IgdGhlIG92ZXJsYXAgYnVmZmVyIOKAlCBza2lwIHRoZVxuICAvLyBmaWxlIHJlYWQgZW50aXJlbHkgYW5kIGRpZmYgdGhlIGlucHV0cyB3ZSBhbHJlYWR5IGhhdmUuXG4gIGlmIChzaW5nbGUgJiYgc2luZ2xlLm9sZF9zdHJpbmcubGVuZ3RoID49IENIVU5LX1NJWkUpIHtcbiAgICByZXR1cm4gZGlmZlRvb2xJbnB1dHNPbmx5KGZpbGVfcGF0aCwgW3NpbmdsZV0pXG4gIH1cblxuICB0cnkge1xuICAgIGNvbnN0IGhhbmRsZSA9IGF3YWl0IG9wZW5Gb3JTY2FuKGZpbGVfcGF0aClcbiAgICBpZiAoaGFuZGxlID09PSBudWxsKSByZXR1cm4gZGlmZlRvb2xJbnB1dHNPbmx5KGZpbGVfcGF0aCwgdmFsaWQpXG4gICAgdHJ5IHtcbiAgICAgIC8vIE11bHRpLWVkaXQgYW5kIGVtcHR5IG9sZF9zdHJpbmcgZ2VudWluZWx5IG5lZWQgZnVsbC1maWxlIGZvciBzZXF1ZW50aWFsXG4gICAgICAvLyByZXBsYWNlbWVudHMg4oCUIHN0cnVjdHVyZWRQYXRjaCBuZWVkcyBiZWZvcmUvYWZ0ZXIgc3RyaW5ncy4gcmVwbGFjZV9hbGxcbiAgICAgIC8vIHJvdXRlcyB0aHJvdWdoIHRoZSBjaHVua2VkIHBhdGggYmVsb3cgKHNob3dzIGZpcnN0LW9jY3VycmVuY2Ugd2luZG93O1xuICAgICAgLy8gbWF0Y2hlcyB3aXRoaW4gdGhlIHNsaWNlIHN0aWxsIHJlcGxhY2UgdmlhIGVkaXQucmVwbGFjZV9hbGwpLlxuICAgICAgaWYgKCFzaW5nbGUgfHwgc2luZ2xlLm9sZF9zdHJpbmcgPT09ICcnKSB7XG4gICAgICAgIGNvbnN0IGZpbGUgPSBhd2FpdCByZWFkQ2FwcGVkKGhhbmRsZSlcbiAgICAgICAgaWYgKGZpbGUgPT09IG51bGwpIHJldHVybiBkaWZmVG9vbElucHV0c09ubHkoZmlsZV9wYXRoLCB2YWxpZClcbiAgICAgICAgY29uc3Qgbm9ybWFsaXplZCA9IHZhbGlkLm1hcChlID0+IG5vcm1hbGl6ZUVkaXQoZmlsZSwgZSkpXG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgcGF0Y2g6IGdldFBhdGNoRm9yRGlzcGxheSh7XG4gICAgICAgICAgICBmaWxlUGF0aDogZmlsZV9wYXRoLFxuICAgICAgICAgICAgZmlsZUNvbnRlbnRzOiBmaWxlLFxuICAgICAgICAgICAgZWRpdHM6IG5vcm1hbGl6ZWQsXG4gICAgICAgICAgfSksXG4gICAgICAgICAgZmlyc3RMaW5lOiBmaXJzdExpbmVPZihmaWxlKSxcbiAgICAgICAgICBmaWxlQ29udGVudDogZmlsZSxcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBjb25zdCBjdHggPSBhd2FpdCBzY2FuRm9yQ29udGV4dChoYW5kbGUsIHNpbmdsZS5vbGRfc3RyaW5nLCBDT05URVhUX0xJTkVTKVxuICAgICAgaWYgKGN0eC50cnVuY2F0ZWQgfHwgY3R4LmNvbnRlbnQgPT09ICcnKSB7XG4gICAgICAgIHJldHVybiBkaWZmVG9vbElucHV0c09ubHkoZmlsZV9wYXRoLCBbc2luZ2xlXSlcbiAgICAgIH1cbiAgICAgIGNvbnN0IG5vcm1hbGl6ZWQgPSBub3JtYWxpemVFZGl0KGN0eC5jb250ZW50LCBzaW5nbGUpXG4gICAgICBjb25zdCBodW5rcyA9IGdldFBhdGNoRm9yRGlzcGxheSh7XG4gICAgICAgIGZpbGVQYXRoOiBmaWxlX3BhdGgsXG4gICAgICAgIGZpbGVDb250ZW50czogY3R4LmNvbnRlbnQsXG4gICAgICAgIGVkaXRzOiBbbm9ybWFsaXplZF0sXG4gICAgICB9KVxuICAgICAgcmV0dXJuIHtcbiAgICAgICAgcGF0Y2g6IGFkanVzdEh1bmtMaW5lTnVtYmVycyhodW5rcywgY3R4LmxpbmVPZmZzZXQgLSAxKSxcbiAgICAgICAgZmlyc3RMaW5lOiBjdHgubGluZU9mZnNldCA9PT0gMSA/IGZpcnN0TGluZU9mKGN0eC5jb250ZW50KSA6IG51bGwsXG4gICAgICAgIGZpbGVDb250ZW50OiBjdHguY29udGVudCxcbiAgICAgIH1cbiAgICB9IGZpbmFsbHkge1xuICAgICAgYXdhaXQgaGFuZGxlLmNsb3NlKClcbiAgICB9XG4gIH0gY2F0Y2ggKGUpIHtcbiAgICBsb2dFcnJvcihlIGFzIEVycm9yKVxuICAgIHJldHVybiBkaWZmVG9vbElucHV0c09ubHkoZmlsZV9wYXRoLCB2YWxpZClcbiAgfVxufVxuXG5mdW5jdGlvbiBkaWZmVG9vbElucHV0c09ubHkoZmlsZVBhdGg6IHN0cmluZywgZWRpdHM6IEZpbGVFZGl0W10pOiBEaWZmRGF0YSB7XG4gIHJldHVybiB7XG4gICAgcGF0Y2g6IGVkaXRzLmZsYXRNYXAoZSA9PlxuICAgICAgZ2V0UGF0Y2hGb3JEaXNwbGF5KHtcbiAgICAgICAgZmlsZVBhdGgsXG4gICAgICAgIGZpbGVDb250ZW50czogZS5vbGRfc3RyaW5nLFxuICAgICAgICBlZGl0czogW2VdLFxuICAgICAgfSksXG4gICAgKSxcbiAgICBmaXJzdExpbmU6IG51bGwsXG4gICAgZmlsZUNvbnRlbnQ6IHVuZGVmaW5lZCxcbiAgfVxufVxuXG5mdW5jdGlvbiBub3JtYWxpemVFZGl0KGZpbGVDb250ZW50OiBzdHJpbmcsIGVkaXQ6IEZpbGVFZGl0KTogRmlsZUVkaXQge1xuICBjb25zdCBhY3R1YWxPbGQgPVxuICAgIGZpbmRBY3R1YWxTdHJpbmcoZmlsZUNvbnRlbnQsIGVkaXQub2xkX3N0cmluZykgfHwgZWRpdC5vbGRfc3RyaW5nXG4gIGNvbnN0IGFjdHVhbE5ldyA9IHByZXNlcnZlUXVvdGVTdHlsZShcbiAgICBlZGl0Lm9sZF9zdHJpbmcsXG4gICAgYWN0dWFsT2xkLFxuICAgIGVkaXQubmV3X3N0cmluZyxcbiAgKVxuICByZXR1cm4geyAuLi5lZGl0LCBvbGRfc3RyaW5nOiBhY3R1YWxPbGQsIG5ld19zdHJpbmc6IGFjdHVhbE5ldyB9XG59XG4iXSwibWFwcGluZ3MiOiI7QUFBQSxjQUFjQSxtQkFBbUIsUUFBUSxNQUFNO0FBQy9DLE9BQU8sS0FBS0MsS0FBSyxNQUFNLE9BQU87QUFDOUIsU0FBU0MsUUFBUSxFQUFFQyxHQUFHLEVBQUVDLFFBQVEsUUFBUSxPQUFPO0FBQy9DLFNBQVNDLGVBQWUsUUFBUSw2QkFBNkI7QUFDN0QsU0FBU0MsR0FBRyxFQUFFQyxJQUFJLFFBQVEsV0FBVztBQUNyQyxjQUFjQyxRQUFRLFFBQVEsZ0NBQWdDO0FBQzlELFNBQ0VDLGdCQUFnQixFQUNoQkMsa0JBQWtCLFFBQ2IsZ0NBQWdDO0FBQ3ZDLFNBQ0VDLHFCQUFxQixFQUNyQkMsYUFBYSxFQUNiQyxrQkFBa0IsUUFDYixrQkFBa0I7QUFDekIsU0FBU0MsUUFBUSxRQUFRLGlCQUFpQjtBQUMxQyxTQUNFQyxVQUFVLEVBQ1ZDLFdBQVcsRUFDWEMsVUFBVSxFQUNWQyxjQUFjLFFBQ1QsNkJBQTZCO0FBQ3BDLFNBQVNDLFdBQVcsUUFBUSx5QkFBeUI7QUFDckQsU0FBU0Msa0JBQWtCLFFBQVEseUJBQXlCO0FBRTVELEtBQUtDLEtBQUssR0FBRztFQUNYQyxTQUFTLEVBQUUsTUFBTTtFQUNqQkMsS0FBSyxFQUFFZixRQUFRLEVBQUU7QUFDbkIsQ0FBQztBQUVELEtBQUtnQixRQUFRLEdBQUc7RUFDZEMsS0FBSyxFQUFFekIsbUJBQW1CLEVBQUU7RUFDNUIwQixTQUFTLEVBQUUsTUFBTSxHQUFHLElBQUk7RUFDeEJDLFdBQVcsRUFBRSxNQUFNLEdBQUcsU0FBUztBQUNqQyxDQUFDO0FBRUQsT0FBTyxTQUFBQyxpQkFBQUMsS0FBQTtFQUFBLE1BQUFDLENBQUEsR0FBQUMsRUFBQTtFQUFBLElBQUFDLEVBQUE7RUFBQSxJQUFBRixDQUFBLFFBQUFELEtBQUEsQ0FBQU4sS0FBQSxJQUFBTyxDQUFBLFFBQUFELEtBQUEsQ0FBQVAsU0FBQTtJQUkwQlUsRUFBQSxHQUFBQSxDQUFBLEtBQzdCQyxZQUFZLENBQUNKLEtBQUssQ0FBQVAsU0FBVSxFQUFFTyxLQUFLLENBQUFOLEtBQU0sQ0FBQztJQUFBTyxDQUFBLE1BQUFELEtBQUEsQ0FBQU4sS0FBQTtJQUFBTyxDQUFBLE1BQUFELEtBQUEsQ0FBQVAsU0FBQTtJQUFBUSxDQUFBLE1BQUFFLEVBQUE7RUFBQTtJQUFBQSxFQUFBLEdBQUFGLENBQUE7RUFBQTtFQUQ1QyxPQUFBSSxXQUFBLElBQXNCOUIsUUFBUSxDQUFDNEIsRUFFL0IsQ0FBQztFQUFBLElBQUFHLEVBQUE7RUFBQSxJQUFBTCxDQUFBLFFBQUFNLE1BQUEsQ0FBQUMsR0FBQTtJQUVxQkYsRUFBQSxJQUFDLFNBQVMsQ0FBQyxXQUFXLENBQVgsS0FBVSxDQUFDLEdBQUc7SUFBQUwsQ0FBQSxNQUFBSyxFQUFBO0VBQUE7SUFBQUEsRUFBQSxHQUFBTCxDQUFBO0VBQUE7RUFBQSxJQUFBUSxFQUFBO0VBQUEsSUFBQVIsQ0FBQSxRQUFBSSxXQUFBLElBQUFKLENBQUEsUUFBQUQsS0FBQSxDQUFBUCxTQUFBO0lBQTdDZ0IsRUFBQSxJQUFDLFFBQVEsQ0FBVyxRQUF5QixDQUF6QixDQUFBSCxFQUF3QixDQUFDLENBQzNDLENBQUMsUUFBUSxDQUFVRCxPQUFXLENBQVhBLFlBQVUsQ0FBQyxDQUFhLFNBQWUsQ0FBZixDQUFBTCxLQUFLLENBQUFQLFNBQVMsQ0FBQyxHQUM1RCxFQUZDLFFBQVEsQ0FFRTtJQUFBUSxDQUFBLE1BQUFJLFdBQUE7SUFBQUosQ0FBQSxNQUFBRCxLQUFBLENBQUFQLFNBQUE7SUFBQVEsQ0FBQSxNQUFBUSxFQUFBO0VBQUE7SUFBQUEsRUFBQSxHQUFBUixDQUFBO0VBQUE7RUFBQSxPQUZYUSxFQUVXO0FBQUE7QUFJZixTQUFBQyxTQUFBUCxFQUFBO0VBQUEsTUFBQUYsQ0FBQSxHQUFBQyxFQUFBO0VBQWtCO0lBQUFTLE9BQUE7SUFBQWxCO0VBQUEsSUFBQVUsRUFNakI7RUFDQztJQUFBUCxLQUFBO0lBQUFDLFNBQUE7SUFBQUM7RUFBQSxJQUEwQ3hCLEdBQUcsQ0FBQ3FDLE9BQU8sQ0FBQztFQUN0RDtJQUFBQztFQUFBLElBQW9CcEMsZUFBZSxDQUFDLENBQUM7RUFBQSxJQUFBOEIsRUFBQTtFQUFBLElBQUFMLENBQUEsUUFBQVcsT0FBQSxJQUFBWCxDQUFBLFFBQUFILFdBQUEsSUFBQUcsQ0FBQSxRQUFBUixTQUFBLElBQUFRLENBQUEsUUFBQUosU0FBQSxJQUFBSSxDQUFBLFFBQUFMLEtBQUE7SUFFbkNVLEVBQUEsSUFBQyxTQUFTLENBQ1IsQ0FBQyxrQkFBa0IsQ0FDVlYsS0FBSyxDQUFMQSxNQUFJLENBQUMsQ0FDUCxHQUFLLENBQUwsTUFBSSxDQUFDLENBQ0hnQixLQUFPLENBQVBBLFFBQU0sQ0FBQyxDQUNKbkIsUUFBUyxDQUFUQSxVQUFRLENBQUMsQ0FDUkksU0FBUyxDQUFUQSxVQUFRLENBQUMsQ0FDUEMsV0FBVyxDQUFYQSxZQUFVLENBQUMsR0FFNUIsRUFUQyxTQUFTLENBU0U7SUFBQUcsQ0FBQSxNQUFBVyxPQUFBO0lBQUFYLENBQUEsTUFBQUgsV0FBQTtJQUFBRyxDQUFBLE1BQUFSLFNBQUE7SUFBQVEsQ0FBQSxNQUFBSixTQUFBO0lBQUFJLENBQUEsTUFBQUwsS0FBQTtJQUFBSyxDQUFBLE1BQUFLLEVBQUE7RUFBQTtJQUFBQSxFQUFBLEdBQUFMLENBQUE7RUFBQTtFQUFBLE9BVFpLLEVBU1k7QUFBQTtBQUloQixTQUFBTyxVQUFBVixFQUFBO0VBQUEsTUFBQUYsQ0FBQSxHQUFBQyxFQUFBO0VBQW1CO0lBQUFZLFFBQUE7SUFBQUM7RUFBQSxJQUFBWixFQU1sQjtFQUFBLElBQUFHLEVBQUE7RUFBQSxJQUFBTCxDQUFBLFFBQUFhLFFBQUEsSUFBQWIsQ0FBQSxRQUFBYyxXQUFBO0lBVVFULEVBQUEsR0FBQVMsV0FBVyxHQUFHLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBUixLQUFPLENBQUMsQ0FBQyxDQUFDLEVBQWYsSUFBSSxDQUE2QixHQUFoREQsUUFBZ0Q7SUFBQWIsQ0FBQSxNQUFBYSxRQUFBO0lBQUFiLENBQUEsTUFBQWMsV0FBQTtJQUFBZCxDQUFBLE1BQUFLLEVBQUE7RUFBQTtJQUFBQSxFQUFBLEdBQUFMLENBQUE7RUFBQTtFQUFBLElBQUFRLEVBQUE7RUFBQSxJQUFBUixDQUFBLFFBQUFLLEVBQUE7SUFSckRHLEVBQUEsSUFBQyxHQUFHLENBQWUsYUFBUSxDQUFSLFFBQVEsQ0FDekIsQ0FBQyxHQUFHLENBQ1UsV0FBUSxDQUFSLFFBQVEsQ0FDUixXQUFRLENBQVIsUUFBUSxDQUNOLGFBQVEsQ0FBUixRQUFRLENBQ1YsVUFBSyxDQUFMLE1BQUksQ0FBQyxDQUNKLFdBQUssQ0FBTCxNQUFJLENBQUMsQ0FFakIsQ0FBQUgsRUFBK0MsQ0FDbEQsRUFSQyxHQUFHLENBU04sRUFWQyxHQUFHLENBVUU7SUFBQUwsQ0FBQSxNQUFBSyxFQUFBO0lBQUFMLENBQUEsTUFBQVEsRUFBQTtFQUFBO0lBQUFBLEVBQUEsR0FBQVIsQ0FBQTtFQUFBO0VBQUEsT0FWTlEsRUFVTTtBQUFBO0FBSVYsZUFBZUwsWUFBWUEsQ0FDekJYLFNBQVMsRUFBRSxNQUFNLEVBQ2pCQyxLQUFLLEVBQUVmLFFBQVEsRUFBRSxDQUNsQixFQUFFcUMsT0FBTyxDQUFDckIsUUFBUSxDQUFDLENBQUM7RUFDbkIsTUFBTXNCLEtBQUssR0FBR3ZCLEtBQUssQ0FBQ3dCLE1BQU0sQ0FBQ0MsQ0FBQyxJQUFJQSxDQUFDLENBQUNDLFVBQVUsSUFBSSxJQUFJLElBQUlELENBQUMsQ0FBQ0UsVUFBVSxJQUFJLElBQUksQ0FBQztFQUM3RSxNQUFNQyxNQUFNLEdBQUdMLEtBQUssQ0FBQ00sTUFBTSxLQUFLLENBQUMsR0FBR04sS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLEdBQUdPLFNBQVM7O0VBRXpEO0VBQ0E7RUFDQTtFQUNBLElBQUlGLE1BQU0sSUFBSUEsTUFBTSxDQUFDRixVQUFVLENBQUNHLE1BQU0sSUFBSXJDLFVBQVUsRUFBRTtJQUNwRCxPQUFPdUMsa0JBQWtCLENBQUNoQyxTQUFTLEVBQUUsQ0FBQzZCLE1BQU0sQ0FBQyxDQUFDO0VBQ2hEO0VBRUEsSUFBSTtJQUNGLE1BQU1JLE1BQU0sR0FBRyxNQUFNdkMsV0FBVyxDQUFDTSxTQUFTLENBQUM7SUFDM0MsSUFBSWlDLE1BQU0sS0FBSyxJQUFJLEVBQUUsT0FBT0Qsa0JBQWtCLENBQUNoQyxTQUFTLEVBQUV3QixLQUFLLENBQUM7SUFDaEUsSUFBSTtNQUNGO01BQ0E7TUFDQTtNQUNBO01BQ0EsSUFBSSxDQUFDSyxNQUFNLElBQUlBLE1BQU0sQ0FBQ0YsVUFBVSxLQUFLLEVBQUUsRUFBRTtRQUN2QyxNQUFNTyxJQUFJLEdBQUcsTUFBTXZDLFVBQVUsQ0FBQ3NDLE1BQU0sQ0FBQztRQUNyQyxJQUFJQyxJQUFJLEtBQUssSUFBSSxFQUFFLE9BQU9GLGtCQUFrQixDQUFDaEMsU0FBUyxFQUFFd0IsS0FBSyxDQUFDO1FBQzlELE1BQU1XLFVBQVUsR0FBR1gsS0FBSyxDQUFDWSxHQUFHLENBQUNWLENBQUMsSUFBSVcsYUFBYSxDQUFDSCxJQUFJLEVBQUVSLENBQUMsQ0FBQyxDQUFDO1FBQ3pELE9BQU87VUFDTHZCLEtBQUssRUFBRVosa0JBQWtCLENBQUM7WUFDeEIrQyxRQUFRLEVBQUV0QyxTQUFTO1lBQ25CdUMsWUFBWSxFQUFFTCxJQUFJO1lBQ2xCakMsS0FBSyxFQUFFa0M7VUFDVCxDQUFDLENBQUM7VUFDRi9CLFNBQVMsRUFBRVAsV0FBVyxDQUFDcUMsSUFBSSxDQUFDO1VBQzVCN0IsV0FBVyxFQUFFNkI7UUFDZixDQUFDO01BQ0g7TUFFQSxNQUFNTSxHQUFHLEdBQUcsTUFBTTVDLGNBQWMsQ0FBQ3FDLE1BQU0sRUFBRUosTUFBTSxDQUFDRixVQUFVLEVBQUVyQyxhQUFhLENBQUM7TUFDMUUsSUFBSWtELEdBQUcsQ0FBQ0MsU0FBUyxJQUFJRCxHQUFHLENBQUNFLE9BQU8sS0FBSyxFQUFFLEVBQUU7UUFDdkMsT0FBT1Ysa0JBQWtCLENBQUNoQyxTQUFTLEVBQUUsQ0FBQzZCLE1BQU0sQ0FBQyxDQUFDO01BQ2hEO01BQ0EsTUFBTU0sVUFBVSxHQUFHRSxhQUFhLENBQUNHLEdBQUcsQ0FBQ0UsT0FBTyxFQUFFYixNQUFNLENBQUM7TUFDckQsTUFBTWMsS0FBSyxHQUFHcEQsa0JBQWtCLENBQUM7UUFDL0IrQyxRQUFRLEVBQUV0QyxTQUFTO1FBQ25CdUMsWUFBWSxFQUFFQyxHQUFHLENBQUNFLE9BQU87UUFDekJ6QyxLQUFLLEVBQUUsQ0FBQ2tDLFVBQVU7TUFDcEIsQ0FBQyxDQUFDO01BQ0YsT0FBTztRQUNMaEMsS0FBSyxFQUFFZCxxQkFBcUIsQ0FBQ3NELEtBQUssRUFBRUgsR0FBRyxDQUFDSSxVQUFVLEdBQUcsQ0FBQyxDQUFDO1FBQ3ZEeEMsU0FBUyxFQUFFb0MsR0FBRyxDQUFDSSxVQUFVLEtBQUssQ0FBQyxHQUFHL0MsV0FBVyxDQUFDMkMsR0FBRyxDQUFDRSxPQUFPLENBQUMsR0FBRyxJQUFJO1FBQ2pFckMsV0FBVyxFQUFFbUMsR0FBRyxDQUFDRTtNQUNuQixDQUFDO0lBQ0gsQ0FBQyxTQUFTO01BQ1IsTUFBTVQsTUFBTSxDQUFDWSxLQUFLLENBQUMsQ0FBQztJQUN0QjtFQUNGLENBQUMsQ0FBQyxPQUFPbkIsQ0FBQyxFQUFFO0lBQ1ZsQyxRQUFRLENBQUNrQyxDQUFDLElBQUlvQixLQUFLLENBQUM7SUFDcEIsT0FBT2Qsa0JBQWtCLENBQUNoQyxTQUFTLEVBQUV3QixLQUFLLENBQUM7RUFDN0M7QUFDRjtBQUVBLFNBQVNRLGtCQUFrQkEsQ0FBQ00sUUFBUSxFQUFFLE1BQU0sRUFBRXJDLEtBQUssRUFBRWYsUUFBUSxFQUFFLENBQUMsRUFBRWdCLFFBQVEsQ0FBQztFQUN6RSxPQUFPO0lBQ0xDLEtBQUssRUFBRUYsS0FBSyxDQUFDOEMsT0FBTyxDQUFDckIsQ0FBQyxJQUNwQm5DLGtCQUFrQixDQUFDO01BQ2pCK0MsUUFBUTtNQUNSQyxZQUFZLEVBQUViLENBQUMsQ0FBQ0MsVUFBVTtNQUMxQjFCLEtBQUssRUFBRSxDQUFDeUIsQ0FBQztJQUNYLENBQUMsQ0FDSCxDQUFDO0lBQ0R0QixTQUFTLEVBQUUsSUFBSTtJQUNmQyxXQUFXLEVBQUUwQjtFQUNmLENBQUM7QUFDSDtBQUVBLFNBQVNNLGFBQWFBLENBQUNoQyxXQUFXLEVBQUUsTUFBTSxFQUFFMkMsSUFBSSxFQUFFOUQsUUFBUSxDQUFDLEVBQUVBLFFBQVEsQ0FBQztFQUNwRSxNQUFNK0QsU0FBUyxHQUNiOUQsZ0JBQWdCLENBQUNrQixXQUFXLEVBQUUyQyxJQUFJLENBQUNyQixVQUFVLENBQUMsSUFBSXFCLElBQUksQ0FBQ3JCLFVBQVU7RUFDbkUsTUFBTXVCLFNBQVMsR0FBRzlELGtCQUFrQixDQUNsQzRELElBQUksQ0FBQ3JCLFVBQVUsRUFDZnNCLFNBQVMsRUFDVEQsSUFBSSxDQUFDcEIsVUFDUCxDQUFDO0VBQ0QsT0FBTztJQUFFLEdBQUdvQixJQUFJO0lBQUVyQixVQUFVLEVBQUVzQixTQUFTO0lBQUVyQixVQUFVLEVBQUVzQjtFQUFVLENBQUM7QUFDbEUiLCJpZ25vcmVMaXN0IjpbXX0= \ No newline at end of file diff --git a/src/components/FileEditToolUpdatedMessage.tsx b/src/components/FileEditToolUpdatedMessage.tsx new file mode 100644 index 0000000..909889a --- /dev/null +++ b/src/components/FileEditToolUpdatedMessage.tsx @@ -0,0 +1,124 @@ +import { c as _c } from "react/compiler-runtime"; +import type { StructuredPatchHunk } from 'diff'; +import * as React from 'react'; +import { useTerminalSize } from '../hooks/useTerminalSize.js'; +import { Box, Text } from '../ink.js'; +import { count } from '../utils/array.js'; +import { MessageResponse } from './MessageResponse.js'; +import { StructuredDiffList } from './StructuredDiffList.js'; +type Props = { + filePath: string; + structuredPatch: StructuredPatchHunk[]; + firstLine: string | null; + fileContent?: string; + style?: 'condensed'; + verbose: boolean; + previewHint?: string; +}; +export function FileEditToolUpdatedMessage(t0) { + const $ = _c(22); + const { + filePath, + structuredPatch, + firstLine, + fileContent, + style, + verbose, + previewHint + } = t0; + const { + columns + } = useTerminalSize(); + const numAdditions = structuredPatch.reduce(_temp2, 0); + const numRemovals = structuredPatch.reduce(_temp4, 0); + let t1; + if ($[0] !== numAdditions) { + t1 = numAdditions > 0 ? <>Added {numAdditions}{" "}{numAdditions > 1 ? "lines" : "line"} : null; + $[0] = numAdditions; + $[1] = t1; + } else { + t1 = $[1]; + } + const t2 = numAdditions > 0 && numRemovals > 0 ? ", " : null; + let t3; + if ($[2] !== numAdditions || $[3] !== numRemovals) { + t3 = numRemovals > 0 ? <>{numAdditions === 0 ? "R" : "r"}emoved {numRemovals}{" "}{numRemovals > 1 ? "lines" : "line"} : null; + $[2] = numAdditions; + $[3] = numRemovals; + $[4] = t3; + } else { + t3 = $[4]; + } + let t4; + if ($[5] !== t1 || $[6] !== t2 || $[7] !== t3) { + t4 = {t1}{t2}{t3}; + $[5] = t1; + $[6] = t2; + $[7] = t3; + $[8] = t4; + } else { + t4 = $[8]; + } + const text = t4; + if (previewHint) { + if (style !== "condensed" && !verbose) { + let t5; + if ($[9] !== previewHint) { + t5 = {previewHint}; + $[9] = previewHint; + $[10] = t5; + } else { + t5 = $[10]; + } + return t5; + } + } else { + if (style === "condensed" && !verbose) { + return text; + } + } + let t5; + if ($[11] !== text) { + t5 = {text}; + $[11] = text; + $[12] = t5; + } else { + t5 = $[12]; + } + const t6 = columns - 12; + let t7; + if ($[13] !== fileContent || $[14] !== filePath || $[15] !== firstLine || $[16] !== structuredPatch || $[17] !== t6) { + t7 = ; + $[13] = fileContent; + $[14] = filePath; + $[15] = firstLine; + $[16] = structuredPatch; + $[17] = t6; + $[18] = t7; + } else { + t7 = $[18]; + } + let t8; + if ($[19] !== t5 || $[20] !== t7) { + t8 = {t5}{t7}; + $[19] = t5; + $[20] = t7; + $[21] = t8; + } else { + t8 = $[21]; + } + return t8; +} +function _temp4(acc_0, hunk_0) { + return acc_0 + count(hunk_0.lines, _temp3); +} +function _temp3(__0) { + return __0.startsWith("-"); +} +function _temp2(acc, hunk) { + return acc + count(hunk.lines, _temp); +} +function _temp(_) { + return _.startsWith("+"); +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJTdHJ1Y3R1cmVkUGF0Y2hIdW5rIiwiUmVhY3QiLCJ1c2VUZXJtaW5hbFNpemUiLCJCb3giLCJUZXh0IiwiY291bnQiLCJNZXNzYWdlUmVzcG9uc2UiLCJTdHJ1Y3R1cmVkRGlmZkxpc3QiLCJQcm9wcyIsImZpbGVQYXRoIiwic3RydWN0dXJlZFBhdGNoIiwiZmlyc3RMaW5lIiwiZmlsZUNvbnRlbnQiLCJzdHlsZSIsInZlcmJvc2UiLCJwcmV2aWV3SGludCIsIkZpbGVFZGl0VG9vbFVwZGF0ZWRNZXNzYWdlIiwidDAiLCIkIiwiX2MiLCJjb2x1bW5zIiwibnVtQWRkaXRpb25zIiwicmVkdWNlIiwiX3RlbXAyIiwibnVtUmVtb3ZhbHMiLCJfdGVtcDQiLCJ0MSIsInQyIiwidDMiLCJ0NCIsInRleHQiLCJ0NSIsInQ2IiwidDciLCJ0OCIsImFjY18wIiwiaHVua18wIiwiYWNjIiwiaHVuayIsImxpbmVzIiwiX3RlbXAzIiwiX18wIiwiXyIsInN0YXJ0c1dpdGgiLCJfdGVtcCJdLCJzb3VyY2VzIjpbIkZpbGVFZGl0VG9vbFVwZGF0ZWRNZXNzYWdlLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IFN0cnVjdHVyZWRQYXRjaEh1bmsgfSBmcm9tICdkaWZmJ1xuaW1wb3J0ICogYXMgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyB1c2VUZXJtaW5hbFNpemUgfSBmcm9tICcuLi9ob29rcy91c2VUZXJtaW5hbFNpemUuanMnXG5pbXBvcnQgeyBCb3gsIFRleHQgfSBmcm9tICcuLi9pbmsuanMnXG5pbXBvcnQgeyBjb3VudCB9IGZyb20gJy4uL3V0aWxzL2FycmF5LmpzJ1xuaW1wb3J0IHsgTWVzc2FnZVJlc3BvbnNlIH0gZnJvbSAnLi9NZXNzYWdlUmVzcG9uc2UuanMnXG5pbXBvcnQgeyBTdHJ1Y3R1cmVkRGlmZkxpc3QgfSBmcm9tICcuL1N0cnVjdHVyZWREaWZmTGlzdC5qcydcblxudHlwZSBQcm9wcyA9IHtcbiAgZmlsZVBhdGg6IHN0cmluZ1xuICBzdHJ1Y3R1cmVkUGF0Y2g6IFN0cnVjdHVyZWRQYXRjaEh1bmtbXVxuICBmaXJzdExpbmU6IHN0cmluZyB8IG51bGxcbiAgZmlsZUNvbnRlbnQ/OiBzdHJpbmdcbiAgc3R5bGU/OiAnY29uZGVuc2VkJ1xuICB2ZXJib3NlOiBib29sZWFuXG4gIHByZXZpZXdIaW50Pzogc3RyaW5nXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBGaWxlRWRpdFRvb2xVcGRhdGVkTWVzc2FnZSh7XG4gIGZpbGVQYXRoLFxuICBzdHJ1Y3R1cmVkUGF0Y2gsXG4gIGZpcnN0TGluZSxcbiAgZmlsZUNvbnRlbnQsXG4gIHN0eWxlLFxuICB2ZXJib3NlLFxuICBwcmV2aWV3SGludCxcbn06IFByb3BzKTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgY29uc3QgeyBjb2x1bW5zIH0gPSB1c2VUZXJtaW5hbFNpemUoKVxuICBjb25zdCBudW1BZGRpdGlvbnMgPSBzdHJ1Y3R1cmVkUGF0Y2gucmVkdWNlKFxuICAgIChhY2MsIGh1bmspID0+IGFjYyArIGNvdW50KGh1bmsubGluZXMsIF8gPT4gXy5zdGFydHNXaXRoKCcrJykpLFxuICAgIDAsXG4gIClcbiAgY29uc3QgbnVtUmVtb3ZhbHMgPSBzdHJ1Y3R1cmVkUGF0Y2gucmVkdWNlKFxuICAgIChhY2MsIGh1bmspID0+IGFjYyArIGNvdW50KGh1bmsubGluZXMsIF8gPT4gXy5zdGFydHNXaXRoKCctJykpLFxuICAgIDAsXG4gIClcblxuICBjb25zdCB0ZXh0ID0gKFxuICAgIDxUZXh0PlxuICAgICAge251bUFkZGl0aW9ucyA+IDAgPyAoXG4gICAgICAgIDw+XG4gICAgICAgICAgQWRkZWQgPFRleHQgYm9sZD57bnVtQWRkaXRpb25zfTwvVGV4dD57JyAnfVxuICAgICAgICAgIHtudW1BZGRpdGlvbnMgPiAxID8gJ2xpbmVzJyA6ICdsaW5lJ31cbiAgICAgICAgPC8+XG4gICAgICApIDogbnVsbH1cbiAgICAgIHtudW1BZGRpdGlvbnMgPiAwICYmIG51bVJlbW92YWxzID4gMCA/ICcsICcgOiBudWxsfVxuICAgICAge251bVJlbW92YWxzID4gMCA/IChcbiAgICAgICAgPD5cbiAgICAgICAgICB7bnVtQWRkaXRpb25zID09PSAwID8gJ1InIDogJ3InfWVtb3ZlZCA8VGV4dCBib2xkPntudW1SZW1vdmFsc308L1RleHQ+eycgJ31cbiAgICAgICAgICB7bnVtUmVtb3ZhbHMgPiAxID8gJ2xpbmVzJyA6ICdsaW5lJ31cbiAgICAgICAgPC8+XG4gICAgICApIDogbnVsbH1cbiAgICA8L1RleHQ+XG4gIClcblxuICAvLyBQbGFuIGZpbGVzOiBpbnZlcnQgY29uZGVuc2VkIGJlaGF2aW9yXG4gIC8vIC0gUmVndWxhciBtb2RlOiBqdXN0IHNob3cgdGhlIGhpbnQgKHVzZXIgY2FuIHR5cGUgL3BsYW4gdG8gc2VlIGZ1bGwgY29udGVudClcbiAgLy8gLSBDb25kZW5zZWQgbW9kZSAoc3ViYWdlbnQgdmlldyk6IHNob3cgdGhlIGRpZmZcbiAgaWYgKHByZXZpZXdIaW50KSB7XG4gICAgaWYgKHN0eWxlICE9PSAnY29uZGVuc2VkJyAmJiAhdmVyYm9zZSkge1xuICAgICAgcmV0dXJuIChcbiAgICAgICAgPE1lc3NhZ2VSZXNwb25zZT5cbiAgICAgICAgICA8VGV4dCBkaW1Db2xvcj57cHJldmlld0hpbnR9PC9UZXh0PlxuICAgICAgICA8L01lc3NhZ2VSZXNwb25zZT5cbiAgICAgIClcbiAgICB9XG4gIH0gZWxzZSBpZiAoc3R5bGUgPT09ICdjb25kZW5zZWQnICYmICF2ZXJib3NlKSB7XG4gICAgcmV0dXJuIHRleHRcbiAgfVxuXG4gIHJldHVybiAoXG4gICAgPE1lc3NhZ2VSZXNwb25zZT5cbiAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiPlxuICAgICAgICA8VGV4dD57dGV4dH08L1RleHQ+XG4gICAgICAgIDxTdHJ1Y3R1cmVkRGlmZkxpc3RcbiAgICAgICAgICBodW5rcz17c3RydWN0dXJlZFBhdGNofVxuICAgICAgICAgIGRpbT17ZmFsc2V9XG4gICAgICAgICAgd2lkdGg9e2NvbHVtbnMgLSAxMn1cbiAgICAgICAgICBmaWxlUGF0aD17ZmlsZVBhdGh9XG4gICAgICAgICAgZmlyc3RMaW5lPXtmaXJzdExpbmV9XG4gICAgICAgICAgZmlsZUNvbnRlbnQ9e2ZpbGVDb250ZW50fVxuICAgICAgICAvPlxuICAgICAgPC9Cb3g+XG4gICAgPC9NZXNzYWdlUmVzcG9uc2U+XG4gIClcbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLGNBQWNBLG1CQUFtQixRQUFRLE1BQU07QUFDL0MsT0FBTyxLQUFLQyxLQUFLLE1BQU0sT0FBTztBQUM5QixTQUFTQyxlQUFlLFFBQVEsNkJBQTZCO0FBQzdELFNBQVNDLEdBQUcsRUFBRUMsSUFBSSxRQUFRLFdBQVc7QUFDckMsU0FBU0MsS0FBSyxRQUFRLG1CQUFtQjtBQUN6QyxTQUFTQyxlQUFlLFFBQVEsc0JBQXNCO0FBQ3RELFNBQVNDLGtCQUFrQixRQUFRLHlCQUF5QjtBQUU1RCxLQUFLQyxLQUFLLEdBQUc7RUFDWEMsUUFBUSxFQUFFLE1BQU07RUFDaEJDLGVBQWUsRUFBRVYsbUJBQW1CLEVBQUU7RUFDdENXLFNBQVMsRUFBRSxNQUFNLEdBQUcsSUFBSTtFQUN4QkMsV0FBVyxDQUFDLEVBQUUsTUFBTTtFQUNwQkMsS0FBSyxDQUFDLEVBQUUsV0FBVztFQUNuQkMsT0FBTyxFQUFFLE9BQU87RUFDaEJDLFdBQVcsQ0FBQyxFQUFFLE1BQU07QUFDdEIsQ0FBQztBQUVELE9BQU8sU0FBQUMsMkJBQUFDLEVBQUE7RUFBQSxNQUFBQyxDQUFBLEdBQUFDLEVBQUE7RUFBb0M7SUFBQVYsUUFBQTtJQUFBQyxlQUFBO0lBQUFDLFNBQUE7SUFBQUMsV0FBQTtJQUFBQyxLQUFBO0lBQUFDLE9BQUE7SUFBQUM7RUFBQSxJQUFBRSxFQVFuQztFQUNOO0lBQUFHO0VBQUEsSUFBb0JsQixlQUFlLENBQUMsQ0FBQztFQUNyQyxNQUFBbUIsWUFBQSxHQUFxQlgsZUFBZSxDQUFBWSxNQUFPLENBQ3pDQyxNQUE4RCxFQUM5RCxDQUNGLENBQUM7RUFDRCxNQUFBQyxX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QTtFQUFBLElBQUFmLENBQUEsU0FBQU4sV0FBQSxJQUFBTSxDQUFBLFNBQUFULFFBQUEsSUFBQVMsQ0FBQSxTQUFBUCxTQUFBLElBQUFPLENBQUEsU0FBQVIsZUFBQSxJQUFBUSxDQUFBLFNBQUFjLEVBQUE7SUFIckJDLEVBQUEsSUFBQyxrQkFBa0IsQ0FDVnZCLEtBQWUsQ0FBZkEsZ0JBQWMsQ0FBQyxDQUNqQixHQUFLLENBQUwsTUFBSSxDQUFDLENBQ0gsS0FBWSxDQUFaLENBQUFzQixFQUFXLENBQUMsQ0FDVHZCLFFBQVEsQ0FBUkEsU0FBTyxDQUFDLENBQ1BFLFNBQVMsQ0FBVEEsVUFBUSxDQUFDLENBQ1BDLFdBQVcsQ0FBWEEsWUFBVSxDQUFDLEdBQ3hCO0lBQUFNLENBQUEsT0FBQU4sV0FBQTtJQUFBTSxDQUFBLE9BQUFULFFBQUE7SUFBQVMsQ0FBQSxPQUFBUCxTQUFBO0lBQUFPLENBQUEsT0FBQVIsZUFBQTtJQUFBUSxDQUFBLE9BQUFjLEVBQUE7SUFBQWQsQ0FBQSxPQUFBZSxFQUFBO0VBQUE7SUFBQUEsRUFBQSxHQUFBZixDQUFBO0VBQUE7RUFBQSxJQUFBZ0IsRUFBQTtFQUFBLElBQUFoQixDQUFBLFNBQUFhLEVBQUEsSUFBQWIsQ0FBQSxTQUFBZSxFQUFBO0lBVk5DLEVBQUEsSUFBQyxlQUFlLENBQ2QsQ0FBQyxHQUFHLENBQWUsYUFBUSxDQUFSLFFBQVEsQ0FDekIsQ0FBQUgsRUFBa0IsQ0FDbEIsQ0FBQUUsRUFPQyxDQUNILEVBVkMsR0FBRyxDQVdOLEVBWkMsZUFBZSxDQVlFO0lBQUFmLENBQUEsT0FBQWEsRUFBQTtJQUFBYixDQUFBLE9BQUFlLEVBQUE7SUFBQWYsQ0FBQSxPQUFBZ0IsRUFBQTtFQUFBO0lBQUFBLEVBQUEsR0FBQWhCLENBQUE7RUFBQTtFQUFBLE9BWmxCZ0IsRUFZa0I7QUFBQTtBQWpFZixTQUFBVCxPQUFBVSxLQUFBLEVBQUFDLE1BQUE7RUFBQSxPQWVZQyxLQUFHLEdBQUdoQyxLQUFLLENBQUNpQyxNQUFJLENBQUFDLEtBQU0sRUFBRUMsTUFBc0IsQ0FBQztBQUFBO0FBZjNELFNBQUFBLE9BQUFDLEdBQUE7RUFBQSxPQWV5Q0MsR0FBQyxDQUFBQyxVQUFXLENBQUMsR0FBRyxDQUFDO0FBQUE7QUFmMUQsU0FBQXBCLE9BQUFjLEdBQUEsRUFBQUMsSUFBQTtFQUFBLE9BV1lELEdBQUcsR0FBR2hDLEtBQUssQ0FBQ2lDLElBQUksQ0FBQUMsS0FBTSxFQUFFSyxLQUFzQixDQUFDO0FBQUE7QUFYM0QsU0FBQUEsTUFBQUYsQ0FBQTtFQUFBLE9BV3lDQSxDQUFDLENBQUFDLFVBQVcsQ0FBQyxHQUFHLENBQUM7QUFBQSIsImlnbm9yZUxpc3QiOltdfQ== \ No newline at end of file diff --git a/src/components/FileEditToolUseRejectedMessage.tsx b/src/components/FileEditToolUseRejectedMessage.tsx new file mode 100644 index 0000000..23bfd12 --- /dev/null +++ b/src/components/FileEditToolUseRejectedMessage.tsx @@ -0,0 +1,170 @@ +import { c as _c } from "react/compiler-runtime"; +import type { StructuredPatchHunk } from 'diff'; +import { relative } from 'path'; +import * as React from 'react'; +import { useTerminalSize } from 'src/hooks/useTerminalSize.js'; +import { getCwd } from 'src/utils/cwd.js'; +import { Box, Text } from '../ink.js'; +import { HighlightedCode } from './HighlightedCode.js'; +import { MessageResponse } from './MessageResponse.js'; +import { StructuredDiffList } from './StructuredDiffList.js'; +const MAX_LINES_TO_RENDER = 10; +type Props = { + file_path: string; + operation: 'write' | 'update'; + // For updates - show diff + patch?: StructuredPatchHunk[]; + firstLine: string | null; + fileContent?: string; + // For new file creation - show content preview + content?: string; + style?: 'condensed'; + verbose: boolean; +}; +export function FileEditToolUseRejectedMessage(t0) { + const $ = _c(38); + const { + file_path, + operation, + patch, + firstLine, + fileContent, + content, + style, + verbose + } = t0; + const { + columns + } = useTerminalSize(); + let t1; + if ($[0] !== operation) { + t1 = User rejected {operation} to ; + $[0] = operation; + $[1] = t1; + } else { + t1 = $[1]; + } + let t2; + if ($[2] !== file_path || $[3] !== verbose) { + t2 = verbose ? file_path : relative(getCwd(), file_path); + $[2] = file_path; + $[3] = verbose; + $[4] = t2; + } else { + t2 = $[4]; + } + let t3; + if ($[5] !== t2) { + t3 = {t2}; + $[5] = t2; + $[6] = t3; + } else { + t3 = $[6]; + } + let t4; + if ($[7] !== t1 || $[8] !== t3) { + t4 = {t1}{t3}; + $[7] = t1; + $[8] = t3; + $[9] = t4; + } else { + t4 = $[9]; + } + const text = t4; + if (style === "condensed" && !verbose) { + let t5; + if ($[10] !== text) { + t5 = {text}; + $[10] = text; + $[11] = t5; + } else { + t5 = $[11]; + } + return t5; + } + if (operation === "write" && content !== undefined) { + let plusLines; + let t5; + if ($[12] !== content || $[13] !== verbose) { + const lines = content.split("\n"); + const numLines = lines.length; + plusLines = numLines - MAX_LINES_TO_RENDER; + t5 = verbose ? content : lines.slice(0, MAX_LINES_TO_RENDER).join("\n"); + $[12] = content; + $[13] = verbose; + $[14] = plusLines; + $[15] = t5; + } else { + plusLines = $[14]; + t5 = $[15]; + } + const truncatedContent = t5; + const t6 = truncatedContent || "(No content)"; + const t7 = columns - 12; + let t8; + if ($[16] !== file_path || $[17] !== t6 || $[18] !== t7) { + t8 = ; + $[16] = file_path; + $[17] = t6; + $[18] = t7; + $[19] = t8; + } else { + t8 = $[19]; + } + let t9; + if ($[20] !== plusLines || $[21] !== verbose) { + t9 = !verbose && plusLines > 0 && … +{plusLines} lines; + $[20] = plusLines; + $[21] = verbose; + $[22] = t9; + } else { + t9 = $[22]; + } + let t10; + if ($[23] !== t8 || $[24] !== t9 || $[25] !== text) { + t10 = {text}{t8}{t9}; + $[23] = t8; + $[24] = t9; + $[25] = text; + $[26] = t10; + } else { + t10 = $[26]; + } + return t10; + } + if (!patch || patch.length === 0) { + let t5; + if ($[27] !== text) { + t5 = {text}; + $[27] = text; + $[28] = t5; + } else { + t5 = $[28]; + } + return t5; + } + const t5 = columns - 12; + let t6; + if ($[29] !== fileContent || $[30] !== file_path || $[31] !== firstLine || $[32] !== patch || $[33] !== t5) { + t6 = ; + $[29] = fileContent; + $[30] = file_path; + $[31] = firstLine; + $[32] = patch; + $[33] = t5; + $[34] = t6; + } else { + t6 = $[34]; + } + let t7; + if ($[35] !== t6 || $[36] !== text) { + t7 = {text}{t6}; + $[35] = t6; + $[36] = text; + $[37] = t7; + } else { + t7 = $[37]; + } + return t7; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJTdHJ1Y3R1cmVkUGF0Y2hIdW5rIiwicmVsYXRpdmUiLCJSZWFjdCIsInVzZVRlcm1pbmFsU2l6ZSIsImdldEN3ZCIsIkJveCIsIlRleHQiLCJIaWdobGlnaHRlZENvZGUiLCJNZXNzYWdlUmVzcG9uc2UiLCJTdHJ1Y3R1cmVkRGlmZkxpc3QiLCJNQVhfTElORVNfVE9fUkVOREVSIiwiUHJvcHMiLCJmaWxlX3BhdGgiLCJvcGVyYXRpb24iLCJwYXRjaCIsImZpcnN0TGluZSIsImZpbGVDb250ZW50IiwiY29udGVudCIsInN0eWxlIiwidmVyYm9zZSIsIkZpbGVFZGl0VG9vbFVzZVJlamVjdGVkTWVzc2FnZSIsInQwIiwiJCIsIl9jIiwiY29sdW1ucyIsInQxIiwidDIiLCJ0MyIsInQ0IiwidGV4dCIsInQ1IiwidW5kZWZpbmVkIiwicGx1c0xpbmVzIiwibGluZXMiLCJzcGxpdCIsIm51bUxpbmVzIiwibGVuZ3RoIiwic2xpY2UiLCJqb2luIiwidHJ1bmNhdGVkQ29udGVudCIsInQ2IiwidDciLCJ0OCIsInQ5IiwidDEwIl0sInNvdXJjZXMiOlsiRmlsZUVkaXRUb29sVXNlUmVqZWN0ZWRNZXNzYWdlLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IFN0cnVjdHVyZWRQYXRjaEh1bmsgfSBmcm9tICdkaWZmJ1xuaW1wb3J0IHsgcmVsYXRpdmUgfSBmcm9tICdwYXRoJ1xuaW1wb3J0ICogYXMgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyB1c2VUZXJtaW5hbFNpemUgfSBmcm9tICdzcmMvaG9va3MvdXNlVGVybWluYWxTaXplLmpzJ1xuaW1wb3J0IHsgZ2V0Q3dkIH0gZnJvbSAnc3JjL3V0aWxzL2N3ZC5qcydcbmltcG9ydCB7IEJveCwgVGV4dCB9IGZyb20gJy4uL2luay5qcydcbmltcG9ydCB7IEhpZ2hsaWdodGVkQ29kZSB9IGZyb20gJy4vSGlnaGxpZ2h0ZWRDb2RlLmpzJ1xuaW1wb3J0IHsgTWVzc2FnZVJlc3BvbnNlIH0gZnJvbSAnLi9NZXNzYWdlUmVzcG9uc2UuanMnXG5pbXBvcnQgeyBTdHJ1Y3R1cmVkRGlmZkxpc3QgfSBmcm9tICcuL1N0cnVjdHVyZWREaWZmTGlzdC5qcydcblxuY29uc3QgTUFYX0xJTkVTX1RPX1JFTkRFUiA9IDEwXG5cbnR5cGUgUHJvcHMgPSB7XG4gIGZpbGVfcGF0aDogc3RyaW5nXG4gIG9wZXJhdGlvbjogJ3dyaXRlJyB8ICd1cGRhdGUnXG4gIC8vIEZvciB1cGRhdGVzIC0gc2hvdyBkaWZmXG4gIHBhdGNoPzogU3RydWN0dXJlZFBhdGNoSHVua1tdXG4gIGZpcnN0TGluZTogc3RyaW5nIHwgbnVsbFxuICBmaWxlQ29udGVudD86IHN0cmluZ1xuICAvLyBGb3IgbmV3IGZpbGUgY3JlYXRpb24gLSBzaG93IGNvbnRlbnQgcHJldmlld1xuICBjb250ZW50Pzogc3RyaW5nXG4gIHN0eWxlPzogJ2NvbmRlbnNlZCdcbiAgdmVyYm9zZTogYm9vbGVhblxufVxuXG5leHBvcnQgZnVuY3Rpb24gRmlsZUVkaXRUb29sVXNlUmVqZWN0ZWRNZXNzYWdlKHtcbiAgZmlsZV9wYXRoLFxuICBvcGVyYXRpb24sXG4gIHBhdGNoLFxuICBmaXJzdExpbmUsXG4gIGZpbGVDb250ZW50LFxuICBjb250ZW50LFxuICBzdHlsZSxcbiAgdmVyYm9zZSxcbn06IFByb3BzKTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgY29uc3QgeyBjb2x1bW5zIH0gPSB1c2VUZXJtaW5hbFNpemUoKVxuICBjb25zdCB0ZXh0ID0gKFxuICAgIDxCb3ggZmxleERpcmVjdGlvbj1cInJvd1wiPlxuICAgICAgPFRleHQgY29sb3I9XCJzdWJ0bGVcIj5Vc2VyIHJlamVjdGVkIHtvcGVyYXRpb259IHRvIDwvVGV4dD5cbiAgICAgIDxUZXh0IGJvbGQgY29sb3I9XCJzdWJ0bGVcIj5cbiAgICAgICAge3ZlcmJvc2UgPyBmaWxlX3BhdGggOiByZWxhdGl2ZShnZXRDd2QoKSwgZmlsZV9wYXRoKX1cbiAgICAgIDwvVGV4dD5cbiAgICA8L0JveD5cbiAgKVxuXG4gIC8vIEZvciBjb25kZW5zZWQgc3R5bGUsIGp1c3Qgc2hvdyB0aGUgdGV4dFxuICBpZiAoc3R5bGUgPT09ICdjb25kZW5zZWQnICYmICF2ZXJib3NlKSB7XG4gICAgcmV0dXJuIDxNZXNzYWdlUmVzcG9uc2U+e3RleHR9PC9NZXNzYWdlUmVzcG9uc2U+XG4gIH1cblxuICAvLyBGb3IgbmV3IGZpbGUgY3JlYXRpb24sIHNob3cgY29udGVudCBwcmV2aWV3IChkaW1tZWQpXG4gIGlmIChvcGVyYXRpb24gPT09ICd3cml0ZScgJiYgY29udGVudCAhPT0gdW5kZWZpbmVkKSB7XG4gICAgY29uc3QgbGluZXMgPSBjb250ZW50LnNwbGl0KCdcXG4nKVxuICAgIGNvbnN0IG51bUxpbmVzID0gbGluZXMubGVuZ3RoXG4gICAgY29uc3QgcGx1c0xpbmVzID0gbnVtTGluZXMgLSBNQVhfTElORVNfVE9fUkVOREVSXG4gICAgY29uc3QgdHJ1bmNhdGVkQ29udGVudCA9IHZlcmJvc2VcbiAgICAgID8gY29udGVudFxuICAgICAgOiBsaW5lcy5zbGljZSgwLCBNQVhfTElORVNfVE9fUkVOREVSKS5qb2luKCdcXG4nKVxuXG4gICAgcmV0dXJuIChcbiAgICAgIDxNZXNzYWdlUmVzcG9uc2U+XG4gICAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiPlxuICAgICAgICAgIHt0ZXh0fVxuICAgICAgICAgIDxIaWdobGlnaHRlZENvZGVcbiAgICAgICAgICAgIGNvZGU9e3RydW5jYXRlZENvbnRlbnQgfHwgJyhObyBjb250ZW50KSd9XG4gICAgICAgICAgICBmaWxlUGF0aD17ZmlsZV9wYXRofVxuICAgICAgICAgICAgd2lkdGg9e2NvbHVtbnMgLSAxMn1cbiAgICAgICAgICAgIGRpbVxuICAgICAgICAgIC8+XG4gICAgICAgICAgeyF2ZXJib3NlICYmIHBsdXNMaW5lcyA+IDAgJiYgKFxuICAgICAgICAgICAgPFRleHQgZGltQ29sb3I+4oCmICt7cGx1c0xpbmVzfSBsaW5lczwvVGV4dD5cbiAgICAgICAgICApfVxuICAgICAgICA8L0JveD5cbiAgICAgIDwvTWVzc2FnZVJlc3BvbnNlPlxuICAgIClcbiAgfVxuXG4gIC8vIEZvciB1cGRhdGVzLCBzaG93IGRpZmZcbiAgaWYgKCFwYXRjaCB8fCBwYXRjaC5sZW5ndGggPT09IDApIHtcbiAgICByZXR1cm4gPE1lc3NhZ2VSZXNwb25zZT57dGV4dH08L01lc3NhZ2VSZXNwb25zZT5cbiAgfVxuXG4gIHJldHVybiAoXG4gICAgPE1lc3NhZ2VSZXNwb25zZT5cbiAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiPlxuICAgICAgICB7dGV4dH1cbiAgICAgICAgPFN0cnVjdHVyZWREaWZmTGlzdFxuICAgICAgICAgIGh1bmtzPXtwYXRjaH1cbiAgICAgICAgICBkaW1cbiAgICAgICAgICB3aWR0aD17Y29sdW1ucyAtIDEyfVxuICAgICAgICAgIGZpbGVQYXRoPXtmaWxlX3BhdGh9XG4gICAgICAgICAgZmlyc3RMaW5lPXtmaXJzdExpbmV9XG4gICAgICAgICAgZmlsZUNvbnRlbnQ9e2ZpbGVDb250ZW50fVxuICAgICAgICAvPlxuICAgICAgPC9Cb3g+XG4gICAgPC9NZXNzYWdlUmVzcG9uc2U+XG4gIClcbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLGNBQWNBLG1CQUFtQixRQUFRLE1BQU07QUFDL0MsU0FBU0MsUUFBUSxRQUFRLE1BQU07QUFDL0IsT0FBTyxLQUFLQyxLQUFLLE1BQU0sT0FBTztBQUM5QixTQUFTQyxlQUFlLFFBQVEsOEJBQThCO0FBQzlELFNBQVNDLE1BQU0sUUFBUSxrQkFBa0I7QUFDekMsU0FBU0MsR0FBRyxFQUFFQyxJQUFJLFFBQVEsV0FBVztBQUNyQyxTQUFTQyxlQUFlLFFBQVEsc0JBQXNCO0FBQ3RELFNBQVNDLGVBQWUsUUFBUSxzQkFBc0I7QUFDdEQsU0FBU0Msa0JBQWtCLFFBQVEseUJBQXlCO0FBRTVELE1BQU1DLG1CQUFtQixHQUFHLEVBQUU7QUFFOUIsS0FBS0MsS0FBSyxHQUFHO0VBQ1hDLFNBQVMsRUFBRSxNQUFN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 \ No newline at end of file diff --git a/src/components/FilePathLink.tsx b/src/components/FilePathLink.tsx new file mode 100644 index 0000000..5b2917a --- /dev/null +++ b/src/components/FilePathLink.tsx @@ -0,0 +1,43 @@ +import { c as _c } from "react/compiler-runtime"; +import React from 'react'; +import { pathToFileURL } from 'url'; +import Link from '../ink/components/Link.js'; +type Props = { + /** The absolute file path */ + filePath: string; + /** Optional display text (defaults to filePath) */ + children?: React.ReactNode; +}; + +/** + * Renders a file path as an OSC 8 hyperlink. + * This helps terminals like iTerm correctly identify file paths + * even when they appear inside parentheses or other text. + */ +export function FilePathLink(t0) { + const $ = _c(5); + const { + filePath, + children + } = t0; + let t1; + if ($[0] !== filePath) { + t1 = pathToFileURL(filePath); + $[0] = filePath; + $[1] = t1; + } else { + t1 = $[1]; + } + const t2 = children ?? filePath; + let t3; + if ($[2] !== t1.href || $[3] !== t2) { + t3 = {t2}; + $[2] = t1.href; + $[3] = t2; + $[4] = t3; + } else { + t3 = $[4]; + } + return t3; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsInBhdGhUb0ZpbGVVUkwiLCJMaW5rIiwiUHJvcHMiLCJmaWxlUGF0aCIsImNoaWxkcmVuIiwiUmVhY3ROb2RlIiwiRmlsZVBhdGhMaW5rIiwidDAiLCIkIiwiX2MiLCJ0MSIsInQyIiwidDMiLCJocmVmIl0sInNvdXJjZXMiOlsiRmlsZVBhdGhMaW5rLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBwYXRoVG9GaWxlVVJMIH0gZnJvbSAndXJsJ1xuaW1wb3J0IExpbmsgZnJvbSAnLi4vaW5rL2NvbXBvbmVudHMvTGluay5qcydcblxudHlwZSBQcm9wcyA9IHtcbiAgLyoqIFRoZSBhYnNvbHV0ZSBmaWxlIHBhdGggKi9cbiAgZmlsZVBhdGg6IHN0cmluZ1xuICAvKiogT3B0aW9uYWwgZGlzcGxheSB0ZXh0IChkZWZhdWx0cyB0byBmaWxlUGF0aCkgKi9cbiAgY2hpbGRyZW4/OiBSZWFjdC5SZWFjdE5vZGVcbn1cblxuLyoqXG4gKiBSZW5kZXJzIGEgZmlsZSBwYXRoIGFzIGFuIE9TQyA4IGh5cGVybGluay5cbiAqIFRoaXMgaGVscHMgdGVybWluYWxzIGxpa2UgaVRlcm0gY29ycmVjdGx5IGlkZW50aWZ5IGZpbGUgcGF0aHNcbiAqIGV2ZW4gd2hlbiB0aGV5IGFwcGVhciBpbnNpZGUgcGFyZW50aGVzZXMgb3Igb3RoZXIgdGV4dC5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIEZpbGVQYXRoTGluayh7IGZpbGVQYXRoLCBjaGlsZHJlbiB9OiBQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIHJldHVybiA8TGluayB1cmw9e3BhdGhUb0ZpbGVVUkwoZmlsZVBhdGgpLmhyZWZ9PntjaGlsZHJlbiA/PyBmaWxlUGF0aH08L0xpbms+XG59XG4iXSwibWFwcGluZ3MiOiI7QUFBQSxPQUFPQSxLQUFLLE1BQU0sT0FBTztBQUN6QixTQUFTQyxhQUFhLFFBQVEsS0FBSztBQUNuQyxPQUFPQyxJQUFJLE1BQU0sMkJBQTJCO0FBRTVDLEtBQUtDLEtBQUssR0FBRztFQUNYO0VBQ0FDLFFBQVEsRUFBRSxNQUFNO0VBQ2hCO0VBQ0FDLFFBQVEsQ0FBQyxFQUFFTCxLQUFLLENBQUNNLFNBQVM7QUFDNUIsQ0FBQzs7QUFFRDtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsT0FBTyxTQUFBQyxhQUFBQyxFQUFBO0VBQUEsTUFBQUMsQ0FBQSxHQUFBQyxFQUFBO0VBQXNCO0lBQUFOLFFBQUE7SUFBQUM7RUFBQSxJQUFBRyxFQUE2QjtFQUFBLElBQUFHLEVBQUE7RUFBQSxJQUFBRixDQUFBLFFBQUFMLFFBQUE7SUFDdENPLEVBQUEsR0FBQVYsYUFBYSxDQUFDRyxRQUFRLENBQUM7SUFBQUssQ0FBQSxNQUFBTCxRQUFBO0lBQUFLLENBQUEsTUFBQUUsRUFBQTtFQUFBO0lBQUFBLEVBQUEsR0FBQUYsQ0FBQTtFQUFBO0VBQVEsTUFBQUcsRUFBQSxHQUFBUCxRQUFvQixJQUFwQkQsUUFBb0I7RUFBQSxJQUFBUyxFQUFBO0VBQUEsSUFBQUosQ0FBQSxRQUFBRSxFQUFBLENBQUFHLElBQUEsSUFBQUwsQ0FBQSxRQUFBRyxFQUFBO0lBQTlEQyxFQUFBLElBQUMsSUFBSSxDQUFNLEdBQTRCLENBQTVCLENBQUFGLEVBQXVCLENBQUFHLElBQUksQ0FBQyxDQUFHLENBQUFGLEVBQW1CLENBQUUsRUFBOUQsSUFBSSxDQUFpRTtJQUFBSCxDQUFBLE1BQUFFLEVBQUEsQ0FBQUcsSUFBQTtJQUFBTCxDQUFBLE1BQUFHLEVBQUE7SUFBQUgsQ0FBQSxNQUFBSSxFQUFBO0VBQUE7SUFBQUEsRUFBQSxHQUFBSixDQUFBO0VBQUE7RUFBQSxPQUF0RUksRUFBc0U7QUFBQSIsImlnbm9yZUxpc3QiOltdfQ== \ No newline at end of file diff --git a/src/components/FullscreenLayout.tsx b/src/components/FullscreenLayout.tsx new file mode 100644 index 0000000..2475ec6 --- /dev/null +++ b/src/components/FullscreenLayout.tsx @@ -0,0 +1,637 @@ +import { c as _c } from "react/compiler-runtime"; +import figures from 'figures'; +import React, { createContext, type ReactNode, type RefObject, useCallback, useEffect, useLayoutEffect, useMemo, useRef, useState, useSyncExternalStore } from 'react'; +import { fileURLToPath } from 'url'; +import { ModalContext } from '../context/modalContext.js'; +import { PromptOverlayProvider, usePromptOverlay, usePromptOverlayDialog } from '../context/promptOverlayContext.js'; +import { useTerminalSize } from '../hooks/useTerminalSize.js'; +import ScrollBox, { type ScrollBoxHandle } from '../ink/components/ScrollBox.js'; +import instances from '../ink/instances.js'; +import { Box, Text } from '../ink.js'; +import type { Message } from '../types/message.js'; +import { openBrowser, openPath } from '../utils/browser.js'; +import { isFullscreenEnvEnabled } from '../utils/fullscreen.js'; +import { plural } from '../utils/stringUtils.js'; +import { isNullRenderingAttachment } from './messages/nullRenderingAttachments.js'; +import PromptInputFooterSuggestions from './PromptInput/PromptInputFooterSuggestions.js'; +import type { StickyPrompt } from './VirtualMessageList.js'; + +/** Rows of transcript context kept visible above the modal pane's ▔ divider. */ +const MODAL_TRANSCRIPT_PEEK = 2; + +/** Context for scroll-derived chrome (sticky header, pill). StickyTracker + * in VirtualMessageList writes via this instead of threading a callback + * up through Messages → REPL → FullscreenLayout. The setter is stable so + * consuming this context never causes re-renders. */ +export const ScrollChromeContext = createContext<{ + setStickyPrompt: (p: StickyPrompt | null) => void; +}>({ + setStickyPrompt: () => {} +}); +type Props = { + /** Content that scrolls (messages, tool output) */ + scrollable: ReactNode; + /** Content pinned to the bottom (spinner, prompt, permissions) */ + bottom: ReactNode; + /** Content rendered inside the ScrollBox after messages — user can scroll + * up to see context while it's showing (used by PermissionRequest). */ + overlay?: ReactNode; + /** Absolute-positioned content anchored at the bottom-right of the + * ScrollBox area, floating over scrollback. Rendered inside the flexGrow + * region (not the bottom slot) so the overflowY:hidden cap doesn't clip + * it. Fullscreen only — used for the companion speech bubble. */ + bottomFloat?: ReactNode; + /** Slash-command dialog content. Rendered in an absolute-positioned + * bottom-anchored pane (▔ divider, paddingX=2) that paints over the + * ScrollBox AND bottom slot. Provides ModalContext so Pane/Dialog inside + * skip their own frame. Fullscreen only; inline after overlay otherwise. */ + modal?: ReactNode; + /** Ref passed via ModalContext so Tabs (or any scroll-owning descendant) + * can attach it to their own ScrollBox for tall content. */ + modalScrollRef?: React.RefObject; + /** Ref to the scroll box for keyboard scrolling. RefObject (not Ref) so + * pillVisible's useSyncExternalStore can subscribe to scroll changes. */ + scrollRef?: RefObject; + /** Y-position (scrollHeight at snapshot) of the unseen-divider. Pill + * shows while viewport bottom hasn't reached this. Ref so REPL doesn't + * re-render on the one-shot snapshot write. */ + dividerYRef?: RefObject; + /** Force-hide the pill (e.g. viewing a sub-agent task). */ + hidePill?: boolean; + /** Force-hide the sticky prompt header (e.g. viewing a teammate task). */ + hideSticky?: boolean; + /** Count for the pill text. 0 → "Jump to bottom", >0 → "N new messages". */ + newMessageCount?: number; + /** Called when the user clicks the "N new" pill. */ + onPillClick?: () => void; +}; + +/** + * Tracks the in-transcript "N new messages" divider position while the + * user is scrolled up. Snapshots message count AND scrollHeight the first + * time sticky breaks. scrollHeight ≈ the y-position of the divider in the + * scroll content (it renders right after the last message that existed at + * snapshot time). + * + * `pillVisible` lives in FullscreenLayout (not here) — it subscribes + * directly to ScrollBox via useSyncExternalStore with a boolean snapshot + * against `dividerYRef`, so per-frame scroll never re-renders REPL. + * `dividerIndex` stays here because REPL needs it for computeUnseenDivider + * → Messages' divider line; it changes only ~twice/scroll-session + * (first scroll-away + repin), acceptable REPL re-render cost. + * + * `onScrollAway` must be called by every scroll-away action with the + * handle; `onRepin` by submit/scroll-to-bottom. + */ +export function useUnseenDivider(messageCount: number): { + /** Index into messages[] where the divider line renders. Cleared on + * sticky-resume (scroll back to bottom) so the "N new" line doesn't + * linger once everything is visible. */ + dividerIndex: number | null; + /** scrollHeight snapshot at first scroll-away — the divider's y-position. + * FullscreenLayout subscribes to ScrollBox and compares viewport bottom + * against this for pillVisible. Ref so writes don't re-render REPL. */ + dividerYRef: RefObject; + onScrollAway: (handle: ScrollBoxHandle) => void; + onRepin: () => void; + /** Scroll the handle so the divider line is at the top of the viewport. */ + jumpToNew: (handle: ScrollBoxHandle | null) => void; + /** Shift dividerIndex and dividerYRef when messages are prepended + * (infinite scroll-back). indexDelta = number of messages prepended; + * heightDelta = content height growth in rows. */ + shiftDivider: (indexDelta: number, heightDelta: number) => void; +} { + const [dividerIndex, setDividerIndex] = useState(null); + // Ref holds the current count for onScrollAway to snapshot. Written in + // the render body (not useEffect) so wheel events arriving between a + // message-append render and its effect flush don't capture a stale + // count (off-by-one in the baseline). React Compiler bails out here — + // acceptable for a hook instantiated once in REPL. + const countRef = useRef(messageCount); + countRef.current = messageCount; + // scrollHeight snapshot — the divider's y in content coords. Ref-only: + // read synchronously in onScrollAway (setState is batched, can't + // read-then-write in the same callback) AND by FullscreenLayout's + // pillVisible subscription. null = pinned to bottom. + const dividerYRef = useRef(null); + const onRepin = useCallback(() => { + // Don't clear dividerYRef here — a trackpad momentum wheel event + // racing in the same stdin batch would see null and re-snapshot, + // overriding the setDividerIndex(null) below. The useEffect below + // clears the ref after React commits the null dividerIndex, so the + // ref stays non-null until the state settles. + setDividerIndex(null); + }, []); + const onScrollAway = useCallback((handle: ScrollBoxHandle) => { + // Nothing below the viewport → nothing to jump to. Covers both: + // • empty/short session: scrollUp calls scrollTo(0) which breaks sticky + // even at scrollTop=0 (wheel-up on fresh session showed the pill) + // • click-to-select at bottom: useDragToScroll.check() calls + // scrollTo(current) to break sticky so streaming content doesn't shift + // under the selection, then onScroll(false, …) — but scrollTop is still + // at max (Sarah Deaton, #claude-code-feedback 2026-03-15) + // pendingDelta: scrollBy accumulates without updating scrollTop. Without + // it, wheeling up from max would see scrollTop==max and suppress the pill. + const max = Math.max(0, handle.getScrollHeight() - handle.getViewportHeight()); + if (handle.getScrollTop() + handle.getPendingDelta() >= max) return; + // Snapshot only on the FIRST scroll-away. onScrollAway fires on EVERY + // scroll action (not just the initial break from sticky) — this guard + // preserves the original baseline so the count doesn't reset on the + // second PageUp. Subsequent calls are ref-only no-ops (no REPL re-render). + if (dividerYRef.current === null) { + dividerYRef.current = handle.getScrollHeight(); + // New scroll-away session → move the divider here (replaces old one) + setDividerIndex(countRef.current); + } + }, []); + const jumpToNew = useCallback((handle_0: ScrollBoxHandle | null) => { + if (!handle_0) return; + // scrollToBottom (not scrollTo(dividerY)): sets stickyScroll=true so + // useVirtualScroll mounts the tail and render-node-to-output pins + // scrollTop=maxScroll. scrollTo sets stickyScroll=false → the clamp + // (still at top-range bounds before React re-renders) pins scrollTop + // back, stopping short. The divider stays rendered (dividerIndex + // unchanged) so users see where new messages started; the clear on + // next submit/explicit scroll-to-bottom handles cleanup. + handle_0.scrollToBottom(); + }, []); + + // Sync dividerYRef with dividerIndex. When onRepin fires (submit, + // scroll-to-bottom), it sets dividerIndex=null but leaves the ref + // non-null — a wheel event racing in the same stdin batch would + // otherwise see null and re-snapshot. Deferring the ref clear to + // useEffect guarantees the ref stays non-null until React has committed + // the null dividerIndex, blocking the if-null guard in onScrollAway. + // + // Also handles /clear, rewind, teammate-view swap — if the count drops + // below the divider index, the divider would point at nothing. + useEffect(() => { + if (dividerIndex === null) { + dividerYRef.current = null; + } else if (messageCount < dividerIndex) { + dividerYRef.current = null; + setDividerIndex(null); + } + }, [messageCount, dividerIndex]); + const shiftDivider = useCallback((indexDelta: number, heightDelta: number) => { + setDividerIndex(idx => idx === null ? null : idx + indexDelta); + if (dividerYRef.current !== null) { + dividerYRef.current += heightDelta; + } + }, []); + return { + dividerIndex, + dividerYRef, + onScrollAway, + onRepin, + jumpToNew, + shiftDivider + }; +} + +/** + * Counts assistant turns in messages[dividerIndex..end). A "turn" is what + * users think of as "a new message from Claude" — not raw assistant entries + * (one turn yields multiple entries: tool_use blocks + text blocks). We count + * non-assistant→assistant transitions, but only for entries that actually + * carry text — tool-use-only entries are skipped (like progress messages) + * so "⏺ Searched for 13 patterns, read 6 files" doesn't tick the pill. + */ +export function countUnseenAssistantTurns(messages: readonly Message[], dividerIndex: number): number { + let count = 0; + let prevWasAssistant = false; + for (let i = dividerIndex; i < messages.length; i++) { + const m = messages[i]!; + if (m.type === 'progress') continue; + // Tool-use-only assistant entries aren't "new messages" to the user — + // skip them the same way we skip progress. prevWasAssistant is NOT + // updated, so a text block immediately following still counts as the + // same turn (tool_use + text from one API response = 1). + if (m.type === 'assistant' && !assistantHasVisibleText(m)) continue; + const isAssistant = m.type === 'assistant'; + if (isAssistant && !prevWasAssistant) count++; + prevWasAssistant = isAssistant; + } + return count; +} +function assistantHasVisibleText(m: Message): boolean { + if (m.type !== 'assistant') return false; + for (const b of m.message.content) { + if (b.type === 'text' && b.text.trim() !== '') return true; + } + return false; +} +export type UnseenDivider = { + firstUnseenUuid: Message['uuid']; + count: number; +}; + +/** + * Builds the unseenDivider object REPL passes to Messages + the pill. + * Returns undefined only when no content has arrived past the divider + * yet (messages[dividerIndex] doesn't exist). Once ANY message arrives + * — including tool_use-only assistant entries and tool_result user entries + * that countUnseenAssistantTurns skips — count floors at 1 so the pill + * flips from "Jump to bottom" to "1 new message". Without the floor, + * the pill stays "Jump to bottom" through an entire tool-call sequence + * until Claude's text response lands. + */ +export function computeUnseenDivider(messages: readonly Message[], dividerIndex: number | null): UnseenDivider | undefined { + if (dividerIndex === null) return undefined; + // Skip progress and null-rendering attachments when picking the divider + // anchor — Messages.tsx filters these out of renderableMessages before the + // dividerBeforeIndex search, so their UUID wouldn't be found (CC-724). + // Hook attachments use randomUUID() so nothing shares their 24-char prefix. + let anchorIdx = dividerIndex; + while (anchorIdx < messages.length && (messages[anchorIdx]?.type === 'progress' || isNullRenderingAttachment(messages[anchorIdx]!))) { + anchorIdx++; + } + const uuid = messages[anchorIdx]?.uuid; + if (!uuid) return undefined; + const count = countUnseenAssistantTurns(messages, dividerIndex); + return { + firstUnseenUuid: uuid, + count: Math.max(1, count) + }; +} + +/** + * Layout wrapper for the REPL. In fullscreen mode, puts scrollable + * content in a sticky-scroll box and pins bottom content via flexbox. + * Outside fullscreen mode, renders content sequentially so the existing + * main-screen scrollback rendering works unchanged. + * + * Fullscreen mode defaults on for ants (CLAUDE_CODE_NO_FLICKER=0 to opt out) + * and off for external users (CLAUDE_CODE_NO_FLICKER=1 to opt in). + * The wrapper + * (alt buffer + mouse tracking + height constraint) lives at REPL's root + * so nothing can accidentally render outside it. + */ +export function FullscreenLayout(t0) { + const $ = _c(47); + const { + scrollable, + bottom, + overlay, + bottomFloat, + modal, + modalScrollRef, + scrollRef, + dividerYRef, + hidePill: t1, + hideSticky: t2, + newMessageCount: t3, + onPillClick + } = t0; + const hidePill = t1 === undefined ? false : t1; + const hideSticky = t2 === undefined ? false : t2; + const newMessageCount = t3 === undefined ? 0 : t3; + const { + rows: terminalRows, + columns + } = useTerminalSize(); + const [stickyPrompt, setStickyPrompt] = useState(null); + let t4; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t4 = { + setStickyPrompt + }; + $[0] = t4; + } else { + t4 = $[0]; + } + const chromeCtx = t4; + let t5; + if ($[1] !== scrollRef) { + t5 = listener => scrollRef?.current?.subscribe(listener) ?? _temp; + $[1] = scrollRef; + $[2] = t5; + } else { + t5 = $[2]; + } + const subscribe = t5; + let t6; + if ($[3] !== dividerYRef || $[4] !== scrollRef) { + t6 = () => { + const s = scrollRef?.current; + const dividerY = dividerYRef?.current; + if (!s || dividerY == null) { + return false; + } + return s.getScrollTop() + s.getPendingDelta() + s.getViewportHeight() < dividerY; + }; + $[3] = dividerYRef; + $[4] = scrollRef; + $[5] = t6; + } else { + t6 = $[5]; + } + const pillVisible = useSyncExternalStore(subscribe, t6); + let t7; + if ($[6] === Symbol.for("react.memo_cache_sentinel")) { + t7 = []; + $[6] = t7; + } else { + t7 = $[6]; + } + useLayoutEffect(_temp3, t7); + if (isFullscreenEnvEnabled()) { + const sticky = hideSticky ? null : stickyPrompt; + const headerPrompt = sticky != null && sticky !== "clicked" && overlay == null ? sticky : null; + const padCollapsed = sticky != null && overlay == null; + let t8; + if ($[7] !== headerPrompt) { + t8 = headerPrompt && ; + $[7] = headerPrompt; + $[8] = t8; + } else { + t8 = $[8]; + } + const t9 = padCollapsed ? 0 : 1; + let t10; + if ($[9] !== scrollable) { + t10 = {scrollable}; + $[9] = scrollable; + $[10] = t10; + } else { + t10 = $[10]; + } + let t11; + if ($[11] !== overlay || $[12] !== scrollRef || $[13] !== t10 || $[14] !== t9) { + t11 = {t10}{overlay}; + $[11] = overlay; + $[12] = scrollRef; + $[13] = t10; + $[14] = t9; + $[15] = t11; + } else { + t11 = $[15]; + } + let t12; + if ($[16] !== hidePill || $[17] !== newMessageCount || $[18] !== onPillClick || $[19] !== overlay || $[20] !== pillVisible) { + t12 = !hidePill && pillVisible && overlay == null && ; + $[16] = hidePill; + $[17] = newMessageCount; + $[18] = onPillClick; + $[19] = overlay; + $[20] = pillVisible; + $[21] = t12; + } else { + t12 = $[21]; + } + let t13; + if ($[22] !== bottomFloat) { + t13 = bottomFloat != null && {bottomFloat}; + $[22] = bottomFloat; + $[23] = t13; + } else { + t13 = $[23]; + } + let t14; + if ($[24] !== t11 || $[25] !== t12 || $[26] !== t13 || $[27] !== t8) { + t14 = {t8}{t11}{t12}{t13}; + $[24] = t11; + $[25] = t12; + $[26] = t13; + $[27] = t8; + $[28] = t14; + } else { + t14 = $[28]; + } + let t15; + let t16; + if ($[29] === Symbol.for("react.memo_cache_sentinel")) { + t15 = ; + t16 = ; + $[29] = t15; + $[30] = t16; + } else { + t15 = $[29]; + t16 = $[30]; + } + let t17; + if ($[31] !== bottom) { + t17 = {t15}{t16}{bottom}; + $[31] = bottom; + $[32] = t17; + } else { + t17 = $[32]; + } + let t18; + if ($[33] !== columns || $[34] !== modal || $[35] !== modalScrollRef || $[36] !== terminalRows) { + t18 = modal != null && {"\u2594".repeat(columns)}{modal}; + $[33] = columns; + $[34] = modal; + $[35] = modalScrollRef; + $[36] = terminalRows; + $[37] = t18; + } else { + t18 = $[37]; + } + let t19; + if ($[38] !== t14 || $[39] !== t17 || $[40] !== t18) { + t19 = {t14}{t17}{t18}; + $[38] = t14; + $[39] = t17; + $[40] = t18; + $[41] = t19; + } else { + t19 = $[41]; + } + return t19; + } + let t8; + if ($[42] !== bottom || $[43] !== modal || $[44] !== overlay || $[45] !== scrollable) { + t8 = <>{scrollable}{bottom}{overlay}{modal}; + $[42] = bottom; + $[43] = modal; + $[44] = overlay; + $[45] = scrollable; + $[46] = t8; + } else { + t8 = $[46]; + } + return t8; +} + +// Slack-style pill. Absolute overlay at bottom={0} of the scrollwrap — floats +// over the ScrollBox's last content row, only obscuring the centered pill +// text (the rest of the row shows ScrollBox content). Scroll-smear from +// DECSTBM shifting the pill's pixels is repaired at the Ink layer +// (absoluteRectsPrev third-pass in render-node-to-output.ts, #23939). Shows +// "Jump to bottom" when count is 0 (scrolled away but no new messages yet — +// the dead zone where users previously thought chat stalled). +function _temp3() { + if (!isFullscreenEnvEnabled()) { + return; + } + const ink = instances.get(process.stdout); + if (!ink) { + return; + } + ink.onHyperlinkClick = _temp2; + return () => { + ink.onHyperlinkClick = undefined; + }; +} +function _temp2(url) { + if (url.startsWith("file:")) { + try { + openPath(fileURLToPath(url)); + } catch {} + } else { + openBrowser(url); + } +} +function _temp() {} +function NewMessagesPill(t0) { + const $ = _c(10); + const { + count, + onClick + } = t0; + const [hover, setHover] = useState(false); + let t1; + let t2; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t1 = () => setHover(true); + t2 = () => setHover(false); + $[0] = t1; + $[1] = t2; + } else { + t1 = $[0]; + t2 = $[1]; + } + const t3 = hover ? "userMessageBackgroundHover" : "userMessageBackground"; + let t4; + if ($[2] !== count) { + t4 = count > 0 ? `${count} new ${plural(count, "message")}` : "Jump to bottom"; + $[2] = count; + $[3] = t4; + } else { + t4 = $[3]; + } + let t5; + if ($[4] !== t3 || $[5] !== t4) { + t5 = {" "}{t4}{" "}{figures.arrowDown}{" "}; + $[4] = t3; + $[5] = t4; + $[6] = t5; + } else { + t5 = $[6]; + } + let t6; + if ($[7] !== onClick || $[8] !== t5) { + t6 = {t5}; + $[7] = onClick; + $[8] = t5; + $[9] = t6; + } else { + t6 = $[9]; + } + return t6; +} + +// Context breadcrumb: when scrolled up into history, pin the current +// conversation turn's prompt above the viewport so you know what Claude was +// responding to. Normal-flow sibling BEFORE the ScrollBox (mirrors the pill +// below it) — shrinks the ScrollBox by exactly 1 row via flex, stays outside +// the DECSTBM scroll region. Click jumps back to the prompt. +// +// Height is FIXED at 1 row (truncate-end for long prompts). A variable-height +// header (1 when short, 2 when wrapped) shifts the ScrollBox by 1 row every +// time the sticky prompt switches during scroll — content jumps on screen +// even with scrollTop unchanged (the DECSTBM region top shifts with the +// ScrollBox, and the diff engine sees "everything moved"). Fixed height +// keeps the ScrollBox anchored; only the header TEXT changes, not its box. +function StickyPromptHeader(t0) { + const $ = _c(8); + const { + text, + onClick + } = t0; + const [hover, setHover] = useState(false); + const t1 = hover ? "userMessageBackgroundHover" : "userMessageBackground"; + let t2; + let t3; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t2 = () => setHover(true); + t3 = () => setHover(false); + $[0] = t2; + $[1] = t3; + } else { + t2 = $[0]; + t3 = $[1]; + } + let t4; + if ($[2] !== text) { + t4 = {figures.pointer} {text}; + $[2] = text; + $[3] = t4; + } else { + t4 = $[3]; + } + let t5; + if ($[4] !== onClick || $[5] !== t1 || $[6] !== t4) { + t5 = {t4}; + $[4] = onClick; + $[5] = t1; + $[6] = t4; + $[7] = t5; + } else { + t5 = $[7]; + } + return t5; +} + +// Slash-command suggestion overlay — see promptOverlayContext.tsx for why +// it's portaled. Scroll-smear from floating over the DECSTBM region is +// repaired at the Ink layer (absoluteRectsPrev in render-node-to-output.ts). +// The renderer clamps negative y to 0 for absolute elements (see +// render-node-to-output.ts), so the top rows (best matches) stay visible +// even when the overlay extends above the viewport. We omit minHeight and +// flex-end here: they would create empty padding rows that shift visible +// items down into the prompt area when the list has fewer items than max. +function SuggestionsOverlay() { + const $ = _c(4); + const data = usePromptOverlay(); + if (!data || data.suggestions.length === 0) { + return null; + } + let t0; + if ($[0] !== data.maxColumnWidth || $[1] !== data.selectedSuggestion || $[2] !== data.suggestions) { + t0 = ; + $[0] = data.maxColumnWidth; + $[1] = data.selectedSuggestion; + $[2] = data.suggestions; + $[3] = t0; + } else { + t0 = $[3]; + } + return t0; +} + +// Dialog portaled from PromptInput (AutoModeOptInDialog) — same clip-escape +// pattern as SuggestionsOverlay. Renders later in tree order so it paints +// over suggestions if both are ever up (they shouldn't be). +function DialogOverlay() { + const $ = _c(2); + const node = usePromptOverlayDialog(); + if (!node) { + return null; + } + let t0; + if ($[0] !== node) { + t0 = {node}; + $[0] = node; + $[1] = t0; + } else { + t0 = $[1]; + } + return t0; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJmaWd1cmVzIiwiUmVhY3QiLCJjcmVhdGVDb250ZXh0IiwiUmVhY3ROb2RlIiwiUmVmT2JqZWN0IiwidXNlQ2FsbGJhY2siLCJ1c2VFZmZlY3QiLCJ1c2VMYXlvdXRFZmZlY3QiLCJ1c2VNZW1vIiwidXNlUmVmIiwidXNlU3RhdGUiLCJ1c2VTeW5jRXh0ZXJuYWxTdG9yZSIsImZpbGVVUkxUb1BhdGgiLCJNb2RhbENvbnRleHQiLCJQcm9tcHRPdmVybGF5UHJvdmlkZXIiLCJ1c2VQcm9tcHRPdmVybGF5IiwidXNlUHJvbXB0T3ZlcmxheURpYWxvZyIsInVzZVRlcm1pbmFsU2l6ZSIsIlNjcm9sbEJveCIsIlNjcm9sbEJveEhhbmRsZSIsImluc3RhbmNlcyIsIkJveCIsIlRleHQiLCJNZXNzYWdlIiwib3BlbkJyb3dzZXIiLCJvcGVuUGF0aCIsImlzRnVsbHNjcmVlbkVudkVuYWJsZWQiLCJwbHVyYWwiLCJpc051bGxSZW5kZXJpbmdBdHRhY2htZW50IiwiUHJvbXB0SW5wdXRGb290ZXJTdWdnZXN0aW9ucyIsIlN0aWNreVByb21wdCIsIk1PREFMX1RSQU5TQ1JJUFRfUEVFSyIsIlNjcm9sbENocm9tZUNvbnRleHQiLCJzZXRTdGlja3lQcm9tcHQiLCJwIiwiUHJvcHMiLCJzY3JvbGxhYmxlIiwiYm90dG9tIiwib3ZlcmxheSIsImJvdHRvbUZsb2F0IiwibW9kYWwiLCJtb2RhbFNjcm9sbFJlZiIsInNjcm9sbFJlZiIsImRpdmlkZXJZUmVmIiwiaGlkZVBpbGwiLCJoaWRlU3RpY2t5IiwibmV3TWVzc2FnZUNvdW50Iiwib25QaWxsQ2xpY2siLCJ1c2VVbnNlZW5EaXZpZGVyIiwibWVzc2FnZUNvdW50IiwiZGl2aWRlckluZGV4Iiwib25TY3JvbGxBd2F5IiwiaGFuZGxlIiwib25SZXBpbiIsImp1bXBUb05ldyIsInNoaWZ0RGl2aWRlciIsImluZGV4RGVsdGEiLCJoZWlnaHREZWx0YSIsInNldERpdmlkZXJJbmRleCIsImNvdW50UmVmIiwiY3VycmVudCIsIm1heCIsIk1hdGgiLCJnZXRTY3JvbGxIZWlnaHQiLCJnZXRWaWV3cG9ydEhlaWdodCIsImdldFNjcm9sbFRvcCIsImdldFBlbmRpbmdEZWx0YSIsInNjcm9sbFRvQm90dG9tIiwiaWR4IiwiY291bnRVbnNlZW5Bc3Npc3RhbnRUdXJucyIsIm1lc3NhZ2VzIiwiY291bnQiLCJwcmV2V2FzQXNzaXN0YW50IiwiaSIsImxlbmd0aCIsIm0iLCJ0eXBlIiwiYXNzaXN0YW50SGFzVmlzaWJsZVRleHQiLCJpc0Fzc2lzdGFudCIsImIiLCJtZXNzYWdlIiwiY29udGVudCIsInRleHQiLCJ0cmltIiwiVW5zZWVuRGl2aWRlciIsImZpcnN0VW5zZWVuVXVpZCIsImNvbXB1dGVVbnNlZW5EaXZpZGVyIiwidW5kZWZpbmVkIiwiYW5jaG9ySWR4IiwidXVpZCIsIkZ1bGxzY3JlZW5MYXlvdXQiLCJ0MCIsIiQiLCJfYyIsInQxIiwidDIiLCJ0MyIsInJvd3MiLCJ0ZXJtaW5hbFJvd3MiLCJjb2x1bW5zIiwic3RpY2t5UHJvbXB0IiwidDQiLCJTeW1ib2wiLCJmb3IiLCJjaHJvbWVDdHgiLCJ0NSIsImxpc3RlbmVyIiwic3Vic2NyaWJlIiwiX3RlbXAiLCJ0NiIsInMiLCJkaXZpZGVyWSIsInBpbGxWaXNpYmxlIiwidDciLCJfdGVtcDMiLCJzdGlja3kiLCJoZWFkZXJQcm9tcHQiLCJwYWRDb2xsYXBzZWQiLCJ0OCIsInNjcm9sbFRvIiwidDkiLCJ0MTAiLCJ0MTEiLCJ0MTIiLCJ0MTMiLCJ0MTQiLCJ0MTUiLCJ0MTYiLCJ0MTciLCJ0MTgiLCJyZXBlYXQiLCJ0MTkiLCJpbmsiLCJnZXQiLCJwcm9jZXNzIiwic3Rkb3V0Iiwib25IeXBlcmxpbmtDbGljayIsIl90ZW1wMiIsInVybCIsInN0YXJ0c1dpdGgiLCJOZXdNZXNzYWdlc1BpbGwiLCJvbkNsaWNrIiwiaG92ZXIiLCJzZXRIb3ZlciIsImFycm93RG93biIsIlN0aWNreVByb21wdEhlYWRlciIsInBvaW50ZXIiLCJTdWdnZXN0aW9uc092ZXJsYXkiLCJkYXRhIiwic3VnZ2VzdGlvbnMiLCJtYXhDb2x1bW5XaWR0aCIsInNlbGVjdGVkU3VnZ2VzdGlvbiIsIkRpYWxvZ092ZXJsYXkiLCJub2RlIl0sInNvdXJjZXMiOlsiRnVsbHNjcmVlbkxheW91dC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGZpZ3VyZXMgZnJvbSAnZmlndXJlcydcbmltcG9ydCBSZWFjdCwge1xuICBjcmVhdGVDb250ZXh0LFxuICB0eXBlIFJlYWN0Tm9kZSxcbiAgdHlwZSBSZWZPYmplY3QsXG4gIHVzZUNhbGxiYWNrLFxuICB1c2VFZmZlY3QsXG4gIHVzZUxheW91dEVmZmVjdCxcbiAgdXNlTWVtbyxcbiAgdXNlUmVmLFxuICB1c2VTdGF0ZSxcbiAgdXNlU3luY0V4dGVybmFsU3RvcmUsXG59IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgZmlsZVVSTFRvUGF0aCB9IGZyb20gJ3VybCdcbmltcG9ydCB7IE1vZGFsQ29udGV4dCB9IGZyb20gJy4uL2NvbnRleHQvbW9kYWxDb250ZXh0LmpzJ1xuaW1wb3J0IHtcbiAgUHJvbXB0T3ZlcmxheVByb3ZpZGVyLFxuICB1c2VQcm9tcHRPdmVybGF5LFxuICB1c2VQcm9tcHRPdmVybGF5RGlhbG9nLFxufSBmcm9tICcuLi9jb250ZXh0L3Byb21wdE92ZXJsYXlDb250ZXh0LmpzJ1xuaW1wb3J0IHsgdXNlVGVybWluYWxTaXplIH0gZnJvbSAnLi4vaG9va3MvdXNlVGVybWluYWxTaXplLmpzJ1xuaW1wb3J0IFNjcm9sbEJveCwgeyB0eXBlIFNjcm9sbEJveEhhbmRsZSB9IGZyb20gJy4uL2luay9jb21wb25lbnRzL1Njcm9sbEJveC5qcydcbmltcG9ydCBpbnN0YW5jZXMgZnJvbSAnLi4vaW5rL2luc3RhbmNlcy5qcydcbmltcG9ydCB7IEJveCwgVGV4dCB9IGZyb20gJy4uL2luay5qcydcbmltcG9ydCB0eXBlIHsgTWVzc2FnZSB9IGZyb20gJy4uL3R5cGVzL21lc3NhZ2UuanMnXG5pbXBvcnQgeyBvcGVuQnJvd3Nlciwgb3BlblBhdGggfSBmcm9tICcuLi91dGlscy9icm93c2VyLmpzJ1xuaW1wb3J0IHsgaXNGdWxsc2NyZWVuRW52RW5hYmxlZCB9IGZyb20gJy4uL3V0aWxzL2Z1bGxzY3JlZW4uanMnXG5pbXBvcnQgeyBwbHVyYWwgfSBmcm9tICcuLi91dGlscy9zdHJpbmdVdGlscy5qcydcbmltcG9ydCB7IGlzTnVsbFJlbmRlcmluZ0F0dGFjaG1lbnQgfSBmcm9tICcuL21lc3NhZ2VzL251bGxSZW5kZXJpbmdBdHRhY2htZW50cy5qcydcbmltcG9ydCBQcm9tcHRJbnB1dEZvb3RlclN1Z2dlc3Rpb25zIGZyb20gJy4vUHJvbXB0SW5wdXQvUHJvbXB0SW5wdXRGb290ZXJTdWdnZXN0aW9ucy5qcydcbmltcG9ydCB0eXBlIHsgU3RpY2t5UHJvbXB0IH0gZnJvbSAnLi9WaXJ0dWFsTWVzc2FnZUxpc3QuanMnXG5cbi8qKiBSb3dzIG9mIHRyYW5zY3JpcHQgY29udGV4dCBrZXB0IHZpc2libGUgYWJvdmUgdGhlIG1vZGFsIHBhbmUncyDilpQgZGl2aWRlci4gKi9cbmNvbnN0IE1PREFMX1RSQU5TQ1JJUFRfUEVFSyA9IDJcblxuLyoqIENvbnRleHQgZm9yIHNjcm9sbC1kZXJpdmVkIGNocm9tZSAoc3RpY2t5IGhlYWRlciwgcGlsbCkuIFN0aWNreVRyYWNrZXJcbiAqICBpbiBWaXJ0dWFsTWVzc2FnZUxpc3Qgd3JpdGVzIHZpYSB0aGlzIGluc3RlYWQgb2YgdGhyZWFkaW5nIGEgY2FsbGJhY2tcbiAqICB1cCB0aHJvdWdoIE1lc3NhZ2VzIOKGkiBSRVBMIOKGkiBGdWxsc2NyZWVuTGF5b3V0LiBUaGUgc2V0dGVyIGlzIHN0YWJsZSBzb1xuICogIGNvbnN1bWluZyB0aGlzIGNvbnRleHQgbmV2ZXIgY2F1c2VzIHJlLXJlbmRlcnMuICovXG5leHBvcnQgY29uc3QgU2Nyb2xsQ2hyb21lQ29udGV4dCA9IGNyZWF0ZUNvbnRleHQ8e1xuICBzZXRTdGlja3lQcm9tcHQ6IChwOiBTdGlja3lQcm9tcHQgfCBudWxsKSA9PiB2b2lkXG59Pih7IHNldFN0aWNreVByb21wdDogKCkgPT4ge30gfSlcblxudHlwZSBQcm9wcyA9IHtcbiAgLyoqIENvbnRlbnQgdGhhdCBzY3JvbGxzIChtZXNzYWdlcywgdG9vbCBvdXRwdXQpICovXG4gIHNjcm9sbGFibGU6IFJlYWN0Tm9kZVxuICAvKiogQ29udGVudCBwaW5uZWQgdG8gdGhlIGJvdHRvbSAoc3Bpbm5lciwgcHJvbXB0LCBwZXJtaXNzaW9ucykgKi9cbiAgYm90dG9tOiBSZWFjdE5vZGVcbiAgLyoqIENvbnRlbnQgcmVuZGVyZWQgaW5zaWRlIHRoZSBTY3JvbGxCb3ggYWZ0ZXIgbWVzc2FnZXMg4oCUIHVzZXIgY2FuIHNjcm9sbFxuICAgKiAgdXAgdG8gc2VlIGNvbnRleHQgd2hpbGUgaXQncyBzaG93aW5nICh1c2VkIGJ5IFBlcm1pc3Npb25SZXF1ZXN0KS4gKi9cbiAgb3ZlcmxheT86IFJlYWN0Tm9kZVxuICAvKiogQWJzb2x1dGUtcG9zaXRpb25lZCBjb250ZW50IGFuY2hvcmVkIGF0IHRoZSBib3R0b20tcmlnaHQgb2YgdGhlXG4gICAqICBTY3JvbGxCb3ggYXJlYSwgZmxvYXRpbmcgb3ZlciBzY3JvbGxiYWNrLiBSZW5kZXJlZCBpbnNpZGUgdGhlIGZsZXhHcm93XG4gICAqICByZWdpb24gKG5vdCB0aGUgYm90dG9tIHNsb3QpIHNvIHRoZSBvdmVyZmxvd1k6aGlkZGVuIGNhcCBkb2Vzbid0IGNsaXBcbiAgICogIGl0LiBGdWxsc2NyZWVuIG9ubHkg4oCUIHVzZWQgZm9yIHRoZSBjb21wYW5pb24gc3BlZWNoIGJ1YmJsZS4gKi9cbiAgYm90dG9tRmxvYXQ/OiBSZWFjdE5vZGVcbiAgLyoqIFNsYXNoLWNvbW1hbmQgZGlhbG9nIGNvbnRlbnQuIFJlbmRlcmVkIGluIGFuIGFic29sdXRlLXBvc2l0aW9uZWRcbiAgICogIGJvdHRvbS1hbmNob3JlZCBwYW5lICjilpQgZGl2aWRlciwgcGFkZGluZ1g9MikgdGhhdCBwYWludHMgb3ZlciB0aGVcbiAgICogIFNjcm9sbEJveCBBTkQgYm90dG9tIHNsb3QuIFByb3ZpZGVzIE1vZGFsQ29udGV4dCBzbyBQYW5lL0RpYWxvZyBpbnNpZGVcbiAgICogIHNraXAgdGhlaXIgb3duIGZyYW1lLiBGdWxsc2NyZWVuIG9ubHk7IGlubGluZSBhZnRlciBvdmVybGF5IG90aGVyd2lzZS4gKi9cbiAgbW9kYWw/OiBSZWFjdE5vZGVcbiAgLyoqIFJlZiBwYXNzZWQgdmlhIE1vZGFsQ29udGV4dCBzbyBUYWJzIChvciBhbnkgc2Nyb2xsLW93bmluZyBkZXNjZW5kYW50KVxuICAgKiAgY2FuIGF0dGFjaCBpdCB0byB0aGVpciBvd24gU2Nyb2xsQm94IGZvciB0YWxsIGNvbnRlbnQuICovXG4gIG1vZGFsU2Nyb2xsUmVmPzogUmVhY3QuUmVmT2JqZWN0PFNjcm9sbEJveEhhbmRsZSB8IG51bGw+XG4gIC8qKiBSZWYgdG8gdGhlIHNjcm9sbCBib3ggZm9yIGtleWJvYXJkIHNjcm9sbGluZy4gUmVmT2JqZWN0IChub3QgUmVmKSBzb1xuICAgKiAgcGlsbFZpc2libGUncyB1c2VTeW5jRXh0ZXJuYWxTdG9yZSBjYW4gc3Vic2NyaWJlIHRvIHNjcm9sbCBjaGFuZ2VzLiAqL1xuICBzY3JvbGxSZWY/OiBSZWZPYmplY3Q8U2Nyb2xsQm94SGFuZGxlIHwgbnVsbD5cbiAgLyoqIFktcG9zaXRpb24gKHNjcm9sbEhlaWdodCBhdCBzbmFwc2hvdCkgb2YgdGhlIHVuc2Vlbi1kaXZpZGVyLiBQaWxsXG4gICAqICBzaG93cyB3aGlsZSB2aWV3cG9ydCBib3R0b20gaGFzbid0IHJlYWNoZWQgdGhpcy4gUmVmIHNvIFJFUEwgZG9lc24ndFxuICAgKiAgcmUtcmVuZGVyIG9uIHRoZSBvbmUtc2hvdCBzbmFwc2hvdCB3cml0ZS4gKi9cbiAgZGl2aWRlcllSZWY/OiBSZWZPYmplY3Q8bnVtYmVyIHwgbnVsbD5cbiAgLyoqIEZvcmNlLWhpZGUgdGhlIHBpbGwgKGUuZy4gdmlld2luZyBhIHN1Yi1hZ2VudCB0YXNrKS4gKi9cbiAgaGlkZVBpbGw/OiBib29sZWFuXG4gIC8qKiBGb3JjZS1oaWRlIHRoZSBzdGlja3kgcHJvbXB0IGhlYWRlciAoZS5nLiB2aWV3aW5nIGEgdGVhbW1hdGUgdGFzaykuICovXG4gIGhpZGVTdGlja3k/OiBib29sZWFuXG4gIC8qKiBDb3VudCBmb3IgdGhlIHBpbGwgdGV4dC4gMCDihpIgXCJKdW1wIHRvIGJvdHRvbVwiLCA+MCDihpIgXCJOIG5ldyBtZXNzYWdlc1wiLiAqL1xuICBuZXdNZXNzYWdlQ291bnQ/OiBudW1iZXJcbiAgLyoqIENhbGxlZCB3aGVuIHRoZSB1c2VyIGNsaWNrcyB0aGUgXCJOIG5ld1wiIHBpbGwuICovXG4gIG9uUGlsbENsaWNrPzogKCkgPT4gdm9pZFxufVxuXG4vKipcbiAqIFRyYWNrcyB0aGUgaW4tdHJhbnNjcmlwdCBcIk4gbmV3IG1lc3NhZ2VzXCIgZGl2aWRlciBwb3NpdGlvbiB3aGlsZSB0aGVcbiAqIHVzZXIgaXMgc2Nyb2xsZWQgdXAuIFNuYXBzaG90cyBtZXNzYWdlIGNvdW50IEFORCBzY3JvbGxIZWlnaHQgdGhlIGZpcnN0XG4gKiB0aW1lIHN0aWNreSBicmVha3MuIHNjcm9sbEhlaWdodCDiiYggdGhlIHktcG9zaXRpb24gb2YgdGhlIGRpdmlkZXIgaW4gdGhlXG4gKiBzY3JvbGwgY29udGVudCAoaXQgcmVuZGVycyByaWdodCBhZnRlciB0aGUgbGFzdCBtZXNzYWdlIHRoYXQgZXhpc3RlZCBhdFxuICogc25hcHNob3QgdGltZSkuXG4gKlxuICogYHBpbGxWaXNpYmxlYCBsaXZlcyBpbiBGdWxsc2NyZWVuTGF5b3V0IChub3QgaGVyZSkg4oCUIGl0IHN1YnNjcmliZXNcbiAqIGRpcmVjdGx5IHRvIFNjcm9sbEJveCB2aWEgdXNlU3luY0V4dGVybmFsU3RvcmUgd2l0aCBhIGJvb2xlYW4gc25hcHNob3RcbiAqIGFnYWluc3QgYGRpdmlkZXJZUmVmYCwgc28gcGVyLWZyYW1lIHNjcm9sbCBuZXZlciByZS1yZW5kZXJzIFJFUEwuXG4gKiBgZGl2aWRlckluZGV4YCBzdGF5cyBoZXJlIGJlY2F1c2UgUkVQTCBuZWVkcyBpdCBmb3IgY29tcHV0ZVVuc2VlbkRpdmlkZXJcbiAqIOKGkiBNZXNzYWdlcycgZGl2aWRlciBsaW5lOyBpdCBjaGFuZ2VzIG9ubHkgfnR3aWNlL3Njcm9sbC1zZXNzaW9uXG4gKiAoZmlyc3Qgc2Nyb2xsLWF3YXkgKyByZXBpbiksIGFjY2VwdGFibGUgUkVQTCByZS1yZW5kZXIgY29zdC5cbiAqXG4gKiBgb25TY3JvbGxBd2F5YCBtdXN0IGJlIGNhbGxlZCBieSBldmVyeSBzY3JvbGwtYXdheSBhY3Rpb24gd2l0aCB0aGVcbiAqIGhhbmRsZTsgYG9uUmVwaW5gIGJ5IHN1Ym1pdC9zY3JvbGwtdG8tYm90dG9tLlxuICovXG5leHBvcnQgZnVuY3Rpb24gdXNlVW5zZWVuRGl2aWRlcihtZXNzYWdlQ291bnQ6IG51bWJlcik6IHtcbiAgLyoqIEluZGV4IGludG8gbWVzc2FnZXNbXSB3aGVyZSB0aGUgZGl2aWRlciBsaW5lIHJlbmRlcnMuIENsZWFyZWQgb25cbiAgICogIHN0aWNreS1yZXN1bWUgKHNjcm9sbCBiYWNrIHRvIGJvdHRvbSkgc28gdGhlIFwiTiBuZXdcIiBsaW5lIGRvZXNuJ3RcbiAgICogIGxpbmdlciBvbmNlIGV2ZXJ5dGhpbmcgaXMgdmlzaWJsZS4gKi9cbiAgZGl2aWRlckluZGV4OiBudW1iZXIgfCBudWxsXG4gIC8qKiBzY3JvbGxIZWlnaHQgc25hcHNob3QgYXQgZmlyc3Qgc2Nyb2xsLWF3YXkg4oCUIHRoZSBkaXZpZGVyJ3MgeS1wb3NpdGlvbi5cbiAgICogIEZ1bGxzY3JlZW5MYXlvdXQgc3Vic2NyaWJlcyB0byBTY3JvbGxCb3ggYW5kIGNvbXBhcmVzIHZpZXdwb3J0IGJvdHRvbVxuICAgKiAgYWdhaW5zdCB0aGlzIGZvciBwaWxsVmlzaWJsZS4gUmVmIHNvIHdyaXRlcyBkb24ndCByZS1yZW5kZXIgUkVQTC4gKi9cbiAgZGl2aWRlcllSZWY6IFJlZk9iamVjdDxudW1iZXIgfCBudWxsPlxuICBvblNjcm9sbEF3YXk6IChoYW5kbGU6IFNjcm9sbEJveEhhbmRsZSkgPT4gdm9pZFxuICBvblJlcGluOiAoKSA9PiB2b2lkXG4gIC8qKiBTY3JvbGwgdGhlIGhhbmRsZSBzbyB0aGUgZGl2aWRlciBsaW5lIGlzIGF0IHRoZSB0b3Agb2YgdGhlIHZpZXdwb3J0LiAqL1xuICBqdW1wVG9OZXc6IChoYW5kbGU6IFNjcm9sbEJveEhhbmRsZSB8IG51bGwpID0+IHZvaWRcbiAgLyoqIFNoaWZ0IGRpdmlkZXJJbmRleCBhbmQgZGl2aWRlcllSZWYgd2hlbiBtZXNzYWdlcyBhcmUgcHJlcGVuZGVkXG4gICAqICAoaW5maW5pdGUgc2Nyb2xsLWJhY2spLiBpbmRleERlbHRhID0gbnVtYmVyIG9mIG1lc3NhZ2VzIHByZXBlbmRlZDtcbiAgICogIGhlaWdodERlbHRhID0gY29udGVudCBoZWlnaHQgZ3Jvd3RoIGluIHJvd3MuICovXG4gIHNoaWZ0RGl2aWRlcjogKGluZGV4RGVsdGE6IG51bWJlciwgaGVpZ2h0RGVsdGE6IG51bWJlcikgPT4gdm9pZFxufSB7XG4gIGNvbnN0IFtkaXZpZGVySW5kZXgsIHNldERpdmlkZXJJbmRleF0gPSB1c2VTdGF0ZTxudW1iZXIgfCBudWxsPihudWxsKVxuICAvLyBSZWYgaG9sZHMgdGhlIGN1cnJlbnQgY291bnQgZm9yIG9uU2Nyb2xsQXdheSB0byBzbmFwc2hvdC4gV3JpdHRlbiBpblxuICAvLyB0aGUgcmVuZGVyIGJvZHkgKG5vdCB1c2VFZmZlY3QpIHNvIHdoZWVsIGV2ZW50cyBhcnJpdmluZyBiZXR3ZWVuIGFcbiAgLy8gbWVzc2FnZS1hcHBlbmQgcmVuZGVyIGFuZCBpdHMgZWZmZWN0IGZsdXNoIGRvbid0IGNhcHR1cmUgYSBzdGFsZVxuICAvLyBjb3VudCAob2ZmLWJ5LW9uZSBpbiB0aGUgYmFzZWxpbmUpLiBSZWFjdCBDb21waWxlciBiYWlscyBvdXQgaGVyZSDigJRcbiAgLy8gYWNjZXB0YWJsZSBmb3IgYSBob29rIGluc3RhbnRpYXRlZCBvbmNlIGluIFJFUEwuXG4gIGNvbnN0IGNvdW50UmVmID0gdXNlUmVmKG1lc3NhZ2VDb3VudClcbiAgY291bnRSZWYuY3VycmVudCA9IG1lc3NhZ2VDb3VudFxuICAvLyBzY3JvbGxIZWlnaHQgc25hcHNob3Qg4oCUIHRoZSBkaXZpZGVyJ3MgeSBpbiBjb250ZW50IGNvb3Jkcy4gUmVmLW9ubHk6XG4gIC8vIHJlYWQgc3luY2hyb25vdXNseSBpbiBvblNjcm9sbEF3YXkgKHNldFN0YXRlIGlzIGJhdGNoZWQsIGNhbid0XG4gIC8vIHJlYWQtdGhlbi13cml0ZSBpbiB0aGUgc2FtZSBjYWxsYmFjaykgQU5EIGJ5IEZ1bGxzY3JlZW5MYXlvdXQnc1xuICAvLyBwaWxsVmlzaWJsZSBzdWJzY3JpcHRpb24uIG51bGwgPSBwaW5uZWQgdG8gYm90dG9tLlxuICBjb25zdCBkaXZpZGVyWVJlZiA9IHVzZVJlZjxudW1iZXIgfCBudWxsPihudWxsKVxuXG4gIGNvbnN0IG9uUmVwaW4gPSB1c2VDYWxsYmFjaygoKSA9PiB7XG4gICAgLy8gRG9uJ3QgY2xlYXIgZGl2aWRlcllSZWYgaGVyZSDigJQgYSB0cmFja3BhZCBtb21lbnR1bSB3aGVlbCBldmVudFxuICAgIC8vIHJhY2luZyBpbiB0aGUgc2FtZSBzdGRpbiBiYXRjaCB3b3VsZCBzZWUgbnVsbCBhbmQgcmUtc25hcHNob3QsXG4gICAgLy8gb3ZlcnJpZGluZyB0aGUgc2V0RGl2aWRlckluZGV4KG51bGwpIGJlbG93LiBUaGUgdXNlRWZmZWN0IGJlbG93XG4gICAgLy8gY2xlYXJzIHRoZSByZWYgYWZ0ZXIgUmVhY3QgY29tbWl0cyB0aGUgbnVsbCBkaXZpZGVySW5kZXgsIHNvIHRoZVxuICAgIC8vIHJlZiBzdGF5cyBub24tbnVsbCB1bnRpbCB0aGUgc3RhdGUgc2V0dGxlcy5cbiAgICBzZXREaXZpZGVySW5kZXgobnVsbClcbiAgfSwgW10pXG5cbiAgY29uc3Qgb25TY3JvbGxBd2F5ID0gdXNlQ2FsbGJhY2soKGhhbmRsZTogU2Nyb2xsQm94SGFuZGxlKSA9PiB7XG4gICAgLy8gTm90aGluZyBiZWxvdyB0aGUgdmlld3BvcnQg4oaSIG5vdGhpbmcgdG8ganVtcCB0by4gQ292ZXJzIGJvdGg6XG4gICAgLy8g4oCiIGVtcHR5L3Nob3J0IHNlc3Npb246IHNjcm9sbFVwIGNhbGxzIHNjcm9sbFRvKDApIHdoaWNoIGJyZWFrcyBzdGlja3lcbiAgICAvLyAgIGV2ZW4gYXQgc2Nyb2xsVG9wPTAgKHdoZWVsLXVwIG9uIGZyZXNoIHNlc3Npb24gc2hvd2VkIHRoZSBwaWxsKVxuICAgIC8vIOKAoiBjbGljay10by1zZWxlY3QgYXQgYm90dG9tOiB1c2VEcmFnVG9TY3JvbGwuY2hlY2soKSBjYWxsc1xuICAgIC8vICAgc2Nyb2xsVG8oY3VycmVudCkgdG8gYnJlYWsgc3RpY2t5IHNvIHN0cmVhbWluZyBjb250ZW50IGRvZXNuJ3Qgc2hpZnRcbiAgICAvLyAgIHVuZGVyIHRoZSBzZWxlY3Rpb24sIHRoZW4gb25TY3JvbGwoZmFsc2UsIOKApikg4oCUIGJ1dCBzY3JvbGxUb3AgaXMgc3RpbGxcbiAgICAvLyAgIGF0IG1heCAoU2FyYWggRGVhdG9uLCAjY2xhdWRlLWNvZGUtZmVlZGJhY2sgMjAyNi0wMy0xNSlcbiAgICAvLyBwZW5kaW5nRGVsdGE6IHNjcm9sbEJ5IGFjY3VtdWxhdGVzIHdpdGhvdXQgdXBkYXRpbmcgc2Nyb2xsVG9wLiBXaXRob3V0XG4gICAgLy8gaXQsIHdoZWVsaW5nIHVwIGZyb20gbWF4IHdvdWxkIHNlZSBzY3JvbGxUb3A9PW1heCBhbmQgc3VwcHJlc3MgdGhlIHBpbGwuXG4gICAgY29uc3QgbWF4ID0gTWF0aC5tYXgoXG4gICAgICAwLFxuICAgICAgaGFuZGxlLmdldFNjcm9sbEhlaWdodCgpIC0gaGFuZGxlLmdldFZpZXdwb3J0SGVpZ2h0KCksXG4gICAgKVxuICAgIGlmIChoYW5kbGUuZ2V0U2Nyb2xsVG9wKCkgKyBoYW5kbGUuZ2V0UGVuZGluZ0RlbHRhKCkgPj0gbWF4KSByZXR1cm5cbiAgICAvLyBTbmFwc2hvdCBvbmx5IG9uIHRoZSBGSVJTVCBzY3JvbGwtYXdheS4gb25TY3JvbGxBd2F5IGZpcmVzIG9uIEVWRVJZXG4gICAgLy8gc2Nyb2xsIGFjdGlvbiAobm90IGp1c3QgdGhlIGluaXRpYWwgYnJlYWsgZnJvbSBzdGlja3kpIOKAlCB0aGlzIGd1YXJkXG4gICAgLy8gcHJlc2VydmVzIHRoZSBvcmlnaW5hbCBiYXNlbGluZSBzbyB0aGUgY291bnQgZG9lc24ndCByZXNldCBvbiB0aGVcbiAgICAvLyBzZWNvbmQgUGFnZVVwLiBTdWJzZXF1ZW50IGNhbGxzIGFyZSByZWYtb25seSBuby1vcHMgKG5vIFJFUEwgcmUtcmVuZGVyKS5cbiAgICBpZiAoZGl2aWRlcllSZWYuY3VycmVudCA9PT0gbnVsbCkge1xuICAgICAgZGl2aWRlcllSZWYuY3VycmVudCA9IGhhbmRsZS5nZXRTY3JvbGxIZWlnaHQoKVxuICAgICAgLy8gTmV3IHNjcm9sbC1hd2F5IHNlc3Npb24g4oaSIG1vdmUgdGhlIGRpdmlkZXIgaGVyZSAocmVwbGFjZXMgb2xkIG9uZSlcbiAgICAgIHNldERpdmlkZXJJbmRleChjb3VudFJlZi5jdXJyZW50KVxuICAgIH1cbiAgfSwgW10pXG5cbiAgY29uc3QganVtcFRvTmV3ID0gdXNlQ2FsbGJhY2soKGhhbmRsZTogU2Nyb2xsQm94SGFuZGxlIHwgbnVsbCkgPT4ge1xuICAgIGlmICghaGFuZGxlKSByZXR1cm5cbiAgICAvLyBzY3JvbGxUb0JvdHRvbSAobm90IHNjcm9sbFRvKGRpdmlkZXJZKSk6IHNldHMgc3RpY2t5U2Nyb2xsPXRydWUgc29cbiAgICAvLyB1c2VWaXJ0dWFsU2Nyb2xsIG1vdW50cyB0aGUgdGFpbCBhbmQgcmVuZGVyLW5vZGUtdG8tb3V0cHV0IHBpbnNcbiAgICAvLyBzY3JvbGxUb3A9bWF4U2Nyb2xsLiBzY3JvbGxUbyBzZXRzIHN0aWNreVNjcm9sbD1mYWxzZSDihpIgdGhlIGNsYW1wXG4gICAgLy8gKHN0aWxsIGF0IHRvcC1yYW5nZSBib3VuZHMgYmVmb3JlIFJlYWN0IHJlLXJlbmRlcnMpIHBpbnMgc2Nyb2xsVG9wXG4gICAgLy8gYmFjaywgc3RvcHBpbmcgc2hvcnQuIFRoZSBkaXZpZGVyIHN0YXlzIHJlbmRlcmVkIChkaXZpZGVySW5kZXhcbiAgICAvLyB1bmNoYW5nZWQpIHNvIHVzZXJzIHNlZSB3aGVyZSBuZXcgbWVzc2FnZXMgc3RhcnRlZDsgdGhlIGNsZWFyIG9uXG4gICAgLy8gbmV4dCBzdWJtaXQvZXhwbGljaXQgc2Nyb2xsLXRvLWJvdHRvbSBoYW5kbGVzIGNsZWFudXAuXG4gICAgaGFuZGxlLnNjcm9sbFRvQm90dG9tKClcbiAgfSwgW10pXG5cbiAgLy8gU3luYyBkaXZpZGVyWVJlZiB3aXRoIGRpdmlkZXJJbmRleC4gV2hlbiBvblJlcGluIGZpcmVzIChzdWJtaXQsXG4gIC8vIHNjcm9sbC10by1ib3R0b20pLCBpdCBzZXRzIGRpdmlkZXJJbmRleD1udWxsIGJ1dCBsZWF2ZXMgdGhlIHJlZlxuICAvLyBub24tbnVsbCDigJQgYSB3aGVlbCBldmVudCByYWNpbmcgaW4gdGhlIHNhbWUgc3RkaW4gYmF0Y2ggd291bGRcbiAgLy8gb3RoZXJ3aXNlIHNlZSBudWxsIGFuZCByZS1zbmFwc2hvdC4gRGVmZXJyaW5nIHRoZSByZWYgY2xlYXIgdG9cbiAgLy8gdXNlRWZmZWN0IGd1YXJhbnRlZXMgdGhlIHJlZiBzdGF5cyBub24tbnVsbCB1bnRpbCBSZWFjdCBoYXMgY29tbWl0dGVkXG4gIC8vIHRoZSBudWxsIGRpdmlkZXJJbmRleCwgYmxvY2tpbmcgdGhlIGlmLW51bGwgZ3VhcmQgaW4gb25TY3JvbGxBd2F5LlxuICAvL1xuICAvLyBBbHNvIGhhbmRsZXMgL2NsZWFyLCByZXdpbmQsIHRlYW1tYXRlLXZpZXcgc3dhcCDigJQgaWYgdGhlIGNvdW50IGRyb3BzXG4gIC8vIGJlbG93IHRoZSBkaXZpZGVyIGluZGV4LCB0aGUgZGl2aWRlciB3b3VsZCBwb2ludCBhdCBub3RoaW5nLlxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmIChkaXZpZGVySW5kZXggPT09IG51bGwpIHtcbiAgICAgIGRpdmlkZXJZUmVmLmN1cnJlbnQgPSBudWxsXG4gICAgfSBlbHNlIGlmIChtZXNzYWdlQ291bnQgPCBkaXZpZGVySW5kZXgpIHtcbiAgICAgIGRpdmlkZXJZUmVmLmN1cnJlbnQgPSBudWxsXG4gICAgICBzZXREaXZpZGVySW5kZXgobnVsbClcbiAgICB9XG4gIH0sIFttZXNzYWdlQ291bnQsIGRpdmlkZXJJbmRleF0pXG5cbiAgY29uc3Qgc2hpZnREaXZpZGVyID0gdXNlQ2FsbGJhY2soXG4gICAgKGluZGV4RGVsdGE6IG51bWJlciwgaGVpZ2h0RGVsdGE6IG51bWJlcikgPT4ge1xuICAgICAgc2V0RGl2aWRlckluZGV4KGlkeCA9PiAoaWR4ID09PSBudWxsID8gbnVsbCA6IGlkeCArIGluZGV4RGVsdGEpKVxuICAgICAgaWYgKGRpdmlkZXJZUmVmLmN1cnJlbnQgIT09IG51bGwpIHtcbiAgICAgICAgZGl2aWRlcllSZWYuY3VycmVudCArPSBoZWlnaHREZWx0YVxuICAgICAgfVxuICAgIH0sXG4gICAgW10sXG4gIClcblxuICByZXR1cm4ge1xuICAgIGRpdmlkZXJJbmRleCxcbiAgICBkaXZpZGVyWVJlZixcbiAgICBvblNjcm9sbEF3YXksXG4gICAgb25SZXBpbixcbiAgICBqdW1wVG9OZXcsXG4gICAgc2hpZnREaXZpZGVyLFxuICB9XG59XG5cbi8qKlxuICogQ291bnRzIGFzc2lzdGFudCB0dXJucyBpbiBtZXNzYWdlc1tkaXZpZGVySW5kZXguLmVuZCkuIEEgXCJ0dXJuXCIgaXMgd2hhdFxuICogdXNlcnMgdGhpbmsgb2YgYXMgXCJhIG5ldyBtZXNzYWdlIGZyb20gQ2xhdWRlXCIg4oCUIG5vdCByYXcgYXNzaXN0YW50IGVudHJpZXNcbiAqIChvbmUgdHVybiB5aWVsZHMgbXVsdGlwbGUgZW50cmllczogdG9vbF91c2UgYmxvY2tzICsgdGV4dCBibG9ja3MpLiBXZSBjb3VudFxuICogbm9uLWFzc2lzdGFudOKGkmFzc2lzdGFudCB0cmFuc2l0aW9ucywgYnV0IG9ubHkgZm9yIGVudHJpZXMgdGhhdCBhY3R1YWxseVxuICogY2FycnkgdGV4dCDigJQgdG9vbC11c2Utb25seSBlbnRyaWVzIGFyZSBza2lwcGVkIChsaWtlIHByb2dyZXNzIG1lc3NhZ2VzKVxuICogc28gXCLij7ogU2VhcmNoZWQgZm9yIDEzIHBhdHRlcm5zLCByZWFkIDYgZmlsZXNcIiBkb2Vzbid0IHRpY2sgdGhlIHBpbGwuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjb3VudFVuc2VlbkFzc2lzdGFudFR1cm5zKFxuICBtZXNzYWdlczogcmVhZG9ubHkgTWVzc2FnZVtdLFxuICBkaXZpZGVySW5kZXg6IG51bWJlcixcbik6IG51bWJlciB7XG4gIGxldCBjb3VudCA9IDBcbiAgbGV0IHByZXZXYXNBc3Npc3RhbnQgPSBmYWxzZVxuICBmb3IgKGxldCBpID0gZGl2aWRlckluZGV4OyBpIDwgbWVzc2FnZXMubGVuZ3RoOyBpKyspIHtcbiAgICBjb25zdCBtID0gbWVzc2FnZXNbaV0hXG4gICAgaWYgKG0udHlwZSA9PT0gJ3Byb2dyZXNzJykgY29udGludWVcbiAgICAvLyBUb29sLXVzZS1vbmx5IGFzc2lzdGFudCBlbnRyaWVzIGFyZW4ndCBcIm5ldyBtZXNzYWdlc1wiIHRvIHRoZSB1c2VyIOKAlFxuICAgIC8vIHNraXAgdGhlbSB0aGUgc2FtZSB3YXkgd2Ugc2tpcCBwcm9ncmVzcy4gcHJldldhc0Fzc2lzdGFudCBpcyBOT1RcbiAgICAvLyB1cGRhdGVkLCBzbyBhIHRleHQgYmxvY2sgaW1tZWRpYXRlbHkgZm9sbG93aW5nIHN0aWxsIGNvdW50cyBhcyB0aGVcbiAgICAvLyBzYW1lIHR1cm4gKHRvb2xfdXNlICsgdGV4dCBmcm9tIG9uZSBBUEkgcmVzcG9uc2UgPSAxKS5cbiAgICBpZiAobS50eXBlID09PSAnYXNzaXN0YW50JyAmJiAhYXNzaXN0YW50SGFzVmlzaWJsZVRleHQobSkpIGNvbnRpbnVlXG4gICAgY29uc3QgaXNBc3Npc3RhbnQgPSBtLnR5cGUgPT09ICdhc3Npc3RhbnQnXG4gICAgaWYgKGlzQXNzaXN0YW50ICYmICFwcmV2V2FzQXNzaXN0YW50KSBjb3VudCsrXG4gICAgcHJldldhc0Fzc2lzdGFudCA9IGlzQXNzaXN0YW50XG4gIH1cbiAgcmV0dXJuIGNvdW50XG59XG5cbmZ1bmN0aW9uIGFzc2lzdGFudEhhc1Zpc2libGVUZXh0KG06IE1lc3NhZ2UpOiBib29sZWFuIHtcbiAgaWYgKG0udHlwZSAhPT0gJ2Fzc2lzdGFudCcpIHJldHVybiBmYWxzZVxuICBmb3IgKGNvbnN0IGIgb2YgbS5tZXNzYWdlLmNvbnRlbnQpIHtcbiAgICBpZiAoYi50eXBlID09PSAndGV4dCcgJiYgYi50ZXh0LnRyaW0oKSAhPT0gJycpIHJldHVybiB0cnVlXG4gIH1cbiAgcmV0dXJuIGZhbHNlXG59XG5cbmV4cG9ydCB0eXBlIFVuc2VlbkRpdmlkZXIgPSB7IGZpcnN0VW5zZWVuVXVpZDogTWVzc2FnZVsndXVpZCddOyBjb3VudDogbnVtYmVyIH1cblxuLyoqXG4gKiBCdWlsZHMgdGhlIHVuc2VlbkRpdmlkZXIgb2JqZWN0IFJFUEwgcGFzc2VzIHRvIE1lc3NhZ2VzICsgdGhlIHBpbGwuXG4gKiBSZXR1cm5zIHVuZGVmaW5lZCBvbmx5IHdoZW4gbm8gY29udGVudCBoYXMgYXJyaXZlZCBwYXN0IHRoZSBkaXZpZGVyXG4gKiB5ZXQgKG1lc3NhZ2VzW2RpdmlkZXJJbmRleF0gZG9lc24ndCBleGlzdCkuIE9uY2UgQU5ZIG1lc3NhZ2UgYXJyaXZlc1xuICog4oCUIGluY2x1ZGluZyB0b29sX3VzZS1vbmx5IGFzc2lzdGFudCBlbnRyaWVzIGFuZCB0b29sX3Jlc3VsdCB1c2VyIGVudHJpZXNcbiAqIHRoYXQgY291bnRVbnNlZW5Bc3Npc3RhbnRUdXJucyBza2lwcyDigJQgY291bnQgZmxvb3JzIGF0IDEgc28gdGhlIHBpbGxcbiAqIGZsaXBzIGZyb20gXCJKdW1wIHRvIGJvdHRvbVwiIHRvIFwiMSBuZXcgbWVzc2FnZVwiLiBXaXRob3V0IHRoZSBmbG9vcixcbiAqIHRoZSBwaWxsIHN0YXlzIFwiSnVtcCB0byBib3R0b21cIiB0aHJvdWdoIGFuIGVudGlyZSB0b29sLWNhbGwgc2VxdWVuY2VcbiAqIHVudGlsIENsYXVkZSdzIHRleHQgcmVzcG9uc2UgbGFuZHMuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjb21wdXRlVW5zZWVuRGl2aWRlcihcbiAgbWVzc2FnZXM6IHJlYWRvbmx5IE1lc3NhZ2VbXSxcbiAgZGl2aWRlckluZGV4OiBudW1iZXIgfCBudWxsLFxuKTogVW5zZWVuRGl2aWRlciB8IHVuZGVmaW5lZCB7XG4gIGlmIChkaXZpZGVySW5kZXggPT09IG51bGwpIHJldHVybiB1bmRlZmluZWRcbiAgLy8gU2tpcCBwcm9ncmVzcyBhbmQgbnVsbC1yZW5kZXJpbmcgYXR0YWNobWVudHMgd2hlbiBwaWNraW5nIHRoZSBkaXZpZGVyXG4gIC8vIGFuY2hvciDigJQgTWVzc2FnZXMudHN4IGZpbHRlcnMgdGhlc2Ugb3V0IG9mIHJlbmRlcmFibGVNZXNzYWdlcyBiZWZvcmUgdGhlXG4gIC8vIGRpdmlkZXJCZWZvcmVJbmRleCBzZWFyY2gsIHNvIHRoZWlyIFVVSUQgd291bGRuJ3QgYmUgZm91bmQgKENDLTcyNCkuXG4gIC8vIEhvb2sgYXR0YWNobWVudHMgdXNlIHJhbmRvbVVVSUQoKSBzbyBub3RoaW5nIHNoYXJlcyB0aGVpciAyNC1jaGFyIHByZWZpeC5cbiAgbGV0IGFuY2hvcklkeCA9IGRpdmlkZXJJbmRleFxuICB3aGlsZSAoXG4gICAgYW5jaG9ySWR4IDwgbWVzc2FnZXMubGVuZ3RoICYmXG4gICAgKG1lc3NhZ2VzW2FuY2hvcklkeF0/LnR5cGUgPT09ICdwcm9ncmVzcycgfHxcbiAgICAgIGlzTnVsbFJlbmRlcmluZ0F0dGFjaG1lbnQobWVzc2FnZXNbYW5jaG9ySWR4XSEpKVxuICApIHtcbiAgICBhbmNob3JJZHgrK1xuICB9XG4gIGNvbnN0IHV1aWQgPSBtZXNzYWdlc1thbmNob3JJZHhdPy51dWlkXG4gIGlmICghdXVpZCkgcmV0dXJuIHVuZGVmaW5lZFxuICBjb25zdCBjb3VudCA9IGNvdW50VW5zZWVuQXNzaXN0YW50VHVybnMobWVzc2FnZXMsIGRpdmlkZXJJbmRleClcbiAgcmV0dXJuIHsgZmlyc3RVbnNlZW5VdWlkOiB1dWlkLCBjb3VudDogTWF0aC5tYXgoMSwgY291bnQpIH1cbn1cblxuLyoqXG4gKiBMYXlvdXQgd3JhcHBlciBmb3IgdGhlIFJFUEwuIEluIGZ1bGxzY3JlZW4gbW9kZSwgcHV0cyBzY3JvbGxhYmxlXG4gKiBjb250ZW50IGluIGEgc3RpY2t5LXNjcm9sbCBib3ggYW5kIHBpbnMgYm90dG9tIGNvbnRlbnQgdmlhIGZsZXhib3guXG4gKiBPdXRzaWRlIGZ1bGxzY3JlZW4gbW9kZSwgcmVuZGVycyBjb250ZW50IHNlcXVlbnRpYWxseSBzbyB0aGUgZXhpc3RpbmdcbiAqIG1haW4tc2NyZWVuIHNjcm9sbGJhY2sgcmVuZGVyaW5nIHdvcmtzIHVuY2hhbmdlZC5cbiAqXG4gKiBGdWxsc2NyZWVuIG1vZGUgZGVmYXVsdHMgb24gZm9yIGFudHMgKENMQVVERV9DT0RFX05PX0ZMSUNLRVI9MCB0byBvcHQgb3V0KVxuICogYW5kIG9mZiBmb3IgZXh0ZXJuYWwgdXNlcnMgKENMQVVERV9DT0RFX05PX0ZMSUNLRVI9MSB0byBvcHQgaW4pLlxuICogVGhlIDxBbHRlcm5hdGVTY3JlZW4+IHdyYXBwZXJcbiAqIChhbHQgYnVmZmVyICsgbW91c2UgdHJhY2tpbmcgKyBoZWlnaHQgY29uc3RyYWludCkgbGl2ZXMgYXQgUkVQTCdzIHJvb3RcbiAqIHNvIG5vdGhpbmcgY2FuIGFjY2lkZW50YWxseSByZW5kZXIgb3V0c2lkZSBpdC5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIEZ1bGxzY3JlZW5MYXlvdXQoe1xuICBzY3JvbGxhYmxlLFxuICBib3R0b20sXG4gIG92ZXJsYXksXG4gIGJvdHRvbUZsb2F0LFxuICBtb2RhbCxcbiAgbW9kYWxTY3JvbGxSZWYsXG4gIHNjcm9sbFJlZixcbiAgZGl2aWRlcllSZWYsXG4gIGhpZGVQaWxsID0gZmFsc2UsXG4gIGhpZGVTdGlja3kgPSBmYWxzZSxcbiAgbmV3TWVzc2FnZUNvdW50ID0gMCxcbiAgb25QaWxsQ2xpY2ssXG59OiBQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IHsgcm93czogdGVybWluYWxSb3dzLCBjb2x1bW5zIH0gPSB1c2VUZXJtaW5hbFNpemUoKVxuICAvLyBTY3JvbGwtZGVyaXZlZCBjaHJvbWUgc3RhdGUgbGl2ZXMgSEVSRSwgbm90IGluIFJFUEwuIFN0aWNreVRyYWNrZXJcbiAgLy8gd3JpdGVzIHZpYSBTY3JvbGxDaHJvbWVDb250ZXh0OyBwaWxsVmlzaWJsZSBzdWJzY3JpYmVzIGRpcmVjdGx5IHRvXG4gIC8vIFNjcm9sbEJveC4gQm90aCBjaGFuZ2UgcmFyZWx5IChwaWxsIGZsaXBzIG9uY2UgcGVyIHRocmVzaG9sZCBjcm9zc2luZyxcbiAgLy8gc3RpY2t5IGNoYW5nZXMgfjUtMjDDly90cmFuc2NyaXB0KSDigJQgcmUtcmVuZGVyaW5nIEZ1bGxzY3JlZW5MYXlvdXQgb25cbiAgLy8gdGhvc2UgaXMgZmluZTsgcmUtcmVuZGVyaW5nIHRoZSA2OTY2LWxpbmUgUkVQTCArIGl0cyAyMisgdXNlQXBwU3RhdGVcbiAgLy8gc2VsZWN0b3JzIHBlci1zY3JvbGwtZnJhbWUgd2FzIG5vdC5cbiAgY29uc3QgW3N0aWNreVByb21wdCwgc2V0U3RpY2t5UHJvbXB0XSA9IHVzZVN0YXRlPFN0aWNreVByb21wdCB8IG51bGw+KG51bGwpXG4gIGNvbnN0IGNocm9tZUN0eCA9IHVzZU1lbW8oKCkgPT4gKHsgc2V0U3RpY2t5UHJvbXB0IH0pLCBbXSlcbiAgLy8gQm9vbGVhbi1xdWFudGl6ZWQgc2Nyb2xsIHN1YnNjcmlwdGlvbi4gU25hcHNob3QgaXMgXCJpcyB2aWV3cG9ydCBib3R0b21cbiAgLy8gYWJvdmUgdGhlIGRpdmlkZXIgeT9cIiDigJQgT2JqZWN0LmlzIG9uIGEgYm9vbGVhbiDihpIgRnVsbHNjcmVlbkxheW91dCBvbmx5XG4gIC8vIHJlLXJlbmRlcnMgd2hlbiB0aGUgcGlsbCBzaG91bGQgYWN0dWFsbHkgZmxpcCwgbm90IHBlci1mcmFtZS5cbiAgY29uc3Qgc3Vic2NyaWJlID0gdXNlQ2FsbGJhY2soXG4gICAgKGxpc3RlbmVyOiAoKSA9PiB2b2lkKSA9PlxuICAgICAgc2Nyb2xsUmVmPy5jdXJyZW50Py5zdWJzY3JpYmUobGlzdGVuZXIpID8/ICgoKSA9PiB7fSksXG4gICAgW3Njcm9sbFJlZl0sXG4gIClcbiAgY29uc3QgcGlsbFZpc2libGUgPSB1c2VTeW5jRXh0ZXJuYWxTdG9yZShzdWJzY3JpYmUsICgpID0+IHtcbiAgICBjb25zdCBzID0gc2Nyb2xsUmVmPy5jdXJyZW50XG4gICAgY29uc3QgZGl2aWRlclkgPSBkaXZpZGVyWVJlZj8uY3VycmVudFxuICAgIGlmICghcyB8fCBkaXZpZGVyWSA9PSBudWxsKSByZXR1cm4gZmFsc2VcbiAgICByZXR1cm4gKFxuICAgICAgcy5nZXRTY3JvbGxUb3AoKSArIHMuZ2V0UGVuZGluZ0RlbHRhKCkgKyBzLmdldFZpZXdwb3J0SGVpZ2h0KCkgPCBkaXZpZGVyWVxuICAgIClcbiAgfSlcbiAgLy8gV2lyZSB1cCBoeXBlcmxpbmsgY2xpY2sgaGFuZGxpbmcg4oCUIGluIGZ1bGxzY3JlZW4gbW9kZSwgbW91c2UgdHJhY2tpbmdcbiAgLy8gaW50ZXJjZXB0cyBjbGlja3MgYmVmb3JlIHRoZSB0ZXJtaW5hbCBjYW4gb3BlbiBPU0MgOCBsaW5rcyBuYXRpdmVseS5cbiAgdXNlTGF5b3V0RWZmZWN0KCgpID0+IHtcbiAgICBpZiAoIWlzRnVsbHNjcmVlbkVudkVuYWJsZWQoKSkgcmV0dXJuXG4gICAgY29uc3QgaW5rID0gaW5zdGFuY2VzLmdldChwcm9jZXNzLnN0ZG91dClcbiAgICBpZiAoIWluaykgcmV0dXJuXG4gICAgaW5rLm9uSHlwZXJsaW5rQ2xpY2sgPSB1cmwgPT4ge1xuICAgICAgLy8gTW9zdCBPU0MgOCBsaW5rcyBlbWl0dGVkIGJ5IENsYXVkZSBDb2RlIGFyZSBmaWxlOi8vIFVSTHMgZnJvbVxuICAgICAgLy8gRmlsZVBhdGhMaW5rIChGaWxlRWRpdC9GaWxlV3JpdGUvRmlsZVJlYWQgdG9vbCBvdXRwdXQpLiBvcGVuQnJvd3NlclxuICAgICAgLy8gcmVqZWN0cyBub24taHR0cChzKSBwcm90b2NvbHMg4oCUIHJvdXRlIGZpbGU6IHRvIG9wZW5QYXRoIGluc3RlYWQuXG4gICAgICBpZiAodXJsLnN0YXJ0c1dpdGgoJ2ZpbGU6JykpIHtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICB2b2lkIG9wZW5QYXRoKGZpbGVVUkxUb1BhdGgodXJsKSlcbiAgICAgICAgfSBjYXRjaCB7XG4gICAgICAgICAgLy8gTWFsZm9ybWVkIGZpbGU6IFVSTHMgKGUuZy4gZmlsZTovL2hvc3QvcGF0aCBmcm9tIHBsYWluLXRleHRcbiAgICAgICAgICAvLyBkZXRlY3Rpb24pIGNhdXNlIGZpbGVVUkxUb1BhdGggdG8gdGhyb3cg4oCUIGlnbm9yZSBzaWxlbnRseS5cbiAgICAgICAgfVxuICAgICAgfSBlbHNlIHtcbiAgICAgICAgdm9pZCBvcGVuQnJvd3Nlcih1cmwpXG4gICAgICB9XG4gICAgfVxuICAgIHJldHVybiAoKSA9PiB7XG4gICAgICBpbmsub25IeXBlcmxpbmtDbGljayA9IHVuZGVmaW5lZFxuICAgIH1cbiAgfSwgW10pXG5cbiAgaWYgKGlzRnVsbHNjcmVlbkVudkVuYWJsZWQoKSkge1xuICAgIC8vIE92ZXJsYXkgcmVuZGVycyBCRUxPVyBtZXNzYWdlcyBpbnNpZGUgdGhlIHNhbWUgU2Nyb2xsQm94IOKAlCB1c2VyIGNhblxuICAgIC8vIHNjcm9sbCB1cCB0byBzZWUgcHJpb3IgY29udGV4dCB3aGlsZSBhIHBlcm1pc3Npb24gZGlhbG9nIGlzIHNob3dpbmcuXG4gICAgLy8gVGhlIFNjcm9sbEJveCBuZXZlciB1bm1vdW50cyBhY3Jvc3Mgb3ZlcmxheSB0cmFuc2l0aW9ucywgc28gc2Nyb2xsXG4gICAgLy8gcG9zaXRpb24gaXMgcHJlc2VydmVkIHdpdGhvdXQgc2F2ZS9yZXN0b3JlLiBzdGlja3lTY3JvbGwgYXV0by1zY3JvbGxzXG4gICAgLy8gdG8gdGhlIGFwcGVuZGVkIG92ZXJsYXkgd2hlbiBpdCBtb3VudHMgKGlmIHVzZXIgd2FzIGFscmVhZHkgYXRcbiAgICAvLyBib3R0b20pOyBSRVBMIHJlLXBpbnMgb24gdGhlIG92ZXJsYXkgYXBwZWFyL2Rpc21pc3MgdHJhbnNpdGlvbiBmb3JcbiAgICAvLyB0aGUgY2FzZSB3aGVyZSBzdGlja3kgd2FzIGJyb2tlbi4gVGFsbCBkaWFsb2dzIChGaWxlRWRpdCBkaWZmcykgc3RpbGxcbiAgICAvLyBnZXQgUGdVcC9QZ0RuL3doZWVsIOKAlCBzYW1lIHNjcm9sbFJlZiBkcml2ZXMgdGhlIHNhbWUgU2Nyb2xsQm94LlxuICAgIC8vIFRocmVlIHN0aWNreSBzdGF0ZXM6IG51bGwgKGF0IGJvdHRvbSksIHt0ZXh0LHNjcm9sbFRvfSAoc2Nyb2xsZWQgdXAsXG4gICAgLy8gaGVhZGVyIHNob3dzKSwgJ2NsaWNrZWQnIChqdXN0IGNsaWNrZWQgaGVhZGVyIOKAlCBoaWRlIGl0IHNvIHRoZVxuICAgIC8vIGNvbnRlbnQg4p2vIHRha2VzIHJvdyAwKS4gcGFkQ29sbGFwc2VkIGNvdmVycyB0aGUgbGF0dGVyIHR3bzogb25jZVxuICAgIC8vIHNjcm9sbGVkIGF3YXkgZnJvbSBib3R0b20sIHBhZGRpbmcgZHJvcHMgdG8gMCBhbmQgc3RheXMgdGhlcmUgdW50aWxcbiAgICAvLyByZXBpbi4gaGVhZGVyVmlzaWJsZSBpcyBvbmx5IHRoZSBtaWRkbGUgc3RhdGUuIEFmdGVyIGNsaWNrOlxuICAgIC8vIHNjcm9sbEJveF95PTAgKGhlYWRlciBnb25lKSArIHBhZGRpbmc9MCDihpIgdmlld3BvcnRUb3A9MCDihpIg4p2vIGF0XG4gICAgLy8gcm93IDAuIE9uIG5leHQgc2Nyb2xsIHRoZSBvbkNoYW5nZSBmaXJlcyB3aXRoIGEgZnJlc2gge3RleHR9IGFuZFxuICAgIC8vIGhlYWRlciBjb21lcyBiYWNrICh2aWV3cG9ydFRvcCAw4oaSMSwgYSBzaW5nbGUgMS1yb3cgc2hpZnQg4oCUXG4gICAgLy8gYWNjZXB0YWJsZSBzaW5jZSB1c2VyIGV4cGxpY2l0bHkgc2Nyb2xsZWQpLlxuICAgIGNvbnN0IHN0aWNreSA9IGhpZGVTdGlja3kgPyBudWxsIDogc3RpY2t5UHJvbXB0XG4gICAgY29uc3QgaGVhZGVyUHJvbXB0ID1cbiAgICAgIHN0aWNreSAhPSBudWxsICYmIHN0aWNreSAhPT0gJ2NsaWNrZWQnICYmIG92ZXJsYXkgPT0gbnVsbCA/IHN0aWNreSA6IG51bGxcbiAgICBjb25zdCBwYWRDb2xsYXBzZWQgPSBzdGlja3kgIT0gbnVsbCAmJiBvdmVybGF5ID09IG51bGxcbiAgICByZXR1cm4gKFxuICAgICAgPFByb21wdE92ZXJsYXlQcm92aWRlcj5cbiAgICAgICAgPEJveCBmbGV4R3Jvdz17MX0gZmxleERpcmVjdGlvbj1cImNvbHVtblwiIG92ZXJmbG93PVwiaGlkZGVuXCI+XG4gICAgICAgICAge2hlYWRlclByb21wdCAmJiAoXG4gICAgICAgICAgICA8U3RpY2t5UHJvbXB0SGVhZGVyXG4gICAgICAgICAgICAgIHRleHQ9e2hlYWRlclByb21wdC50ZXh0fVxuICAgICAgICAgICAgICBvbkNsaWNrPXtoZWFkZXJQcm9tcHQuc2Nyb2xsVG99XG4gICAgICAgICAgICAvPlxuICAgICAgICAgICl9XG4gICAgICAgICAgPFNjcm9sbEJveFxuICAgICAgICAgICAgcmVmPXtzY3JvbGxSZWZ9XG4gICAgICAgICAgICBmbGV4R3Jvdz17MX1cbiAgICAgICAgICAgIGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIlxuICAgICAgICAgICAgcGFkZGluZ1RvcD17cGFkQ29sbGFwc2VkID8gMCA6IDF9XG4gICAgICAgICAgICBzdGlja3lTY3JvbGxcbiAgICAgICAgICA+XG4gICAgICAgICAgICA8U2Nyb2xsQ2hyb21lQ29udGV4dCB2YWx1ZT17Y2hyb21lQ3R4fT5cbiAgICAgICAgICAgICAge3Njcm9sbGFibGV9XG4gICAgICAgICAgICA8L1Njcm9sbENocm9tZUNvbnRleHQ+XG4gICAgICAgICAgICB7b3ZlcmxheX1cbiAgICAgICAgICA8L1Njcm9sbEJveD5cbiAgICAgICAgICB7IWhpZGVQaWxsICYmIHBpbGxWaXNpYmxlICYmIG92ZXJsYXkgPT0gbnVsbCAmJiAoXG4gICAgICAgICAgICA8TmV3TWVzc2FnZXNQaWxsIGNvdW50PXtuZXdNZXNzYWdlQ291bnR9IG9uQ2xpY2s9e29uUGlsbENsaWNrfSAvPlxuICAgICAgICAgICl9XG4gICAgICAgICAge2JvdHRvbUZsb2F0ICE9IG51bGwgJiYgKFxuICAgICAgICAgICAgPEJveCBwb3NpdGlvbj1cImFic29sdXRlXCIgYm90dG9tPXswfSByaWdodD17MH0gb3BhcXVlPlxuICAgICAgICAgICAgICB7Ym90dG9tRmxvYXR9XG4gICAgICAgICAgICA8L0JveD5cbiAgICAgICAgICApfVxuICAgICAgICA8L0JveD5cbiAgICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCIgZmxleFNocmluaz17MH0gd2lkdGg9XCIxMDAlXCIgbWF4SGVpZ2h0PVwiNTAlXCI+XG4gICAgICAgICAgPFN1Z2dlc3Rpb25zT3ZlcmxheSAvPlxuICAgICAgICAgIDxEaWFsb2dPdmVybGF5IC8+XG4gICAgICAgICAgPEJveFxuICAgICAgICAgICAgZmxleERpcmVjdGlvbj1cImNvbHVtblwiXG4gICAgICAgICAgICB3aWR0aD1cIjEwMCVcIlxuICAgICAgICAgICAgZmxleEdyb3c9ezF9XG4gICAgICAgICAgICBvdmVyZmxvd1k9XCJoaWRkZW5cIlxuICAgICAgICAgID5cbiAgICAgICAgICAgIHtib3R0b219XG4gICAgICAgICAgPC9Cb3g+XG4gICAgICAgIDwvQm94PlxuICAgICAgICB7bW9kYWwgIT0gbnVsbCAmJiAoXG4gICAgICAgICAgPE1vZGFsQ29udGV4dFxuICAgICAgICAgICAgdmFsdWU9e3tcbiAgICAgICAgICAgICAgcm93czogdGVybWluYWxSb3dzIC0gTU9EQUxfVFJBTlNDUklQVF9QRUVLIC0gMSxcbiAgICAgICAgICAgICAgY29sdW1uczogY29sdW1ucyAtIDQsXG4gICAgICAgICAgICAgIHNjcm9sbFJlZjogbW9kYWxTY3JvbGxSZWYgPz8gbnVsbCxcbiAgICAgICAgICAgIH19XG4gICAgICAgICAgPlxuICAgICAgICAgICAgey8qIEJvdHRvbS1hbmNob3JlZCwgZ3Jvd3MgdXB3YXJkIHRvIGZpdCBjb250ZW50LiBtYXhIZWlnaHQga2VlcHMgYVxuICAgICAgICAgICAgICAgIGZldyByb3dzIG9mIHRyYW5zY3JpcHQgcGVlayBhYm92ZSB0aGUg4paUIGRpdmlkZXIuIFNob3J0IG1vZGFsc1xuICAgICAgICAgICAgICAgICgvbW9kZWwpIHNpdCBzbWFsbCBhdCB0aGUgYm90dG9tIHdpdGggbG90cyBvZiB0cmFuc2NyaXB0IGFib3ZlO1xuICAgICAgICAgICAgICAgIHRhbGwgbW9kYWxzICgvYnVkZHkgQ2FyZCkgZ3JvdyBhcyBuZWVkZWQsIGNsaXBwZWQgYnkgb3ZlcmZsb3cuXG4gICAgICAgICAgICAgICAgUHJldmlvdXNseSBmaXhlZC1oZWlnaHQgKHRvcCtib3R0b20gYW5jaG9yZWQpIOKAlCBhbnkgZml4ZWQgY2FwXG4gICAgICAgICAgICAgICAgZWl0aGVyIGNsaXBwZWQgdGFsbCBjb250ZW50IG9yIGxlZnQgc2hvcnQgY29udGVudCBmbG9hdGluZyBpblxuICAgICAgICAgICAgICAgIGEgbW9zdGx5LWVtcHR5IHBhbmUuXG5cbiAgICAgICAgICAgICAgICBmbGV4U2hyaW5rPTAgb24gdGhlIGlubmVyIEJveCBpcyBsb2FkLWJlYXJpbmc6IHdpdGggU2hyaW5rPTEsXG4gICAgICAgICAgICAgICAgeW9nYSBzcXVlZXplcyBkZWVwIGNoaWxkcmVuIHRvIGg9MCB3aGVuIGNvbnRlbnQgPiBtYXhIZWlnaHQsXG4gICAgICAgICAgICAgICAgYW5kIHNpYmxpbmcgVGV4dHMgbGFuZCBvbiB0aGUgc2FtZSByb3cg4oaSIGdob3N0IG92ZXJsYXBcbiAgICAgICAgICAgICAgICAoXCI1IHNlcnZlcnNQIHNlcnZlcnNcIikuIENsaXBwaW5nIGF0IHRoZSBvdXRlciBCb3gncyBtYXhIZWlnaHRcbiAgICAgICAgICAgICAgICBrZWVwcyBjaGlsZHJlbiBhdCBuYXR1cmFsIHNpemUuXG5cbiAgICAgICAgICAgICAgICBEaXZpZGVyIHdyYXBwZWQgaW4gZmxleFNocmluaz0wOiB3aGVuIHRoZSBpbm5lciBib3ggb3ZlcmZsb3dzXG4gICAgICAgICAgICAgICAgKHRhbGwgL2NvbmZpZyBvcHRpb24gbGlzdCksIHlvZ2Egc2hyaW5rcyB0aGUgZGl2aWRlciBUZXh0IHRvXG4gICAgICAgICAgICAgICAgaD0wIHRvIGFic29yYiB0aGUgZGVmaWNpdCDigJQgaXQncyB0aGUgb25seSBzaHJpbmthYmxlIHNpYmxpbmcuXG4gICAgICAgICAgICAgICAgVGhlIHdyYXBwZXIga2VlcHMgaXQgYXQgMSByb3c7IG92ZXJmbG93IHBhc3QgbWF4SGVpZ2h0IGlzXG4gICAgICAgICAgICAgICAgY2xpcHBlZCBhdCB0aGUgYm90dG9tIGJ5IG92ZXJmbG93PWhpZGRlbiBpbnN0ZWFkLiAqL31cbiAgICAgICAgICAgIDxCb3hcbiAgICAgICAgICAgICAgcG9zaXRpb249XCJhYnNvbHV0ZVwiXG4gICAgICAgICAgICAgIGJvdHRvbT17MH1cbiAgICAgICAgICAgICAgbGVmdD17MH1cbiAgICAgICAgICAgICAgcmlnaHQ9ezB9XG4gICAgICAgICAgICAgIG1heEhlaWdodD17dGVybWluYWxSb3dzIC0gTU9EQUxfVFJBTlNDUklQVF9QRUVLfVxuICAgICAgICAgICAgICBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCJcbiAgICAgICAgICAgICAgb3ZlcmZsb3c9XCJoaWRkZW5cIlxuICAgICAgICAgICAgICBvcGFxdWVcbiAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgPEJveCBmbGV4U2hyaW5rPXswfT5cbiAgICAgICAgICAgICAgICA8VGV4dCBjb2xvcj1cInBlcm1pc3Npb25cIj57J+KWlCcucmVwZWF0KGNvbHVtbnMpfTwvVGV4dD5cbiAgICAgICAgICAgICAgPC9Cb3g+XG4gICAgICAgICAgICAgIDxCb3hcbiAgICAgICAgICAgICAgICBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCJcbiAgICAgICAgICAgICAgICBwYWRkaW5nWD17Mn1cbiAgICAgICAgICAgICAgICBmbGV4U2hyaW5rPXswfVxuICAgICAgICAgICAgICAgIG92ZXJmbG93PVwiaGlkZGVuXCJcbiAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgIHttb2RhbH1cbiAgICAgICAgICAgICAgPC9Cb3g+XG4gICAgICAgICAgICA8L0JveD5cbiAgICAgICAgICA8L01vZGFsQ29udGV4dD5cbiAgICAgICAgKX1cbiAgICAgIDwvUHJvbXB0T3ZlcmxheVByb3ZpZGVyPlxuICAgIClcbiAgfVxuXG4gIHJldHVybiAoXG4gICAgPD5cbiAgICAgIHtzY3JvbGxhYmxlfVxuICAgICAge2JvdHRvbX1cbiAgICAgIHtvdmVybGF5fVxuICAgICAge21vZGFsfVxuICAgIDwvPlxuICApXG59XG5cbi8vIFNsYWNrLXN0eWxlIHBpbGwuIEFic29sdXRlIG92ZXJsYXkgYXQgYm90dG9tPXswfSBvZiB0aGUgc2Nyb2xsd3JhcCDigJQgZmxvYXRzXG4vLyBvdmVyIHRoZSBTY3JvbGxCb3gncyBsYXN0IGNvbnRlbnQgcm93LCBvbmx5IG9ic2N1cmluZyB0aGUgY2VudGVyZWQgcGlsbFxuLy8gdGV4dCAodGhlIHJlc3Qgb2YgdGhlIHJvdyBzaG93cyBTY3JvbGxCb3ggY29udGVudCkuIFNjcm9sbC1zbWVhciBmcm9tXG4vLyBERUNTVEJNIHNoaWZ0aW5nIHRoZSBwaWxsJ3MgcGl4ZWxzIGlzIHJlcGFpcmVkIGF0IHRoZSBJbmsgbGF5ZXJcbi8vIChhYnNvbHV0ZVJlY3RzUHJldiB0aGlyZC1wYXNzIGluIHJlbmRlci1ub2RlLXRvLW91dHB1dC50cywgIzIzOTM5KS4gU2hvd3Ncbi8vIFwiSnVtcCB0byBib3R0b21cIiB3aGVuIGNvdW50IGlzIDAgKHNjcm9sbGVkIGF3YXkgYnV0IG5vIG5ldyBtZXNzYWdlcyB5ZXQg4oCUXG4vLyB0aGUgZGVhZCB6b25lIHdoZXJlIHVzZXJzIHByZXZpb3VzbHkgdGhvdWdodCBjaGF0IHN0YWxsZWQpLlxuZnVuY3Rpb24gTmV3TWVzc2FnZXNQaWxsKHtcbiAgY291bnQsXG4gIG9uQ2xpY2ssXG59OiB7XG4gIGNvdW50OiBudW1iZXJcbiAgb25DbGljaz86ICgpID0+IHZvaWRcbn0pOiBSZWFjdC5SZWFjdE5vZGUge1xuICBjb25zdCBbaG92ZXIsIHNldEhvdmVyXSA9IHVzZVN0YXRlKGZhbHNlKVxuICByZXR1cm4gKFxuICAgIDxCb3hcbiAgICAgIHBvc2l0aW9uPVwiYWJzb2x1dGVcIlxuICAgICAgYm90dG9tPXswfVxuICAgICAgbGVmdD17MH1cbiAgICAgIHJpZ2h0PXswfVxuICAgICAganVzdGlmeUNvbnRlbnQ9XCJjZW50ZXJcIlxuICAgID5cbiAgICAgIDxCb3hcbiAgICAgICAgb25DbGljaz17b25DbGlja31cbiAgICAgICAgb25Nb3VzZUVudGVyPXsoKSA9PiBzZXRIb3Zlcih0cnVlKX1cbiAgICAgICAgb25Nb3VzZUxlYXZlPXsoKSA9PiBzZXRIb3ZlcihmYWxzZSl9XG4gICAgICA+XG4gICAgICAgIDxUZXh0XG4gICAgICAgICAgYmFja2dyb3VuZENvbG9yPXtcbiAgICAgICAgICAgIGhvdmVyID8gJ3VzZXJNZXNzYWdlQmFja2dyb3VuZEhvdmVyJyA6ICd1c2VyTWVzc2FnZUJhY2tncm91bmQnXG4gICAgICAgICAgfVxuICAgICAgICAgIGRpbUNvbG9yXG4gICAgICAgID5cbiAgICAgICAgICB7JyAnfVxuICAgICAgICAgIHtjb3VudCA+IDBcbiAgICAgICAgICAgID8gYCR7Y291bnR9IG5ldyAke3BsdXJhbChjb3VudCwgJ21lc3NhZ2UnKX1gXG4gICAgICAgICAgICA6ICdKdW1wIHRvIGJvdHRvbSd9eycgJ31cbiAgICAgICAgICB7ZmlndXJlcy5hcnJvd0Rvd259eycgJ31cbiAgICAgICAgPC9UZXh0PlxuICAgICAgPC9Cb3g+XG4gICAgPC9Cb3g+XG4gIClcbn1cblxuLy8gQ29udGV4dCBicmVhZGNydW1iOiB3aGVuIHNjcm9sbGVkIHVwIGludG8gaGlzdG9yeSwgcGluIHRoZSBjdXJyZW50XG4vLyBjb252ZXJzYXRpb24gdHVybidzIHByb21wdCBhYm92ZSB0aGUgdmlld3BvcnQgc28geW91IGtub3cgd2hhdCBDbGF1ZGUgd2FzXG4vLyByZXNwb25kaW5nIHRvLiBOb3JtYWwtZmxvdyBzaWJsaW5nIEJFRk9SRSB0aGUgU2Nyb2xsQm94IChtaXJyb3JzIHRoZSBwaWxsXG4vLyBiZWxvdyBpdCkg4oCUIHNocmlua3MgdGhlIFNjcm9sbEJveCBieSBleGFjdGx5IDEgcm93IHZpYSBmbGV4LCBzdGF5cyBvdXRzaWRlXG4vLyB0aGUgREVDU1RCTSBzY3JvbGwgcmVnaW9uLiBDbGljayBqdW1wcyBiYWNrIHRvIHRoZSBwcm9tcHQuXG4vL1xuLy8gSGVpZ2h0IGlzIEZJWEVEIGF0IDEgcm93ICh0cnVuY2F0ZS1lbmQgZm9yIGxvbmcgcHJvbXB0cykuIEEgdmFyaWFibGUtaGVpZ2h0XG4vLyBoZWFkZXIgKDEgd2hlbiBzaG9ydCwgMiB3aGVuIHdyYXBwZWQpIHNoaWZ0cyB0aGUgU2Nyb2xsQm94IGJ5IDEgcm93IGV2ZXJ5XG4vLyB0aW1lIHRoZSBzdGlja3kgcHJvbXB0IHN3aXRjaGVzIGR1cmluZyBzY3JvbGwg4oCUIGNvbnRlbnQganVtcHMgb24gc2NyZWVuXG4vLyBldmVuIHdpdGggc2Nyb2xsVG9wIHVuY2hhbmdlZCAodGhlIERFQ1NUQk0gcmVnaW9uIHRvcCBzaGlmdHMgd2l0aCB0aGVcbi8vIFNjcm9sbEJveCwgYW5kIHRoZSBkaWZmIGVuZ2luZSBzZWVzIFwiZXZlcnl0aGluZyBtb3ZlZFwiKS4gRml4ZWQgaGVpZ2h0XG4vLyBrZWVwcyB0aGUgU2Nyb2xsQm94IGFuY2hvcmVkOyBvbmx5IHRoZSBoZWFkZXIgVEVYVCBjaGFuZ2VzLCBub3QgaXRzIGJveC5cbmZ1bmN0aW9uIFN0aWNreVByb21wdEhlYWRlcih7XG4gIHRleHQsXG4gIG9uQ2xpY2ssXG59OiB7XG4gIHRleHQ6IHN0cmluZ1xuICBvbkNsaWNrOiAoKSA9PiB2b2lkXG59KTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgY29uc3QgW2hvdmVyLCBzZXRIb3Zlcl0gPSB1c2VTdGF0ZShmYWxzZSlcbiAgcmV0dXJuIChcbiAgICA8Qm94XG4gICAgICBmbGV4U2hyaW5rPXswfVxuICAgICAgd2lkdGg9XCIxMDAlXCJcbiAgICAgIGhlaWdodD17MX1cbiAgICAgIHBhZGRpbmdSaWdodD17MX1cbiAgICAgIGJhY2tncm91bmRDb2xvcj17XG4gICAgICAgIGhvdmVyID8gJ3VzZXJNZXNzYWdlQmFja2dyb3VuZEhvdmVyJyA6ICd1c2VyTWVzc2FnZUJhY2tncm91bmQnXG4gICAgICB9XG4gICAgICBvbkNsaWNrPXtvbkNsaWNrfVxuICAgICAgb25Nb3VzZUVudGVyPXsoKSA9PiBzZXRIb3Zlcih0cnVlKX1cbiAgICAgIG9uTW91c2VMZWF2ZT17KCkgPT4gc2V0SG92ZXIoZmFsc2UpfVxuICAgID5cbiAgICAgIDxUZXh0IGNvbG9yPVwic3VidGxlXCIgd3JhcD1cInRydW5jYXRlLWVuZFwiPlxuICAgICAgICB7ZmlndXJlcy5wb2ludGVyfSB7dGV4dH1cbiAgICAgIDwvVGV4dD5cbiAgICA8L0JveD5cbiAgKVxufVxuXG4vLyBTbGFzaC1jb21tYW5kIHN1Z2dlc3Rpb24gb3ZlcmxheSDigJQgc2VlIHByb21wdE92ZXJsYXlDb250ZXh0LnRzeCBmb3Igd2h5XG4vLyBpdCdzIHBvcnRhbGVkLiBTY3JvbGwtc21lYXIgZnJvbSBmbG9hdGluZyBvdmVyIHRoZSBERUNTVEJNIHJlZ2lvbiBpc1xuLy8gcmVwYWlyZWQgYXQgdGhlIEluayBsYXllciAoYWJzb2x1dGVSZWN0c1ByZXYgaW4gcmVuZGVyLW5vZGUtdG8tb3V0cHV0LnRzKS5cbi8vIFRoZSByZW5kZXJlciBjbGFtcHMgbmVnYXRpdmUgeSB0byAwIGZvciBhYnNvbHV0ZSBlbGVtZW50cyAoc2VlXG4vLyByZW5kZXItbm9kZS10by1vdXRwdXQudHMpLCBzbyB0aGUgdG9wIHJvd3MgKGJlc3QgbWF0Y2hlcykgc3RheSB2aXNpYmxlXG4vLyBldmVuIHdoZW4gdGhlIG92ZXJsYXkgZXh0ZW5kcyBhYm92ZSB0aGUgdmlld3BvcnQuIFdlIG9taXQgbWluSGVpZ2h0IGFuZFxuLy8gZmxleC1lbmQgaGVyZTogdGhleSB3b3VsZCBjcmVhdGUgZW1wdHkgcGFkZGluZyByb3dzIHRoYXQgc2hpZnQgdmlzaWJsZVxuLy8gaXRlbXMgZG93biBpbnRvIHRoZSBwcm9tcHQgYXJlYSB3aGVuIHRoZSBsaXN0IGhhcyBmZXdlciBpdGVtcyB0aGFuIG1heC5cbmZ1bmN0aW9uIFN1Z2dlc3Rpb25zT3ZlcmxheSgpOiBSZWFjdC5SZWFjdE5vZGUge1xuICBjb25zdCBkYXRhID0gdXNlUHJvbXB0T3ZlcmxheSgpXG4gIGlmICghZGF0YSB8fCBkYXRhLnN1Z2dlc3Rpb25zLmxlbmd0aCA9PT0gMCkgcmV0dXJuIG51bGxcbiAgcmV0dXJuIChcbiAgICA8Qm94XG4gICAgICBwb3NpdGlvbj1cImFic29sdXRlXCJcbiAgICAgIGJvdHRvbT1cIjEwMCVcIlxuICAgICAgbGVmdD17MH1cbiAgICAgIHJpZ2h0PXswfVxuICAgICAgcGFkZGluZ1g9ezJ9XG4gICAgICBwYWRkaW5nVG9wPXsxfVxuICAgICAgZmxleERpcmVjdGlvbj1cImNvbHVtblwiXG4gICAgICBvcGFxdWVcbiAgICA+XG4gICAgICA8UHJvbXB0SW5wdXRGb290ZXJTdWdnZXN0aW9uc1xuICAgICAgICBzdWdnZXN0aW9ucz17ZGF0YS5zdWdnZXN0aW9uc31cbiAgICAgICAgc2VsZWN0ZWRTdWdnZXN0aW9uPXtkYXRhLnNlbGVjdGVkU3VnZ2VzdGlvbn1cbiAgICAgICAgbWF4Q29sdW1uV2lkdGg9e2RhdGEubWF4Q29sdW1uV2lkdGh9XG4gICAgICAgIG92ZXJsYXlcbiAgICAgIC8+XG4gICAgPC9Cb3g+XG4gIClcbn1cblxuLy8gRGlhbG9nIHBvcnRhbGVkIGZyb20gUHJvbXB0SW5wdXQgKEF1dG9Nb2RlT3B0SW5EaWFsb2cpIOKAlCBzYW1lIGNsaXAtZXNjYXBlXG4vLyBwYXR0ZXJuIGFzIFN1Z2dlc3Rpb25zT3ZlcmxheS4gUmVuZGVycyBsYXRlciBpbiB0cmVlIG9yZGVyIHNvIGl0IHBhaW50c1xuLy8gb3ZlciBzdWdnZXN0aW9ucyBpZiBib3RoIGFyZSBldmVyIHVwICh0aGV5IHNob3VsZG4ndCBiZSkuXG5mdW5jdGlvbiBEaWFsb2dPdmVybGF5KCk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IG5vZGUgPSB1c2VQcm9tcHRPdmVybGF5RGlhbG9nKClcbiAgaWYgKCFub2RlKSByZXR1cm4gbnVsbFxuICByZXR1cm4gKFxuICAgIDxCb3ggcG9zaXRpb249XCJhYnNvbHV0ZVwiIGJvdHRvbT1cIjEwMCVcIiBsZWZ0PXswfSByaWdodD17MH0gb3BhcXVlPlxuICAgICAge25vZGV9XG4gICAgPC9Cb3g+XG4gIClcbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU9BLE9BQU8sTUFBTSxTQUFTO0FBQzdCLE9BQU9DLEtBQUssSUFDVkMsYUFBYSxFQUNiLEtBQUtDLFNBQVMsRUFDZCxLQUFLQyxTQUFTLEVBQ2RDLFdBQVcsRUFDWEMsU0FBUyxFQUNUQyxlQUFlLEVBQ2ZDLE9BQU8sRUFDUEMsTUFBTSxFQUNOQyxRQUFRLEVBQ1JDLG9CQUFvQixRQUNmLE9BQU87QUFDZCxTQUFTQyxhQUFhLFFBQVEsS0FBSztBQUNuQyxTQUFTQyxZQUFZLFFBQVEsNEJBQTRCO0FBQ3pELFNBQ0VDLHFCQUFxQixFQUNyQkMsZ0JBQWdCLEVBQ2hCQyxzQkFBc0IsUUFDakIsb0NBQW9DO0FBQzNDLFNBQVNDLGVBQWUsUUFBUSw2QkFBNkI7QUFDN0QsT0FBT0MsU0FBUyxJQUFJLEtBQUtDLGVBQWUsUUFBUSxnQ0FBZ0M7QUFDaEYsT0FBT0MsU0FBUyxNQUFNLHFCQUFxQjtBQUMzQyxTQUFTQyxHQUFHLEVBQUVDLElBQUksUUFBUSxXQUFXO0FBQ3JDLGNBQWNDLE9BQU8sUUFBUSxxQkFBcUI7QUFDbEQsU0FBU0MsV0FBVyxFQUFFQyxRQUFRLFFBQVEscUJBQXFCO0FBQzNELFNBQVNDLHNCQUFzQixRQUFRLHdCQUF3QjtBQUMvRCxTQUFTQyxNQUFNLFFBQVEseUJBQXlCO0FBQ2hELFNBQVNDLHlCQUF5QixRQUFRLHdDQUF3QztBQUNsRixPQUFPQyw0QkFBNEIsTUFBTSwrQ0FBK0M7QUFDeEYsY0FBY0MsWUFBWSxRQUFRLHlCQUF5Qjs7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 \ No newline at end of file diff --git a/src/components/GlobalSearchDialog.tsx b/src/components/GlobalSearchDialog.tsx new file mode 100644 index 0000000..b4551e2 --- /dev/null +++ b/src/components/GlobalSearchDialog.tsx @@ -0,0 +1,343 @@ +import { c as _c } from "react/compiler-runtime"; +import { resolve as resolvePath } from 'path'; +import * as React from 'react'; +import { useEffect, useRef, useState } from 'react'; +import { useRegisterOverlay } from '../context/overlayContext.js'; +import { useTerminalSize } from '../hooks/useTerminalSize.js'; +import { Text } from '../ink.js'; +import { logEvent } from '../services/analytics/index.js'; +import { getCwd } from '../utils/cwd.js'; +import { openFileInExternalEditor } from '../utils/editor.js'; +import { truncatePathMiddle, truncateToWidth } from '../utils/format.js'; +import { highlightMatch } from '../utils/highlightMatch.js'; +import { relativePath } from '../utils/permissions/filesystem.js'; +import { readFileInRange } from '../utils/readFileInRange.js'; +import { ripGrepStream } from '../utils/ripgrep.js'; +import { FuzzyPicker } from './design-system/FuzzyPicker.js'; +import { LoadingState } from './design-system/LoadingState.js'; +type Props = { + onDone: () => void; + onInsert: (text: string) => void; +}; +type Match = { + file: string; + line: number; + text: string; +}; +const VISIBLE_RESULTS = 12; +const DEBOUNCE_MS = 100; +const PREVIEW_CONTEXT_LINES = 4; +// rg -m is per-file; we also cap the parsed array to keep memory bounded. +const MAX_MATCHES_PER_FILE = 10; +const MAX_TOTAL_MATCHES = 500; + +/** + * Global Search dialog (ctrl+shift+f / cmd+shift+f). + * Debounced ripgrep search across the workspace. + */ +export function GlobalSearchDialog(t0) { + const $ = _c(40); + const { + onDone, + onInsert + } = t0; + useRegisterOverlay("global-search"); + const { + columns, + rows + } = useTerminalSize(); + const previewOnRight = columns >= 140; + const visibleResults = Math.min(VISIBLE_RESULTS, Math.max(4, rows - 14)); + let t1; + if ($[0] === Symbol.for("react.memo_cache_sentinel")) { + t1 = []; + $[0] = t1; + } else { + t1 = $[0]; + } + const [matches, setMatches] = useState(t1); + const [truncated, setTruncated] = useState(false); + const [isSearching, setIsSearching] = useState(false); + const [query, setQuery] = useState(""); + const [focused, setFocused] = useState(undefined); + const [preview, setPreview] = useState(null); + const abortRef = useRef(null); + const timeoutRef = useRef(null); + let t2; + let t3; + if ($[1] === Symbol.for("react.memo_cache_sentinel")) { + t2 = () => () => { + if (timeoutRef.current) { + clearTimeout(timeoutRef.current); + } + abortRef.current?.abort(); + }; + t3 = []; + $[1] = t2; + $[2] = t3; + } else { + t2 = $[1]; + t3 = $[2]; + } + useEffect(t2, t3); + let t4; + let t5; + if ($[3] !== focused) { + t4 = () => { + if (!focused) { + setPreview(null); + return; + } + const controller = new AbortController(); + const absolute = resolvePath(getCwd(), focused.file); + const start = Math.max(0, focused.line - PREVIEW_CONTEXT_LINES - 1); + readFileInRange(absolute, start, PREVIEW_CONTEXT_LINES * 2 + 1, undefined, controller.signal).then(r => { + if (controller.signal.aborted) { + return; + } + setPreview({ + file: focused.file, + line: focused.line, + content: r.content + }); + }).catch(() => { + if (controller.signal.aborted) { + return; + } + setPreview({ + file: focused.file, + line: focused.line, + content: "(preview unavailable)" + }); + }); + return () => controller.abort(); + }; + t5 = [focused]; + $[3] = focused; + $[4] = t4; + $[5] = t5; + } else { + t4 = $[4]; + t5 = $[5]; + } + useEffect(t4, t5); + let t6; + if ($[6] === Symbol.for("react.memo_cache_sentinel")) { + t6 = q => { + setQuery(q); + if (timeoutRef.current) { + clearTimeout(timeoutRef.current); + } + abortRef.current?.abort(); + if (!q.trim()) { + setMatches(_temp); + setIsSearching(false); + setTruncated(false); + return; + } + const controller_0 = new AbortController(); + abortRef.current = controller_0; + setIsSearching(true); + setTruncated(false); + const queryLower = q.toLowerCase(); + setMatches(m_0 => { + const filtered = m_0.filter(match => match.text.toLowerCase().includes(queryLower)); + return filtered.length === m_0.length ? m_0 : filtered; + }); + timeoutRef.current = setTimeout(_temp4, DEBOUNCE_MS, q, controller_0, setMatches, setTruncated, setIsSearching); + }; + $[6] = t6; + } else { + t6 = $[6]; + } + const handleQueryChange = t6; + const listWidth = previewOnRight ? Math.floor((columns - 10) * 0.5) : columns - 8; + const maxPathWidth = Math.max(20, Math.floor(listWidth * 0.4)); + const maxTextWidth = Math.max(20, listWidth - maxPathWidth - 4); + const previewWidth = previewOnRight ? Math.max(40, columns - listWidth - 14) : columns - 6; + let t7; + if ($[7] !== matches.length || $[8] !== onDone) { + t7 = m_3 => { + const opened = openFileInExternalEditor(resolvePath(getCwd(), m_3.file), m_3.line); + logEvent("tengu_global_search_select", { + result_count: matches.length, + opened_editor: opened + }); + onDone(); + }; + $[7] = matches.length; + $[8] = onDone; + $[9] = t7; + } else { + t7 = $[9]; + } + const handleOpen = t7; + let t8; + if ($[10] !== matches.length || $[11] !== onDone || $[12] !== onInsert) { + t8 = (m_4, mention) => { + onInsert(mention ? `@${m_4.file}#L${m_4.line} ` : `${m_4.file}:${m_4.line} `); + logEvent("tengu_global_search_insert", { + result_count: matches.length, + mention + }); + onDone(); + }; + $[10] = matches.length; + $[11] = onDone; + $[12] = onInsert; + $[13] = t8; + } else { + t8 = $[13]; + } + const handleInsert = t8; + const matchLabel = matches.length > 0 ? `${matches.length}${truncated ? "+" : ""} matches${isSearching ? "\u2026" : ""}` : " "; + const t9 = previewOnRight ? "right" : "bottom"; + let t10; + if ($[14] !== handleInsert) { + t10 = { + action: "mention", + handler: m_5 => handleInsert(m_5, true) + }; + $[14] = handleInsert; + $[15] = t10; + } else { + t10 = $[15]; + } + let t11; + if ($[16] !== handleInsert) { + t11 = { + action: "insert path", + handler: m_6 => handleInsert(m_6, false) + }; + $[16] = handleInsert; + $[17] = t11; + } else { + t11 = $[17]; + } + let t12; + if ($[18] !== isSearching) { + t12 = q_0 => isSearching ? "Searching\u2026" : q_0 ? "No matches" : "Type to search\u2026"; + $[18] = isSearching; + $[19] = t12; + } else { + t12 = $[19]; + } + let t13; + if ($[20] !== maxPathWidth || $[21] !== maxTextWidth || $[22] !== query) { + t13 = (m_7, isFocused) => {truncatePathMiddle(m_7.file, maxPathWidth)}:{m_7.line}{" "}{highlightMatch(truncateToWidth(m_7.text.trimStart(), maxTextWidth), query)}; + $[20] = maxPathWidth; + $[21] = maxTextWidth; + $[22] = query; + $[23] = t13; + } else { + t13 = $[23]; + } + let t14; + if ($[24] !== preview || $[25] !== previewWidth || $[26] !== query) { + t14 = m_8 => preview?.file === m_8.file && preview.line === m_8.line ? <>{truncatePathMiddle(m_8.file, previewWidth)}:{m_8.line}{preview.content.split("\n").map((line_0, i) => {highlightMatch(truncateToWidth(line_0, previewWidth), query)})} : ; + $[24] = preview; + $[25] = previewWidth; + $[26] = query; + $[27] = t14; + } else { + t14 = $[27]; + } + let t15; + if ($[28] !== handleOpen || $[29] !== matchLabel || $[30] !== matches || $[31] !== onDone || $[32] !== t10 || $[33] !== t11 || $[34] !== t12 || $[35] !== t13 || $[36] !== t14 || $[37] !== t9 || $[38] !== visibleResults) { + t15 = ; + $[28] = handleOpen; + $[29] = matchLabel; + $[30] = matches; + $[31] = onDone; + $[32] = t10; + $[33] = t11; + $[34] = t12; + $[35] = t13; + $[36] = t14; + $[37] = t9; + $[38] = visibleResults; + $[39] = t15; + } else { + t15 = $[39]; + } + return t15; +} +function _temp4(query_0, controller_1, setMatches_0, setTruncated_0, setIsSearching_0) { + const cwd = getCwd(); + let collected = 0; + ripGrepStream(["-n", "--no-heading", "-i", "-m", String(MAX_MATCHES_PER_FILE), "-F", "-e", query_0], cwd, controller_1.signal, lines => { + if (controller_1.signal.aborted) { + return; + } + const parsed = []; + for (const line of lines) { + const m_1 = parseRipgrepLine(line); + if (!m_1) { + continue; + } + const rel = relativePath(cwd, m_1.file); + parsed.push({ + ...m_1, + file: rel.startsWith("..") ? m_1.file : rel + }); + } + if (!parsed.length) { + return; + } + collected = collected + parsed.length; + collected; + setMatches_0(prev => { + const seen = new Set(prev.map(matchKey)); + const fresh = parsed.filter(p => !seen.has(matchKey(p))); + if (!fresh.length) { + return prev; + } + const next = prev.concat(fresh); + return next.length > MAX_TOTAL_MATCHES ? next.slice(0, MAX_TOTAL_MATCHES) : next; + }); + if (collected >= MAX_TOTAL_MATCHES) { + controller_1.abort(); + setTruncated_0(true); + setIsSearching_0(false); + } + }).catch(_temp2).finally(() => { + if (controller_1.signal.aborted) { + return; + } + if (collected === 0) { + setMatches_0(_temp3); + } + setIsSearching_0(false); + }); +} +function _temp3(m_2) { + return m_2.length ? [] : m_2; +} +function _temp2() {} +function _temp(m) { + return m.length ? [] : m; +} +function matchKey(m: Match): string { + return `${m.file}:${m.line}`; +} + +/** + * Parse a ripgrep -n --no-heading output line: "path:line:text". + * Windows paths may contain a drive letter ("C:\..."), so a simple split on + * the first colon would mangle the path — use a regex that captures up to + * the first :: instead. + * @internal exported for testing + */ +export function parseRipgrepLine(line: string): Match | null { + const m = /^(.*?):(\d+):(.*)$/.exec(line); + if (!m) return null; + const [, file, lineStr, text] = m; + const lineNum = Number(lineStr); + if (!file || !Number.isFinite(lineNum)) return null; + return { + file, + line: lineNum, + text: text ?? '' + }; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJyZXNvbHZlIiwicmVzb2x2ZVBhdGgiLCJSZWFjdCIsInVzZUVmZmVjdCIsInVzZVJlZiIsInVzZVN0YXRlIiwidXNlUmVnaXN0ZXJPdmVybGF5IiwidXNlVGVybWluYWxTaXplIiwiVGV4dCIsImxvZ0V2ZW50IiwiZ2V0Q3dkIiwib3BlbkZpbGVJbkV4dGVybmFsRWRpdG9yIiwidHJ1bmNhdGVQYXRoTWlkZGxlIiwidHJ1bmNhdGVUb1dpZHRoIiwiaGlnaGxpZ2h0TWF0Y2giLCJyZWxhdGl2ZVBhdGgiLCJyZWFkRmlsZUluUmFuZ2UiLCJyaXBHcmVwU3RyZWFtIiwiRnV6enlQaWNrZXIiLCJMb2FkaW5nU3RhdGUiLCJQcm9wcyIsIm9uRG9uZSIsIm9uSW5zZXJ0IiwidGV4dCIsIk1hdGNoIiwiZmlsZSIsImxpbmUiLCJWSVNJQkxFX1JFU1VMVFMiLCJERUJPVU5DRV9NUyIsIlBSRVZJRVdfQ09OVEVYVF9MSU5FUyIsIk1BWF9NQVRDSEVTX1BFUl9GSUxFIiwiTUFYX1RPVEFMX01BVENIRVMiLCJHbG9iYWxTZWFyY2hEaWFsb2ciLCJ0MCIsIiQiLCJfYyIsImNvbHVtbnMiLCJyb3dzIiwicHJldmlld09uUmlnaHQiLCJ2aXNpYmxlUmVzdWx0cyIsIk1hdGgiLCJtaW4iLCJtYXgiLCJ0MSIsIlN5bWJvbCIsImZvciIsIm1hdGNoZXMiLCJzZXRNYXRjaGVzIiwidHJ1bmNhdGVkIiwic2V0VHJ1bmNhdGVkIiwiaXNTZWFyY2hpbmciLCJzZXRJc1NlYXJjaGluZyIsInF1ZXJ5Iiwic2V0UXVlcnkiLCJmb2N1c2VkIiwic2V0Rm9jdXNlZCIsInVuZGVmaW5lZCIsInByZXZpZXciLCJzZXRQcmV2aWV3IiwiYWJvcnRSZWYiLCJ0aW1lb3V0UmVmIiwidDIiLCJ0MyIsImN1cnJlbnQiLCJjbGVhclRpbWVvdXQiLCJhYm9ydCIsInQ0IiwidDUiLCJjb250cm9sbGVyIiwiQWJvcnRDb250cm9sbGVyIiwiYWJzb2x1dGUiLCJzdGFydCIsInNpZ25hbCIsInRoZW4iLCJyIiwiYWJvcnRlZCIsImNvbnRlbnQiLCJjYXRjaCIsInQ2IiwicSIsInRyaW0iLCJfdGVtcCIsImNvbnRyb2xsZXJfMCIsInF1ZXJ5TG93ZXIiLCJ0b0xvd2VyQ2FzZSIsIm1fMCIsImZpbHRlcmVkIiwibSIsImZpbHRlciIsIm1hdGNoIiwiaW5jbHVkZXMiLCJsZW5ndGgiLCJzZXRUaW1lb3V0IiwiX3RlbXA0IiwiaGFuZGxlUXVlcnlDaGFuZ2UiLCJsaXN0V2lkdGgiLCJmbG9vciIsIm1heFBhdGhXaWR0aCIsIm1heFRleHRXaWR0aCIsInByZXZpZXdXaWR0aCIsInQ3IiwibV8zIiwib3BlbmVkIiwicmVzdWx0X2NvdW50Iiwib3BlbmVkX2VkaXRvciIsImhhbmRsZU9wZW4iLCJ0OCIsIm1fNCIsIm1lbnRpb24iLCJoYW5kbGVJbnNlcnQiLCJtYXRjaExhYmVsIiwidDkiLCJ0MTAiLCJhY3Rpb24iLCJoYW5kbGVyIiwibV81IiwidDExIiwibV82IiwidDEyIiwicV8wIiwidDEzIiwibV83IiwiaXNGb2N1c2VkIiwidHJpbVN0YXJ0IiwidDE0IiwibV84Iiwic3BsaXQiLCJtYXAiLCJsaW5lXzAiLCJpIiwidDE1IiwibWF0Y2hLZXkiLCJxdWVyeV8wIiwiY29udHJvbGxlcl8xIiwic2V0TWF0Y2hlc18wIiwic2V0VHJ1bmNhdGVkXzAiLCJzZXRJc1NlYXJjaGluZ18wIiwiY3dkIiwiY29sbGVjdGVkIiwiU3RyaW5nIiwibGluZXMiLCJwYXJzZWQiLCJtXzEiLCJwYXJzZVJpcGdyZXBMaW5lIiwicmVsIiwicHVzaCIsInN0YXJ0c1dpdGgiLCJwcmV2Iiwic2VlbiIsIlNldCIsImZyZXNoIiwicCIsImhhcyIsIm5leHQiLCJjb25jYXQiLCJzbGljZSIsIl90ZW1wMiIsImZpbmFsbHkiLCJfdGVtcDMiLCJtXzIiLCJleGVjIiwibGluZVN0ciIsImxpbmVOdW0iLCJOdW1iZXIiLCJpc0Zpbml0ZSJdLCJzb3VyY2VzIjpbIkdsb2JhbFNlYXJjaERpYWxvZy50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgcmVzb2x2ZSBhcyByZXNvbHZlUGF0aCB9IGZyb20gJ3BhdGgnXG5pbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZUVmZmVjdCwgdXNlUmVmLCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgdXNlUmVnaXN0ZXJPdmVybGF5IH0gZnJvbSAnLi4vY29udGV4dC9vdmVybGF5Q29udGV4dC5qcydcbmltcG9ydCB7IHVzZVRlcm1pbmFsU2l6ZSB9IGZyb20gJy4uL2hvb2tzL3VzZVRlcm1pbmFsU2l6ZS5qcydcbmltcG9ydCB7IFRleHQgfSBmcm9tICcuLi9pbmsuanMnXG5pbXBvcnQgeyBsb2dFdmVudCB9IGZyb20gJy4uL3NlcnZpY2VzL2FuYWx5dGljcy9pbmRleC5qcydcbmltcG9ydCB7IGdldEN3ZCB9IGZyb20gJy4uL3V0aWxzL2N3ZC5qcydcbmltcG9ydCB7IG9wZW5GaWxlSW5FeHRlcm5hbEVkaXRvciB9IGZyb20gJy4uL3V0aWxzL2VkaXRvci5qcydcbmltcG9ydCB7IHRydW5jYXRlUGF0aE1pZGRsZSwgdHJ1bmNhdGVUb1dpZHRoIH0gZnJvbSAnLi4vdXRpbHMvZm9ybWF0LmpzJ1xuaW1wb3J0IHsgaGlnaGxpZ2h0TWF0Y2ggfSBmcm9tICcuLi91dGlscy9oaWdobGlnaHRNYXRjaC5qcydcbmltcG9ydCB7IHJlbGF0aXZlUGF0aCB9IGZyb20gJy4uL3V0aWxzL3Blcm1pc3Npb25zL2ZpbGVzeXN0ZW0uanMnXG5pbXBvcnQgeyByZWFkRmlsZUluUmFuZ2UgfSBmcm9tICcuLi91dGlscy9yZWFkRmlsZUluUmFuZ2UuanMnXG5pbXBvcnQgeyByaXBHcmVwU3RyZWFtIH0gZnJvbSAnLi4vdXRpbHMvcmlwZ3JlcC5qcydcbmltcG9ydCB7IEZ1enp5UGlja2VyIH0gZnJvbSAnLi9kZXNpZ24tc3lzdGVtL0Z1enp5UGlja2VyLmpzJ1xuaW1wb3J0IHsgTG9hZGluZ1N0YXRlIH0gZnJvbSAnLi9kZXNpZ24tc3lzdGVtL0xvYWRpbmdTdGF0ZS5qcydcblxudHlwZSBQcm9wcyA9IHtcbiAgb25Eb25lOiAoKSA9PiB2b2lkXG4gIG9uSW5zZXJ0OiAodGV4dDogc3RyaW5nKSA9PiB2b2lkXG59XG5cbnR5cGUgTWF0Y2ggPSB7XG4gIGZpbGU6IHN0cmluZ1xuICBsaW5lOiBudW1iZXJcbiAgdGV4dDogc3RyaW5nXG59XG5cbmNvbnN0IFZJU0lCTEVfUkVTVUxUUyA9IDEyXG5jb25zdCBERUJPVU5DRV9NUyA9IDEwMFxuY29uc3QgUFJFVklFV19DT05URVhUX0xJTkVTID0gNFxuLy8gcmcgLW0gaXMgcGVyLWZpbGU7IHdlIGFsc28gY2FwIHRoZSBwYXJzZWQgYXJyYXkgdG8ga2VlcCBtZW1vcnkgYm91bmRlZC5cbmNvbnN0IE1BWF9NQVRDSEVTX1BFUl9GSUxFID0gMTBcbmNvbnN0IE1BWF9UT1RBTF9NQVRDSEVTID0gNTAwXG5cbi8qKlxuICogR2xvYmFsIFNlYXJjaCBkaWFsb2cgKGN0cmwrc2hpZnQrZiAvIGNtZCtzaGlmdCtmKS5cbiAqIERlYm91bmNlZCByaXBncmVwIHNlYXJjaCBhY3Jvc3MgdGhlIHdvcmtzcGFjZS5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIEdsb2JhbFNlYXJjaERpYWxvZyh7XG4gIG9uRG9uZSxcbiAgb25JbnNlcnQsXG59OiBQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIHVzZVJlZ2lzdGVyT3ZlcmxheSgnZ2xvYmFsLXNlYXJjaCcpXG4gIGNvbnN0IHsgY29sdW1ucywgcm93cyB9ID0gdXNlVGVybWluYWxTaXplKClcbiAgY29uc3QgcHJldmlld09uUmlnaHQgPSBjb2x1bW5zID49IDE0MFxuICAvLyBDaHJvbWUgKHRpdGxlICsgc2VhcmNoICsgbWF0Y2hMYWJlbCArIGhpbnRzICsgcGFuZSBib3JkZXIgKyBnYXBzKSBlYXRzXG4gIC8vIH4xNCByb3dzLiBTaHJpbmsgdGhlIGxpc3Qgb24gc2hvcnQgdGVybWluYWxzIHNvIHRoZSBkaWFsb2cgZG9lc24ndCBjbGlwLlxuICBjb25zdCB2aXNpYmxlUmVzdWx0cyA9IE1hdGgubWluKFZJU0lCTEVfUkVTVUxUUywgTWF0aC5tYXgoNCwgcm93cyAtIDE0KSlcblxuICBjb25zdCBbbWF0Y2hlcywgc2V0TWF0Y2hlc10gPSB1c2VTdGF0ZTxNYXRjaFtdPihbXSlcbiAgY29uc3QgW3RydW5jYXRlZCwgc2V0VHJ1bmNhdGVkXSA9IHVzZVN0YXRlKGZhbHNlKVxuICBjb25zdCBbaXNTZWFyY2hpbmcsIHNldElzU2VhcmNoaW5nXSA9IHVzZVN0YXRlKGZhbHNlKVxuICBjb25zdCBbcXVlcnksIHNldFF1ZXJ5XSA9IHVzZVN0YXRlKCcnKVxuICBjb25zdCBbZm9jdXNlZCwgc2V0Rm9jdXNlZF0gPSB1c2VTdGF0ZTxNYXRjaCB8IHVuZGVmaW5lZD4odW5kZWZpbmVkKVxuICBjb25zdCBbcHJldmlldywgc2V0UHJldmlld10gPSB1c2VTdGF0ZTx7XG4gICAgZmlsZTogc3RyaW5nXG4gICAgbGluZTogbnVtYmVyXG4gICAgY29udGVudDogc3RyaW5nXG4gIH0gfCBudWxsPihudWxsKVxuICBjb25zdCBhYm9ydFJlZiA9IHVzZVJlZjxBYm9ydENvbnRyb2xsZXIgfCBudWxsPihudWxsKVxuICBjb25zdCB0aW1lb3V0UmVmID0gdXNlUmVmPFJldHVyblR5cGU8dHlwZW9mIHNldFRpbWVvdXQ+IHwgbnVsbD4obnVsbClcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIHJldHVybiAoKSA9PiB7XG4gICAgICBpZiAodGltZW91dFJlZi5jdXJyZW50KSBjbGVhclRpbWVvdXQodGltZW91dFJlZi5jdXJyZW50KVxuICAgICAgYWJvcnRSZWYuY3VycmVudD8uYWJvcnQoKVxuICAgIH1cbiAgfSwgW10pXG5cbiAgLy8gTG9hZCBjb250ZXh0IGxpbmVzIGFyb3VuZCB0aGUgZm9jdXNlZCBtYXRjaC4gQWJvcnRDb250cm9sbGVyIHByZXZlbnRzXG4gIC8vIGhvbGRpbmcg4oaTIGZyb20gcGlsaW5nIHVwIHJlYWRzLlxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmICghZm9jdXNlZCkge1xuICAgICAgc2V0UHJldmlldyhudWxsKVxuICAgICAgcmV0dXJuXG4gICAgfVxuICAgIGNvbnN0IGNvbnRyb2xsZXIgPSBuZXcgQWJvcnRDb250cm9sbGVyKClcbiAgICBjb25zdCBhYnNvbHV0ZSA9IHJlc29sdmVQYXRoKGdldEN3ZCgpLCBmb2N1c2VkLmZpbGUpXG4gICAgY29uc3Qgc3RhcnQgPSBNYXRoLm1heCgwLCBmb2N1c2VkLmxpbmUgLSBQUkVWSUVXX0NPTlRFWFRfTElORVMgLSAxKVxuICAgIHZvaWQgcmVhZEZpbGVJblJhbmdlKFxuICAgICAgYWJzb2x1dGUsXG4gICAgICBzdGFydCxcbiAgICAgIFBSRVZJRVdfQ09OVEVYVF9MSU5FUyAqIDIgKyAxLFxuICAgICAgdW5kZWZpbmVkLFxuICAgICAgY29udHJvbGxlci5zaWduYWwsXG4gICAgKVxuICAgICAgLnRoZW4ociA9PiB7XG4gICAgICAgIGlmIChjb250cm9sbGVyLnNpZ25hbC5hYm9ydGVkKSByZXR1cm5cbiAgICAgICAgc2V0UHJldmlldyh7XG4gICAgICAgICAgZmlsZTogZm9jdXNlZC5maWxlLFxuICAgICAgICAgIGxpbmU6IGZvY3VzZWQubGluZSxcbiAgICAgICAgICBjb250ZW50OiByLmNvbnRlbnQsXG4gICAgICAgIH0pXG4gICAgICB9KVxuICAgICAgLmNhdGNoKCgpID0+IHtcbiAgICAgICAgaWYgKGNvbnRyb2xsZXIuc2lnbmFsLmFib3J0ZWQpIHJldHVyblxuICAgICAgICBzZXRQcmV2aWV3KHtcbiAgICAgICAgICBmaWxlOiBmb2N1c2VkLmZpbGUsXG4gICAgICAgICAgbGluZTogZm9jdXNlZC5saW5lLFxuICAgICAgICAgIGNvbnRlbnQ6ICcocHJldmlldyB1bmF2YWlsYWJsZSknLFxuICAgICAgICB9KVxuICAgICAgfSlcbiAgICByZXR1cm4gKCkgPT4gY29udHJvbGxlci5hYm9ydCgpXG4gIH0sIFtmb2N1c2VkXSlcblxuICBjb25zdCBoYW5kbGVRdWVyeUNoYW5nZSA9IChxOiBzdHJpbmcpID0+IHtcbiAgICBzZXRRdWVyeShxKVxuICAgIGlmICh0aW1lb3V0UmVmLmN1cnJlbnQpIGNsZWFyVGltZW91dCh0aW1lb3V0UmVmLmN1cnJlbnQpXG4gICAgYWJvcnRSZWYuY3VycmVudD8uYWJvcnQoKVxuXG4gICAgaWYgKCFxLnRyaW0oKSkge1xuICAgICAgc2V0TWF0Y2hlcyhtID0+IChtLmxlbmd0aCA/IFtdIDogbSkpXG4gICAgICBzZXRJc1NlYXJjaGluZyhmYWxzZSlcbiAgICAgIHNldFRydW5jYXRlZChmYWxzZSlcbiAgICAgIHJldHVyblxuICAgIH1cbiAgICBjb25zdCBjb250cm9sbGVyID0gbmV3IEFib3J0Q29udHJvbGxlcigpXG4gICAgYWJvcnRSZWYuY3VycmVudCA9IGNvbnRyb2xsZXJcbiAgICBzZXRJc1NlYXJjaGluZyh0cnVlKVxuICAgIHNldFRydW5jYXRlZChmYWxzZSlcbiAgICAvLyBDbGllbnQtZmlsdGVyIGV4aXN0aW5nIHJlc3VsdHMgd2hpbGUgcmcgd2Fsa3Mg4oCUIGtlZXBzIHNvbWV0aGluZyBvblxuICAgIC8vIHNjcmVlbiBpbnN0ZWFkIG9mIGZsYXNoaW5nIGJsYW5rLiByZyByZXN1bHRzIGFyZSBtZXJnZWQgaW4gKGRlZHVwZWQgYnlcbiAgICAvLyBmaWxlOmxpbmUpIHJhdGhlciB0aGFuIHJlcGxhY2VkLCBzbyB0aGUgY291bnQgaXMgbW9ub3RvbmljIHdpdGhpbiBhXG4gICAgLy8gcXVlcnk6IGl0IG9ubHkgZ3Jvd3MgYXMgcmcgc3RyZWFtcywgbmV2ZXIgZGlwcyB0byB0aGUgZmlyc3QgY2h1bmsnc1xuICAgIC8vIHNpemUuIE5hcnJvd2luZyAobmV3IHF1ZXJ5IGV4dGVuZHMgb2xkKTogZmlsdGVyIGlzIGV4YWN0IOKAlCBhbnkgbGluZVxuICAgIC8vIHRoYXQgbWF0Y2hlZCB0aGUgb2xkIC1GIC1pIGxpdGVyYWwgY29udGFpbnMgdGhlIG5ldyBvbmUgaWZmIGl0cyB0ZXh0XG4gICAgLy8gaW5jbHVkZXMgdGhlIG5ldyBxdWVyeSBsb3dlcmVkLiBOb24tbmFycm93aW5nIChicm9hZGVuaW5nL2RpZmZlcmVudCk6XG4gICAgLy8gZmlsdGVyIGlzIGJlc3QtZWZmb3J0IOKAlCBtYXkgYnJpZWZseSBzaG93IGEgc3Vic2V0IHVudGlsIHJnIGZpbGxzIGluXG4gICAgLy8gdGhlIHJlc3QuXG4gICAgY29uc3QgcXVlcnlMb3dlciA9IHEudG9Mb3dlckNhc2UoKVxuICAgIHNldE1hdGNoZXMobSA9PiB7XG4gICAgICBjb25zdCBmaWx0ZXJlZCA9IG0uZmlsdGVyKG1hdGNoID0+XG4gICAgICAgIG1hdGNoLnRleHQudG9Mb3dlckNhc2UoKS5pbmNsdWRlcyhxdWVyeUxvd2VyKSxcbiAgICAgIClcbiAgICAgIHJldHVybiBmaWx0ZXJlZC5sZW5ndGggPT09IG0ubGVuZ3RoID8gbSA6IGZpbHRlcmVkXG4gICAgfSlcblxuICAgIHRpbWVvdXRSZWYuY3VycmVudCA9IHNldFRpbWVvdXQoXG4gICAgICAocXVlcnksIGNvbnRyb2xsZXIsIHNldE1hdGNoZXMsIHNldFRydW5jYXRlZCwgc2V0SXNTZWFyY2hpbmcpID0+IHtcbiAgICAgICAgLy8gcmlwZ3JlcCBvdXRwdXRzIGFic29sdXRlIHBhdGhzIHdoZW4gZ2l2ZW4gYW4gYWJzb2x1dGUgdGFyZ2V0LCBzb1xuICAgICAgICAvLyByZWxhdGl2aXplIGFnYWluc3QgY3dkIHRvIHByZXNlcnZlIGRpcmVjdG9yeSBjb250ZXh0IGluIHRoZSB0cnVuY2F0ZWRcbiAgICAgICAgLy8gZGlzcGxheSAob3RoZXJ3aXNlIHRoZSBjd2QgcHJlZml4IGVhdHMgdGhlIHdpZHRoIGJ1ZGdldCkuXG4gICAgICAgIC8vIHJlbGF0aXZlUGF0aCgpIHJldHVybnMgUE9TSVgtbm9ybWFsaXplZCBvdXRwdXQgc28gdHJ1bmNhdGVQYXRoTWlkZGxlXG4gICAgICAgIC8vICh3aGljaCB1c2VzIGxhc3RJbmRleE9mKCcvJykpIHdvcmtzIG9uIFdpbmRvd3MgdG9vLlxuICAgICAgICBjb25zdCBjd2QgPSBnZXRDd2QoKVxuICAgICAgICBsZXQgY29sbGVjdGVkID0gMFxuICAgICAgICB2b2lkIHJpcEdyZXBTdHJlYW0oXG4gICAgICAgICAgLy8gLWUgZGlzYW1iaWd1YXRlcyBwYXR0ZXJuIGZyb20gb3B0aW9ucyB3aGVuIHRoZSBxdWVyeSBzdGFydHMgd2l0aCAnLSdcbiAgICAgICAgICAvLyAoZS5nLiBzZWFyY2hpbmcgZm9yIFwiLS12ZXJib3NlXCIgb3IgXCItcmZcIikuIFNlZSBHcmVwVG9vbC50cyBmb3IgdGhlXG4gICAgICAgICAgLy8gc2FtZSBwcmVjYXV0aW9uLlxuICAgICAgICAgIFtcbiAgICAgICAgICAgICctbicsXG4gICAgICAgICAgICAnLS1uby1oZWFkaW5nJyxcbiAgICAgICAgICAgICctaScsXG4gICAgICAgICAgICAnLW0nLFxuICAgICAgICAgICAgU3RyaW5nKE1BWF9NQVRDSEVTX1BFUl9GSUxFKSxcbiAgICAgICAgICAgICctRicsXG4gICAgICAgICAgICAnLWUnLFxuICAgICAgICAgICAgcXVlcnksXG4gICAgICAgICAgXSxcbiAgICAgICAgICBjd2QsXG4gICAgICAgICAgY29udHJvbGxlci5zaWduYWwsXG4gICAgICAgICAgbGluZXMgPT4ge1xuICAgICAgICAgICAgaWYgKGNvbnRyb2xsZXIuc2lnbmFsLmFib3J0ZWQpIHJldHVyblxuICAgICAgICAgICAgY29uc3QgcGFyc2VkOiBNYXRjaFtdID0gW11cbiAgICAgICAgICAgIGZvciAoY29uc3QgbGluZSBvZiBsaW5lcykge1xuICAgICAgICAgICAgICBjb25zdCBtID0gcGFyc2VSaXBncmVwTGluZShsaW5lKVxuICAgICAgICAgICAgICBpZiAoIW0pIGNvbnRpbnVlXG4gICAgICAgICAgICAgIGNvbnN0IHJlbCA9IHJlbGF0aXZlUGF0aChjd2QsIG0uZmlsZSlcbiAgICAgICAgICAgICAgcGFyc2VkLnB1c2goeyAuLi5tLCBmaWxlOiByZWwuc3RhcnRzV2l0aCgnLi4nKSA/IG0uZmlsZSA6IHJlbCB9KVxuICAgICAgICAgICAgfVxuICAgICAgICAgICAgaWYgKCFwYXJzZWQubGVuZ3RoKSByZXR1cm5cbiAgICAgICAgICAgIGNvbGxlY3RlZCArPSBwYXJzZWQubGVuZ3RoXG4gICAgICAgICAgICBzZXRNYXRjaGVzKHByZXYgPT4ge1xuICAgICAgICAgICAgICAvLyBBcHBlbmQrZGVkdXBlIGluc3RlYWQgb2YgcmVwbGFjZTogcHJldiBtYXkgaG9sZCBjbGllbnQtXG4gICAgICAgICAgICAgIC8vIGZpbHRlcmVkIHJlc3VsdHMgdGhhdCBhcmUgdmFsaWQgbWF0Y2hlcyBmb3IgdGhpcyBxdWVyeS5cbiAgICAgICAgICAgICAgLy8gUmVwbGFjaW5nIHdvdWxkIGRyb3AgdGhlIGNvdW50IHRvIHRoaXMgY2h1bmsncyBzaXplIHRoZW5cbiAgICAgICAgICAgICAgLy8gZ3JvdyBpdCBiYWNrIOKAlCB2aXNpYmxlIGFzIGEgZmxpY2tlci5cbiAgICAgICAgICAgICAgY29uc3Qgc2VlbiA9IG5ldyBTZXQocHJldi5tYXAobWF0Y2hLZXkpKVxuICAgICAgICAgICAgICBjb25zdCBmcmVzaCA9IHBhcnNlZC5maWx0ZXIocCA9PiAhc2Vlbi5oYXMobWF0Y2hLZXkocCkpKVxuICAgICAgICAgICAgICBpZiAoIWZyZXNoLmxlbmd0aCkgcmV0dXJuIHByZXZcbiAgICAgICAgICAgICAgY29uc3QgbmV4dCA9IHByZXYuY29uY2F0KGZyZXNoKVxuICAgICAgICAgICAgICByZXR1cm4gbmV4dC5sZW5ndGggPiBNQVhfVE9UQUxfTUFUQ0hFU1xuICAgICAgICAgICAgICAgID8gbmV4dC5zbGljZSgwLCBNQVhfVE9UQUxfTUFUQ0hFUylcbiAgICAgICAgICAgICAgICA6IG5leHRcbiAgICAgICAgICAgIH0pXG4gICAgICAgICAgICBpZiAoY29sbGVjdGVkID49IE1BWF9UT1RBTF9NQVRDSEVTKSB7XG4gICAgICAgICAgICAgIGNvbnRyb2xsZXIuYWJvcnQoKVxuICAgICAgICAgICAgICBzZXRUcnVuY2F0ZWQodHJ1ZSlcbiAgICAgICAgICAgICAgc2V0SXNTZWFyY2hpbmcoZmFsc2UpXG4gICAgICAgICAgICB9XG4gICAgICAgICAgfSxcbiAgICAgICAgKVxuICAgICAgICAgIC5jYXRjaCgoKSA9PiB7fSlcbiAgICAgICAgICAvLyBTdHJlYW0gY2xvc2VkIHdpdGggemVybyBjaHVua3Mg4oCUIGNsZWFyIHN0YWxlIHJlc3VsdHMgc29cbiAgICAgICAgICAvLyBcIk5vIG1hdGNoZXNcIiByZW5kZXJzIGluc3RlYWQgb2YgdGhlIHByZXZpb3VzIHF1ZXJ5J3MgbGlzdC5cbiAgICAgICAgICAuZmluYWxseSgoKSA9PiB7XG4gICAgICAgICAgICBpZiAoY29udHJvbGxlci5zaWduYWwuYWJvcnRlZCkgcmV0dXJuXG4gICAgICAgICAgICBpZiAoY29sbGVjdGVkID09PSAwKSBzZXRNYXRjaGVzKG0gPT4gKG0ubGVuZ3RoID8gW10gOiBtKSlcbiAgICAgICAgICAgIHNldElzU2VhcmNoaW5nKGZhbHNlKVxuICAgICAgICAgIH0pXG4gICAgICB9LFxuICAgICAgREVCT1VOQ0VfTVMsXG4gICAgICBxLFxuICAgICAgY29udHJvbGxlcixcbiAgICAgIHNldE1hdGNoZXMsXG4gICAgICBzZXRUcnVuY2F0ZWQsXG4gICAgICBzZXRJc1NlYXJjaGluZyxcbiAgICApXG4gIH1cblxuICBjb25zdCBsaXN0V2lkdGggPSBwcmV2aWV3T25SaWdodFxuICAgID8gTWF0aC5mbG9vcigoY29sdW1ucyAtIDEwKSAqIDAuNSlcbiAgICA6IGNvbHVtbnMgLSA4XG4gIGNvbnN0IG1heFBhdGhXaWR0aCA9IE1hdGgubWF4KDIwLCBNYXRoLmZsb29yKGxpc3RXaWR0aCAqIDAuNCkpXG4gIGNvbnN0IG1heFRleHRXaWR0aCA9IE1hdGgubWF4KDIwLCBsaXN0V2lkdGggLSBtYXhQYXRoV2lkdGggLSA0KVxuICBjb25zdCBwcmV2aWV3V2lkdGggPSBwcmV2aWV3T25SaWdodFxuICAgID8gTWF0aC5tYXgoNDAsIGNvbHVtbnMgLSBsaXN0V2lkdGggLSAxNClcbiAgICA6IGNvbHVtbnMgLSA2XG5cbiAgY29uc3QgaGFuZGxlT3BlbiA9IChtOiBNYXRjaCkgPT4ge1xuICAgIGNvbnN0IG9wZW5lZCA9IG9wZW5GaWxlSW5FeHRlcm5hbEVkaXRvcihcbiAgICAgIHJlc29sdmVQYXRoKGdldEN3ZCgpLCBtLmZpbGUpLFxuICAgICAgbS5saW5lLFxuICAgIClcbiAgICBsb2dFdmVudCgndGVuZ3VfZ2xvYmFsX3NlYXJjaF9zZWxlY3QnLCB7XG4gICAgICByZXN1bHRfY291bnQ6IG1hdGNoZXMubGVuZ3RoLFxuICAgICAgb3BlbmVkX2VkaXRvcjogb3BlbmVkLFxuICAgIH0pXG4gICAgb25Eb25lKClcbiAgfVxuXG4gIGNvbnN0IGhhbmRsZUluc2VydCA9IChtOiBNYXRjaCwgbWVudGlvbjogYm9vbGVhbikgPT4ge1xuICAgIG9uSW5zZXJ0KG1lbnRpb24gPyBgQCR7bS5maWxlfSNMJHttLmxpbmV9IGAgOiBgJHttLmZpbGV9OiR7bS5saW5lfSBgKVxuICAgIGxvZ0V2ZW50KCd0ZW5ndV9nbG9iYWxfc2VhcmNoX2luc2VydCcsIHtcbiAgICAgIHJlc3VsdF9jb3VudDogbWF0Y2hlcy5sZW5ndGgsXG4gICAgICBtZW50aW9uLFxuICAgIH0pXG4gICAgb25Eb25lKClcbiAgfVxuXG4gIC8vIEFsd2F5cyBwYXNzIGEgbm9uLWVtcHR5IHN0cmluZyBzbyB0aGUgbGluZSBpcyByZXNlcnZlZCDigJQgcHJldmVudHMgdGhlXG4gIC8vIHNlYXJjaEJveCBmcm9tIGJvdW5jaW5nIHdoZW4gdGhlIGNvdW50IGFwcGVhcnMvZGlzYXBwZWFycy5cbiAgY29uc3QgbWF0Y2hMYWJlbCA9XG4gICAgbWF0Y2hlcy5sZW5ndGggPiAwXG4gICAgICA/IGAke21hdGNoZXMubGVuZ3RofSR7dHJ1bmNhdGVkID8gJysnIDogJyd9IG1hdGNoZXMke2lzU2VhcmNoaW5nID8gJ+KApicgOiAnJ31gXG4gICAgICA6ICcgJ1xuXG4gIHJldHVybiAoXG4gICAgPEZ1enp5UGlja2VyXG4gICAgICB0aXRsZT1cIkdsb2JhbCBTZWFyY2hcIlxuICAgICAgcGxhY2Vob2xkZXI9XCJUeXBlIHRvIHNlYXJjaOKAplwiXG4gICAgICBpdGVtcz17bWF0Y2hlc31cbiAgICAgIGdldEtleT17bWF0Y2hLZXl9XG4gICAgICB2aXNpYmxlQ291bnQ9e3Zpc2libGVSZXN1bHRzfVxuICAgICAgZGlyZWN0aW9uPVwidXBcIlxuICAgICAgcHJldmlld1Bvc2l0aW9uPXtwcmV2aWV3T25SaWdodCA/ICdyaWdodCcgOiAnYm90dG9tJ31cbiAgICAgIG9uUXVlcnlDaGFuZ2U9e2hhbmRsZVF1ZXJ5Q2hhbmdlfVxuICAgICAgb25Gb2N1cz17c2V0Rm9jdXNlZH1cbiAgICAgIG9uU2VsZWN0PXtoYW5kbGVPcGVufVxuICAgICAgb25UYWI9e3sgYWN0aW9uOiAnbWVudGlvbicsIGhhbmRsZXI6IG0gPT4gaGFuZGxlSW5zZXJ0KG0sIHRydWUpIH19XG4gICAgICBvblNoaWZ0VGFiPXt7XG4gICAgICAgIGFjdGlvbjogJ2luc2VydCBwYXRoJyxcbiAgICAgICAgaGFuZGxlcjogbSA9PiBoYW5kbGVJbnNlcnQobSwgZmFsc2UpLFxuICAgICAgfX1cbiAgICAgIG9uQ2FuY2VsPXtvbkRvbmV9XG4gICAgICBlbXB0eU1lc3NhZ2U9e3EgPT5cbiAgICAgICAgaXNTZWFyY2hpbmcgPyAnU2VhcmNoaW5n4oCmJyA6IHEgPyAnTm8gbWF0Y2hlcycgOiAnVHlwZSB0byBzZWFyY2jigKYnXG4gICAgICB9XG4gICAgICBtYXRjaExhYmVsPXttYXRjaExhYmVsfVxuICAgICAgc2VsZWN0QWN0aW9uPVwib3BlbiBpbiBlZGl0b3JcIlxuICAgICAgcmVuZGVySXRlbT17KG0sIGlzRm9jdXNlZCkgPT4gKFxuICAgICAgICA8VGV4dCBjb2xvcj17aXNGb2N1c2VkID8gJ3N1Z2dlc3Rpb24nIDogdW5kZWZpbmVkfT5cbiAgICAgICAgICA8VGV4dCBkaW1Db2xvcj5cbiAgICAgICAgICAgIHt0cnVuY2F0ZVBhdGhNaWRkbGUobS5maWxlLCBtYXhQYXRoV2lkdGgpfTp7bS5saW5lfVxuICAgICAgICAgIDwvVGV4dD57JyAnfVxuICAgICAgICAgIHtoaWdobGlnaHRNYXRjaChcbiAgICAgICAgICAgIHRydW5jYXRlVG9XaWR0aChtLnRleHQudHJpbVN0YXJ0KCksIG1heFRleHRXaWR0aCksXG4gICAgICAgICAgICBxdWVyeSxcbiAgICAgICAgICApfVxuICAgICAgICA8L1RleHQ+XG4gICAgICApfVxuICAgICAgcmVuZGVyUHJldmlldz17bSA9PlxuICAgICAgICBwcmV2aWV3Py5maWxlID09PSBtLmZpbGUgJiYgcHJldmlldy5saW5lID09PSBtLmxpbmUgPyAoXG4gICAgICAgICAgPD5cbiAgICAgICAgICAgIDxUZXh0IGRpbUNvbG9yPlxuICAgICAgICAgICAgICB7dHJ1bmNhdGVQYXRoTWlkZGxlKG0uZmlsZSwgcHJldmlld1dpZHRoKX06e20ubGluZX1cbiAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgIHtwcmV2aWV3LmNvbnRlbnQuc3BsaXQoJ1xcbicpLm1hcCgobGluZSwgaSkgPT4gKFxuICAgICAgICAgICAgICA8VGV4dCBrZXk9e2l9PlxuICAgICAgICAgICAgICAgIHtoaWdobGlnaHRNYXRjaCh0cnVuY2F0ZVRvV2lkdGgobGluZSwgcHJldmlld1dpZHRoKSwgcXVlcnkpfVxuICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICApKX1cbiAgICAgICAgICA8Lz5cbiAgICAgICAgKSA6IChcbiAgICAgICAgICA8TG9hZGluZ1N0YXRlIG1lc3NhZ2U9XCJMb2FkaW5n4oCmXCIgZGltQ29sb3IgLz5cbiAgICAgICAgKVxuICAgICAgfVxuICAgIC8+XG4gIClcbn1cblxuZnVuY3Rpb24gbWF0Y2hLZXkobTogTWF0Y2gpOiBzdHJpbmcge1xuICByZXR1cm4gYCR7bS5maWxlfToke20ubGluZX1gXG59XG5cbi8qKlxuICogUGFyc2UgYSByaXBncmVwIC1uIC0tbm8taGVhZGluZyBvdXRwdXQgbGluZTogXCJwYXRoOmxpbmU6dGV4dFwiLlxuICogV2luZG93cyBwYXRocyBtYXkgY29udGFpbiBhIGRyaXZlIGxldHRlciAoXCJDOlxcLi4uXCIpLCBzbyBhIHNpbXBsZSBzcGxpdCBvblxuICogdGhlIGZpcnN0IGNvbG9uIHdvdWxkIG1hbmdsZSB0aGUgcGF0aCDigJQgdXNlIGEgcmVnZXggdGhhdCBjYXB0dXJlcyB1cCB0b1xuICogdGhlIGZpcnN0IDo8ZGlnaXRzPjogaW5zdGVhZC5cbiAqIEBpbnRlcm5hbCBleHBvcnRlZCBmb3IgdGVzdGluZ1xuICovXG5leHBvcnQgZnVuY3Rpb24gcGFyc2VSaXBncmVwTGluZShsaW5lOiBzdHJpbmcpOiBNYXRjaCB8IG51bGwge1xuICBjb25zdCBtID0gL14oLio/KTooXFxkKyk6KC4qKSQvLmV4ZWMobGluZSlcbiAgaWYgKCFtKSByZXR1cm4gbnVsbFxuICBjb25zdCBbLCBmaWxlLCBsaW5lU3RyLCB0ZXh0XSA9IG1cbiAgY29uc3QgbGluZU51bSA9IE51bWJlcihsaW5lU3RyKVxuICBpZiAoIWZpbGUgfHwgIU51bWJlci5pc0Zpbml0ZShsaW5lTnVtKSkgcmV0dXJuIG51bGxcbiAgcmV0dXJuIHsgZmlsZSwgbGluZTogbGluZU51bSwgdGV4dDogdGV4dCA/PyAnJyB9XG59XG4iXSwibWFwcGluZ3MiOiI7QUFBQSxTQUFTQSxPQUFPLElBQUlDLFdBQVcsUUFBUSxNQUFNO0FBQzdDLE9BQU8sS0FBS0MsS0FBSyxNQUFNLE9BQU87QUFDOUIsU0FBU0MsU0FBUyxFQUFFQyxNQUFNLEVBQUVDLFFBQVEsUUFBUSxPQUFPO0FBQ25ELFNBQVNDLGtCQUFrQixRQUFRLDhCQUE4QjtBQUNqRSxTQUFTQyxlQUFlLFFBQVEsNkJBQTZCO0FBQzdELFNBQVNDLElBQUksUUFBUSxXQUFXO0FBQ2hDLFNBQVNDLFFBQVEsUUFBUSxnQ0FBZ0M7QUFDekQsU0FBU0MsTUFBTSxRQUFRLGlCQUFpQjtBQUN4QyxTQUFTQyx3QkFBd0IsUUFBUSxvQkFBb0I7QUFDN0QsU0FBU0Msa0JBQWtCLEVBQUVDLGVBQWUsUUFBUSxvQkFBb0I7QUFDeEUsU0FBU0MsY0FBYyxRQUFRLDRCQUE0QjtBQUMzRCxTQUFTQyxZQUFZLFFBQVEsb0NBQW9D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 \ No newline at end of file diff --git a/src/components/HelpV2/Commands.tsx b/src/components/HelpV2/Commands.tsx new file mode 100644 index 0000000..525ef1b --- /dev/null +++ b/src/components/HelpV2/Commands.tsx @@ -0,0 +1,82 @@ +import { c as _c } from "react/compiler-runtime"; +import * as React from 'react'; +import { useMemo } from 'react'; +import { type Command, formatDescriptionWithSource } from '../../commands.js'; +import { Box, Text } from '../../ink.js'; +import { truncate } from '../../utils/format.js'; +import { Select } from '../CustomSelect/select.js'; +import { useTabHeaderFocus } from '../design-system/Tabs.js'; +type Props = { + commands: Command[]; + maxHeight: number; + columns: number; + title: string; + onCancel: () => void; + emptyMessage?: string; +}; +export function Commands(t0) { + const $ = _c(14); + const { + commands, + maxHeight, + columns, + title, + onCancel, + emptyMessage + } = t0; + const { + headerFocused, + focusHeader + } = useTabHeaderFocus(); + const maxWidth = Math.max(1, columns - 10); + const visibleCount = Math.max(1, Math.floor((maxHeight - 10) / 2)); + let t1; + if ($[0] !== commands || $[1] !== maxWidth) { + const seen = new Set(); + let t2; + if ($[3] !== maxWidth) { + t2 = cmd_0 => ({ + label: `/${cmd_0.name}`, + value: cmd_0.name, + description: truncate(formatDescriptionWithSource(cmd_0), maxWidth, true) + }); + $[3] = maxWidth; + $[4] = t2; + } else { + t2 = $[4]; + } + t1 = commands.filter(cmd => { + if (seen.has(cmd.name)) { + return false; + } + seen.add(cmd.name); + return true; + }).sort(_temp).map(t2); + $[0] = commands; + $[1] = maxWidth; + $[2] = t1; + } else { + t1 = $[2]; + } + const options = t1; + let t2; + if ($[5] !== commands.length || $[6] !== emptyMessage || $[7] !== focusHeader || $[8] !== headerFocused || $[9] !== onCancel || $[10] !== options || $[11] !== title || $[12] !== visibleCount) { + t2 = {commands.length === 0 && emptyMessage ? {emptyMessage} : <>{title}; + $[3] = handleSelect; + $[4] = t3; + } else { + t3 = $[4]; + } + let t4; + if ($[5] === Symbol.for("react.memo_cache_sentinel")) { + t4 = You can also configure this in /config or with the --ide flag; + $[5] = t4; + } else { + t4 = $[5]; + } + let t5; + if ($[6] !== onComplete || $[7] !== t3) { + t5 = {t3}{t4}; + $[6] = onComplete; + $[7] = t3; + $[8] = t5; + } else { + t5 = $[8]; + } + return t5; +} +export function shouldShowAutoConnectDialog(): boolean { + const config = getGlobalConfig(); + return !isSupportedTerminal() && config.autoConnectIde !== true && config.hasIdeAutoConnectDialogBeenShown !== true; +} +type IdeDisableAutoConnectDialogProps = { + onComplete: (disableAutoConnect: boolean) => void; +}; +export function IdeDisableAutoConnectDialog(t0) { + const $ = _c(10); + const { + onComplete + } = t0; + let t1; + if ($[0] !== onComplete) { + t1 = value => { + const disableAutoConnect = value === "yes"; + if (disableAutoConnect) { + saveGlobalConfig(_temp); + } + onComplete(disableAutoConnect); + }; + $[0] = onComplete; + $[1] = t1; + } else { + t1 = $[1]; + } + const handleSelect = t1; + let t2; + if ($[2] !== onComplete) { + t2 = () => { + onComplete(false); + }; + $[2] = onComplete; + $[3] = t2; + } else { + t2 = $[3]; + } + const handleCancel = t2; + let t3; + if ($[4] === Symbol.for("react.memo_cache_sentinel")) { + t3 = [{ + label: "No", + value: "no" + }, { + label: "Yes", + value: "yes" + }]; + $[4] = t3; + } else { + t3 = $[4]; + } + const options = t3; + let t4; + if ($[5] !== handleSelect) { + t4 = onDone(value)} />; + $[10] = onDone; + $[11] = t9; + } else { + t9 = $[11]; + } + let t10; + if ($[12] !== t3 || $[13] !== t4 || $[14] !== t9) { + t10 = {t5}{t9}; + $[12] = t3; + $[13] = t4; + $[14] = t9; + $[15] = t10; + } else { + t10 = $[15]; + } + return t10; +} +function formatIdleDuration(minutes: number): string { + if (minutes < 1) { + return '< 1m'; + } + if (minutes < 60) { + return `${Math.floor(minutes)}m`; + } + const hours = Math.floor(minutes / 60); + const remainingMinutes = Math.floor(minutes % 60); + if (remainingMinutes === 0) { + return `${hours}h`; + } + return `${hours}h ${remainingMinutes}m`; +} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkJveCIsIlRleHQiLCJmb3JtYXRUb2tlbnMiLCJTZWxlY3QiLCJEaWFsb2ciLCJJZGxlUmV0dXJuQWN0aW9uIiwiUHJvcHMiLCJpZGxlTWludXRlcyIsInRvdGFsSW5wdXRUb2tlbnMiLCJvbkRvbmUiLCJhY3Rpb24iLCJJZGxlUmV0dXJuRGlhbG9nIiwidDAiLCIkIiwiX2MiLCJ0MSIsImZvcm1hdElkbGVEdXJhdGlvbiIsImZvcm1hdHRlZElkbGUiLCJ0MiIsImZvcm1hdHRlZFRva2VucyIsInQzIiwidDQiLCJ0NSIsIlN5bWJvbCIsImZvciIsInQ2IiwidmFsdWUiLCJjb25zdCIsImxhYmVsIiwidDciLCJ0OCIsInQ5IiwidDEwIiwibWludXRlcyIsIk1hdGgiLCJmbG9vciIs