1. Shell.ts / imagePaste.ts / filesystem.ts: Linux/macOS 默认 tmp 路径 从硬编码 '/tmp' 改为 os.tmpdir(),自动适配 Termux/Android 等无 /tmp 的环境;macOS 桌面零变化;CLAUDE_CODE_TMPDIR 仍优先级最高。 2. ripgrep.ts: builtin rg 二进制缺失时(Android/Termux、不完整安装) 自动 fallback 到 PATH 上的系统 rg,通过 note 字段携带人读提示; /doctor 渲染 note;init 启动时写一行 stderr warning。 Co-Authored-By: glm-5.2 <zai-org@claude-code-best.win>
7.1 KiB
Ripgrep System Fallback — Design
Date: 2026-06-15
Status: Approved (pending spec review)
Topic: Make ripgrep gracefully degrade to system rg when the bundled/builtin binary is unavailable on the current platform (e.g. Android/Termux).
Problem
src/utils/ripgrep.ts getRipgrepConfig() has three resolution branches:
USE_BUILTIN_RIPGREP=0→ look uprgonPATHisInBundledMode()→ bun-internal embedded rg- Otherwise →
vendor/ripgrep/<arch>-<platform>/rg(builtin)
On Android/Termux, all three fail:
- The user has not opted into system rg.
- Bun does not publish Android builds, so
isInBundledMode()is false. scripts/postinstall.cjs:81throwsUnsupported platform: android, so no builtin binary is ever downloaded.vendor/ripgrep/contains noarm64-androiddirectory.
Net effect: spawn of a nonexistent path → ENOENT → user sees "ripgrep 缺失" with no recovery path other than manually setting USE_BUILTIN_RIPGREP=0. The discovery pipeline (Grep/Glob tools, file suggestions, hooks) all fail in the same way.
More generally, the same breakage occurs on any platform where the builtin binary is missing for any reason (incomplete install, custom platform, deleted vendor directory). The current code has no graceful degradation.
Goals
- On any platform, when the builtin/bundled ripgrep is unavailable, automatically fall back to
rgonPATH. - Surface the fallback clearly to the user via
/doctorand a one-line startup warning, so they understand why they are not on the bundled rg and what to do if the system rg is also missing. - Do not change behavior on platforms where the builtin rg works (macOS, Linux, Windows).
Non-Goals
- Downloading or shipping an Android-native ripgrep binary.
- Adding a REPL persistent status indicator.
- Touching
USE_BUILTIN_RIPGREPsemantics for users who already opt into system rg. - Modifying build /
postinstall.cjsplatform mapping.
Design
Decision chain (getRipgrepConfig)
The function gains an existence check and a system-rg fallback. The order of existing branches is preserved.
1. USE_BUILTIN_RIPGREP=0 (user-opt) → system rg mode='system' note=undefined
2. isInBundledMode() → bun embedded rg mode='embedded' note=undefined
3. Compute builtin path; existsSync(rgPath)?
✓ true → builtin rg mode='builtin' note=undefined
✓ false → findExecutable('rg', [])
✓ found → system rg (auto fallback) mode='system' note='fallback: builtin rg unavailable on <platform>, using system rg'
✗ missing → keep builtin path (let upper layer ENOENT) mode='builtin' note='no ripgrep available on <platform>; install via apt/pkg/brew/...'
Rationale for the missing-system-rg branch returning the (nonexistent) builtin path: it preserves the historical spawn behavior so existing error-handling paths in ripGrepRaw and callers continue to see ENOENT. The new note field carries the human-readable explanation; the spawn itself still fails the same way.
existsSync is a single synchronous syscall; getRipgrepConfig is already memoized via lodash, so the cost is paid once per process.
Status API (getRipgrepStatus)
type RipgrepStatus = {
mode: 'system' | 'builtin' | 'embedded' // unchanged
path: string // unchanged
working: boolean | null // unchanged
note?: string // NEW — human-readable hint
}
The internal ripgrepStatus singleton also gains note?: string. testRipgrepOnFirstUse propagates the note from the active config.
The note value is sourced from getRipgrepConfig() (the source of truth), so the API remains a single read; no second lookup.
UI — /doctor
src/screens/Doctor.tsx renders the existing Search: line plus the note when present. Two example outputs:
Search: OK (system rg fallback — builtin ripgrep unavailable on android)
Search: Not working (no ripgrep available on android — install via apt/pkg/brew)
src/utils/doctorDiagnostic.ts extends the ripgrepStatus object it returns to include note.
UI — startup warning
A single check near the end of src/entrypoints/init.ts reads getRipgrepStatus(). If note is set, it writes one line to stderr:
[ripgrep] fallback: builtin rg unavailable on android, using system rg
Constraints:
- Non-blocking — does not throw or exit.
- Fires at most once per process (memoized config + idempotent init).
- Goes to stderr so it does not corrupt pipe mode (
-p) stdout. - No retry, no telemetry beyond existing
tengu_ripgrep_availability.
Testing
New test file src/utils/__tests__/ripgrepDecision.test.ts (or extend an existing one) covers the five branches:
USE_BUILTIN_RIPGREP=0andrgon PATH →mode='system',note=undefined.isInBundledMode()→mode='embedded',note=undefined.- Builtin path exists →
mode='builtin',note=undefined. - Builtin path missing,
rgon PATH →mode='system',noteset. - Builtin path missing,
rgnot on PATH →mode='builtin',noteset (path is the nonexistent builtin path).
Mocks: existsSync (via fs module), findExecutable, isInBundledMode, process.env.USE_BUILTIN_RIPGREP, process.platform. Follow the project's mock conventions (see tests/mocks/); no business-module mocking.
Existing doctorDiagnostic tests: extend to assert note is propagated; update any snapshots.
Risks
- Behavior preservation on supported platforms: the
existsSynccheck only changes the path when the builtin file is genuinely absent. On macOS/Linux/Windows the builtin binary always exists post-install, so the decision chain resolves tomode='builtin'exactly as today. Verified by the test for branch 3. notefield addition is backward-compatible: optional field; existing consumers ignore it.- Memoization:
getRipgrepConfigis memoized for the process lifetime. If a user installs ripgrep mid-session, the fallback will not trigger until restart. Acceptable — matches existing behavior forUSE_BUILTIN_RIPGREPchanges. - Platform string in
note: usesprocess.platformdirectly ('android','linux','darwin','win32'). No translation; the message is diagnostic, not user-facing marketing copy.
Out of Scope (YAGNI)
- Android prebuilt binary download.
- Persistent REPL status indicator.
- Build-time vendor changes.
- Telemetry beyond what
testRipgrepOnFirstUsealready emits.
Acceptance Criteria
- On a platform where the builtin rg binary is missing and
rgis onPATH,getRipgrepStatus()returnsmode='system',path=<resolved system rg>,noteset to a non-empty human-readable string. - On a platform where neither builtin nor system rg is available,
/doctordisplaysNot workingplus the install hint. - The startup warning fires exactly once per session when
noteis set. - All existing ripgrep tests pass unchanged on macOS/Linux dev machines.
bun run precheckis green.