@tonyclaw/agent-inspector 2.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/.output/cli.js +1611 -0
- package/.output/nitro.json +17 -0
- package/.output/public/assets/CompareDrawer-CU5ZrWcL.js +1 -0
- package/.output/public/assets/ProxyViewerContainer-pEBqVp1d.js +101 -0
- package/.output/public/assets/ReplayDialog-F58yNg5j.js +1 -0
- package/.output/public/assets/RequestAnatomy-C9lT0qE_.js +1 -0
- package/.output/public/assets/ResponseView-DHJq6bnz.js +1 -0
- package/.output/public/assets/StreamingChunkSequence-BTgfpFUT.js +1 -0
- package/.output/public/assets/_sessionId-DsNRbnNm.js +1 -0
- package/.output/public/assets/alibaba-TTwafVwX.svg +1 -0
- package/.output/public/assets/index-CpWG2hFn.css +1 -0
- package/.output/public/assets/index-DmBV8Gve.js +1 -0
- package/.output/public/assets/json-viewer-CZVYLR8j.js +14 -0
- package/.output/public/assets/main-DHs7FBK3.js +18 -0
- package/.output/public/assets/minimax-BPMzvuL-.jpeg +0 -0
- package/.output/public/assets/qwen-CONDcHqt.png +0 -0
- package/.output/public/assets/zhipuai-BPNAnxo-.svg +219 -0
- package/.output/server/_chunks/ssr-renderer.mjs +22 -0
- package/.output/server/_libs/@radix-ui/react-accessible-icon+[...].mjs +1 -0
- package/.output/server/_libs/@radix-ui/react-dismissable-layer+[...].mjs +210 -0
- package/.output/server/_libs/@radix-ui/react-navigation-menu+[...].mjs +2 -0
- package/.output/server/_libs/@radix-ui/react-one-time-password-field+[...].mjs +2 -0
- package/.output/server/_libs/@radix-ui/react-password-toggle-field+[...].mjs +2 -0
- package/.output/server/_libs/@radix-ui/react-use-callback-ref+[...].mjs +11 -0
- package/.output/server/_libs/@radix-ui/react-use-controllable-state+[...].mjs +69 -0
- package/.output/server/_libs/@radix-ui/react-use-effect-event+[...].mjs +1 -0
- package/.output/server/_libs/@radix-ui/react-use-escape-keydown+[...].mjs +17 -0
- package/.output/server/_libs/@radix-ui/react-use-is-hydrated+[...].mjs +1 -0
- package/.output/server/_libs/@radix-ui/react-use-layout-effect+[...].mjs +6 -0
- package/.output/server/_libs/@radix-ui/react-visually-hidden+[...].mjs +34 -0
- package/.output/server/_libs/ajv-formats.mjs +330 -0
- package/.output/server/_libs/ajv.mjs +11444 -0
- package/.output/server/_libs/aria-hidden.mjs +122 -0
- package/.output/server/_libs/atomically.mjs +152 -0
- package/.output/server/_libs/bail.mjs +8 -0
- package/.output/server/_libs/cfworker__json-schema.mjs +1 -0
- package/.output/server/_libs/character-entities.mjs +2130 -0
- package/.output/server/_libs/class-variance-authority.mjs +44 -0
- package/.output/server/_libs/clsx.mjs +16 -0
- package/.output/server/_libs/comma-separated-tokens.mjs +10 -0
- package/.output/server/_libs/conf.mjs +635 -0
- package/.output/server/_libs/cookie-es.mjs +58 -0
- package/.output/server/_libs/core-util-is.mjs +75 -0
- package/.output/server/_libs/croner.mjs +1 -0
- package/.output/server/_libs/crossws.mjs +1 -0
- package/.output/server/_libs/debounce-fn.mjs +69 -0
- package/.output/server/_libs/decode-named-character-reference+[...].mjs +8 -0
- package/.output/server/_libs/dequal.mjs +27 -0
- package/.output/server/_libs/detect-node-es.mjs +1 -0
- package/.output/server/_libs/devlop.mjs +8 -0
- package/.output/server/_libs/diff.mjs +320 -0
- package/.output/server/_libs/dot-prop.mjs +265 -0
- package/.output/server/_libs/env-paths.mjs +57 -0
- package/.output/server/_libs/estree-util-is-identifier-name.mjs +11 -0
- package/.output/server/_libs/extend.mjs +97 -0
- package/.output/server/_libs/fast-deep-equal.mjs +38 -0
- package/.output/server/_libs/fast-uri.mjs +812 -0
- package/.output/server/_libs/floating-ui__core.mjs +725 -0
- package/.output/server/_libs/floating-ui__dom.mjs +622 -0
- package/.output/server/_libs/floating-ui__react-dom.mjs +292 -0
- package/.output/server/_libs/floating-ui__utils.mjs +320 -0
- package/.output/server/_libs/get-nonce.mjs +9 -0
- package/.output/server/_libs/h3-v2.mjs +276 -0
- package/.output/server/_libs/h3.mjs +408 -0
- package/.output/server/_libs/hast-util-to-jsx-runtime.mjs +388 -0
- package/.output/server/_libs/hast-util-whitespace.mjs +10 -0
- package/.output/server/_libs/hookable.mjs +1 -0
- package/.output/server/_libs/html-url-attributes.mjs +26 -0
- package/.output/server/_libs/immediate.mjs +74 -0
- package/.output/server/_libs/inherits.mjs +50 -0
- package/.output/server/_libs/inline-style-parser.mjs +142 -0
- package/.output/server/_libs/is-plain-obj.mjs +10 -0
- package/.output/server/_libs/isarray.mjs +14 -0
- package/.output/server/_libs/isbot.mjs +20 -0
- package/.output/server/_libs/json-schema-traverse.mjs +180 -0
- package/.output/server/_libs/jszip.mjs +3051 -0
- package/.output/server/_libs/lie.mjs +273 -0
- package/.output/server/_libs/lucide-react.mjs +492 -0
- package/.output/server/_libs/mdast-util-from-markdown.mjs +717 -0
- package/.output/server/_libs/mdast-util-to-hast.mjs +710 -0
- package/.output/server/_libs/mdast-util-to-string.mjs +38 -0
- package/.output/server/_libs/micromark-core-commonmark.mjs +2259 -0
- package/.output/server/_libs/micromark-factory-destination.mjs +94 -0
- package/.output/server/_libs/micromark-factory-label.mjs +63 -0
- package/.output/server/_libs/micromark-factory-space.mjs +24 -0
- package/.output/server/_libs/micromark-factory-title.mjs +65 -0
- package/.output/server/_libs/micromark-factory-whitespace.mjs +22 -0
- package/.output/server/_libs/micromark-util-character.mjs +44 -0
- package/.output/server/_libs/micromark-util-chunked.mjs +36 -0
- package/.output/server/_libs/micromark-util-classify-character+[...].mjs +12 -0
- package/.output/server/_libs/micromark-util-combine-extensions+[...].mjs +41 -0
- package/.output/server/_libs/micromark-util-decode-numeric-character-reference+[...].mjs +19 -0
- package/.output/server/_libs/micromark-util-decode-string.mjs +21 -0
- package/.output/server/_libs/micromark-util-encode.mjs +1 -0
- package/.output/server/_libs/micromark-util-html-tag-name.mjs +69 -0
- package/.output/server/_libs/micromark-util-normalize-identifier+[...].mjs +6 -0
- package/.output/server/_libs/micromark-util-resolve-all.mjs +15 -0
- package/.output/server/_libs/micromark-util-sanitize-uri.mjs +41 -0
- package/.output/server/_libs/micromark-util-subtokenize.mjs +346 -0
- package/.output/server/_libs/micromark.mjs +906 -0
- package/.output/server/_libs/mimic-function.mjs +47 -0
- package/.output/server/_libs/modelcontextprotocol__server.mjs +9738 -0
- package/.output/server/_libs/ocache.mjs +1 -0
- package/.output/server/_libs/ohash.mjs +1 -0
- package/.output/server/_libs/pako.mjs +4223 -0
- package/.output/server/_libs/process-nextick-args.mjs +48 -0
- package/.output/server/_libs/property-information.mjs +1209 -0
- package/.output/server/_libs/radix-ui.mjs +1 -0
- package/.output/server/_libs/radix-ui__number.mjs +6 -0
- package/.output/server/_libs/radix-ui__primitive.mjs +11 -0
- package/.output/server/_libs/radix-ui__react-accordion.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-alert-dialog.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-arrow.mjs +23 -0
- package/.output/server/_libs/radix-ui__react-aspect-ratio.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-avatar.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-checkbox.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-collapsible.mjs +144 -0
- package/.output/server/_libs/radix-ui__react-collection.mjs +69 -0
- package/.output/server/_libs/radix-ui__react-compose-refs.mjs +39 -0
- package/.output/server/_libs/radix-ui__react-context-menu.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-context.mjs +78 -0
- package/.output/server/_libs/radix-ui__react-dialog.mjs +325 -0
- package/.output/server/_libs/radix-ui__react-direction.mjs +9 -0
- package/.output/server/_libs/radix-ui__react-dropdown-menu.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-focus-guards.mjs +29 -0
- package/.output/server/_libs/radix-ui__react-focus-scope.mjs +206 -0
- package/.output/server/_libs/radix-ui__react-form.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-hover-card.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-id.mjs +14 -0
- package/.output/server/_libs/radix-ui__react-label.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-menu.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-menubar.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-popover.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-popper.mjs +286 -0
- package/.output/server/_libs/radix-ui__react-portal.mjs +16 -0
- package/.output/server/_libs/radix-ui__react-presence.mjs +128 -0
- package/.output/server/_libs/radix-ui__react-primitive.mjs +42 -0
- package/.output/server/_libs/radix-ui__react-progress.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-radio-group.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-roving-focus.mjs +224 -0
- package/.output/server/_libs/radix-ui__react-scroll-area.mjs +721 -0
- package/.output/server/_libs/radix-ui__react-select.mjs +1163 -0
- package/.output/server/_libs/radix-ui__react-separator.mjs +28 -0
- package/.output/server/_libs/radix-ui__react-slider.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-slot.mjs +99 -0
- package/.output/server/_libs/radix-ui__react-switch.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-tabs.mjs +189 -0
- package/.output/server/_libs/radix-ui__react-toast.mjs +2 -0
- package/.output/server/_libs/radix-ui__react-toggle-group.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-toggle.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-toolbar.mjs +1 -0
- package/.output/server/_libs/radix-ui__react-tooltip.mjs +495 -0
- package/.output/server/_libs/radix-ui__react-use-previous.mjs +14 -0
- package/.output/server/_libs/radix-ui__react-use-size.mjs +39 -0
- package/.output/server/_libs/react-dom.mjs +10781 -0
- package/.output/server/_libs/react-markdown.mjs +147 -0
- package/.output/server/_libs/react-remove-scroll-bar.mjs +82 -0
- package/.output/server/_libs/react-remove-scroll.mjs +328 -0
- package/.output/server/_libs/react-style-singleton.mjs +69 -0
- package/.output/server/_libs/react.mjs +515 -0
- package/.output/server/_libs/readable-stream.mjs +1518 -0
- package/.output/server/_libs/remark-parse.mjs +19 -0
- package/.output/server/_libs/remark-rehype.mjs +21 -0
- package/.output/server/_libs/rou3.mjs +14 -0
- package/.output/server/_libs/safe-buffer.mjs +64 -0
- package/.output/server/_libs/semver.mjs +1938 -0
- package/.output/server/_libs/seroval-plugins.mjs +58 -0
- package/.output/server/_libs/seroval.mjs +1765 -0
- package/.output/server/_libs/setimmediate.mjs +152 -0
- package/.output/server/_libs/space-separated-tokens.mjs +6 -0
- package/.output/server/_libs/srvx.mjs +1029 -0
- package/.output/server/_libs/stubborn-fs.mjs +91 -0
- package/.output/server/_libs/stubborn-utils.mjs +66 -0
- package/.output/server/_libs/style-to-js.mjs +72 -0
- package/.output/server/_libs/style-to-object.mjs +38 -0
- package/.output/server/_libs/swr.mjs +939 -0
- package/.output/server/_libs/tailwind-merge.mjs +3010 -0
- package/.output/server/_libs/tanstack__history.mjs +217 -0
- package/.output/server/_libs/tanstack__react-router.mjs +1480 -0
- package/.output/server/_libs/tanstack__react-store.mjs +1 -0
- package/.output/server/_libs/tanstack__react-virtual.mjs +44 -0
- package/.output/server/_libs/tanstack__router-core.mjs +4827 -0
- package/.output/server/_libs/tanstack__store.mjs +1 -0
- package/.output/server/_libs/tanstack__virtual-core.mjs +1225 -0
- package/.output/server/_libs/tiny-invariant.mjs +12 -0
- package/.output/server/_libs/tiny-warning.mjs +5 -0
- package/.output/server/_libs/trim-lines.mjs +41 -0
- package/.output/server/_libs/trough.mjs +85 -0
- package/.output/server/_libs/tslib.mjs +1 -0
- package/.output/server/_libs/ufo.mjs +54 -0
- package/.output/server/_libs/uint8array-extras.mjs +69 -0
- package/.output/server/_libs/ungap__structured-clone.mjs +212 -0
- package/.output/server/_libs/unified.mjs +661 -0
- package/.output/server/_libs/unist-util-is.mjs +100 -0
- package/.output/server/_libs/unist-util-position.mjs +27 -0
- package/.output/server/_libs/unist-util-stringify-position.mjs +27 -0
- package/.output/server/_libs/unist-util-visit-parents.mjs +82 -0
- package/.output/server/_libs/unist-util-visit.mjs +24 -0
- package/.output/server/_libs/unstorage.mjs +1 -0
- package/.output/server/_libs/use-callback-ref.mjs +66 -0
- package/.output/server/_libs/use-sidecar.mjs +106 -0
- package/.output/server/_libs/use-sync-external-store.mjs +64 -0
- package/.output/server/_libs/util-deprecate.mjs +12 -0
- package/.output/server/_libs/vfile-message.mjs +138 -0
- package/.output/server/_libs/vfile.mjs +467 -0
- package/.output/server/_libs/when-exit.mjs +53 -0
- package/.output/server/_libs/zod.mjs +4524 -0
- package/.output/server/_sessionId-wMLPvC5g.mjs +123 -0
- package/.output/server/_ssr/CompareDrawer-BU4V0uVf.mjs +1041 -0
- package/.output/server/_ssr/ProxyViewerContainer-BnRwFEnn.mjs +5972 -0
- package/.output/server/_ssr/ReplayDialog-C7dn9pd_.mjs +322 -0
- package/.output/server/_ssr/RequestAnatomy-C1rWpe9-.mjs +353 -0
- package/.output/server/_ssr/ResponseView-hGpPaYsf.mjs +602 -0
- package/.output/server/_ssr/StreamingChunkSequence-BRWI1r_G.mjs +302 -0
- package/.output/server/_ssr/index-BKURLVPz.mjs +118 -0
- package/.output/server/_ssr/index.mjs +1184 -0
- package/.output/server/_ssr/json-viewer-BBd2DtQP.mjs +515 -0
- package/.output/server/_ssr/router-BcZ0D6AB.mjs +6317 -0
- package/.output/server/_ssr/start-HYkvq4Ni.mjs +4 -0
- package/.output/server/_tanstack-start-manifest_v-1y8ZVxRI.mjs +4 -0
- package/.output/server/index.mjs +436 -0
- package/.output/server/node_modules/tslib/modules/index.js +70 -0
- package/.output/server/node_modules/tslib/modules/package.json +3 -0
- package/.output/server/node_modules/tslib/package.json +47 -0
- package/.output/server/node_modules/tslib/tslib.js +484 -0
- package/.output/server/package.json +9 -0
- package/LICENSE +21 -0
- package/README.md +52 -0
- package/package.json +110 -0
- package/src/assets/favicon.svg +31 -0
- package/src/assets/logos/alibaba.svg +1 -0
- package/src/assets/logos/anthropic.svg +1 -0
- package/src/assets/logos/claude-code.svg +4 -0
- package/src/assets/logos/deepseek.svg +1 -0
- package/src/assets/logos/mcp.png +0 -0
- package/src/assets/logos/minimax.jpeg +0 -0
- package/src/assets/logos/openai.svg +1 -0
- package/src/assets/logos/opencode.svg +4 -0
- package/src/assets/logos/qwen.png +0 -0
- package/src/assets/logos/zhipuai.svg +219 -0
- package/src/cli/detect-tools.ts +147 -0
- package/src/cli/doctor.ts +521 -0
- package/src/cli/onboard.ts +224 -0
- package/src/cli/templates/command-onboard.ts +17 -0
- package/src/cli/templates/skill-onboard.ts +547 -0
- package/src/cli.ts +345 -0
- package/src/components/OnboardingBanner.tsx +67 -0
- package/src/components/ProxyViewer.tsx +545 -0
- package/src/components/ProxyViewerContainer.tsx +363 -0
- package/src/components/providers/ImportWizardDialog.tsx +349 -0
- package/src/components/providers/ProviderCard.tsx +474 -0
- package/src/components/providers/ProviderForm.tsx +494 -0
- package/src/components/providers/ProviderLogo.tsx +117 -0
- package/src/components/providers/ProvidersPanel.tsx +619 -0
- package/src/components/providers/SettingsDialog.tsx +202 -0
- package/src/components/proxy-viewer/CompareDrawer.tsx +893 -0
- package/src/components/proxy-viewer/ConversationGroup.tsx +107 -0
- package/src/components/proxy-viewer/ConversationHeader.tsx +300 -0
- package/src/components/proxy-viewer/LogEntry.tsx +543 -0
- package/src/components/proxy-viewer/LogEntryHeader.tsx +501 -0
- package/src/components/proxy-viewer/ReplayDialog.tsx +218 -0
- package/src/components/proxy-viewer/ResponseView.tsx +171 -0
- package/src/components/proxy-viewer/StreamingChunkSequence.tsx +188 -0
- package/src/components/proxy-viewer/ThreadConnector.tsx +136 -0
- package/src/components/proxy-viewer/TurnGroup.tsx +337 -0
- package/src/components/proxy-viewer/anatomy/RequestAnatomy.tsx +98 -0
- package/src/components/proxy-viewer/anatomy/SegmentBar.tsx +196 -0
- package/src/components/proxy-viewer/anatomy/tokenEstimate.ts +53 -0
- package/src/components/proxy-viewer/anatomy/types.ts +39 -0
- package/src/components/proxy-viewer/anatomy/useAnatomyJump.ts +114 -0
- package/src/components/proxy-viewer/cacheTrend.ts +50 -0
- package/src/components/proxy-viewer/diff/DiffView.tsx +321 -0
- package/src/components/proxy-viewer/diff/computeDiff.ts +178 -0
- package/src/components/proxy-viewer/diff/index.ts +3 -0
- package/src/components/proxy-viewer/formats/anthropic/ContentBlocks.tsx +157 -0
- package/src/components/proxy-viewer/formats/anthropic/ResponseView.tsx +66 -0
- package/src/components/proxy-viewer/formats/anthropic/thinkingExtract.ts +21 -0
- package/src/components/proxy-viewer/formats/index.tsx +33 -0
- package/src/components/proxy-viewer/formats/openai/ResponseView.tsx +170 -0
- package/src/components/proxy-viewer/index.ts +9 -0
- package/src/components/proxy-viewer/lazy.ts +37 -0
- package/src/components/proxy-viewer/log-formats/anthropic.ts +194 -0
- package/src/components/proxy-viewer/log-formats/index.ts +23 -0
- package/src/components/proxy-viewer/log-formats/openai.ts +167 -0
- package/src/components/proxy-viewer/log-formats/types.ts +40 -0
- package/src/components/proxy-viewer/log-formats/unknown.ts +18 -0
- package/src/components/proxy-viewer/logEntryVisibility.ts +39 -0
- package/src/components/proxy-viewer/requestDiff.ts +277 -0
- package/src/components/proxy-viewer/useCopyFeedback.ts +36 -0
- package/src/components/proxy-viewer/useKeyboardNavigation.ts +190 -0
- package/src/components/proxy-viewer/viewerState.ts +66 -0
- package/src/components/ui/badge.tsx +47 -0
- package/src/components/ui/button.tsx +47 -0
- package/src/components/ui/collapsible.tsx +21 -0
- package/src/components/ui/confirm-dialog.tsx +51 -0
- package/src/components/ui/crab-logo.tsx +95 -0
- package/src/components/ui/crab-variants.tsx +467 -0
- package/src/components/ui/dialog.tsx +129 -0
- package/src/components/ui/json-expansion-button.tsx +56 -0
- package/src/components/ui/json-viewer-bulk.ts +97 -0
- package/src/components/ui/json-viewer.tsx +494 -0
- package/src/components/ui/mcp-logo.tsx +20 -0
- package/src/components/ui/scroll-area.tsx +54 -0
- package/src/components/ui/select.tsx +178 -0
- package/src/components/ui/separator.tsx +28 -0
- package/src/components/ui/tabs.tsx +88 -0
- package/src/components/ui/tooltip.tsx +51 -0
- package/src/index.css +11 -0
- package/src/knowledge/candidateStore.ts +63 -0
- package/src/knowledge/distiller.ts +98 -0
- package/src/knowledge/openclawClient.ts +118 -0
- package/src/knowledge/redactor.ts +80 -0
- package/src/knowledge/types.ts +84 -0
- package/src/lib/apiClient.ts +49 -0
- package/src/lib/export-logs.ts +51 -0
- package/src/lib/mask.ts +4 -0
- package/src/lib/objectUtils.ts +22 -0
- package/src/lib/providerContract.ts +26 -0
- package/src/lib/providerTestContract.ts +107 -0
- package/src/lib/runtimeConfig.ts +25 -0
- package/src/lib/serverPort.ts +41 -0
- package/src/lib/sessionUrl.ts +44 -0
- package/src/lib/stopReason.ts +58 -0
- package/src/lib/useOnboarding.ts +80 -0
- package/src/lib/useProviders.ts +30 -0
- package/src/lib/useStripConfig.ts +108 -0
- package/src/lib/utils.ts +21 -0
- package/src/mcp/loopback.ts +76 -0
- package/src/mcp/previewExtractor.ts +166 -0
- package/src/mcp/server.ts +396 -0
- package/src/mcp/toolHandlers.ts +341 -0
- package/src/proxy/chunkStorage.ts +112 -0
- package/src/proxy/claudeCodeStrip.ts +99 -0
- package/src/proxy/config.ts +172 -0
- package/src/proxy/constants.ts +47 -0
- package/src/proxy/dataDir.ts +86 -0
- package/src/proxy/formats/anthropic/anthropicProvider.ts +75 -0
- package/src/proxy/formats/anthropic/handler.ts +71 -0
- package/src/proxy/formats/anthropic/index.ts +14 -0
- package/src/proxy/formats/anthropic/register.ts +4 -0
- package/src/proxy/formats/anthropic/schemas.ts +237 -0
- package/src/proxy/formats/anthropic/stream.ts +205 -0
- package/src/proxy/formats/handler.ts +46 -0
- package/src/proxy/formats/index.ts +12 -0
- package/src/proxy/formats/jsonSchema.ts +36 -0
- package/src/proxy/formats/openai/alibabaProvider.ts +38 -0
- package/src/proxy/formats/openai/handler.ts +96 -0
- package/src/proxy/formats/openai/index.ts +25 -0
- package/src/proxy/formats/openai/provider.ts +50 -0
- package/src/proxy/formats/openai/register.ts +4 -0
- package/src/proxy/formats/openai/schemas.ts +187 -0
- package/src/proxy/formats/openai/stream.ts +206 -0
- package/src/proxy/formats/protocol.ts +50 -0
- package/src/proxy/formats/providerRegistry.ts +51 -0
- package/src/proxy/formats/providers/index.ts +3 -0
- package/src/proxy/formats/registry.ts +66 -0
- package/src/proxy/handler.ts +334 -0
- package/src/proxy/logFinalizer.ts +305 -0
- package/src/proxy/logFinalizer.worker.ts +24 -0
- package/src/proxy/logIndex.ts +268 -0
- package/src/proxy/logger.ts +179 -0
- package/src/proxy/openaiOrphanToolStrip.ts +142 -0
- package/src/proxy/providerImporters.ts +491 -0
- package/src/proxy/providers.ts +613 -0
- package/src/proxy/schemas.ts +209 -0
- package/src/proxy/sessionProcess.ts +140 -0
- package/src/proxy/sessionRuntime.ts +85 -0
- package/src/proxy/sessionSupervisor.ts +283 -0
- package/src/proxy/sessionWorkerEntry.ts +26 -0
- package/src/proxy/socketTracker.ts +255 -0
- package/src/proxy/store.ts +412 -0
- package/src/proxy/upstream.ts +90 -0
- package/src/router.tsx +16 -0
- package/src/routes/__root.tsx +45 -0
- package/src/routes/api/config.paths.ts +14 -0
- package/src/routes/api/config.ts +53 -0
- package/src/routes/api/health.ts +15 -0
- package/src/routes/api/knowledge.candidates.$candidateId.promote.ts +32 -0
- package/src/routes/api/knowledge.candidates.ts +10 -0
- package/src/routes/api/knowledge.project-context.ts +18 -0
- package/src/routes/api/knowledge.search.ts +31 -0
- package/src/routes/api/knowledge.sessions.$sessionId.candidates.ts +16 -0
- package/src/routes/api/logs.$id.chunks.ts +36 -0
- package/src/routes/api/logs.$id.replay.ts +191 -0
- package/src/routes/api/logs.$id.ts +22 -0
- package/src/routes/api/logs.stream.ts +74 -0
- package/src/routes/api/logs.ts +59 -0
- package/src/routes/api/mcp.ts +25 -0
- package/src/routes/api/models.ts +10 -0
- package/src/routes/api/providers.$providerId.test.log.ts +293 -0
- package/src/routes/api/providers.$providerId.ts +50 -0
- package/src/routes/api/providers.export.ts +26 -0
- package/src/routes/api/providers.import.ts +47 -0
- package/src/routes/api/providers.scan.ts +23 -0
- package/src/routes/api/providers.ts +45 -0
- package/src/routes/api/sessions.ts +17 -0
- package/src/routes/index.tsx +6 -0
- package/src/routes/proxy/$.ts +15 -0
- package/src/routes/session/$sessionId.tsx +23 -0
- package/styles/globals.css +188 -0
|
@@ -0,0 +1,167 @@
|
|
|
1
|
+
import { OpenAIRequestSchema, parseOpenAIResponse } from "../../../proxy/formats/openai/schemas";
|
|
2
|
+
import type { AnatomySegment } from "../anatomy/types";
|
|
3
|
+
import { countCharacters, estimateTokens } from "../anatomy/tokenEstimate";
|
|
4
|
+
import type { LogFormatAdapter } from "./types";
|
|
5
|
+
import { emptyRequestAnalysis, EMPTY_RESPONSE_ANALYSIS } from "./types";
|
|
6
|
+
|
|
7
|
+
type OpenAIMessageContent = string | ReadonlyArray<Record<string, unknown>> | null | undefined;
|
|
8
|
+
|
|
9
|
+
/** Flatten an OpenAI `content` field to its text representation. */
|
|
10
|
+
function contentToText(content: unknown): string {
|
|
11
|
+
if (typeof content === "string") return content;
|
|
12
|
+
if (!Array.isArray(content)) return "";
|
|
13
|
+
const parts: string[] = [];
|
|
14
|
+
for (const block of content) {
|
|
15
|
+
if (block === null || typeof block !== "object") continue;
|
|
16
|
+
// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
|
|
17
|
+
const b = block as Record<string, unknown>;
|
|
18
|
+
const type = b.type;
|
|
19
|
+
if (type === "text" && typeof b.text === "string") {
|
|
20
|
+
parts.push(b.text);
|
|
21
|
+
} else if (type === "image_url") {
|
|
22
|
+
parts.push("[image]");
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
return parts.join("\n");
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
/** Flatten an OpenAI message to text — content plus optional tool calls plus optional reasoning. */
|
|
29
|
+
function messageToText(message: Record<string, unknown>): string {
|
|
30
|
+
const parts: string[] = [];
|
|
31
|
+
const text = contentToText(message.content);
|
|
32
|
+
if (text.length > 0) parts.push(text);
|
|
33
|
+
const reasoning =
|
|
34
|
+
typeof message.reasoning_content === "string"
|
|
35
|
+
? message.reasoning_content
|
|
36
|
+
: typeof message.thinking === "string"
|
|
37
|
+
? message.thinking
|
|
38
|
+
: typeof message.think === "string"
|
|
39
|
+
? message.think
|
|
40
|
+
: "";
|
|
41
|
+
if (reasoning.length > 0) parts.push(reasoning);
|
|
42
|
+
const toolCalls = message.tool_calls;
|
|
43
|
+
if (Array.isArray(toolCalls)) {
|
|
44
|
+
for (const call of toolCalls) {
|
|
45
|
+
if (call === null || typeof call !== "object") continue;
|
|
46
|
+
// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
|
|
47
|
+
const fn = (call as { function?: { name?: unknown; arguments?: unknown } }).function;
|
|
48
|
+
if (fn === undefined) continue;
|
|
49
|
+
const name = typeof fn.name === "string" ? fn.name : "";
|
|
50
|
+
const args = typeof fn.arguments === "string" ? fn.arguments : "";
|
|
51
|
+
parts.push(`${name} ${args}`.trim());
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
return parts.join("\n");
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
/** Flatten an OpenAI `tools` array to its schema text. */
|
|
58
|
+
function toolsToText(tools: unknown): string {
|
|
59
|
+
if (!Array.isArray(tools)) return "";
|
|
60
|
+
const parts: string[] = [];
|
|
61
|
+
for (const tool of tools) {
|
|
62
|
+
if (tool === null || typeof tool !== "object") continue;
|
|
63
|
+
// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
|
|
64
|
+
const t = tool as {
|
|
65
|
+
function?: { name?: unknown; description?: unknown; parameters?: unknown };
|
|
66
|
+
};
|
|
67
|
+
const fn = t.function;
|
|
68
|
+
if (fn === undefined) continue;
|
|
69
|
+
if (typeof fn.name === "string") parts.push(fn.name);
|
|
70
|
+
if (typeof fn.description === "string") parts.push(fn.description);
|
|
71
|
+
if (fn.parameters !== undefined) parts.push(JSON.stringify(fn.parameters));
|
|
72
|
+
}
|
|
73
|
+
return parts.join("\n");
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
function segment(
|
|
77
|
+
role: AnatomySegment["role"],
|
|
78
|
+
label: string,
|
|
79
|
+
text: string,
|
|
80
|
+
path: string,
|
|
81
|
+
): AnatomySegment {
|
|
82
|
+
return {
|
|
83
|
+
role,
|
|
84
|
+
label,
|
|
85
|
+
text,
|
|
86
|
+
size: estimateTokens(text),
|
|
87
|
+
characters: countCharacters(text),
|
|
88
|
+
path,
|
|
89
|
+
};
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
export const openAILogFormatAdapter: LogFormatAdapter = {
|
|
93
|
+
format: "openai",
|
|
94
|
+
|
|
95
|
+
analyzeRequest(rawBody) {
|
|
96
|
+
if (rawBody === null) return emptyRequestAnalysis(rawBody);
|
|
97
|
+
try {
|
|
98
|
+
const result = OpenAIRequestSchema.safeParse(JSON.parse(rawBody));
|
|
99
|
+
if (!result.success) return emptyRequestAnalysis(rawBody);
|
|
100
|
+
return {
|
|
101
|
+
parsed: result.data,
|
|
102
|
+
comparisonValue: result.data,
|
|
103
|
+
messageCount: result.data.messages.length,
|
|
104
|
+
toolCount:
|
|
105
|
+
result.data.tools !== undefined && result.data.tools.length > 0
|
|
106
|
+
? result.data.tools.length
|
|
107
|
+
: null,
|
|
108
|
+
};
|
|
109
|
+
} catch {
|
|
110
|
+
return emptyRequestAnalysis(rawBody);
|
|
111
|
+
}
|
|
112
|
+
},
|
|
113
|
+
|
|
114
|
+
analyzeResponse(responseText) {
|
|
115
|
+
if (responseText === null) return EMPTY_RESPONSE_ANALYSIS;
|
|
116
|
+
const parsed = parseOpenAIResponse(responseText);
|
|
117
|
+
if (parsed === null) return EMPTY_RESPONSE_ANALYSIS;
|
|
118
|
+
const toolNames =
|
|
119
|
+
parsed.choices[0]?.message?.tool_calls
|
|
120
|
+
?.map((toolCall) => toolCall.function?.name ?? "")
|
|
121
|
+
.filter((name) => name !== "") ?? [];
|
|
122
|
+
return {
|
|
123
|
+
parsed,
|
|
124
|
+
toolNames: toolNames.length > 0 ? toolNames : null,
|
|
125
|
+
};
|
|
126
|
+
},
|
|
127
|
+
|
|
128
|
+
anatomySegments(parsed) {
|
|
129
|
+
if (parsed === null || typeof parsed !== "object") return null;
|
|
130
|
+
// We deliberately skip OpenAIRequestSchema validation here for the
|
|
131
|
+
// same reason as the Anthropic adapter: the Anatomy view should
|
|
132
|
+
// render even when the body shape is slightly off-schema. We only
|
|
133
|
+
// need the top-level `messages` and `tools` arrays.
|
|
134
|
+
// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
|
|
135
|
+
const body = parsed as { messages?: unknown; tools?: unknown };
|
|
136
|
+
const segments: AnatomySegment[] = [];
|
|
137
|
+
|
|
138
|
+
// Promote a leading role: "system" message into a "system" segment.
|
|
139
|
+
// Spec: derive system role from a leading role: "system" message, then
|
|
140
|
+
// a segment per remaining message. We still keep the system message in
|
|
141
|
+
// the segment list (not skipped) when there is no separate system field
|
|
142
|
+
// to compare against — the leading role: "system" message IS the system
|
|
143
|
+
// content for OpenAI Chat Completions. If a separate `system` field
|
|
144
|
+
// existed and matched the message, we would dedupe, but Chat
|
|
145
|
+
// Completions has no such field, so we never have to skip.
|
|
146
|
+
if (Array.isArray(body.messages)) {
|
|
147
|
+
body.messages.forEach((message, index) => {
|
|
148
|
+
if (message === null || typeof message !== "object") return;
|
|
149
|
+
// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
|
|
150
|
+
const m = message as { role?: unknown; content?: unknown };
|
|
151
|
+
const role =
|
|
152
|
+
m.role === "user" || m.role === "assistant" || m.role === "system" || m.role === "tool"
|
|
153
|
+
? m.role
|
|
154
|
+
: "user";
|
|
155
|
+
const text = messageToText(m);
|
|
156
|
+
segments.push(segment(role, `[${index}] ${role}`, text, `/messages/${index}`));
|
|
157
|
+
});
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
if (Array.isArray(body.tools) && body.tools.length > 0) {
|
|
161
|
+
const text = toolsToText(body.tools);
|
|
162
|
+
segments.push(segment("tools", "tools", text, "/tools"));
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
return segments.length > 0 ? segments : null;
|
|
166
|
+
},
|
|
167
|
+
};
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
import type { RequestFormat } from "../../../proxy/schemas";
|
|
2
|
+
import type { AnatomySegment } from "../anatomy/types";
|
|
3
|
+
|
|
4
|
+
export type RequestAnalysis = {
|
|
5
|
+
parsed: unknown | null;
|
|
6
|
+
comparisonValue: unknown;
|
|
7
|
+
messageCount: number | null;
|
|
8
|
+
toolCount: number | null;
|
|
9
|
+
};
|
|
10
|
+
|
|
11
|
+
export type ResponseAnalysis = {
|
|
12
|
+
parsed: unknown | null;
|
|
13
|
+
toolNames: string[] | null;
|
|
14
|
+
};
|
|
15
|
+
|
|
16
|
+
export type LogFormatAdapter = {
|
|
17
|
+
readonly format: RequestFormat;
|
|
18
|
+
analyzeRequest(rawBody: string | null): RequestAnalysis;
|
|
19
|
+
analyzeResponse(responseText: string | null): ResponseAnalysis;
|
|
20
|
+
/**
|
|
21
|
+
* Derive the ordered list of segments shown in the Anatomy tab for a
|
|
22
|
+
* parsed request body. Returns `null` when the body is `null` or fails
|
|
23
|
+
* the format's schema (e.g. unknown format).
|
|
24
|
+
*/
|
|
25
|
+
anatomySegments(parsed: unknown): AnatomySegment[] | null;
|
|
26
|
+
};
|
|
27
|
+
|
|
28
|
+
export function emptyRequestAnalysis(rawBody: string | null): RequestAnalysis {
|
|
29
|
+
return {
|
|
30
|
+
parsed: null,
|
|
31
|
+
comparisonValue: rawBody,
|
|
32
|
+
messageCount: null,
|
|
33
|
+
toolCount: null,
|
|
34
|
+
};
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
export const EMPTY_RESPONSE_ANALYSIS: ResponseAnalysis = {
|
|
38
|
+
parsed: null,
|
|
39
|
+
toolNames: null,
|
|
40
|
+
};
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import type { LogFormatAdapter } from "./types";
|
|
2
|
+
import { emptyRequestAnalysis, EMPTY_RESPONSE_ANALYSIS } from "./types";
|
|
3
|
+
|
|
4
|
+
export const unknownLogFormatAdapter: LogFormatAdapter = {
|
|
5
|
+
format: "unknown",
|
|
6
|
+
|
|
7
|
+
analyzeRequest(rawBody) {
|
|
8
|
+
return emptyRequestAnalysis(rawBody);
|
|
9
|
+
},
|
|
10
|
+
|
|
11
|
+
analyzeResponse() {
|
|
12
|
+
return EMPTY_RESPONSE_ANALYSIS;
|
|
13
|
+
},
|
|
14
|
+
|
|
15
|
+
anatomySegments() {
|
|
16
|
+
return null;
|
|
17
|
+
},
|
|
18
|
+
};
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Pure visibility rule for the "Raw Request" tab. Extracted so it can be
|
|
3
|
+
* unit-tested without rendering React. The tab appears only when all three
|
|
4
|
+
* are true: anthropic format, full view mode, strip toggle enabled.
|
|
5
|
+
*/
|
|
6
|
+
export function shouldShowRawRequestTab(
|
|
7
|
+
apiFormat: string,
|
|
8
|
+
viewMode: "simple" | "full",
|
|
9
|
+
strip: boolean,
|
|
10
|
+
): boolean {
|
|
11
|
+
return apiFormat === "anthropic" && viewMode === "full" && strip;
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
/**
|
|
15
|
+
* Pure visibility rule for the "Diff with Raw" button in the Headers tab.
|
|
16
|
+
* The button only makes sense when the user is in full mode (where the
|
|
17
|
+
* `Raw Headers` tab is shown) and we actually captured raw headers.
|
|
18
|
+
*/
|
|
19
|
+
export function shouldShowHeadersDiffButton(
|
|
20
|
+
viewMode: "simple" | "full",
|
|
21
|
+
hasRawHeaders: boolean,
|
|
22
|
+
): boolean {
|
|
23
|
+
return viewMode === "full" && hasRawHeaders;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* Pure visibility rule for the "Diff with Raw" button in the Request tab.
|
|
28
|
+
* Mirrors the conditions for the `Raw Request` tab itself: full mode plus
|
|
29
|
+
* the strip toggle being on for an anthropic-format request. We also need
|
|
30
|
+
* an actual raw request body to diff against.
|
|
31
|
+
*/
|
|
32
|
+
export function shouldShowRequestDiffButton(
|
|
33
|
+
apiFormat: string,
|
|
34
|
+
viewMode: "simple" | "full",
|
|
35
|
+
strip: boolean,
|
|
36
|
+
hasRawRequest: boolean,
|
|
37
|
+
): boolean {
|
|
38
|
+
return apiFormat === "anthropic" && viewMode === "full" && strip && hasRawRequest;
|
|
39
|
+
}
|
|
@@ -0,0 +1,277 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Path-aligned JSON-tree diff helpers for the "compare two log requests" feature.
|
|
3
|
+
*
|
|
4
|
+
* Given two captured log Request payloads, we:
|
|
5
|
+
* 1. Normalize each payload (deep-clone, sort object keys, keep array order)
|
|
6
|
+
* so that non-semantic differences (key order, string whitespace) do not
|
|
7
|
+
* generate false positives.
|
|
8
|
+
* 2. Walk the two trees in lockstep, emitting a list of path-anchored
|
|
9
|
+
* `DiffOp`s. Equal subtrees collapse into a single op at the subtree
|
|
10
|
+
* root, so the renderer can choose to expand or hide them.
|
|
11
|
+
* 3. Emit ops in path-sorted order (depth-first, sibling order = object-key
|
|
12
|
+
* ascending then array-index ascending) so the renderer can lay them
|
|
13
|
+
* out linearly.
|
|
14
|
+
*
|
|
15
|
+
* No runtime dependencies. Mirrors the `cacheTrend.ts` pure-helper pattern.
|
|
16
|
+
*/
|
|
17
|
+
|
|
18
|
+
export type JsonPrimitive = string | number | boolean | null;
|
|
19
|
+
export type JsonValue = JsonPrimitive | JsonValue[] | { [key: string]: JsonValue };
|
|
20
|
+
|
|
21
|
+
export type JsonNode =
|
|
22
|
+
| { kind: "object"; value: Record<string, JsonNode> }
|
|
23
|
+
| { kind: "array"; value: JsonNode[] }
|
|
24
|
+
| { kind: "primitive"; value: JsonPrimitive };
|
|
25
|
+
|
|
26
|
+
export type DiffOp =
|
|
27
|
+
| { kind: "equal"; path: string; value: JsonNode }
|
|
28
|
+
| { kind: "added"; path: string; value: JsonNode }
|
|
29
|
+
| { kind: "removed"; path: string; value: JsonNode }
|
|
30
|
+
| {
|
|
31
|
+
kind: "changed";
|
|
32
|
+
path: string;
|
|
33
|
+
left: JsonNode;
|
|
34
|
+
right: JsonNode;
|
|
35
|
+
};
|
|
36
|
+
|
|
37
|
+
/** A single segment of a JSON path: an object key or an array index. */
|
|
38
|
+
export type PathSegment = string | number;
|
|
39
|
+
|
|
40
|
+
const ROOT_PATH = "";
|
|
41
|
+
|
|
42
|
+
/** Render a path segment list as a human-readable string.
|
|
43
|
+
*
|
|
44
|
+
* - Root (empty list) → `""` (no label; the renderer hides the gutter)
|
|
45
|
+
* - Object key → `.foo`
|
|
46
|
+
* - Array index → `[3]`
|
|
47
|
+
* - Top-level object key → `messages` (no leading dot)
|
|
48
|
+
*
|
|
49
|
+
* Examples:
|
|
50
|
+
* [] → ""
|
|
51
|
+
* ["messages"] → "messages"
|
|
52
|
+
* ["messages", 3] → "messages[3]"
|
|
53
|
+
* ["messages", 3, "content"] → "messages[3].content"
|
|
54
|
+
* ["messages", 3, "content", 0] → "messages[3].content[0]"
|
|
55
|
+
*/
|
|
56
|
+
export function formatPath(segments: PathSegment[]): string {
|
|
57
|
+
if (segments.length === 0) return ROOT_PATH;
|
|
58
|
+
let out = "";
|
|
59
|
+
for (let i = 0; i < segments.length; i++) {
|
|
60
|
+
const seg = segments[i];
|
|
61
|
+
if (seg === undefined) continue;
|
|
62
|
+
if (typeof seg === "number") {
|
|
63
|
+
out += `[${seg}]`;
|
|
64
|
+
} else if (i === 0) {
|
|
65
|
+
out += seg;
|
|
66
|
+
} else {
|
|
67
|
+
out += `.${seg}`;
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
return out;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
function isPlainObject(value: unknown): value is Record<string, unknown> {
|
|
74
|
+
return typeof value === "object" && value !== null && !Array.isArray(value);
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
/**
|
|
78
|
+
* Normalize a captured Request body for comparison.
|
|
79
|
+
*
|
|
80
|
+
* Captured Request bodies are sometimes already-parsed objects and sometimes
|
|
81
|
+
* JSON-encoded strings (depending on which capture path produced the log).
|
|
82
|
+
* We:
|
|
83
|
+
* 1. Try `JSON.parse` if the input is a string.
|
|
84
|
+
* 2. Deep-walk the value, building a fresh `JsonNode` tree.
|
|
85
|
+
* 3. Sort object keys lexicographically (key order is not semantically
|
|
86
|
+
* meaningful for most LLM SDKs but varies between SDK versions, so we
|
|
87
|
+
* canonicalize it).
|
|
88
|
+
* 4. Keep array order (semantically meaningful: `messages[]` order is
|
|
89
|
+
* conversation order; reordering it would silently destroy the diff).
|
|
90
|
+
* 5. Leave primitives alone (no whitespace trim — those can be meaningful
|
|
91
|
+
* in user-message text).
|
|
92
|
+
*
|
|
93
|
+
* Idempotent at the data level: re-normalizing the same shape (in any key
|
|
94
|
+
* order) yields the same tree.
|
|
95
|
+
*/
|
|
96
|
+
export function normalizeRequest(raw: unknown): JsonNode {
|
|
97
|
+
if (typeof raw === "string") {
|
|
98
|
+
try {
|
|
99
|
+
return toNode(JSON.parse(raw));
|
|
100
|
+
} catch {
|
|
101
|
+
// Unparseable string: wrap as a primitive string node so the diff can
|
|
102
|
+
// still operate on it (changed vs. another string, or added/removed
|
|
103
|
+
// vs. an object).
|
|
104
|
+
return { kind: "primitive", value: raw };
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
return toNode(raw);
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
function toNode(value: unknown): JsonNode {
|
|
111
|
+
if (value === null) return { kind: "primitive", value: null };
|
|
112
|
+
if (typeof value === "string") return { kind: "primitive", value };
|
|
113
|
+
if (typeof value === "number") return { kind: "primitive", value };
|
|
114
|
+
if (typeof value === "boolean") return { kind: "primitive", value };
|
|
115
|
+
if (Array.isArray(value)) {
|
|
116
|
+
return { kind: "array", value: value.map((v) => toNode(v)) };
|
|
117
|
+
}
|
|
118
|
+
if (isPlainObject(value)) {
|
|
119
|
+
const out: Record<string, JsonNode> = {};
|
|
120
|
+
for (const k of Object.keys(value).sort()) {
|
|
121
|
+
out[k] = toNode(value[k]);
|
|
122
|
+
}
|
|
123
|
+
return { kind: "object", value: out };
|
|
124
|
+
}
|
|
125
|
+
// Functions, symbols, bigints, undefined — treat as null in the diff model.
|
|
126
|
+
return { kind: "primitive", value: null };
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
/**
|
|
130
|
+
* Compute the path-aligned diff of two normalized trees.
|
|
131
|
+
*
|
|
132
|
+
* Emits `DiffOp[]` in path-sorted order. For each path:
|
|
133
|
+
* - if both trees have the same value, emit a single `equal` op at the
|
|
134
|
+
* deepest common ancestor (so equal subtrees collapse into one op);
|
|
135
|
+
* - if only the right has it, emit `added`;
|
|
136
|
+
* - if only the left has it, emit `removed`;
|
|
137
|
+
* - if both have it but it differs, emit `changed` (and recurse into
|
|
138
|
+
* objects/arrays so the user sees *which* field changed).
|
|
139
|
+
*
|
|
140
|
+
* Pure: does not mutate `left` or `right`.
|
|
141
|
+
*/
|
|
142
|
+
export function diffTrees(left: JsonNode, right: JsonNode): DiffOp[] {
|
|
143
|
+
const ops: DiffOp[] = [];
|
|
144
|
+
walk([], left, right, ops);
|
|
145
|
+
return ops;
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
function walk(segments: PathSegment[], left: JsonNode, right: JsonNode, out: DiffOp[]): void {
|
|
149
|
+
const path = formatPath(segments);
|
|
150
|
+
|
|
151
|
+
if (nodeEqual(left, right)) {
|
|
152
|
+
out.push({ kind: "equal", path, value: left });
|
|
153
|
+
return;
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
// Type mismatch or primitive change.
|
|
157
|
+
if (left.kind !== right.kind) {
|
|
158
|
+
out.push({ kind: "changed", path, left, right });
|
|
159
|
+
return;
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
if (left.kind === "primitive" && right.kind === "primitive") {
|
|
163
|
+
out.push({ kind: "changed", path, left, right });
|
|
164
|
+
return;
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
if (left.kind === "object" && right.kind === "object") {
|
|
168
|
+
const leftKeys = Object.keys(left.value);
|
|
169
|
+
const rightKeys = Object.keys(right.value);
|
|
170
|
+
const rightKeySet = new Set(rightKeys);
|
|
171
|
+
|
|
172
|
+
for (const k of leftKeys) {
|
|
173
|
+
const lChild = left.value[k];
|
|
174
|
+
if (lChild === undefined) continue;
|
|
175
|
+
if (!rightKeySet.has(k)) {
|
|
176
|
+
out.push({
|
|
177
|
+
kind: "removed",
|
|
178
|
+
path: formatPath([...segments, k]),
|
|
179
|
+
value: lChild,
|
|
180
|
+
});
|
|
181
|
+
} else {
|
|
182
|
+
const rChild = right.value[k];
|
|
183
|
+
if (rChild === undefined) continue;
|
|
184
|
+
walk([...segments, k], lChild, rChild, out);
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
for (const k of rightKeys) {
|
|
188
|
+
if (leftKeys.includes(k)) continue;
|
|
189
|
+
const rChild = right.value[k];
|
|
190
|
+
if (rChild === undefined) continue;
|
|
191
|
+
out.push({
|
|
192
|
+
kind: "added",
|
|
193
|
+
path: formatPath([...segments, k]),
|
|
194
|
+
value: rChild,
|
|
195
|
+
});
|
|
196
|
+
}
|
|
197
|
+
return;
|
|
198
|
+
}
|
|
199
|
+
|
|
200
|
+
if (left.kind === "array" && right.kind === "array") {
|
|
201
|
+
const minLen = Math.min(left.value.length, right.value.length);
|
|
202
|
+
for (let i = 0; i < minLen; i++) {
|
|
203
|
+
const lChild = left.value[i];
|
|
204
|
+
const rChild = right.value[i];
|
|
205
|
+
if (lChild === undefined || rChild === undefined) continue;
|
|
206
|
+
walk([...segments, i], lChild, rChild, out);
|
|
207
|
+
}
|
|
208
|
+
for (let i = minLen; i < right.value.length; i++) {
|
|
209
|
+
const rChild = right.value[i];
|
|
210
|
+
if (rChild === undefined) continue;
|
|
211
|
+
out.push({
|
|
212
|
+
kind: "added",
|
|
213
|
+
path: formatPath([...segments, i]),
|
|
214
|
+
value: rChild,
|
|
215
|
+
});
|
|
216
|
+
}
|
|
217
|
+
for (let i = minLen; i < left.value.length; i++) {
|
|
218
|
+
const lChild = left.value[i];
|
|
219
|
+
if (lChild === undefined) continue;
|
|
220
|
+
out.push({
|
|
221
|
+
kind: "removed",
|
|
222
|
+
path: formatPath([...segments, i]),
|
|
223
|
+
value: lChild,
|
|
224
|
+
});
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
function nodeEqual(a: JsonNode, b: JsonNode): boolean {
|
|
230
|
+
if (a.kind !== b.kind) return false;
|
|
231
|
+
if (a.kind === "primitive" && b.kind === "primitive") {
|
|
232
|
+
return a.value === b.value;
|
|
233
|
+
}
|
|
234
|
+
if (a.kind === "array" && b.kind === "array") {
|
|
235
|
+
if (a.value.length !== b.value.length) return false;
|
|
236
|
+
for (let i = 0; i < a.value.length; i++) {
|
|
237
|
+
const ai = a.value[i];
|
|
238
|
+
const bi = b.value[i];
|
|
239
|
+
if (ai === undefined || bi === undefined) return false;
|
|
240
|
+
if (!nodeEqual(ai, bi)) return false;
|
|
241
|
+
}
|
|
242
|
+
return true;
|
|
243
|
+
}
|
|
244
|
+
if (a.kind === "object" && b.kind === "object") {
|
|
245
|
+
const aKeys = Object.keys(a.value);
|
|
246
|
+
const bKeys = Object.keys(b.value);
|
|
247
|
+
if (aKeys.length !== bKeys.length) return false;
|
|
248
|
+
for (const k of aKeys) {
|
|
249
|
+
const av = a.value[k];
|
|
250
|
+
const bv = b.value[k];
|
|
251
|
+
if (av === undefined || bv === undefined) return false;
|
|
252
|
+
if (!nodeEqual(av, bv)) return false;
|
|
253
|
+
}
|
|
254
|
+
return true;
|
|
255
|
+
}
|
|
256
|
+
return false;
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
/** Render a JsonNode to a compact human-readable preview. Used by the
|
|
260
|
+
* diff view's gutter or change rows where we want a one-line summary
|
|
261
|
+
* without pretty-printing the whole subtree. */
|
|
262
|
+
export function previewNode(node: JsonNode, maxLen = 80): string {
|
|
263
|
+
let s: string;
|
|
264
|
+
switch (node.kind) {
|
|
265
|
+
case "primitive":
|
|
266
|
+
s = node.value === null ? "null" : JSON.stringify(node.value);
|
|
267
|
+
break;
|
|
268
|
+
case "array":
|
|
269
|
+
s = `[… ${node.value.length} items]`;
|
|
270
|
+
break;
|
|
271
|
+
case "object":
|
|
272
|
+
s = `{… ${Object.keys(node.value).length} keys}`;
|
|
273
|
+
break;
|
|
274
|
+
}
|
|
275
|
+
if (s.length > maxLen) s = `${s.slice(0, maxLen - 1)}…`;
|
|
276
|
+
return s;
|
|
277
|
+
}
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
import { useCallback, useEffect, useRef, useState, type MouseEvent } from "react";
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Clipboard write with a 2s "Copied!" indicator. Returns null-safe
|
|
5
|
+
* callbacks so callers can pass `text === null` through unconditionally
|
|
6
|
+
* (the hook still runs; the click handler short-circuits).
|
|
7
|
+
*/
|
|
8
|
+
export function useCopyFeedback(text: string | null): {
|
|
9
|
+
copied: boolean;
|
|
10
|
+
copy: (event: MouseEvent) => void;
|
|
11
|
+
} {
|
|
12
|
+
const [copied, setCopied] = useState(false);
|
|
13
|
+
const timerRef = useRef<ReturnType<typeof setTimeout> | null>(null);
|
|
14
|
+
|
|
15
|
+
useEffect(
|
|
16
|
+
() => () => {
|
|
17
|
+
if (timerRef.current !== null) clearTimeout(timerRef.current);
|
|
18
|
+
},
|
|
19
|
+
[],
|
|
20
|
+
);
|
|
21
|
+
|
|
22
|
+
const copy = useCallback(
|
|
23
|
+
(event: MouseEvent) => {
|
|
24
|
+
event.stopPropagation();
|
|
25
|
+
if (text === null) return;
|
|
26
|
+
void window.navigator.clipboard.writeText(text).then(() => {
|
|
27
|
+
setCopied(true);
|
|
28
|
+
if (timerRef.current !== null) clearTimeout(timerRef.current);
|
|
29
|
+
timerRef.current = setTimeout(() => setCopied(false), 2000);
|
|
30
|
+
});
|
|
31
|
+
},
|
|
32
|
+
[text],
|
|
33
|
+
);
|
|
34
|
+
|
|
35
|
+
return { copied, copy };
|
|
36
|
+
}
|