clawdex-mobile 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.github/workflows/ci.yml +77 -0
- package/.github/workflows/npm-release.yml +67 -0
- package/AGENTS.md +120 -0
- package/LICENSE +21 -0
- package/README.md +648 -0
- package/STATUS.md +115 -0
- package/apps/mobile/.env.example +7 -0
- package/apps/mobile/App.tsx +589 -0
- package/apps/mobile/app.json +33 -0
- package/apps/mobile/assets/brand/adaptive-icon.png +0 -0
- package/apps/mobile/assets/brand/app-icon.png +0 -0
- package/apps/mobile/assets/brand/favicon.png +0 -0
- package/apps/mobile/assets/brand/mark.png +0 -0
- package/apps/mobile/assets/brand/splash-icon.png +0 -0
- package/apps/mobile/babel.config.js +7 -0
- package/apps/mobile/eslint.config.cjs +28 -0
- package/apps/mobile/index.js +5 -0
- package/apps/mobile/metro.config.js +3 -0
- package/apps/mobile/package.json +55 -0
- package/apps/mobile/src/api/__tests__/chatMapping.test.ts +132 -0
- package/apps/mobile/src/api/__tests__/client.test.ts +872 -0
- package/apps/mobile/src/api/__tests__/ws.test.ts +575 -0
- package/apps/mobile/src/api/chatMapping.ts +591 -0
- package/apps/mobile/src/api/client.ts +1087 -0
- package/apps/mobile/src/api/types.ts +267 -0
- package/apps/mobile/src/api/ws.ts +801 -0
- package/apps/mobile/src/components/ActivityBar.tsx +76 -0
- package/apps/mobile/src/components/ApprovalBanner.tsx +207 -0
- package/apps/mobile/src/components/BrandMark.tsx +43 -0
- package/apps/mobile/src/components/ChatHeader.tsx +106 -0
- package/apps/mobile/src/components/ChatInput.tsx +236 -0
- package/apps/mobile/src/components/ChatMessage.tsx +400 -0
- package/apps/mobile/src/components/StatusLine.tsx +52 -0
- package/apps/mobile/src/components/ToolBlock.tsx +65 -0
- package/apps/mobile/src/components/TypingIndicator.tsx +64 -0
- package/apps/mobile/src/config.ts +75 -0
- package/apps/mobile/src/navigation/DrawerContent.tsx +969 -0
- package/apps/mobile/src/screens/GitScreen.tsx +573 -0
- package/apps/mobile/src/screens/MainScreen.tsx +6905 -0
- package/apps/mobile/src/screens/PrivacyScreen.tsx +196 -0
- package/apps/mobile/src/screens/SettingsScreen.tsx +776 -0
- package/apps/mobile/src/screens/TerminalScreen.tsx +251 -0
- package/apps/mobile/src/screens/TermsScreen.tsx +192 -0
- package/apps/mobile/src/theme.ts +112 -0
- package/apps/mobile/src/types/assets.d.ts +4 -0
- package/apps/mobile/tsconfig.json +33 -0
- package/bin/clawdex.js +72 -0
- package/docs/app-review-notes.md +111 -0
- package/docs/open-source-license-requirements.md +32 -0
- package/docs/plans/2026-02-20-codex-desktop-style-redesign.md +190 -0
- package/docs/plans/2026-02-20-codex-mobile-implementation.md +1630 -0
- package/docs/plans/2026-02-21-codex-ui-redesign-design.md +101 -0
- package/docs/plans/2026-02-21-codex-ui-redesign.md +1229 -0
- package/docs/realtime-streaming-limitations.md +77 -0
- package/package.json +47 -0
- package/scripts/setup-secure-dev.sh +169 -0
- package/scripts/setup-wizard.sh +1302 -0
- package/scripts/start-bridge-secure.sh +38 -0
- package/scripts/start-expo.sh +176 -0
- package/scripts/stop-services.sh +61 -0
- package/scripts/teardown.sh +136 -0
- package/services/mac-bridge/.env.example +10 -0
- package/services/mac-bridge/codex-types/AbsolutePathBuf.ts +14 -0
- package/services/mac-bridge/codex-types/AddConversationListenerParams.ts +6 -0
- package/services/mac-bridge/codex-types/AddConversationSubscriptionResponse.ts +5 -0
- package/services/mac-bridge/codex-types/AgentMessageContent.ts +5 -0
- package/services/mac-bridge/codex-types/AgentMessageContentDeltaEvent.ts +5 -0
- package/services/mac-bridge/codex-types/AgentMessageDeltaEvent.ts +5 -0
- package/services/mac-bridge/codex-types/AgentMessageEvent.ts +5 -0
- package/services/mac-bridge/codex-types/AgentMessageItem.ts +21 -0
- package/services/mac-bridge/codex-types/AgentReasoningDeltaEvent.ts +5 -0
- package/services/mac-bridge/codex-types/AgentReasoningEvent.ts +5 -0
- package/services/mac-bridge/codex-types/AgentReasoningRawContentDeltaEvent.ts +5 -0
- package/services/mac-bridge/codex-types/AgentReasoningRawContentEvent.ts +5 -0
- package/services/mac-bridge/codex-types/AgentReasoningSectionBreakEvent.ts +5 -0
- package/services/mac-bridge/codex-types/AgentStatus.ts +8 -0
- package/services/mac-bridge/codex-types/ApplyPatchApprovalParams.ts +21 -0
- package/services/mac-bridge/codex-types/ApplyPatchApprovalRequestEvent.ts +23 -0
- package/services/mac-bridge/codex-types/ApplyPatchApprovalResponse.ts +6 -0
- package/services/mac-bridge/codex-types/ArchiveConversationParams.ts +6 -0
- package/services/mac-bridge/codex-types/ArchiveConversationResponse.ts +5 -0
- package/services/mac-bridge/codex-types/AskForApproval.ts +9 -0
- package/services/mac-bridge/codex-types/AuthMode.ts +8 -0
- package/services/mac-bridge/codex-types/AuthStatusChangeNotification.ts +9 -0
- package/services/mac-bridge/codex-types/BackgroundEventEvent.ts +5 -0
- package/services/mac-bridge/codex-types/ByteRange.ts +13 -0
- package/services/mac-bridge/codex-types/CallToolResult.ts +9 -0
- package/services/mac-bridge/codex-types/CancelLoginChatGptParams.ts +5 -0
- package/services/mac-bridge/codex-types/CancelLoginChatGptResponse.ts +5 -0
- package/services/mac-bridge/codex-types/ClientInfo.ts +5 -0
- package/services/mac-bridge/codex-types/ClientNotification.ts +5 -0
- package/services/mac-bridge/codex-types/ClientRequest.ts +60 -0
- package/services/mac-bridge/codex-types/CodexErrorInfo.ts +8 -0
- package/services/mac-bridge/codex-types/CollabAgentInteractionBeginEvent.ts +23 -0
- package/services/mac-bridge/codex-types/CollabAgentInteractionEndEvent.ts +28 -0
- package/services/mac-bridge/codex-types/CollabAgentSpawnBeginEvent.ts +19 -0
- package/services/mac-bridge/codex-types/CollabAgentSpawnEndEvent.ts +28 -0
- package/services/mac-bridge/codex-types/CollabCloseBeginEvent.ts +18 -0
- package/services/mac-bridge/codex-types/CollabCloseEndEvent.ts +24 -0
- package/services/mac-bridge/codex-types/CollabResumeBeginEvent.ts +18 -0
- package/services/mac-bridge/codex-types/CollabResumeEndEvent.ts +24 -0
- package/services/mac-bridge/codex-types/CollabWaitingBeginEvent.ts +18 -0
- package/services/mac-bridge/codex-types/CollabWaitingEndEvent.ts +19 -0
- package/services/mac-bridge/codex-types/CollaborationMode.ts +10 -0
- package/services/mac-bridge/codex-types/CollaborationModeMask.ts +11 -0
- package/services/mac-bridge/codex-types/ContentItem.ts +5 -0
- package/services/mac-bridge/codex-types/ContextCompactedEvent.ts +5 -0
- package/services/mac-bridge/codex-types/ContextCompactionItem.ts +5 -0
- package/services/mac-bridge/codex-types/ConversationGitInfo.ts +5 -0
- package/services/mac-bridge/codex-types/ConversationSummary.ts +8 -0
- package/services/mac-bridge/codex-types/CreditsSnapshot.ts +5 -0
- package/services/mac-bridge/codex-types/CustomPrompt.ts +5 -0
- package/services/mac-bridge/codex-types/DeprecationNoticeEvent.ts +13 -0
- package/services/mac-bridge/codex-types/DynamicToolCallRequest.ts +6 -0
- package/services/mac-bridge/codex-types/ElicitationRequestEvent.ts +5 -0
- package/services/mac-bridge/codex-types/ErrorEvent.ts +6 -0
- package/services/mac-bridge/codex-types/EventMsg.ts +78 -0
- package/services/mac-bridge/codex-types/ExecApprovalRequestEvent.ts +44 -0
- package/services/mac-bridge/codex-types/ExecCommandApprovalParams.ts +16 -0
- package/services/mac-bridge/codex-types/ExecCommandApprovalResponse.ts +6 -0
- package/services/mac-bridge/codex-types/ExecCommandBeginEvent.ts +35 -0
- package/services/mac-bridge/codex-types/ExecCommandEndEvent.ts +64 -0
- package/services/mac-bridge/codex-types/ExecCommandOutputDeltaEvent.ts +18 -0
- package/services/mac-bridge/codex-types/ExecCommandSource.ts +5 -0
- package/services/mac-bridge/codex-types/ExecCommandStatus.ts +5 -0
- package/services/mac-bridge/codex-types/ExecOneOffCommandParams.ts +6 -0
- package/services/mac-bridge/codex-types/ExecOneOffCommandResponse.ts +5 -0
- package/services/mac-bridge/codex-types/ExecOutputStream.ts +5 -0
- package/services/mac-bridge/codex-types/ExecPolicyAmendment.ts +12 -0
- package/services/mac-bridge/codex-types/ExitedReviewModeEvent.ts +6 -0
- package/services/mac-bridge/codex-types/FileChange.ts +5 -0
- package/services/mac-bridge/codex-types/ForcedLoginMethod.ts +5 -0
- package/services/mac-bridge/codex-types/ForkConversationParams.ts +7 -0
- package/services/mac-bridge/codex-types/ForkConversationResponse.ts +7 -0
- package/services/mac-bridge/codex-types/FunctionCallOutputBody.ts +6 -0
- package/services/mac-bridge/codex-types/FunctionCallOutputContentItem.ts +9 -0
- package/services/mac-bridge/codex-types/FunctionCallOutputPayload.ts +12 -0
- package/services/mac-bridge/codex-types/FuzzyFileSearchParams.ts +5 -0
- package/services/mac-bridge/codex-types/FuzzyFileSearchResponse.ts +6 -0
- package/services/mac-bridge/codex-types/FuzzyFileSearchResult.ts +8 -0
- package/services/mac-bridge/codex-types/FuzzyFileSearchSessionCompletedNotification.ts +5 -0
- package/services/mac-bridge/codex-types/FuzzyFileSearchSessionUpdatedNotification.ts +6 -0
- package/services/mac-bridge/codex-types/GetAuthStatusParams.ts +5 -0
- package/services/mac-bridge/codex-types/GetAuthStatusResponse.ts +6 -0
- package/services/mac-bridge/codex-types/GetConversationSummaryParams.ts +6 -0
- package/services/mac-bridge/codex-types/GetConversationSummaryResponse.ts +6 -0
- package/services/mac-bridge/codex-types/GetHistoryEntryResponseEvent.ts +10 -0
- package/services/mac-bridge/codex-types/GetUserAgentResponse.ts +5 -0
- package/services/mac-bridge/codex-types/GetUserSavedConfigResponse.ts +6 -0
- package/services/mac-bridge/codex-types/GhostCommit.ts +8 -0
- package/services/mac-bridge/codex-types/GitDiffToRemoteParams.ts +5 -0
- package/services/mac-bridge/codex-types/GitDiffToRemoteResponse.ts +6 -0
- package/services/mac-bridge/codex-types/GitSha.ts +5 -0
- package/services/mac-bridge/codex-types/HistoryEntry.ts +5 -0
- package/services/mac-bridge/codex-types/InitializeCapabilities.ts +17 -0
- package/services/mac-bridge/codex-types/InitializeParams.ts +7 -0
- package/services/mac-bridge/codex-types/InitializeResponse.ts +5 -0
- package/services/mac-bridge/codex-types/InputItem.ts +10 -0
- package/services/mac-bridge/codex-types/InputModality.ts +8 -0
- package/services/mac-bridge/codex-types/InterruptConversationParams.ts +6 -0
- package/services/mac-bridge/codex-types/InterruptConversationResponse.ts +6 -0
- package/services/mac-bridge/codex-types/ItemCompletedEvent.ts +7 -0
- package/services/mac-bridge/codex-types/ItemStartedEvent.ts +7 -0
- package/services/mac-bridge/codex-types/ListConversationsParams.ts +5 -0
- package/services/mac-bridge/codex-types/ListConversationsResponse.ts +6 -0
- package/services/mac-bridge/codex-types/ListCustomPromptsResponseEvent.ts +9 -0
- package/services/mac-bridge/codex-types/ListRemoteSkillsResponseEvent.ts +9 -0
- package/services/mac-bridge/codex-types/ListSkillsResponseEvent.ts +9 -0
- package/services/mac-bridge/codex-types/LocalShellAction.ts +6 -0
- package/services/mac-bridge/codex-types/LocalShellExecAction.ts +5 -0
- package/services/mac-bridge/codex-types/LocalShellStatus.ts +5 -0
- package/services/mac-bridge/codex-types/LoginApiKeyParams.ts +5 -0
- package/services/mac-bridge/codex-types/LoginApiKeyResponse.ts +5 -0
- package/services/mac-bridge/codex-types/LoginChatGptCompleteNotification.ts +8 -0
- package/services/mac-bridge/codex-types/LoginChatGptResponse.ts +5 -0
- package/services/mac-bridge/codex-types/LogoutChatGptResponse.ts +5 -0
- package/services/mac-bridge/codex-types/McpAuthStatus.ts +5 -0
- package/services/mac-bridge/codex-types/McpInvocation.ts +18 -0
- package/services/mac-bridge/codex-types/McpListToolsResponseEvent.ts +25 -0
- package/services/mac-bridge/codex-types/McpStartupCompleteEvent.ts +6 -0
- package/services/mac-bridge/codex-types/McpStartupFailure.ts +5 -0
- package/services/mac-bridge/codex-types/McpStartupStatus.ts +5 -0
- package/services/mac-bridge/codex-types/McpStartupUpdateEvent.ts +14 -0
- package/services/mac-bridge/codex-types/McpToolCallBeginEvent.ts +10 -0
- package/services/mac-bridge/codex-types/McpToolCallEndEvent.ts +15 -0
- package/services/mac-bridge/codex-types/MessagePhase.ts +11 -0
- package/services/mac-bridge/codex-types/ModeKind.ts +8 -0
- package/services/mac-bridge/codex-types/ModelRerouteEvent.ts +6 -0
- package/services/mac-bridge/codex-types/ModelRerouteReason.ts +5 -0
- package/services/mac-bridge/codex-types/NetworkAccess.ts +8 -0
- package/services/mac-bridge/codex-types/NetworkApprovalContext.ts +6 -0
- package/services/mac-bridge/codex-types/NetworkApprovalProtocol.ts +5 -0
- package/services/mac-bridge/codex-types/NewConversationParams.ts +8 -0
- package/services/mac-bridge/codex-types/NewConversationResponse.ts +7 -0
- package/services/mac-bridge/codex-types/ParsedCommand.ts +12 -0
- package/services/mac-bridge/codex-types/PatchApplyBeginEvent.ts +23 -0
- package/services/mac-bridge/codex-types/PatchApplyEndEvent.ts +36 -0
- package/services/mac-bridge/codex-types/PatchApplyStatus.ts +5 -0
- package/services/mac-bridge/codex-types/Personality.ts +5 -0
- package/services/mac-bridge/codex-types/PlanDeltaEvent.ts +5 -0
- package/services/mac-bridge/codex-types/PlanItem.ts +5 -0
- package/services/mac-bridge/codex-types/PlanItemArg.ts +6 -0
- package/services/mac-bridge/codex-types/PlanType.ts +5 -0
- package/services/mac-bridge/codex-types/Profile.ts +9 -0
- package/services/mac-bridge/codex-types/RateLimitSnapshot.ts +8 -0
- package/services/mac-bridge/codex-types/RateLimitWindow.ts +17 -0
- package/services/mac-bridge/codex-types/RawResponseItemEvent.ts +6 -0
- package/services/mac-bridge/codex-types/ReadOnlyAccess.ts +19 -0
- package/services/mac-bridge/codex-types/ReasoningContentDeltaEvent.ts +5 -0
- package/services/mac-bridge/codex-types/ReasoningEffort.ts +8 -0
- package/services/mac-bridge/codex-types/ReasoningItem.ts +5 -0
- package/services/mac-bridge/codex-types/ReasoningItemContent.ts +5 -0
- package/services/mac-bridge/codex-types/ReasoningItemReasoningSummary.ts +5 -0
- package/services/mac-bridge/codex-types/ReasoningRawContentDeltaEvent.ts +5 -0
- package/services/mac-bridge/codex-types/ReasoningSummary.ts +10 -0
- package/services/mac-bridge/codex-types/RemoteSkillDownloadedEvent.ts +8 -0
- package/services/mac-bridge/codex-types/RemoteSkillSummary.ts +5 -0
- package/services/mac-bridge/codex-types/RemoveConversationListenerParams.ts +5 -0
- package/services/mac-bridge/codex-types/RemoveConversationSubscriptionResponse.ts +5 -0
- package/services/mac-bridge/codex-types/RequestId.ts +5 -0
- package/services/mac-bridge/codex-types/RequestUserInputEvent.ts +15 -0
- package/services/mac-bridge/codex-types/RequestUserInputQuestion.ts +6 -0
- package/services/mac-bridge/codex-types/RequestUserInputQuestionOption.ts +5 -0
- package/services/mac-bridge/codex-types/Resource.ts +9 -0
- package/services/mac-bridge/codex-types/ResourceTemplate.ts +9 -0
- package/services/mac-bridge/codex-types/ResponseItem.ts +18 -0
- package/services/mac-bridge/codex-types/ResumeConversationParams.ts +8 -0
- package/services/mac-bridge/codex-types/ResumeConversationResponse.ts +7 -0
- package/services/mac-bridge/codex-types/ReviewCodeLocation.ts +9 -0
- package/services/mac-bridge/codex-types/ReviewDecision.ts +9 -0
- package/services/mac-bridge/codex-types/ReviewFinding.ts +9 -0
- package/services/mac-bridge/codex-types/ReviewLineRange.ts +8 -0
- package/services/mac-bridge/codex-types/ReviewOutputEvent.ts +9 -0
- package/services/mac-bridge/codex-types/ReviewRequest.ts +9 -0
- package/services/mac-bridge/codex-types/ReviewTarget.ts +9 -0
- package/services/mac-bridge/codex-types/SandboxMode.ts +5 -0
- package/services/mac-bridge/codex-types/SandboxPolicy.ts +44 -0
- package/services/mac-bridge/codex-types/SandboxSettings.ts +6 -0
- package/services/mac-bridge/codex-types/SendUserMessageParams.ts +7 -0
- package/services/mac-bridge/codex-types/SendUserMessageResponse.ts +5 -0
- package/services/mac-bridge/codex-types/SendUserTurnParams.ts +16 -0
- package/services/mac-bridge/codex-types/SendUserTurnResponse.ts +5 -0
- package/services/mac-bridge/codex-types/ServerNotification.ts +45 -0
- package/services/mac-bridge/codex-types/ServerRequest.ts +16 -0
- package/services/mac-bridge/codex-types/SessionConfiguredEvent.ts +57 -0
- package/services/mac-bridge/codex-types/SessionConfiguredNotification.ts +8 -0
- package/services/mac-bridge/codex-types/SessionNetworkProxyRuntime.ts +5 -0
- package/services/mac-bridge/codex-types/SessionSource.ts +6 -0
- package/services/mac-bridge/codex-types/SetDefaultModelParams.ts +6 -0
- package/services/mac-bridge/codex-types/SetDefaultModelResponse.ts +5 -0
- package/services/mac-bridge/codex-types/Settings.ts +9 -0
- package/services/mac-bridge/codex-types/SkillDependencies.ts +6 -0
- package/services/mac-bridge/codex-types/SkillErrorInfo.ts +5 -0
- package/services/mac-bridge/codex-types/SkillInterface.ts +5 -0
- package/services/mac-bridge/codex-types/SkillMetadata.ts +12 -0
- package/services/mac-bridge/codex-types/SkillScope.ts +5 -0
- package/services/mac-bridge/codex-types/SkillToolDependency.ts +5 -0
- package/services/mac-bridge/codex-types/SkillsListEntry.ts +7 -0
- package/services/mac-bridge/codex-types/StepStatus.ts +5 -0
- package/services/mac-bridge/codex-types/StreamErrorEvent.ts +12 -0
- package/services/mac-bridge/codex-types/SubAgentSource.ts +6 -0
- package/services/mac-bridge/codex-types/TerminalInteractionEvent.ts +17 -0
- package/services/mac-bridge/codex-types/TextElement.ts +14 -0
- package/services/mac-bridge/codex-types/ThreadId.ts +5 -0
- package/services/mac-bridge/codex-types/ThreadNameUpdatedEvent.ts +6 -0
- package/services/mac-bridge/codex-types/ThreadRolledBackEvent.ts +9 -0
- package/services/mac-bridge/codex-types/TokenCountEvent.ts +7 -0
- package/services/mac-bridge/codex-types/TokenUsage.ts +5 -0
- package/services/mac-bridge/codex-types/TokenUsageInfo.ts +6 -0
- package/services/mac-bridge/codex-types/Tool.ts +9 -0
- package/services/mac-bridge/codex-types/Tools.ts +5 -0
- package/services/mac-bridge/codex-types/TurnAbortReason.ts +5 -0
- package/services/mac-bridge/codex-types/TurnAbortedEvent.ts +6 -0
- package/services/mac-bridge/codex-types/TurnCompleteEvent.ts +5 -0
- package/services/mac-bridge/codex-types/TurnDiffEvent.ts +5 -0
- package/services/mac-bridge/codex-types/TurnItem.ts +11 -0
- package/services/mac-bridge/codex-types/TurnStartedEvent.ts +6 -0
- package/services/mac-bridge/codex-types/UndoCompletedEvent.ts +5 -0
- package/services/mac-bridge/codex-types/UndoStartedEvent.ts +5 -0
- package/services/mac-bridge/codex-types/UpdatePlanArgs.ts +10 -0
- package/services/mac-bridge/codex-types/UserInfoResponse.ts +5 -0
- package/services/mac-bridge/codex-types/UserInput.ts +16 -0
- package/services/mac-bridge/codex-types/UserMessageEvent.ts +22 -0
- package/services/mac-bridge/codex-types/UserMessageItem.ts +6 -0
- package/services/mac-bridge/codex-types/UserSavedConfig.ts +14 -0
- package/services/mac-bridge/codex-types/Verbosity.ts +9 -0
- package/services/mac-bridge/codex-types/ViewImageToolCallEvent.ts +13 -0
- package/services/mac-bridge/codex-types/WarningEvent.ts +5 -0
- package/services/mac-bridge/codex-types/WebSearchAction.ts +5 -0
- package/services/mac-bridge/codex-types/WebSearchBeginEvent.ts +5 -0
- package/services/mac-bridge/codex-types/WebSearchEndEvent.ts +6 -0
- package/services/mac-bridge/codex-types/WebSearchItem.ts +6 -0
- package/services/mac-bridge/codex-types/WebSearchMode.ts +5 -0
- package/services/mac-bridge/codex-types/index.ts +234 -0
- package/services/mac-bridge/codex-types/serde_json/JsonValue.ts +5 -0
- package/services/mac-bridge/codex-types/v2/Account.ts +6 -0
- package/services/mac-bridge/codex-types/v2/AccountLoginCompletedNotification.ts +5 -0
- package/services/mac-bridge/codex-types/v2/AccountRateLimitsUpdatedNotification.ts +6 -0
- package/services/mac-bridge/codex-types/v2/AccountUpdatedNotification.ts +6 -0
- package/services/mac-bridge/codex-types/v2/AgentMessageDeltaNotification.ts +5 -0
- package/services/mac-bridge/codex-types/v2/AnalyticsConfig.ts +6 -0
- package/services/mac-bridge/codex-types/v2/AppBranding.ts +8 -0
- package/services/mac-bridge/codex-types/v2/AppDisabledReason.ts +5 -0
- package/services/mac-bridge/codex-types/v2/AppInfo.ts +19 -0
- package/services/mac-bridge/codex-types/v2/AppListUpdatedNotification.ts +9 -0
- package/services/mac-bridge/codex-types/v2/AppMetadata.ts +7 -0
- package/services/mac-bridge/codex-types/v2/AppReview.ts +5 -0
- package/services/mac-bridge/codex-types/v2/AppScreenshot.ts +5 -0
- package/services/mac-bridge/codex-types/v2/AppsConfig.ts +6 -0
- package/services/mac-bridge/codex-types/v2/AppsListParams.ts +24 -0
- package/services/mac-bridge/codex-types/v2/AppsListResponse.ts +14 -0
- package/services/mac-bridge/codex-types/v2/AskForApproval.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ByteRange.ts +5 -0
- package/services/mac-bridge/codex-types/v2/CancelLoginAccountParams.ts +5 -0
- package/services/mac-bridge/codex-types/v2/CancelLoginAccountResponse.ts +6 -0
- package/services/mac-bridge/codex-types/v2/CancelLoginAccountStatus.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ChatgptAuthTokensRefreshParams.ts +16 -0
- package/services/mac-bridge/codex-types/v2/ChatgptAuthTokensRefreshReason.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ChatgptAuthTokensRefreshResponse.ts +5 -0
- package/services/mac-bridge/codex-types/v2/CodexErrorInfo.ts +11 -0
- package/services/mac-bridge/codex-types/v2/CollabAgentState.ts +6 -0
- package/services/mac-bridge/codex-types/v2/CollabAgentStatus.ts +5 -0
- package/services/mac-bridge/codex-types/v2/CollabAgentTool.ts +5 -0
- package/services/mac-bridge/codex-types/v2/CollabAgentToolCallStatus.ts +5 -0
- package/services/mac-bridge/codex-types/v2/CommandAction.ts +5 -0
- package/services/mac-bridge/codex-types/v2/CommandExecParams.ts +6 -0
- package/services/mac-bridge/codex-types/v2/CommandExecResponse.ts +5 -0
- package/services/mac-bridge/codex-types/v2/CommandExecutionApprovalDecision.ts +6 -0
- package/services/mac-bridge/codex-types/v2/CommandExecutionOutputDeltaNotification.ts +5 -0
- package/services/mac-bridge/codex-types/v2/CommandExecutionRequestApprovalParams.ts +37 -0
- package/services/mac-bridge/codex-types/v2/CommandExecutionRequestApprovalResponse.ts +6 -0
- package/services/mac-bridge/codex-types/v2/CommandExecutionStatus.ts +5 -0
- package/services/mac-bridge/codex-types/v2/Config.ts +17 -0
- package/services/mac-bridge/codex-types/v2/ConfigBatchWriteParams.ts +10 -0
- package/services/mac-bridge/codex-types/v2/ConfigEdit.ts +7 -0
- package/services/mac-bridge/codex-types/v2/ConfigLayer.ts +7 -0
- package/services/mac-bridge/codex-types/v2/ConfigLayerMetadata.ts +6 -0
- package/services/mac-bridge/codex-types/v2/ConfigLayerSource.ts +16 -0
- package/services/mac-bridge/codex-types/v2/ConfigReadParams.ts +11 -0
- package/services/mac-bridge/codex-types/v2/ConfigReadResponse.ts +8 -0
- package/services/mac-bridge/codex-types/v2/ConfigRequirements.ts +9 -0
- package/services/mac-bridge/codex-types/v2/ConfigRequirementsReadResponse.ts +10 -0
- package/services/mac-bridge/codex-types/v2/ConfigValueWriteParams.ts +11 -0
- package/services/mac-bridge/codex-types/v2/ConfigWarningNotification.ts +22 -0
- package/services/mac-bridge/codex-types/v2/ConfigWriteResponse.ts +12 -0
- package/services/mac-bridge/codex-types/v2/ContextCompactedNotification.ts +8 -0
- package/services/mac-bridge/codex-types/v2/CreditsSnapshot.ts +5 -0
- package/services/mac-bridge/codex-types/v2/DeprecationNoticeNotification.ts +13 -0
- package/services/mac-bridge/codex-types/v2/DynamicToolCallOutputContentItem.ts +5 -0
- package/services/mac-bridge/codex-types/v2/DynamicToolCallParams.ts +6 -0
- package/services/mac-bridge/codex-types/v2/DynamicToolCallResponse.ts +6 -0
- package/services/mac-bridge/codex-types/v2/DynamicToolSpec.ts +6 -0
- package/services/mac-bridge/codex-types/v2/ErrorNotification.ts +6 -0
- package/services/mac-bridge/codex-types/v2/ExecPolicyAmendment.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ExperimentalFeature.ts +37 -0
- package/services/mac-bridge/codex-types/v2/ExperimentalFeatureListParams.ts +13 -0
- package/services/mac-bridge/codex-types/v2/ExperimentalFeatureListResponse.ts +11 -0
- package/services/mac-bridge/codex-types/v2/ExperimentalFeatureStage.ts +5 -0
- package/services/mac-bridge/codex-types/v2/FeedbackUploadParams.ts +5 -0
- package/services/mac-bridge/codex-types/v2/FeedbackUploadResponse.ts +5 -0
- package/services/mac-bridge/codex-types/v2/FileChangeApprovalDecision.ts +5 -0
- package/services/mac-bridge/codex-types/v2/FileChangeOutputDeltaNotification.ts +5 -0
- package/services/mac-bridge/codex-types/v2/FileChangeRequestApprovalParams.ts +14 -0
- package/services/mac-bridge/codex-types/v2/FileChangeRequestApprovalResponse.ts +6 -0
- package/services/mac-bridge/codex-types/v2/FileUpdateChange.ts +6 -0
- package/services/mac-bridge/codex-types/v2/GetAccountParams.ts +13 -0
- package/services/mac-bridge/codex-types/v2/GetAccountRateLimitsResponse.ts +14 -0
- package/services/mac-bridge/codex-types/v2/GetAccountResponse.ts +6 -0
- package/services/mac-bridge/codex-types/v2/GitInfo.ts +5 -0
- package/services/mac-bridge/codex-types/v2/HazelnutScope.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ItemCompletedNotification.ts +6 -0
- package/services/mac-bridge/codex-types/v2/ItemStartedNotification.ts +6 -0
- package/services/mac-bridge/codex-types/v2/ListMcpServerStatusParams.ts +13 -0
- package/services/mac-bridge/codex-types/v2/ListMcpServerStatusResponse.ts +11 -0
- package/services/mac-bridge/codex-types/v2/LoginAccountParams.ts +21 -0
- package/services/mac-bridge/codex-types/v2/LoginAccountResponse.ts +9 -0
- package/services/mac-bridge/codex-types/v2/LogoutAccountResponse.ts +5 -0
- package/services/mac-bridge/codex-types/v2/McpAuthStatus.ts +5 -0
- package/services/mac-bridge/codex-types/v2/McpServerOauthLoginCompletedNotification.ts +5 -0
- package/services/mac-bridge/codex-types/v2/McpServerOauthLoginParams.ts +5 -0
- package/services/mac-bridge/codex-types/v2/McpServerOauthLoginResponse.ts +5 -0
- package/services/mac-bridge/codex-types/v2/McpServerRefreshResponse.ts +5 -0
- package/services/mac-bridge/codex-types/v2/McpServerStatus.ts +9 -0
- package/services/mac-bridge/codex-types/v2/McpToolCallError.ts +5 -0
- package/services/mac-bridge/codex-types/v2/McpToolCallProgressNotification.ts +5 -0
- package/services/mac-bridge/codex-types/v2/McpToolCallResult.ts +6 -0
- package/services/mac-bridge/codex-types/v2/McpToolCallStatus.ts +5 -0
- package/services/mac-bridge/codex-types/v2/MergeStrategy.ts +5 -0
- package/services/mac-bridge/codex-types/v2/Model.ts +8 -0
- package/services/mac-bridge/codex-types/v2/ModelListParams.ts +17 -0
- package/services/mac-bridge/codex-types/v2/ModelListResponse.ts +11 -0
- package/services/mac-bridge/codex-types/v2/ModelRerouteReason.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ModelReroutedNotification.ts +6 -0
- package/services/mac-bridge/codex-types/v2/NetworkAccess.ts +5 -0
- package/services/mac-bridge/codex-types/v2/NetworkRequirements.ts +5 -0
- package/services/mac-bridge/codex-types/v2/OverriddenMetadata.ts +7 -0
- package/services/mac-bridge/codex-types/v2/PatchApplyStatus.ts +5 -0
- package/services/mac-bridge/codex-types/v2/PatchChangeKind.ts +5 -0
- package/services/mac-bridge/codex-types/v2/PlanDeltaNotification.ts +9 -0
- package/services/mac-bridge/codex-types/v2/ProductSurface.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ProfileV2.ts +11 -0
- package/services/mac-bridge/codex-types/v2/RateLimitSnapshot.ts +8 -0
- package/services/mac-bridge/codex-types/v2/RateLimitWindow.ts +5 -0
- package/services/mac-bridge/codex-types/v2/RawResponseItemCompletedNotification.ts +6 -0
- package/services/mac-bridge/codex-types/v2/ReadOnlyAccess.ts +6 -0
- package/services/mac-bridge/codex-types/v2/ReasoningEffortOption.ts +6 -0
- package/services/mac-bridge/codex-types/v2/ReasoningSummaryPartAddedNotification.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ReasoningSummaryTextDeltaNotification.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ReasoningTextDeltaNotification.ts +5 -0
- package/services/mac-bridge/codex-types/v2/RemoteSkillSummary.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ResidencyRequirement.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ReviewDelivery.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ReviewStartParams.ts +12 -0
- package/services/mac-bridge/codex-types/v2/ReviewStartResponse.ts +13 -0
- package/services/mac-bridge/codex-types/v2/ReviewTarget.ts +9 -0
- package/services/mac-bridge/codex-types/v2/SandboxMode.ts +5 -0
- package/services/mac-bridge/codex-types/v2/SandboxPolicy.ts +8 -0
- package/services/mac-bridge/codex-types/v2/SandboxWorkspaceWrite.ts +5 -0
- package/services/mac-bridge/codex-types/v2/SessionSource.ts +6 -0
- package/services/mac-bridge/codex-types/v2/SkillDependencies.ts +6 -0
- package/services/mac-bridge/codex-types/v2/SkillErrorInfo.ts +5 -0
- package/services/mac-bridge/codex-types/v2/SkillInterface.ts +5 -0
- package/services/mac-bridge/codex-types/v2/SkillMetadata.ts +12 -0
- package/services/mac-bridge/codex-types/v2/SkillScope.ts +5 -0
- package/services/mac-bridge/codex-types/v2/SkillToolDependency.ts +5 -0
- package/services/mac-bridge/codex-types/v2/SkillsConfigWriteParams.ts +5 -0
- package/services/mac-bridge/codex-types/v2/SkillsConfigWriteResponse.ts +5 -0
- package/services/mac-bridge/codex-types/v2/SkillsListEntry.ts +7 -0
- package/services/mac-bridge/codex-types/v2/SkillsListExtraRootsForCwd.ts +5 -0
- package/services/mac-bridge/codex-types/v2/SkillsListParams.ts +18 -0
- package/services/mac-bridge/codex-types/v2/SkillsListResponse.ts +6 -0
- package/services/mac-bridge/codex-types/v2/SkillsRemoteReadParams.ts +7 -0
- package/services/mac-bridge/codex-types/v2/SkillsRemoteReadResponse.ts +6 -0
- package/services/mac-bridge/codex-types/v2/SkillsRemoteWriteParams.ts +5 -0
- package/services/mac-bridge/codex-types/v2/SkillsRemoteWriteResponse.ts +5 -0
- package/services/mac-bridge/codex-types/v2/TerminalInteractionNotification.ts +5 -0
- package/services/mac-bridge/codex-types/v2/TextElement.ts +14 -0
- package/services/mac-bridge/codex-types/v2/TextPosition.ts +13 -0
- package/services/mac-bridge/codex-types/v2/TextRange.ts +6 -0
- package/services/mac-bridge/codex-types/v2/Thread.ts +51 -0
- package/services/mac-bridge/codex-types/v2/ThreadArchiveParams.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ThreadArchiveResponse.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ThreadArchivedNotification.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ThreadCompactStartParams.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ThreadCompactStartResponse.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ThreadForkParams.ts +28 -0
- package/services/mac-bridge/codex-types/v2/ThreadForkResponse.ts +9 -0
- package/services/mac-bridge/codex-types/v2/ThreadItem.ts +81 -0
- package/services/mac-bridge/codex-types/v2/ThreadListParams.ts +39 -0
- package/services/mac-bridge/codex-types/v2/ThreadListResponse.ts +11 -0
- package/services/mac-bridge/codex-types/v2/ThreadLoadedListParams.ts +13 -0
- package/services/mac-bridge/codex-types/v2/ThreadLoadedListResponse.ts +14 -0
- package/services/mac-bridge/codex-types/v2/ThreadNameUpdatedNotification.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ThreadReadParams.ts +9 -0
- package/services/mac-bridge/codex-types/v2/ThreadReadResponse.ts +6 -0
- package/services/mac-bridge/codex-types/v2/ThreadResumeParams.ts +37 -0
- package/services/mac-bridge/codex-types/v2/ThreadResumeResponse.ts +9 -0
- package/services/mac-bridge/codex-types/v2/ThreadRollbackParams.ts +12 -0
- package/services/mac-bridge/codex-types/v2/ThreadRollbackResponse.ts +14 -0
- package/services/mac-bridge/codex-types/v2/ThreadSetNameParams.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ThreadSetNameResponse.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ThreadSortKey.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ThreadSourceKind.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ThreadStartParams.ts +17 -0
- package/services/mac-bridge/codex-types/v2/ThreadStartResponse.ts +9 -0
- package/services/mac-bridge/codex-types/v2/ThreadStartedNotification.ts +6 -0
- package/services/mac-bridge/codex-types/v2/ThreadTokenUsage.ts +6 -0
- package/services/mac-bridge/codex-types/v2/ThreadTokenUsageUpdatedNotification.ts +6 -0
- package/services/mac-bridge/codex-types/v2/ThreadUnarchiveParams.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ThreadUnarchiveResponse.ts +6 -0
- package/services/mac-bridge/codex-types/v2/ThreadUnarchivedNotification.ts +5 -0
- package/services/mac-bridge/codex-types/v2/TokenUsageBreakdown.ts +5 -0
- package/services/mac-bridge/codex-types/v2/ToolRequestUserInputAnswer.ts +8 -0
- package/services/mac-bridge/codex-types/v2/ToolRequestUserInputOption.ts +8 -0
- package/services/mac-bridge/codex-types/v2/ToolRequestUserInputParams.ts +9 -0
- package/services/mac-bridge/codex-types/v2/ToolRequestUserInputQuestion.ts +9 -0
- package/services/mac-bridge/codex-types/v2/ToolRequestUserInputResponse.ts +9 -0
- package/services/mac-bridge/codex-types/v2/ToolsV2.ts +5 -0
- package/services/mac-bridge/codex-types/v2/Turn.ts +18 -0
- package/services/mac-bridge/codex-types/v2/TurnCompletedNotification.ts +6 -0
- package/services/mac-bridge/codex-types/v2/TurnDiffUpdatedNotification.ts +9 -0
- package/services/mac-bridge/codex-types/v2/TurnError.ts +6 -0
- package/services/mac-bridge/codex-types/v2/TurnInterruptParams.ts +5 -0
- package/services/mac-bridge/codex-types/v2/TurnInterruptResponse.ts +5 -0
- package/services/mac-bridge/codex-types/v2/TurnPlanStep.ts +6 -0
- package/services/mac-bridge/codex-types/v2/TurnPlanStepStatus.ts +5 -0
- package/services/mac-bridge/codex-types/v2/TurnPlanUpdatedNotification.ts +6 -0
- package/services/mac-bridge/codex-types/v2/TurnStartParams.ts +44 -0
- package/services/mac-bridge/codex-types/v2/TurnStartResponse.ts +6 -0
- package/services/mac-bridge/codex-types/v2/TurnStartedNotification.ts +6 -0
- package/services/mac-bridge/codex-types/v2/TurnStatus.ts +5 -0
- package/services/mac-bridge/codex-types/v2/TurnSteerParams.ts +11 -0
- package/services/mac-bridge/codex-types/v2/TurnSteerResponse.ts +5 -0
- package/services/mac-bridge/codex-types/v2/UserInput.ts +10 -0
- package/services/mac-bridge/codex-types/v2/WebSearchAction.ts +5 -0
- package/services/mac-bridge/codex-types/v2/WindowsWorldWritableWarningNotification.ts +5 -0
- package/services/mac-bridge/codex-types/v2/WriteStatus.ts +5 -0
- package/services/mac-bridge/codex-types/v2/index.ts +204 -0
- package/services/mac-bridge/eslint.config.cjs +22 -0
- package/services/mac-bridge/package.json +30 -0
- package/services/mac-bridge/schema.ts +0 -0
- package/services/mac-bridge/src/index.ts +18 -0
- package/services/mac-bridge/src/server.ts +426 -0
- package/services/mac-bridge/src/services/__tests__/gitService.test.ts +157 -0
- package/services/mac-bridge/src/services/__tests__/realtimeHub.test.ts +116 -0
- package/services/mac-bridge/src/services/__tests__/terminalService.test.ts +51 -0
- package/services/mac-bridge/src/services/codexAppServerClient.ts +507 -0
- package/services/mac-bridge/src/services/codexCliAdapter.ts +622 -0
- package/services/mac-bridge/src/services/gitService.ts +61 -0
- package/services/mac-bridge/src/services/realtimeHub.ts +25 -0
- package/services/mac-bridge/src/services/terminalService.ts +226 -0
- package/services/mac-bridge/src/types.ts +151 -0
- package/services/mac-bridge/src/utils/__tests__/threadMapping.test.ts +397 -0
- package/services/mac-bridge/src/utils/threadMapping.ts +176 -0
- package/services/mac-bridge/tsconfig.json +16 -0
- package/services/mac-bridge/vitest.config.ts +9 -0
- package/services/rust-bridge/.env.example +11 -0
- package/services/rust-bridge/Cargo.lock +1127 -0
- package/services/rust-bridge/Cargo.toml +14 -0
- package/services/rust-bridge/package.json +13 -0
- package/services/rust-bridge/security_best_practices_report.md +24 -0
- package/services/rust-bridge/src/main.rs +2713 -0
- package/services/rust-bridge/src/services/git.rs +271 -0
- package/services/rust-bridge/src/services/mod.rs +5 -0
- package/services/rust-bridge/src/services/terminal.rs +267 -0
- package/tsconfig.json +4 -0
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import {
|
|
2
|
+
TerminalCommandRejectedError,
|
|
3
|
+
TerminalService
|
|
4
|
+
} from '../terminalService';
|
|
5
|
+
|
|
6
|
+
describe('TerminalService', () => {
|
|
7
|
+
const cwd = process.cwd();
|
|
8
|
+
|
|
9
|
+
it('executes an allowlisted command', async () => {
|
|
10
|
+
const terminal = new TerminalService({
|
|
11
|
+
allowedCommands: ['pwd']
|
|
12
|
+
});
|
|
13
|
+
|
|
14
|
+
const result = await terminal.executeShell('pwd', { cwd });
|
|
15
|
+
|
|
16
|
+
expect(result.code).toBe(0);
|
|
17
|
+
expect(result.command).toBe('pwd');
|
|
18
|
+
expect(result.cwd).toBe(cwd);
|
|
19
|
+
});
|
|
20
|
+
|
|
21
|
+
it('rejects commands outside the allowlist', async () => {
|
|
22
|
+
const terminal = new TerminalService({
|
|
23
|
+
allowedCommands: ['pwd']
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
await expect(
|
|
27
|
+
terminal.executeShell('ls', { cwd })
|
|
28
|
+
).rejects.toBeInstanceOf(TerminalCommandRejectedError);
|
|
29
|
+
});
|
|
30
|
+
|
|
31
|
+
it('rejects shell control characters in user command input', async () => {
|
|
32
|
+
const terminal = new TerminalService({
|
|
33
|
+
allowedCommands: ['pwd', 'ls']
|
|
34
|
+
});
|
|
35
|
+
|
|
36
|
+
await expect(
|
|
37
|
+
terminal.executeShell('pwd && ls', { cwd })
|
|
38
|
+
).rejects.toThrow('Disallowed control character');
|
|
39
|
+
});
|
|
40
|
+
|
|
41
|
+
it('supports quoted arguments', async () => {
|
|
42
|
+
const terminal = new TerminalService({
|
|
43
|
+
allowedCommands: ['echo']
|
|
44
|
+
});
|
|
45
|
+
|
|
46
|
+
const result = await terminal.executeShell('echo "hello world"', { cwd });
|
|
47
|
+
|
|
48
|
+
expect(result.code).toBe(0);
|
|
49
|
+
expect(result.stdout).toBe('hello world');
|
|
50
|
+
});
|
|
51
|
+
});
|
|
@@ -0,0 +1,507 @@
|
|
|
1
|
+
import { spawn, type ChildProcessWithoutNullStreams } from 'node:child_process';
|
|
2
|
+
import { appendFileSync } from 'node:fs';
|
|
3
|
+
|
|
4
|
+
import type { ApprovalDecision, ApprovalKind, PendingApproval } from '../types';
|
|
5
|
+
|
|
6
|
+
const LOG_FILE = '/tmp/codex-app-server.log';
|
|
7
|
+
|
|
8
|
+
function logRpc(direction: 'SEND' | 'RECV' | 'STDERR' | 'INFO', data: string): void {
|
|
9
|
+
const ts = new Date().toISOString();
|
|
10
|
+
try {
|
|
11
|
+
appendFileSync(LOG_FILE, `[${ts}] ${direction}: ${data}\n`);
|
|
12
|
+
} catch {
|
|
13
|
+
// ignore write errors
|
|
14
|
+
}
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
export interface AppServerNotification {
|
|
18
|
+
method: string;
|
|
19
|
+
params: Record<string, unknown> | null;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
interface JsonRpcErrorShape {
|
|
23
|
+
code: number;
|
|
24
|
+
message: string;
|
|
25
|
+
data?: unknown;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
interface JsonRpcResponse {
|
|
29
|
+
id: string | number | null;
|
|
30
|
+
result?: unknown;
|
|
31
|
+
error?: JsonRpcErrorShape;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
interface JsonRpcRequest {
|
|
35
|
+
id: string | number;
|
|
36
|
+
method: string;
|
|
37
|
+
params?: unknown;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
interface PendingRequest {
|
|
41
|
+
resolve: (value: unknown) => void;
|
|
42
|
+
reject: (reason?: unknown) => void;
|
|
43
|
+
timeout: NodeJS.Timeout;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
interface CodexAppServerClientOptions {
|
|
47
|
+
cliBin: string;
|
|
48
|
+
timeoutMs: number;
|
|
49
|
+
onStderr?: (chunk: string) => void;
|
|
50
|
+
onApprovalRequested?: (approval: PendingApproval) => void;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
export interface ThreadListParams {
|
|
54
|
+
limit?: number;
|
|
55
|
+
cursor?: string | null;
|
|
56
|
+
cwd?: string | null;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
interface PendingApprovalRequest {
|
|
60
|
+
requestId: string | number;
|
|
61
|
+
approval: PendingApproval;
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
export class CodexAppServerClient {
|
|
65
|
+
private readonly cliBin: string;
|
|
66
|
+
private readonly timeoutMs: number;
|
|
67
|
+
private readonly onStderr?: (chunk: string) => void;
|
|
68
|
+
private readonly onApprovalRequested?: (approval: PendingApproval) => void;
|
|
69
|
+
|
|
70
|
+
private child: ChildProcessWithoutNullStreams | null = null;
|
|
71
|
+
private stdoutBuffer = '';
|
|
72
|
+
private started = false;
|
|
73
|
+
private starting: Promise<void> | null = null;
|
|
74
|
+
private requestCounter = 0;
|
|
75
|
+
private pending = new Map<string | number, PendingRequest>();
|
|
76
|
+
private listeners = new Set<(notification: AppServerNotification) => void>();
|
|
77
|
+
private approvalCounter = 0;
|
|
78
|
+
private pendingApprovals = new Map<string, PendingApprovalRequest>();
|
|
79
|
+
|
|
80
|
+
constructor(options: CodexAppServerClientOptions) {
|
|
81
|
+
this.cliBin = options.cliBin;
|
|
82
|
+
this.timeoutMs = options.timeoutMs;
|
|
83
|
+
this.onStderr = options.onStderr;
|
|
84
|
+
this.onApprovalRequested = options.onApprovalRequested;
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
onNotification(listener: (notification: AppServerNotification) => void): () => void {
|
|
88
|
+
this.listeners.add(listener);
|
|
89
|
+
return () => {
|
|
90
|
+
this.listeners.delete(listener);
|
|
91
|
+
};
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
listPendingApprovals(): PendingApproval[] {
|
|
95
|
+
return [...this.pendingApprovals.values()]
|
|
96
|
+
.map((entry) => structuredClone(entry.approval))
|
|
97
|
+
.sort((a, b) => b.requestedAt.localeCompare(a.requestedAt));
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
async resolveApproval(
|
|
101
|
+
approvalId: string,
|
|
102
|
+
decision: ApprovalDecision
|
|
103
|
+
): Promise<PendingApproval | null> {
|
|
104
|
+
const pendingApproval = this.pendingApprovals.get(approvalId);
|
|
105
|
+
if (!pendingApproval) {
|
|
106
|
+
return null;
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
this.pendingApprovals.delete(approvalId);
|
|
110
|
+
|
|
111
|
+
try {
|
|
112
|
+
await this.writeJson({
|
|
113
|
+
jsonrpc: '2.0',
|
|
114
|
+
id: pendingApproval.requestId,
|
|
115
|
+
result: { decision }
|
|
116
|
+
});
|
|
117
|
+
} catch (error) {
|
|
118
|
+
this.pendingApprovals.set(approvalId, pendingApproval);
|
|
119
|
+
throw error;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
return structuredClone(pendingApproval.approval);
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
async threadList(params: ThreadListParams = {}): Promise<Record<string, unknown>> {
|
|
126
|
+
return this.request('thread/list', {
|
|
127
|
+
cursor: params.cursor ?? null,
|
|
128
|
+
limit: params.limit ?? 100,
|
|
129
|
+
sortKey: null,
|
|
130
|
+
modelProviders: null,
|
|
131
|
+
sourceKinds: ['cli', 'vscode', 'exec', 'appServer', 'subAgent', 'unknown'],
|
|
132
|
+
archived: false,
|
|
133
|
+
cwd: params.cwd ?? null
|
|
134
|
+
});
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
async threadRead(threadId: string, includeTurns = true): Promise<Record<string, unknown>> {
|
|
138
|
+
return this.request('thread/read', {
|
|
139
|
+
threadId,
|
|
140
|
+
includeTurns
|
|
141
|
+
});
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
async threadStart(params: {
|
|
145
|
+
cwd?: string | null;
|
|
146
|
+
approvalPolicy?: string;
|
|
147
|
+
sandbox?: string;
|
|
148
|
+
}): Promise<Record<string, unknown>> {
|
|
149
|
+
return this.request('thread/start', {
|
|
150
|
+
model: null,
|
|
151
|
+
modelProvider: null,
|
|
152
|
+
cwd: params.cwd ?? null,
|
|
153
|
+
approvalPolicy: params.approvalPolicy ?? 'on-request',
|
|
154
|
+
sandbox: params.sandbox ?? 'workspace-write',
|
|
155
|
+
config: null,
|
|
156
|
+
baseInstructions: null,
|
|
157
|
+
developerInstructions: null,
|
|
158
|
+
personality: null,
|
|
159
|
+
ephemeral: null,
|
|
160
|
+
experimentalRawEvents: false,
|
|
161
|
+
persistExtendedHistory: true
|
|
162
|
+
});
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
async threadResume(threadId: string): Promise<Record<string, unknown>> {
|
|
166
|
+
return this.request('thread/resume', {
|
|
167
|
+
threadId,
|
|
168
|
+
history: null,
|
|
169
|
+
path: null,
|
|
170
|
+
model: null,
|
|
171
|
+
modelProvider: null,
|
|
172
|
+
cwd: null,
|
|
173
|
+
approvalPolicy: 'on-request',
|
|
174
|
+
sandbox: 'workspace-write',
|
|
175
|
+
config: null,
|
|
176
|
+
baseInstructions: null,
|
|
177
|
+
developerInstructions: null,
|
|
178
|
+
personality: null,
|
|
179
|
+
persistExtendedHistory: true
|
|
180
|
+
});
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
async turnStart(threadId: string, userText: string): Promise<Record<string, unknown>> {
|
|
184
|
+
return this.request('turn/start', {
|
|
185
|
+
threadId,
|
|
186
|
+
input: [
|
|
187
|
+
{
|
|
188
|
+
type: 'text',
|
|
189
|
+
text: userText,
|
|
190
|
+
text_elements: []
|
|
191
|
+
}
|
|
192
|
+
],
|
|
193
|
+
cwd: null,
|
|
194
|
+
approvalPolicy: null,
|
|
195
|
+
sandboxPolicy: null,
|
|
196
|
+
model: null,
|
|
197
|
+
effort: null,
|
|
198
|
+
summary: null,
|
|
199
|
+
personality: null,
|
|
200
|
+
outputSchema: null,
|
|
201
|
+
collaborationMode: null
|
|
202
|
+
});
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
private async request(method: string, params: unknown): Promise<Record<string, unknown>> {
|
|
206
|
+
await this.ensureStarted();
|
|
207
|
+
return this.requestInternal(method, params);
|
|
208
|
+
}
|
|
209
|
+
|
|
210
|
+
private async requestInternal(
|
|
211
|
+
method: string,
|
|
212
|
+
params: unknown
|
|
213
|
+
): Promise<Record<string, unknown>> {
|
|
214
|
+
const id = `${Date.now()}-${++this.requestCounter}`;
|
|
215
|
+
const payload = {
|
|
216
|
+
jsonrpc: '2.0',
|
|
217
|
+
id,
|
|
218
|
+
method,
|
|
219
|
+
params
|
|
220
|
+
};
|
|
221
|
+
|
|
222
|
+
const result = await new Promise<unknown>((resolve, reject) => {
|
|
223
|
+
const timeout = setTimeout(() => {
|
|
224
|
+
this.pending.delete(id);
|
|
225
|
+
reject(new Error(`app-server request timeout: ${method}`));
|
|
226
|
+
}, this.timeoutMs);
|
|
227
|
+
|
|
228
|
+
this.pending.set(id, {
|
|
229
|
+
resolve,
|
|
230
|
+
reject,
|
|
231
|
+
timeout
|
|
232
|
+
});
|
|
233
|
+
|
|
234
|
+
this.writeJson(payload).catch((error) => {
|
|
235
|
+
clearTimeout(timeout);
|
|
236
|
+
this.pending.delete(id);
|
|
237
|
+
reject(error);
|
|
238
|
+
});
|
|
239
|
+
});
|
|
240
|
+
|
|
241
|
+
if (!this.isRecord(result)) {
|
|
242
|
+
throw new Error(`app-server returned non-object result for ${method}`);
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
return result;
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
private async ensureStarted(): Promise<void> {
|
|
249
|
+
if (this.started) {
|
|
250
|
+
return;
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
if (this.starting) {
|
|
254
|
+
return this.starting;
|
|
255
|
+
}
|
|
256
|
+
|
|
257
|
+
this.starting = this.startInternal();
|
|
258
|
+
try {
|
|
259
|
+
await this.starting;
|
|
260
|
+
} finally {
|
|
261
|
+
this.starting = null;
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
|
|
265
|
+
private async startInternal(): Promise<void> {
|
|
266
|
+
const child = spawn(this.cliBin, ['app-server', '--listen', 'stdio://'], {
|
|
267
|
+
env: process.env,
|
|
268
|
+
stdio: ['pipe', 'pipe', 'pipe']
|
|
269
|
+
});
|
|
270
|
+
|
|
271
|
+
this.child = child;
|
|
272
|
+
|
|
273
|
+
child.stdout.setEncoding('utf8');
|
|
274
|
+
child.stderr.setEncoding('utf8');
|
|
275
|
+
|
|
276
|
+
child.stdout.on('data', (chunk: string) => {
|
|
277
|
+
this.stdoutBuffer += chunk;
|
|
278
|
+
|
|
279
|
+
let newlineIndex = this.stdoutBuffer.indexOf('\n');
|
|
280
|
+
while (newlineIndex >= 0) {
|
|
281
|
+
const line = this.stdoutBuffer.slice(0, newlineIndex).trim();
|
|
282
|
+
this.stdoutBuffer = this.stdoutBuffer.slice(newlineIndex + 1);
|
|
283
|
+
this.handleLine(line);
|
|
284
|
+
newlineIndex = this.stdoutBuffer.indexOf('\n');
|
|
285
|
+
}
|
|
286
|
+
});
|
|
287
|
+
|
|
288
|
+
child.stderr.on('data', (chunk: string) => {
|
|
289
|
+
logRpc('STDERR', chunk.trimEnd());
|
|
290
|
+
this.onStderr?.(chunk);
|
|
291
|
+
});
|
|
292
|
+
|
|
293
|
+
child.on('error', (error) => {
|
|
294
|
+
logRpc('INFO', `process error: ${error.message}`);
|
|
295
|
+
this.failAllPending(error);
|
|
296
|
+
this.pendingApprovals.clear();
|
|
297
|
+
this.started = false;
|
|
298
|
+
});
|
|
299
|
+
|
|
300
|
+
child.on('close', (code, signal) => {
|
|
301
|
+
logRpc('INFO', `process closed (code=${String(code)} signal=${String(signal)})`);
|
|
302
|
+
this.failAllPending(
|
|
303
|
+
new Error(
|
|
304
|
+
`codex app-server closed (code=${String(code)} signal=${String(signal)})`
|
|
305
|
+
)
|
|
306
|
+
);
|
|
307
|
+
this.pendingApprovals.clear();
|
|
308
|
+
this.started = false;
|
|
309
|
+
this.child = null;
|
|
310
|
+
});
|
|
311
|
+
|
|
312
|
+
const initializeResult = await this.requestInternal('initialize', {
|
|
313
|
+
clientInfo: {
|
|
314
|
+
name: 'codex-mobile-bridge',
|
|
315
|
+
title: 'Codex Mobile Bridge',
|
|
316
|
+
version: '0.1.0'
|
|
317
|
+
},
|
|
318
|
+
capabilities: {
|
|
319
|
+
experimentalApi: true,
|
|
320
|
+
optOutNotificationMethods: null
|
|
321
|
+
}
|
|
322
|
+
});
|
|
323
|
+
|
|
324
|
+
if (!this.isRecord(initializeResult)) {
|
|
325
|
+
throw new Error('app-server initialize returned invalid payload');
|
|
326
|
+
}
|
|
327
|
+
|
|
328
|
+
await this.writeJson({
|
|
329
|
+
jsonrpc: '2.0',
|
|
330
|
+
method: 'initialized',
|
|
331
|
+
params: null
|
|
332
|
+
});
|
|
333
|
+
|
|
334
|
+
this.started = true;
|
|
335
|
+
}
|
|
336
|
+
|
|
337
|
+
private handleLine(line: string): void {
|
|
338
|
+
if (!line) {
|
|
339
|
+
return;
|
|
340
|
+
}
|
|
341
|
+
|
|
342
|
+
logRpc('RECV', line);
|
|
343
|
+
|
|
344
|
+
let parsed: unknown;
|
|
345
|
+
try {
|
|
346
|
+
parsed = JSON.parse(line);
|
|
347
|
+
} catch {
|
|
348
|
+
return;
|
|
349
|
+
}
|
|
350
|
+
|
|
351
|
+
if (!this.isRecord(parsed)) {
|
|
352
|
+
return;
|
|
353
|
+
}
|
|
354
|
+
|
|
355
|
+
const hasMethod = typeof parsed.method === 'string';
|
|
356
|
+
const hasId = typeof parsed.id === 'string' || typeof parsed.id === 'number';
|
|
357
|
+
|
|
358
|
+
if (hasMethod && hasId) {
|
|
359
|
+
this.handleServerRequest({
|
|
360
|
+
id: parsed.id as string | number,
|
|
361
|
+
method: String(parsed.method),
|
|
362
|
+
params: parsed.params
|
|
363
|
+
});
|
|
364
|
+
return;
|
|
365
|
+
}
|
|
366
|
+
|
|
367
|
+
if (hasMethod) {
|
|
368
|
+
this.emitNotification({
|
|
369
|
+
method: String(parsed.method),
|
|
370
|
+
params: this.isRecord(parsed.params) ? parsed.params : null
|
|
371
|
+
});
|
|
372
|
+
return;
|
|
373
|
+
}
|
|
374
|
+
|
|
375
|
+
if ('id' in parsed) {
|
|
376
|
+
this.handleResponse({
|
|
377
|
+
id: (parsed.id as string | number | null) ?? null,
|
|
378
|
+
result: parsed.result,
|
|
379
|
+
error: this.isRecord(parsed.error)
|
|
380
|
+
? {
|
|
381
|
+
code: Number(parsed.error.code),
|
|
382
|
+
message: String(parsed.error.message),
|
|
383
|
+
data: parsed.error.data
|
|
384
|
+
}
|
|
385
|
+
: undefined
|
|
386
|
+
});
|
|
387
|
+
}
|
|
388
|
+
}
|
|
389
|
+
|
|
390
|
+
private handleResponse(response: JsonRpcResponse): void {
|
|
391
|
+
const requestId = response.id;
|
|
392
|
+
if (requestId === null || requestId === undefined) {
|
|
393
|
+
return;
|
|
394
|
+
}
|
|
395
|
+
|
|
396
|
+
const pending = this.pending.get(requestId);
|
|
397
|
+
if (!pending) {
|
|
398
|
+
return;
|
|
399
|
+
}
|
|
400
|
+
|
|
401
|
+
clearTimeout(pending.timeout);
|
|
402
|
+
this.pending.delete(requestId);
|
|
403
|
+
|
|
404
|
+
if (response.error) {
|
|
405
|
+
console.error(`[RPC ERROR]`, JSON.stringify(response.error));
|
|
406
|
+
pending.reject(
|
|
407
|
+
new Error(
|
|
408
|
+
`app-server error ${String(response.error.code)}: ${response.error.message}`
|
|
409
|
+
)
|
|
410
|
+
);
|
|
411
|
+
return;
|
|
412
|
+
}
|
|
413
|
+
|
|
414
|
+
pending.resolve(response.result ?? null);
|
|
415
|
+
}
|
|
416
|
+
|
|
417
|
+
private handleServerRequest(request: JsonRpcRequest): void {
|
|
418
|
+
const method = request.method;
|
|
419
|
+
|
|
420
|
+
if (method === 'item/commandExecution/requestApproval') {
|
|
421
|
+
this.queueApprovalRequest('commandExecution', request);
|
|
422
|
+
return;
|
|
423
|
+
}
|
|
424
|
+
|
|
425
|
+
if (method === 'item/fileChange/requestApproval') {
|
|
426
|
+
this.queueApprovalRequest('fileChange', request);
|
|
427
|
+
return;
|
|
428
|
+
}
|
|
429
|
+
|
|
430
|
+
void this.writeJson({
|
|
431
|
+
jsonrpc: '2.0',
|
|
432
|
+
id: request.id,
|
|
433
|
+
error: {
|
|
434
|
+
code: -32601,
|
|
435
|
+
message: `Unsupported server request method: ${method}`
|
|
436
|
+
}
|
|
437
|
+
});
|
|
438
|
+
}
|
|
439
|
+
|
|
440
|
+
private queueApprovalRequest(kind: ApprovalKind, request: JsonRpcRequest): void {
|
|
441
|
+
const params = this.isRecord(request.params) ? request.params : {};
|
|
442
|
+
const threadId = this.readString(params.threadId) ?? 'unknown-thread';
|
|
443
|
+
const turnId = this.readString(params.turnId) ?? 'unknown-turn';
|
|
444
|
+
const itemId = this.readString(params.itemId) ?? 'unknown-item';
|
|
445
|
+
|
|
446
|
+
const approval: PendingApproval = {
|
|
447
|
+
id: `${Date.now()}-${++this.approvalCounter}`,
|
|
448
|
+
kind,
|
|
449
|
+
threadId,
|
|
450
|
+
turnId,
|
|
451
|
+
itemId,
|
|
452
|
+
requestedAt: new Date().toISOString(),
|
|
453
|
+
reason: this.readString(params.reason) ?? undefined,
|
|
454
|
+
command: this.readString(params.command) ?? undefined,
|
|
455
|
+
cwd: this.readString(params.cwd) ?? undefined,
|
|
456
|
+
grantRoot: this.readString(params.grantRoot) ?? undefined
|
|
457
|
+
};
|
|
458
|
+
|
|
459
|
+
this.pendingApprovals.set(approval.id, {
|
|
460
|
+
requestId: request.id,
|
|
461
|
+
approval
|
|
462
|
+
});
|
|
463
|
+
|
|
464
|
+
this.onApprovalRequested?.(structuredClone(approval));
|
|
465
|
+
}
|
|
466
|
+
|
|
467
|
+
private emitNotification(notification: AppServerNotification): void {
|
|
468
|
+
for (const listener of this.listeners) {
|
|
469
|
+
listener(notification);
|
|
470
|
+
}
|
|
471
|
+
}
|
|
472
|
+
|
|
473
|
+
private async writeJson(payload: Record<string, unknown>): Promise<void> {
|
|
474
|
+
if (!this.child || !this.child.stdin.writable) {
|
|
475
|
+
throw new Error('app-server stdin is not writable');
|
|
476
|
+
}
|
|
477
|
+
|
|
478
|
+
const line = JSON.stringify(payload);
|
|
479
|
+
logRpc('SEND', line);
|
|
480
|
+
await new Promise<void>((resolve, reject) => {
|
|
481
|
+
this.child?.stdin.write(`${line}\n`, (error) => {
|
|
482
|
+
if (error) {
|
|
483
|
+
reject(error);
|
|
484
|
+
return;
|
|
485
|
+
}
|
|
486
|
+
|
|
487
|
+
resolve();
|
|
488
|
+
});
|
|
489
|
+
});
|
|
490
|
+
}
|
|
491
|
+
|
|
492
|
+
private failAllPending(error: Error): void {
|
|
493
|
+
for (const [id, pending] of this.pending.entries()) {
|
|
494
|
+
clearTimeout(pending.timeout);
|
|
495
|
+
pending.reject(error);
|
|
496
|
+
this.pending.delete(id);
|
|
497
|
+
}
|
|
498
|
+
}
|
|
499
|
+
|
|
500
|
+
private isRecord(value: unknown): value is Record<string, unknown> {
|
|
501
|
+
return typeof value === 'object' && value !== null;
|
|
502
|
+
}
|
|
503
|
+
|
|
504
|
+
private readString(value: unknown): string | null {
|
|
505
|
+
return typeof value === 'string' ? value : null;
|
|
506
|
+
}
|
|
507
|
+
}
|