Files
clawtap/server/adapters/interface.ts
T
kuannnn 0fcf66fc22 feat: ClawTap v0.2.0
Interactive Prompts:
- Unified InteractivePrompt type across all 3 adapters (Claude/Codex/Gemini)
- InteractivePromptOverlay component with options, text input, countdown
- Gemini + Codex pane monitors detect tool confirmation, ask user, plan approval
- respondInteractivePrompt routing: permission → respondPermission, options → _selectOption
- Claude AskUserQuestion nested questions[0] structure parsing

Cross-AI Review:
- Client-generated reviewId, removed pendingReview state
- FloatingReviewPanel uses CSS display:none instead of unmount (keeps hooks alive)
- Child review sessions default to YOLO/bypass permission mode
- Send back to parent, send to existing/new review, tab switching, end review
- Collapsed review cards with read-only panel for ended reviews
- Full reconnect support: active + ended reviews restore correctly

AskUserQuestion Tool Card UI:
- Dedicated renderer replaces raw JSON display
- Options shown with selected (green) / unselected (gray) indicators
- Free text answers shown in quoted format with green border
- Collapsed summary: question → answer
- Shared parseAskQuestionInput utility (client + server)
- Historical tool results attached via _result on tool_use blocks

Adapter Fixes:
- Session→adapter mapping persisted in SQLite (survives server restart)
- SESSION_CREATED deferred for pendingRekey adapters (Codex/Gemini)
- session-rekeyed handler sends complete SESSION_CREATED with adapter + cwd
- Gemini: auto-accept folder trust, privacy notice, IDE nudge, YOLO * prompt
- Claude: auto-accept bypass permissions confirmation (v2.1.85+)
- Port fallback (EADDRINUSE → try +1), statusLine shell script wrapper

Other:
- Desktop Enter sends / Shift+Enter newline; Mobile Enter newline
- Strip CLAWTAP_REF marker from session list
- Active sessions tab shows adapter badge
- Rename CLAUDE_UI_PASSWORD → CLAWTAP_PASSWORD

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-27 14:46:00 +08:00

175 lines
6.9 KiB
TypeScript

