fix(servers): make settings form re-render on every keystroke (#1361)#1362
Conversation
The settings modal is fully controlled by App.tsx, but pending edits were stashed in a useRef — refs don't trigger re-renders, so the input's `value` prop stayed stale until the 300ms debounce + PUT + refetch round-tripped. Clicking "Add Header" / "Add Metadata" showed no row, and typing in OAuth fields had no character echo, until that round-trip completed (~500ms). The bug landed in #1353 with the debounced-flush pattern. It surfaced after #1356 because per-keystroke PUTs now also write the keychain, making the round-trip latency more noticeable. Lift the draft into `useState`. Every change re-renders the modal immediately. Keep the 300ms debounce on the PUT side. Initialize the draft from the server-list entry only when the modal opens to a new target (not on every `servers` change) so background refreshes don't clobber the user's in-progress edits. Flush on close to preserve final keystrokes. Verified live against the dev server: header/metadata rows appear on click, OAuth Client ID / Client Secret typing echoes without lag, and the values persist correctly (clientId in mcp.json, clientSecret in the macOS Keychain). Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
|
@claude review |
|
Claude finished @cliffhall's task in 1m 34s —— View job Claude PR Review
SummaryThe fix is correct and the PR description accurately describes the pre-existing bug (ref-based draft producing no re-renders). Swapping to The diff is net +31 lines, much of which is comments. I found no correctness regressions, just a few minor observations. Correctness — looks good
Minor observations1.
|
…1361) Picks up the three review observations on #1362: 1. Extract the draft state + debounce + flush logic from App.tsx into `core/react/useSettingsDraft`. Same behavior; thinner App.tsx; testable in isolation with `renderHook`. Tests pin the regression #1361 was about (synchronous onChange → draft update) plus the debounce window, burst-collapse, flush-on-close, error propagation, and the "background refresh must not clobber in-progress edits" contract via a `resolveInitial` closure swap. 2. Hoist the empty `InspectorServerSettings` fallback to a module-scope `EMPTY_SETTINGS` const. Both the hook's init path and the modal's closed-state fallback share the same object identity; the inline `useMemo` that previously re-derived it is gone. 3. Add a comment on the `resolveInitial` callback noting that we read `servers` from this render's closure under the assumption that the entry point is the "Settings" button on a rendered server card. Documents the contract for future callers (e.g. a keyboard shortcut that opens the modal pre-hydration). Validate clean: 1756 tests pass. New useSettingsDraft.test.tsx adds 13 test cases. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
|
Thanks for the review. All three observations addressed in 1. Regression test (the policy ask). Extracted the draft state + debounce + flush logic into
App.tsx is now thinner — it just wires inputs to the hook plus a tiny 2. 3. CI: 144 files, 1756 tests, all green. |
|
@claude review |
|
Claude finished @cliffhall's task in 1m 41s —— View job Claude PR Review — follow-up on
|
…1361) Two non-blocking observations from the third review on #1362: 1. `flush` identity churned on every keystroke because the useCallback closed over `draft` + `targetId`. Read both through refs instead so the returned callback identity is stable. The behavior is unchanged — but the consumer's `onSettingsModalClose` (which wraps `flush`) no longer re-allocates per keystroke, which means the modal's `onClose` prop identity also stays stable. A new test pins this. 2. Added a comment in `onChange` calling out that callers switching `targetId` mid-debounce must `flush()` first to preserve the pending PUT. The only switch path today (`onSettingsModalClose` in App.tsx) does that already — the comment is for future refactors. 3. New unmount-cleanup test pins the intentional "drop final <debounceMs window on unmount" trade-off — flushing on unmount could fire a PUT against an unmounting component, which is a worse footgun than losing a few hundred ms of typing on route change / HMR. Validate clean: 144 files, 1758 tests pass. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
|
Thanks for the LGTM. Picked up all three minor observations in 1. 2. Latent mid-debounce target switch (very low). Added a comment at the 3. Unmount mid-edit (intentional). Added a test that documents and pins the trade-off — typing → unmount → advance timers past the debounce window → assert Validate clean: 144 files, 1758 tests, all green. |
|
@claude review |
|
Claude finished @cliffhall's task in 1m 11s —— View job Claude PR Review — follow-up on
|
Closes #1361.
Problem
ServerSettingsModalis fully controlled byApp.tsx. Pending edits were stashed in auseRefand a 300ms-debounced PUT was scheduled — but refs don't trigger re-renders, so the form'svalueprop stayed stale until the PUT round-tripped anduseServersrefetched. Result: clicking "Add Header" / "Add Metadata" showed no input row, and typing in OAuth fields had no character echo, until ~500ms after each interaction.The bug landed in #1353 with the debounce pattern; #1356 made it user-visible by adding keychain writes to the per-keystroke PUT path.
Fix
Replace the ref with
useState. Every change re-renders immediately. Keep the 300ms debounce on the PUT side. Initialize the draft from the server-list entry only when the modal opens to a new target id — not on everyserverschange — so background SSE-triggered refreshes don't clobber the user's in-progress edits. Flush pending edits synchronously on modal close so the final keystrokes always land.Files
clients/web/src/App.tsx—pendingSettingsRef→settingsDraftstate;flushPendingSettings→sendSettingsUpdate+ inline flush at close;settingsModalTargetremoved (now derived inside the open-effect).Test plan
npm run validate— passes (format / lint / build / coverage / 1743 tests).clientIdlands in~/.mcp-inspector/mcp.jsonandclientSecretlands in the macOS Keychain undermcp-inspector:${serverId}:oauth-client-secret.🤖 Generated with Claude Code