@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,321 @@
|
|
|
1
|
+
import { useVirtualizer } from "@tanstack/react-virtual";
|
|
2
|
+
import { Columns2, Rows3 } from "lucide-react";
|
|
3
|
+
import { memo, useRef, useState, type JSX } from "react";
|
|
4
|
+
import { cn } from "../../../lib/utils";
|
|
5
|
+
import type { DiffLine, DiffResult } from "./computeDiff";
|
|
6
|
+
|
|
7
|
+
type DiffMode = "unified" | "split";
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* Approximate line height (px) used for the virtualizer's `estimateSize`.
|
|
11
|
+
* Matches the `py-1` + `text-xs` (12px line-height ~ 18px) row padding so
|
|
12
|
+
* the estimate is close enough that we rarely need to re-measure.
|
|
13
|
+
*/
|
|
14
|
+
const ESTIMATED_LINE_HEIGHT = 22;
|
|
15
|
+
|
|
16
|
+
/**
|
|
17
|
+
* Cap on how many diff lines we render. Bodies larger than this still get
|
|
18
|
+
* the diff computed (so the line count is accurate) but only the first
|
|
19
|
+
* `MAX_RENDERED_LINES` are shown, with a notice explaining the truncation.
|
|
20
|
+
* This protects the tab from freezing on accidental large payloads.
|
|
21
|
+
*/
|
|
22
|
+
const MAX_RENDERED_LINES = 10_000;
|
|
23
|
+
|
|
24
|
+
type DiffViewProps = {
|
|
25
|
+
result: DiffResult;
|
|
26
|
+
/** Optional caption shown in the empty state. */
|
|
27
|
+
emptyLabel?: string;
|
|
28
|
+
};
|
|
29
|
+
|
|
30
|
+
function kindPrefix(kind: "context" | "added" | "removed"): string {
|
|
31
|
+
if (kind === "added") return "+";
|
|
32
|
+
if (kind === "removed") return "-";
|
|
33
|
+
return " ";
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
function kindClass(kind: "context" | "added" | "removed"): string {
|
|
37
|
+
if (kind === "added") {
|
|
38
|
+
return "bg-green-500/10 text-green-700 dark:text-green-300";
|
|
39
|
+
}
|
|
40
|
+
if (kind === "removed") {
|
|
41
|
+
return "bg-red-500/10 text-red-700 dark:text-red-300";
|
|
42
|
+
}
|
|
43
|
+
return "text-foreground/80";
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* Render a `DiffResult` in unified or split mode. The unified view is a
|
|
48
|
+
* single column with `+` / `-` / ` ` prefixes; the split view pairs up
|
|
49
|
+
* consecutive removed+added runs into "changed" rows for side-by-side
|
|
50
|
+
* reading.
|
|
51
|
+
*
|
|
52
|
+
* Virtualized because request bodies with full conversation history can be
|
|
53
|
+
* tens of thousands of lines long.
|
|
54
|
+
*/
|
|
55
|
+
const DiffViewInner = function DiffView({ result, emptyLabel }: DiffViewProps): JSX.Element {
|
|
56
|
+
const [mode, setMode] = useState<DiffMode>("unified");
|
|
57
|
+
const scrollRef = useRef<HTMLDivElement>(null);
|
|
58
|
+
|
|
59
|
+
const virtualizer = useVirtualizer({
|
|
60
|
+
count: result.lines.length,
|
|
61
|
+
getScrollElement: () => scrollRef.current,
|
|
62
|
+
estimateSize: () => ESTIMATED_LINE_HEIGHT,
|
|
63
|
+
overscan: 10,
|
|
64
|
+
});
|
|
65
|
+
|
|
66
|
+
if (result.isEmpty) {
|
|
67
|
+
return (
|
|
68
|
+
<div
|
|
69
|
+
className="rounded border border-dashed border-border bg-muted/30 px-4 py-6 text-center text-xs text-muted-foreground"
|
|
70
|
+
data-testid="diff-empty"
|
|
71
|
+
>
|
|
72
|
+
{emptyLabel ?? "No transformation applied — raw and processed are identical."}
|
|
73
|
+
</div>
|
|
74
|
+
);
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
const totalLines = result.lines.length;
|
|
78
|
+
const truncated = totalLines > MAX_RENDERED_LINES;
|
|
79
|
+
const visibleLines = truncated ? result.lines.slice(0, MAX_RENDERED_LINES) : result.lines;
|
|
80
|
+
|
|
81
|
+
return (
|
|
82
|
+
<div className="space-y-2">
|
|
83
|
+
<div className="flex items-center justify-between text-xs text-muted-foreground">
|
|
84
|
+
<span>
|
|
85
|
+
{totalLines.toLocaleString()} line{totalLines === 1 ? "" : "s"}
|
|
86
|
+
{truncated && ` (showing first ${MAX_RENDERED_LINES.toLocaleString()})`}
|
|
87
|
+
</span>
|
|
88
|
+
<div className="inline-flex rounded-md border border-border overflow-hidden">
|
|
89
|
+
<button
|
|
90
|
+
type="button"
|
|
91
|
+
onClick={() => setMode("unified")}
|
|
92
|
+
aria-pressed={mode === "unified"}
|
|
93
|
+
className={cn(
|
|
94
|
+
"flex items-center gap-1 px-2 py-1 transition-colors",
|
|
95
|
+
mode === "unified" ? "bg-muted text-foreground" : "hover:bg-muted/50",
|
|
96
|
+
)}
|
|
97
|
+
title="Unified diff (single column)"
|
|
98
|
+
>
|
|
99
|
+
<Rows3 className="size-3" />
|
|
100
|
+
Unified
|
|
101
|
+
</button>
|
|
102
|
+
<button
|
|
103
|
+
type="button"
|
|
104
|
+
onClick={() => setMode("split")}
|
|
105
|
+
aria-pressed={mode === "split"}
|
|
106
|
+
className={cn(
|
|
107
|
+
"flex items-center gap-1 px-2 py-1 transition-colors border-l border-border",
|
|
108
|
+
mode === "split" ? "bg-muted text-foreground" : "hover:bg-muted/50",
|
|
109
|
+
)}
|
|
110
|
+
title="Split diff (side by side)"
|
|
111
|
+
>
|
|
112
|
+
<Columns2 className="size-3" />
|
|
113
|
+
Split
|
|
114
|
+
</button>
|
|
115
|
+
</div>
|
|
116
|
+
</div>
|
|
117
|
+
<div
|
|
118
|
+
ref={scrollRef}
|
|
119
|
+
className="max-h-[60vh] overflow-auto rounded border border-border bg-background font-mono text-xs"
|
|
120
|
+
data-testid="diff-viewport"
|
|
121
|
+
>
|
|
122
|
+
{mode === "unified" ? (
|
|
123
|
+
<UnifiedRows virtualizer={virtualizer} lines={visibleLines} />
|
|
124
|
+
) : (
|
|
125
|
+
<SplitRows lines={visibleLines} />
|
|
126
|
+
)}
|
|
127
|
+
</div>
|
|
128
|
+
</div>
|
|
129
|
+
);
|
|
130
|
+
};
|
|
131
|
+
|
|
132
|
+
function UnifiedRows({
|
|
133
|
+
virtualizer,
|
|
134
|
+
lines,
|
|
135
|
+
}: {
|
|
136
|
+
virtualizer: ReturnType<typeof useVirtualizer<HTMLDivElement, Element>>;
|
|
137
|
+
lines: ReadonlyArray<DiffLine>;
|
|
138
|
+
}): JSX.Element {
|
|
139
|
+
return (
|
|
140
|
+
<div
|
|
141
|
+
style={{
|
|
142
|
+
height: `${virtualizer.getTotalSize()}px`,
|
|
143
|
+
width: "100%",
|
|
144
|
+
position: "relative",
|
|
145
|
+
}}
|
|
146
|
+
>
|
|
147
|
+
{virtualizer.getVirtualItems().map((virtualRow) => {
|
|
148
|
+
const line = lines[virtualRow.index];
|
|
149
|
+
if (line === undefined) return null;
|
|
150
|
+
return (
|
|
151
|
+
<div
|
|
152
|
+
key={virtualRow.key}
|
|
153
|
+
data-index={virtualRow.index}
|
|
154
|
+
ref={virtualizer.measureElement}
|
|
155
|
+
className={cn("flex items-start gap-3 px-2", kindClass(line.kind))}
|
|
156
|
+
style={{
|
|
157
|
+
position: "absolute",
|
|
158
|
+
top: 0,
|
|
159
|
+
left: 0,
|
|
160
|
+
width: "100%",
|
|
161
|
+
transform: `translateY(${virtualRow.start}px)`,
|
|
162
|
+
}}
|
|
163
|
+
>
|
|
164
|
+
<span className="select-none text-muted-foreground/60 w-10 text-right tabular-nums">
|
|
165
|
+
{line.oldLineNumber ?? ""}
|
|
166
|
+
</span>
|
|
167
|
+
<span className="select-none text-muted-foreground/60 w-10 text-right tabular-nums">
|
|
168
|
+
{line.newLineNumber ?? ""}
|
|
169
|
+
</span>
|
|
170
|
+
<span
|
|
171
|
+
className={cn(
|
|
172
|
+
"select-none w-3 text-right font-bold",
|
|
173
|
+
line.kind === "added" && "text-green-600 dark:text-green-400",
|
|
174
|
+
line.kind === "removed" && "text-red-600 dark:text-red-400",
|
|
175
|
+
)}
|
|
176
|
+
>
|
|
177
|
+
{kindPrefix(line.kind)}
|
|
178
|
+
</span>
|
|
179
|
+
<span className="whitespace-pre-wrap break-all flex-1">{line.text}</span>
|
|
180
|
+
</div>
|
|
181
|
+
);
|
|
182
|
+
})}
|
|
183
|
+
</div>
|
|
184
|
+
);
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
/**
|
|
188
|
+
* Convert a stream of context / removed / added lines into "rows" suitable
|
|
189
|
+
* for a split view:
|
|
190
|
+
* - A run of `removed` followed by a run of `added` is paired line-by-line
|
|
191
|
+
* as a "changed" row (the shorter side is padded with blanks).
|
|
192
|
+
* - A `removed`-only run becomes rows marked as deletion-only.
|
|
193
|
+
* - An `added`-only run becomes rows marked as addition-only.
|
|
194
|
+
* - `context` lines become rows where both sides show the same text.
|
|
195
|
+
*/
|
|
196
|
+
type SplitRow = {
|
|
197
|
+
left: { text: string; lineNumber: number | null } | null;
|
|
198
|
+
right: { text: string; lineNumber: number | null } | null;
|
|
199
|
+
/** "equal" | "changed" | "removed" | "added" */
|
|
200
|
+
kind: "equal" | "changed" | "removed" | "added";
|
|
201
|
+
};
|
|
202
|
+
|
|
203
|
+
function buildSplitRows(lines: ReadonlyArray<DiffLine>): SplitRow[] {
|
|
204
|
+
const rows: SplitRow[] = [];
|
|
205
|
+
let i = 0;
|
|
206
|
+
while (i < lines.length) {
|
|
207
|
+
const line = lines[i];
|
|
208
|
+
if (line === undefined) break;
|
|
209
|
+
if (line.kind === "context") {
|
|
210
|
+
rows.push({
|
|
211
|
+
left: { text: line.text, lineNumber: line.oldLineNumber },
|
|
212
|
+
right: { text: line.text, lineNumber: line.newLineNumber },
|
|
213
|
+
kind: "equal",
|
|
214
|
+
});
|
|
215
|
+
i += 1;
|
|
216
|
+
continue;
|
|
217
|
+
}
|
|
218
|
+
// Collect a run of removes followed by a run of adds.
|
|
219
|
+
const removes: DiffLine[] = [];
|
|
220
|
+
const adds: DiffLine[] = [];
|
|
221
|
+
while (i < lines.length && lines[i]?.kind === "removed") {
|
|
222
|
+
const removed = lines[i];
|
|
223
|
+
if (removed !== undefined) removes.push(removed);
|
|
224
|
+
i += 1;
|
|
225
|
+
}
|
|
226
|
+
while (i < lines.length && lines[i]?.kind === "added") {
|
|
227
|
+
const added = lines[i];
|
|
228
|
+
if (added !== undefined) adds.push(added);
|
|
229
|
+
i += 1;
|
|
230
|
+
}
|
|
231
|
+
const pairCount = Math.max(removes.length, adds.length);
|
|
232
|
+
for (let p = 0; p < pairCount; p += 1) {
|
|
233
|
+
const r = removes[p];
|
|
234
|
+
const a = adds[p];
|
|
235
|
+
if (r !== undefined && a !== undefined) {
|
|
236
|
+
rows.push({
|
|
237
|
+
left: { text: r.text, lineNumber: r.oldLineNumber },
|
|
238
|
+
right: { text: a.text, lineNumber: a.newLineNumber },
|
|
239
|
+
kind: "changed",
|
|
240
|
+
});
|
|
241
|
+
} else if (r !== undefined) {
|
|
242
|
+
rows.push({
|
|
243
|
+
left: { text: r.text, lineNumber: r.oldLineNumber },
|
|
244
|
+
right: null,
|
|
245
|
+
kind: "removed",
|
|
246
|
+
});
|
|
247
|
+
} else if (a !== undefined) {
|
|
248
|
+
rows.push({
|
|
249
|
+
left: null,
|
|
250
|
+
right: { text: a.text, lineNumber: a.newLineNumber },
|
|
251
|
+
kind: "added",
|
|
252
|
+
});
|
|
253
|
+
}
|
|
254
|
+
}
|
|
255
|
+
}
|
|
256
|
+
return rows;
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
function SplitRowClass(kind: SplitRow["kind"]): string {
|
|
260
|
+
switch (kind) {
|
|
261
|
+
case "changed":
|
|
262
|
+
return "bg-amber-500/10";
|
|
263
|
+
case "removed":
|
|
264
|
+
return "bg-red-500/10";
|
|
265
|
+
case "added":
|
|
266
|
+
return "bg-green-500/10";
|
|
267
|
+
case "equal":
|
|
268
|
+
return "";
|
|
269
|
+
}
|
|
270
|
+
}
|
|
271
|
+
|
|
272
|
+
function SplitRows({ lines }: { lines: ReadonlyArray<DiffLine> }): JSX.Element {
|
|
273
|
+
const rows = buildSplitRows(lines);
|
|
274
|
+
return (
|
|
275
|
+
<div data-testid="diff-split">
|
|
276
|
+
{rows.map((row, idx) => (
|
|
277
|
+
<div
|
|
278
|
+
// idx is stable for a given diff result; the rows array is rebuilt
|
|
279
|
+
// when the input `lines` change, so React's reconciliation handles
|
|
280
|
+
// re-renders correctly.
|
|
281
|
+
// biome-ignore lint/suspicious/noArrayIndexKey: see above
|
|
282
|
+
key={idx}
|
|
283
|
+
className={cn("grid grid-cols-2 font-mono text-xs", SplitRowClass(row.kind))}
|
|
284
|
+
>
|
|
285
|
+
<div className="flex items-start gap-2 px-2 border-r border-border/50">
|
|
286
|
+
<span className="select-none text-muted-foreground/60 w-8 text-right tabular-nums">
|
|
287
|
+
{row.left?.lineNumber ?? ""}
|
|
288
|
+
</span>
|
|
289
|
+
<span
|
|
290
|
+
className={cn(
|
|
291
|
+
"select-none w-3 text-right font-bold",
|
|
292
|
+
(row.kind === "removed" || row.kind === "changed") &&
|
|
293
|
+
"text-red-600 dark:text-red-400",
|
|
294
|
+
)}
|
|
295
|
+
>
|
|
296
|
+
{row.left !== null ? "-" : " "}
|
|
297
|
+
</span>
|
|
298
|
+
<span className="whitespace-pre-wrap break-all flex-1">{row.left?.text ?? ""}</span>
|
|
299
|
+
</div>
|
|
300
|
+
<div className="flex items-start gap-2 px-2">
|
|
301
|
+
<span className="select-none text-muted-foreground/60 w-8 text-right tabular-nums">
|
|
302
|
+
{row.right?.lineNumber ?? ""}
|
|
303
|
+
</span>
|
|
304
|
+
<span
|
|
305
|
+
className={cn(
|
|
306
|
+
"select-none w-3 text-right font-bold",
|
|
307
|
+
(row.kind === "added" || row.kind === "changed") &&
|
|
308
|
+
"text-green-600 dark:text-green-400",
|
|
309
|
+
)}
|
|
310
|
+
>
|
|
311
|
+
{row.right !== null ? "+" : " "}
|
|
312
|
+
</span>
|
|
313
|
+
<span className="whitespace-pre-wrap break-all flex-1">{row.right?.text ?? ""}</span>
|
|
314
|
+
</div>
|
|
315
|
+
</div>
|
|
316
|
+
))}
|
|
317
|
+
</div>
|
|
318
|
+
);
|
|
319
|
+
}
|
|
320
|
+
|
|
321
|
+
export const DiffView = memo(DiffViewInner);
|
|
@@ -0,0 +1,178 @@
|
|
|
1
|
+
import { diffJson, diffLines, type Change } from "diff";
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* A single line in a unified diff view.
|
|
5
|
+
*
|
|
6
|
+
* - `kind: "context"` — line present in both sides
|
|
7
|
+
* - `kind: "removed"` — line only in the left (raw) side
|
|
8
|
+
* - `kind: "added"` — line only in the right (processed) side
|
|
9
|
+
*
|
|
10
|
+
* `oldLineNumber` / `newLineNumber` are 1-based and `null` for the side the
|
|
11
|
+
* line is not on. `text` always includes the trailing newline-free content
|
|
12
|
+
* for that line.
|
|
13
|
+
*/
|
|
14
|
+
export type DiffLine = {
|
|
15
|
+
kind: "context" | "removed" | "added";
|
|
16
|
+
text: string;
|
|
17
|
+
oldLineNumber: number | null;
|
|
18
|
+
newLineNumber: number | null;
|
|
19
|
+
};
|
|
20
|
+
|
|
21
|
+
/** Result of computing a diff: the normalized lines plus a quick "no change" flag. */
|
|
22
|
+
export type DiffResult = {
|
|
23
|
+
lines: DiffLine[];
|
|
24
|
+
/** True when the two inputs are byte-equal (or semantically equal for JSON). */
|
|
25
|
+
isEmpty: boolean;
|
|
26
|
+
};
|
|
27
|
+
|
|
28
|
+
/**
|
|
29
|
+
* Normalize an HTTP-header record into a deterministic `[lowercaseKey, value]`
|
|
30
|
+
* tuple. Lowercasing the key matches how `buildProxyHeaders` stores raw
|
|
31
|
+
* headers in `handler.ts`, and sorting by key gives stable diffs across
|
|
32
|
+
* different client header orders.
|
|
33
|
+
*/
|
|
34
|
+
function normalizeHeaders(headers: Record<string, string> | undefined): Array<[string, string]> {
|
|
35
|
+
if (headers === undefined) return [];
|
|
36
|
+
return Object.entries(headers)
|
|
37
|
+
.map((entry): [string, string] => [entry[0].toLowerCase(), entry[1]])
|
|
38
|
+
.sort(([a], [b]) => a.localeCompare(b));
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
/**
|
|
42
|
+
* Render a header entry as a single line in the form `key: value`. Mirrors
|
|
43
|
+
* the layout used by the existing Headers / Raw Headers tabs in
|
|
44
|
+
* `LogEntry.tsx` so the diff view looks like the same data, just with
|
|
45
|
+
* add/remove highlights.
|
|
46
|
+
*/
|
|
47
|
+
function headerLine([key, value]: [string, string]): string {
|
|
48
|
+
return `${key}: ${value}`;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
/**
|
|
52
|
+
* Convert a list of `diff` library `Change` objects into `DiffLine` records
|
|
53
|
+
* with stable 1-based line numbers on each side. Each change's value is
|
|
54
|
+
* split on `\n` so we can produce per-line rows (the `diff` library groups
|
|
55
|
+
* runs of consecutive equal/added/removed lines into one Change).
|
|
56
|
+
*/
|
|
57
|
+
function changesToLines(changes: Change[]): DiffLine[] {
|
|
58
|
+
const lines: DiffLine[] = [];
|
|
59
|
+
let oldNum = 0;
|
|
60
|
+
let newNum = 0;
|
|
61
|
+
for (const change of changes) {
|
|
62
|
+
// Trailing newlines are part of the change's value; dropping the empty
|
|
63
|
+
// string after the final `\n` keeps the line count correct.
|
|
64
|
+
const parts = change.value.split("\n");
|
|
65
|
+
if (parts.length > 0 && parts[parts.length - 1] === "") parts.pop();
|
|
66
|
+
for (const text of parts) {
|
|
67
|
+
if (change.added) {
|
|
68
|
+
newNum += 1;
|
|
69
|
+
lines.push({ kind: "added", text, oldLineNumber: null, newLineNumber: newNum });
|
|
70
|
+
} else if (change.removed) {
|
|
71
|
+
oldNum += 1;
|
|
72
|
+
lines.push({ kind: "removed", text, oldLineNumber: oldNum, newLineNumber: null });
|
|
73
|
+
} else {
|
|
74
|
+
oldNum += 1;
|
|
75
|
+
newNum += 1;
|
|
76
|
+
lines.push({
|
|
77
|
+
kind: "context",
|
|
78
|
+
text,
|
|
79
|
+
oldLineNumber: oldNum,
|
|
80
|
+
newLineNumber: newNum,
|
|
81
|
+
});
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
return lines;
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
/**
|
|
89
|
+
* Compute a diff between two HTTP-header records. Headers are normalized
|
|
90
|
+
* (lower-cased keys, sorted) before comparison so that key reordering or
|
|
91
|
+
* case differences don't show as spurious changes.
|
|
92
|
+
*
|
|
93
|
+
* Returns `{ lines, isEmpty: true }` when the two records contain the same
|
|
94
|
+
* key/value pairs, which the UI uses to show a "No transformation applied"
|
|
95
|
+
* placeholder instead of an empty diff.
|
|
96
|
+
*/
|
|
97
|
+
export function computeHeadersDiff(
|
|
98
|
+
rawHeaders: Record<string, string> | undefined,
|
|
99
|
+
processedHeaders: Record<string, string> | undefined,
|
|
100
|
+
): DiffResult {
|
|
101
|
+
const left = normalizeHeaders(rawHeaders).map(headerLine).join("\n");
|
|
102
|
+
const right = normalizeHeaders(processedHeaders).map(headerLine).join("\n");
|
|
103
|
+
|
|
104
|
+
if (left === right) {
|
|
105
|
+
return { lines: [], isEmpty: true };
|
|
106
|
+
}
|
|
107
|
+
const changes = diffLines(left, right);
|
|
108
|
+
return { lines: changesToLines(changes), isEmpty: false };
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
/**
|
|
112
|
+
* Try to parse a string as JSON. Returns the parsed value on success, or
|
|
113
|
+
* `null` on failure. We use this to decide between semantic (`diffJson`) and
|
|
114
|
+
* line-level (`diffLines`) diffs for request bodies. The return type is a
|
|
115
|
+
* union of `null` and the input types `diffJson` accepts, so the call site
|
|
116
|
+
* can pass the result directly without re-narrowing.
|
|
117
|
+
*/
|
|
118
|
+
function tryParseJson(text: string): null | string | object {
|
|
119
|
+
try {
|
|
120
|
+
const parsed: unknown = JSON.parse(text);
|
|
121
|
+
if (parsed === null) return parsed;
|
|
122
|
+
if (typeof parsed === "string" || typeof parsed === "object") return parsed;
|
|
123
|
+
// Numbers, booleans, etc. — also valid JSON, but the diffJson overloads
|
|
124
|
+
// only accept string | object, so we surface these as line-diff by
|
|
125
|
+
// returning null.
|
|
126
|
+
return null;
|
|
127
|
+
} catch {
|
|
128
|
+
return null;
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
/**
|
|
133
|
+
* Compute a diff between two request bodies.
|
|
134
|
+
*
|
|
135
|
+
* - If both sides parse as JSON, use `diffJson` for a semantic key-level
|
|
136
|
+
* comparison. Reordering fields, or adding/removing a key inside an
|
|
137
|
+
* object, produces cleaner output than a raw text diff.
|
|
138
|
+
* - If either side is not valid JSON, fall back to `diffLines` of the raw
|
|
139
|
+
* strings. This keeps the diff working for non-JSON bodies (rare in this
|
|
140
|
+
* proxy, but possible) and for partial / truncated bodies.
|
|
141
|
+
*
|
|
142
|
+
* The two inputs are compared as-is. Whitespace and key order in JSON are
|
|
143
|
+
* normalized by the parser, so two semantically-equal bodies will report
|
|
144
|
+
* `isEmpty: true` even if their textual forms differ.
|
|
145
|
+
*/
|
|
146
|
+
export function computeRequestDiff(
|
|
147
|
+
rawBody: string | null,
|
|
148
|
+
processedBody: string | null,
|
|
149
|
+
): DiffResult {
|
|
150
|
+
const left = rawBody ?? "";
|
|
151
|
+
const right = processedBody ?? "";
|
|
152
|
+
|
|
153
|
+
if (left === right) {
|
|
154
|
+
return { lines: [], isEmpty: true };
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
const leftJson = tryParseJson(left);
|
|
158
|
+
const rightJson = tryParseJson(right);
|
|
159
|
+
if (leftJson !== null && rightJson !== null) {
|
|
160
|
+
// diffJson runs canonicalize() on both sides, so key order / whitespace
|
|
161
|
+
// differences are handled before the line diff is built. Two JSON values
|
|
162
|
+
// that are deep-equal produce an empty change set. The overload set
|
|
163
|
+
// includes some signatures that return `undefined` (abortable modes);
|
|
164
|
+
// since we pass no options, the result is always defined in practice,
|
|
165
|
+
// but we guard with a fallback to the line diff for type-safety.
|
|
166
|
+
const changes = diffJson(leftJson, rightJson) ?? [];
|
|
167
|
+
if (changes.length === 1) {
|
|
168
|
+
const first = changes[0];
|
|
169
|
+
if (first !== undefined && !first.added && !first.removed) {
|
|
170
|
+
return { lines: [], isEmpty: true };
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
return { lines: changesToLines(changes), isEmpty: false };
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
const changes = diffLines(left, right);
|
|
177
|
+
return { lines: changesToLines(changes), isEmpty: false };
|
|
178
|
+
}
|
|
@@ -0,0 +1,157 @@
|
|
|
1
|
+
import { Brain, ChevronDown, ChevronRight, Terminal } from "lucide-react";
|
|
2
|
+
import { type JSX, memo, useState } from "react";
|
|
3
|
+
import ReactMarkdown from "react-markdown";
|
|
4
|
+
import type { ResponseContentBlockType } from "../../../../proxy/schemas";
|
|
5
|
+
import { Badge } from "../../../ui/badge";
|
|
6
|
+
import { Collapsible, CollapsibleContent, CollapsibleTrigger } from "../../../ui/collapsible";
|
|
7
|
+
import { JsonViewer } from "../../../ui/json-viewer";
|
|
8
|
+
import { safeJsonValue } from "../../../ui/json-viewer-bulk";
|
|
9
|
+
import { ScrollArea } from "../../../ui/scroll-area";
|
|
10
|
+
import { extractThinkingFromContent } from "./thinkingExtract";
|
|
11
|
+
|
|
12
|
+
function assertNever(_value: never): JSX.Element {
|
|
13
|
+
return <></>;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
function SystemReminderBlock({ text }: { text: string }): JSX.Element {
|
|
17
|
+
const [open, setOpen] = useState(false);
|
|
18
|
+
|
|
19
|
+
return (
|
|
20
|
+
<Collapsible open={open} onOpenChange={setOpen}>
|
|
21
|
+
<CollapsibleTrigger className="flex items-center gap-1.5 py-0.5 cursor-pointer hover:opacity-80 transition-opacity group">
|
|
22
|
+
{open ? (
|
|
23
|
+
<ChevronDown className="size-3 text-muted-foreground" />
|
|
24
|
+
) : (
|
|
25
|
+
<ChevronRight className="size-3 text-muted-foreground" />
|
|
26
|
+
)}
|
|
27
|
+
<span className="text-muted-foreground text-xs italic select-none opacity-60">
|
|
28
|
+
[system-reminder]
|
|
29
|
+
</span>
|
|
30
|
+
</CollapsibleTrigger>
|
|
31
|
+
<CollapsibleContent>
|
|
32
|
+
<div className="pl-4 pt-1">
|
|
33
|
+
<div className="prose prose-sm dark:prose-invert max-w-none [&_pre]:bg-muted [&_pre]:text-foreground [&_code]:text-[0.8em] [&_p]:my-1 [&_ul]:my-1 [&_ol]:my-1">
|
|
34
|
+
<ReactMarkdown>{text}</ReactMarkdown>
|
|
35
|
+
</div>
|
|
36
|
+
</div>
|
|
37
|
+
</CollapsibleContent>
|
|
38
|
+
</Collapsible>
|
|
39
|
+
);
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
export const TextBlock = memo(function TextBlock({ text }: { text: string }): JSX.Element {
|
|
43
|
+
if (text.includes("<system-reminder>")) {
|
|
44
|
+
return <SystemReminderBlock text={text} />;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
// Check for <think> tags wrapped in content
|
|
48
|
+
const { thinking, remainingText } = extractThinkingFromContent(text);
|
|
49
|
+
|
|
50
|
+
return (
|
|
51
|
+
<div className="space-y-2">
|
|
52
|
+
{thinking !== null && <ThinkingBlock thinking={thinking} />}
|
|
53
|
+
{remainingText.length > 0 && (
|
|
54
|
+
<div className="prose prose-sm dark:prose-invert max-w-none [&_pre]:bg-muted [&_pre]:text-foreground [&_code]:text-[0.8em] [&_p]:my-1 [&_ul]:my-1 [&_ol]:my-1">
|
|
55
|
+
<ReactMarkdown>{remainingText}</ReactMarkdown>
|
|
56
|
+
</div>
|
|
57
|
+
)}
|
|
58
|
+
{thinking === null && remainingText.length === 0 && (
|
|
59
|
+
<p className="text-xs text-muted-foreground italic">Empty text block</p>
|
|
60
|
+
)}
|
|
61
|
+
</div>
|
|
62
|
+
);
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
export const ThinkingBlock = memo(function ThinkingBlock({
|
|
66
|
+
thinking,
|
|
67
|
+
}: {
|
|
68
|
+
thinking: string;
|
|
69
|
+
}): JSX.Element {
|
|
70
|
+
const [open, setOpen] = useState(false);
|
|
71
|
+
|
|
72
|
+
return (
|
|
73
|
+
<Collapsible open={open} onOpenChange={setOpen}>
|
|
74
|
+
<div className="border-l-2 border-purple-500/40 my-1">
|
|
75
|
+
<CollapsibleTrigger className="flex items-center gap-1.5 px-3 py-1 w-full text-left cursor-pointer hover:bg-purple-500/5 transition-colors rounded-r-sm group">
|
|
76
|
+
<Brain className="size-3.5 text-purple-400 shrink-0" />
|
|
77
|
+
<span className="text-xs font-medium text-purple-400">Thinking</span>
|
|
78
|
+
<Badge
|
|
79
|
+
variant="ghost"
|
|
80
|
+
className="text-[10px] text-muted-foreground px-1.5 py-0 h-4 font-mono"
|
|
81
|
+
>
|
|
82
|
+
{thinking.length.toLocaleString()} chars
|
|
83
|
+
</Badge>
|
|
84
|
+
<span className="flex-1" />
|
|
85
|
+
{open ? (
|
|
86
|
+
<ChevronDown className="size-3 text-muted-foreground" />
|
|
87
|
+
) : (
|
|
88
|
+
<ChevronRight className="size-3 text-muted-foreground" />
|
|
89
|
+
)}
|
|
90
|
+
</CollapsibleTrigger>
|
|
91
|
+
<CollapsibleContent>
|
|
92
|
+
<div className="px-3 pb-2">
|
|
93
|
+
<ScrollArea className="max-h-[60vh]">
|
|
94
|
+
<pre className="text-xs text-muted-foreground whitespace-pre-wrap font-mono leading-relaxed">
|
|
95
|
+
{thinking}
|
|
96
|
+
</pre>
|
|
97
|
+
</ScrollArea>
|
|
98
|
+
</div>
|
|
99
|
+
</CollapsibleContent>
|
|
100
|
+
</div>
|
|
101
|
+
</Collapsible>
|
|
102
|
+
);
|
|
103
|
+
});
|
|
104
|
+
|
|
105
|
+
export const ToolUseBlock = memo(function ToolUseBlock({
|
|
106
|
+
name,
|
|
107
|
+
input,
|
|
108
|
+
}: {
|
|
109
|
+
name: string;
|
|
110
|
+
input: Record<string, unknown>;
|
|
111
|
+
}): JSX.Element {
|
|
112
|
+
const [open, setOpen] = useState(false);
|
|
113
|
+
|
|
114
|
+
return (
|
|
115
|
+
<Collapsible open={open} onOpenChange={setOpen}>
|
|
116
|
+
<div className="border-l-2 border-blue-500/40 my-1">
|
|
117
|
+
<CollapsibleTrigger className="flex items-center gap-1.5 px-3 py-1 w-full text-left cursor-pointer hover:bg-blue-500/5 transition-colors rounded-r-sm group">
|
|
118
|
+
<Terminal className="size-3.5 text-blue-400 shrink-0" />
|
|
119
|
+
<Badge variant="outline" className="text-[10px] font-mono px-1.5 py-0 h-4">
|
|
120
|
+
{name}
|
|
121
|
+
</Badge>
|
|
122
|
+
<span className="flex-1" />
|
|
123
|
+
{open ? (
|
|
124
|
+
<ChevronDown className="size-3 text-muted-foreground" />
|
|
125
|
+
) : (
|
|
126
|
+
<ChevronRight className="size-3 text-muted-foreground" />
|
|
127
|
+
)}
|
|
128
|
+
</CollapsibleTrigger>
|
|
129
|
+
<CollapsibleContent>
|
|
130
|
+
<div className="px-3 pb-2">
|
|
131
|
+
<ScrollArea className="max-h-[60vh]">
|
|
132
|
+
<JsonViewer data={safeJsonValue(input)} defaultExpandDepth={0} />
|
|
133
|
+
</ScrollArea>
|
|
134
|
+
</div>
|
|
135
|
+
</CollapsibleContent>
|
|
136
|
+
</div>
|
|
137
|
+
</Collapsible>
|
|
138
|
+
);
|
|
139
|
+
});
|
|
140
|
+
|
|
141
|
+
export const ResponseContentBlockRenderer = memo(function ResponseContentBlockRenderer({
|
|
142
|
+
block,
|
|
143
|
+
}: {
|
|
144
|
+
block: ResponseContentBlockType;
|
|
145
|
+
}): JSX.Element {
|
|
146
|
+
switch (block.type) {
|
|
147
|
+
case "text":
|
|
148
|
+
return <TextBlock text={block.text} />;
|
|
149
|
+
case "thinking":
|
|
150
|
+
case "think":
|
|
151
|
+
return <ThinkingBlock thinking={block.thinking} />;
|
|
152
|
+
case "tool_use":
|
|
153
|
+
return <ToolUseBlock name={block.name} input={block.input} />;
|
|
154
|
+
default:
|
|
155
|
+
return assertNever(block);
|
|
156
|
+
}
|
|
157
|
+
});
|