// server/adapters/interface.ts
import { EventEmitter } from 'events';
import type { Express } from 'express';
import type { QueryOptions, PermissionBehavior, PermissionMode, ChatMessage } from '../types/messages.js';
import type {
AdapterCapabilities, SessionInfo, ModelInfo,
PermissionModeInfo, EffortLevelInfo, ReconnectState,
} from '../types/adapter.js';
/** Threshold for switching from sendKeys (character-by-character) to pasteBuffer (bulk paste) */
export const PASTE_THRESHOLD = 500;
/** Check if text should be sent via pasteBuffer instead of sendKeys */
export function isLargeContent(text: string): boolean {
return text.length > PASTE_THRESHOLD || text.includes('\n');
}
/** Cached session status for deduplication and reconnect */
export interface CachedStatus {
contextPercent: number | null;
model: string | null;
cost: number | null;
}
/** Directory entry returned by listDirectory */
export interface DirectoryEntry {
name: string;
path: string;
hasChildren: boolean;
}
/** Active session info returned by getActiveSessions */
export interface ActiveSessionInfo {
sessionId: string;
cwd: string;
adapter: string;
permissionMode: string;
lastActivity: number | null;
hasClients: boolean;
hasDesktop: boolean;
isNonInteractive: boolean;
firstPrompt: string | null;
}
/** Messages result from getMessages */
export interface MessagesResult {
messages: unknown[];
lastModified: string | null;
}
/**
* IAdapter — Base class for CLI agent adapters.
*
* Each adapter is a self-contained plugin that:
* - Manages CLI process lifecycle (start, resume, attach, destroy)
* - Registers its own HTTP hook routes via setup(app)
* - Owns its session store (getSessions, getMessages)
* - Emits standardized events for the session manager
*
* Events (all emit sessionId as first arg):
* streaming-text(sessionId, text)
* thinking(sessionId, { text, detail })
* tool-start(sessionId, { toolId, toolName, input })
* tool-done(sessionId, { toolId, toolName, result })
* tool-updates(sessionId, toolsMap)
* new-messages(sessionId, messages[])
* session-idle(sessionId)
* permission-request(sessionId, { requestId, toolName, input })
* ask-question(sessionId, { requestId, toolName, input })
* status-update(sessionId, { contextPercent, model, cost })
*/
export class IAdapter extends EventEmitter {
/** Unique adapter identifier (e.g. 'claude', 'codex') */
static id: string = '';
/** Human-readable name (e.g. 'Claude Code') */
static displayName: string = '';
/** CLI binary name for auto-detection (e.g. 'claude') */
static command: string = '';
protected _clientChecker: ((sessionId: string) => boolean) | null = null;
/**
* Register adapter-specific HTTP routes and configure CLI hooks.
* Called once during server startup.
*/
setup(app: Express): void { throw new Error('Not implemented: setup'); }
/**
* Set a function that checks if WS clients are connected for a session.
*/
setClientChecker(fn: (sessionId: string) => boolean): void { this._clientChecker = fn; }
// --- Session Lifecycle ---
async startSession(cwd: string, options?: QueryOptions): Promise<{ sessionId: string; pendingRekey?: boolean }> { throw new Error('Not implemented: startSession'); }
async resumeSession(sessionId: string, cwd: string, options?: QueryOptions): Promise<{ sessionId: string }> { throw new Error('Not implemented: resumeSession'); }
async attachSession(sessionId: string, cwd: string, options?: QueryOptions): Promise<{ sessionId: string }> { throw new Error('Not implemented: attachSession'); }
async destroySession(sessionId: string): Promise<void> { throw new Error('Not implemented: destroySession'); }
// --- Messaging ---
async sendMessage(sessionId: string, text: string, options?: QueryOptions): Promise<void> { throw new Error('Not implemented: sendMessage'); }
async respondPlan(sessionId: string, optionIndex: number, text?: string): Promise<void> {}
async interrupt(sessionId: string): Promise<void> { throw new Error('Not implemented: interrupt'); }
async switchModel(sessionId: string, model: string): Promise<void> {}
flushMessages(sessionId: string): void {}
syncWatcherPosition(sessionId: string): void {}
getReconnectState(sessionId: string): ReconnectState { return { tools: {} as Record<string, any>, pendingRequests: [] }; }
// --- Session Store ---
async getSessions(dir?: string, limit?: number): Promise<SessionInfo[]> { throw new Error('Not implemented: getSessions'); }
async getMessages(sessionId: string, dir?: string): Promise<MessagesResult> { throw new Error('Not implemented: getMessages'); }
async listDirectory(path?: string): Promise<DirectoryEntry[]> { throw new Error('Not implemented: listDirectory'); }
// --- Permissions ---
async switchPermissionMode(sessionId: string, mode: string): Promise<boolean> { return false; }
respondPermission(requestId: string, behavior: PermissionBehavior): void {}
async respondQuestion(requestId: string, answer: string): Promise<void> {}
releaseAllPending(sessionId: string): void {}
resolveAllPendingAs(sessionId: string, behavior: PermissionBehavior): void {}
// --- Query ---
getSession(sessionId: string): unknown { return null; }
getLastStatus(sessionId: string): { contextPercent: number | null; model: string | null; cost: number | null } | null { return null; }
getActiveSessions(): ActiveSessionInfo[] { return []; }
async hasActiveWindow(sessionId: string): Promise<boolean> { return false; }
// --- Capabilities ---
getModels(): ModelInfo[] { return []; }
getPermissionModes(): PermissionModeInfo[] { return []; }
getEffortLevels(): EffortLevelInfo[] { return []; }
getEffortLabel(): string { return 'Effort'; }
getCapabilities(): AdapterCapabilities {
return {
supportsPlanMode: false,
supportsPermissionModes: false,
supportsInterrupt: false,
supportsResume: false,
supportsAttach: false,
supportsStatusLine: false,
supportsImages: false,
supportsStreaming: true,
maxContextWindow: 0,
};
}
getHookPrefix(): string {
return `/api/hooks/${(this.constructor as any).id}`;
}
// --- Hooks ---
/** Update the port used by hooks (called when port fallback changes the port). */
setHookPort(port: number | string): void {}
/** Install adapter-specific hooks (e.g., write to CLI settings). No server needed. */
installHooks(): void {}
/** Remove adapter-specific hooks. No server needed. */
uninstallHooks(): void {}
/** Respond to an interactive prompt (permission, question, plan, etc). */
respondInteractivePrompt(requestId: string, selectedOption?: string, textValue?: string): void {}
// --- Lifecycle ---
/** Called on server shutdown to clean up external config (e.g. CLI hooks). */
async cleanup(): Promise<void> {}
/** Check if a session is actively processing a request. */
isProcessing(sessionId: string): boolean { return false; }
}