@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.
Files changed (400) hide show
  1. package/.output/cli.js +1611 -0
  2. package/.output/nitro.json +17 -0
  3. package/.output/public/assets/CompareDrawer-CU5ZrWcL.js +1 -0
  4. package/.output/public/assets/ProxyViewerContainer-pEBqVp1d.js +101 -0
  5. package/.output/public/assets/ReplayDialog-F58yNg5j.js +1 -0
  6. package/.output/public/assets/RequestAnatomy-C9lT0qE_.js +1 -0
  7. package/.output/public/assets/ResponseView-DHJq6bnz.js +1 -0
  8. package/.output/public/assets/StreamingChunkSequence-BTgfpFUT.js +1 -0
  9. package/.output/public/assets/_sessionId-DsNRbnNm.js +1 -0
  10. package/.output/public/assets/alibaba-TTwafVwX.svg +1 -0
  11. package/.output/public/assets/index-CpWG2hFn.css +1 -0
  12. package/.output/public/assets/index-DmBV8Gve.js +1 -0
  13. package/.output/public/assets/json-viewer-CZVYLR8j.js +14 -0
  14. package/.output/public/assets/main-DHs7FBK3.js +18 -0
  15. package/.output/public/assets/minimax-BPMzvuL-.jpeg +0 -0
  16. package/.output/public/assets/qwen-CONDcHqt.png +0 -0
  17. package/.output/public/assets/zhipuai-BPNAnxo-.svg +219 -0
  18. package/.output/server/_chunks/ssr-renderer.mjs +22 -0
  19. package/.output/server/_libs/@radix-ui/react-accessible-icon+[...].mjs +1 -0
  20. package/.output/server/_libs/@radix-ui/react-dismissable-layer+[...].mjs +210 -0
  21. package/.output/server/_libs/@radix-ui/react-navigation-menu+[...].mjs +2 -0
  22. package/.output/server/_libs/@radix-ui/react-one-time-password-field+[...].mjs +2 -0
  23. package/.output/server/_libs/@radix-ui/react-password-toggle-field+[...].mjs +2 -0
  24. package/.output/server/_libs/@radix-ui/react-use-callback-ref+[...].mjs +11 -0
  25. package/.output/server/_libs/@radix-ui/react-use-controllable-state+[...].mjs +69 -0
  26. package/.output/server/_libs/@radix-ui/react-use-effect-event+[...].mjs +1 -0
  27. package/.output/server/_libs/@radix-ui/react-use-escape-keydown+[...].mjs +17 -0
  28. package/.output/server/_libs/@radix-ui/react-use-is-hydrated+[...].mjs +1 -0
  29. package/.output/server/_libs/@radix-ui/react-use-layout-effect+[...].mjs +6 -0
  30. package/.output/server/_libs/@radix-ui/react-visually-hidden+[...].mjs +34 -0
  31. package/.output/server/_libs/ajv-formats.mjs +330 -0
  32. package/.output/server/_libs/ajv.mjs +11444 -0
  33. package/.output/server/_libs/aria-hidden.mjs +122 -0
  34. package/.output/server/_libs/atomically.mjs +152 -0
  35. package/.output/server/_libs/bail.mjs +8 -0
  36. package/.output/server/_libs/cfworker__json-schema.mjs +1 -0
  37. package/.output/server/_libs/character-entities.mjs +2130 -0
  38. package/.output/server/_libs/class-variance-authority.mjs +44 -0
  39. package/.output/server/_libs/clsx.mjs +16 -0
  40. package/.output/server/_libs/comma-separated-tokens.mjs +10 -0
  41. package/.output/server/_libs/conf.mjs +635 -0
  42. package/.output/server/_libs/cookie-es.mjs +58 -0
  43. package/.output/server/_libs/core-util-is.mjs +75 -0
  44. package/.output/server/_libs/croner.mjs +1 -0
  45. package/.output/server/_libs/crossws.mjs +1 -0
  46. package/.output/server/_libs/debounce-fn.mjs +69 -0
  47. package/.output/server/_libs/decode-named-character-reference+[...].mjs +8 -0
  48. package/.output/server/_libs/dequal.mjs +27 -0
  49. package/.output/server/_libs/detect-node-es.mjs +1 -0
  50. package/.output/server/_libs/devlop.mjs +8 -0
  51. package/.output/server/_libs/diff.mjs +320 -0
  52. package/.output/server/_libs/dot-prop.mjs +265 -0
  53. package/.output/server/_libs/env-paths.mjs +57 -0
  54. package/.output/server/_libs/estree-util-is-identifier-name.mjs +11 -0
  55. package/.output/server/_libs/extend.mjs +97 -0
  56. package/.output/server/_libs/fast-deep-equal.mjs +38 -0
  57. package/.output/server/_libs/fast-uri.mjs +812 -0
  58. package/.output/server/_libs/floating-ui__core.mjs +725 -0
  59. package/.output/server/_libs/floating-ui__dom.mjs +622 -0
  60. package/.output/server/_libs/floating-ui__react-dom.mjs +292 -0
  61. package/.output/server/_libs/floating-ui__utils.mjs +320 -0
  62. package/.output/server/_libs/get-nonce.mjs +9 -0
  63. package/.output/server/_libs/h3-v2.mjs +276 -0
  64. package/.output/server/_libs/h3.mjs +408 -0
  65. package/.output/server/_libs/hast-util-to-jsx-runtime.mjs +388 -0
  66. package/.output/server/_libs/hast-util-whitespace.mjs +10 -0
  67. package/.output/server/_libs/hookable.mjs +1 -0
  68. package/.output/server/_libs/html-url-attributes.mjs +26 -0
  69. package/.output/server/_libs/immediate.mjs +74 -0
  70. package/.output/server/_libs/inherits.mjs +50 -0
  71. package/.output/server/_libs/inline-style-parser.mjs +142 -0
  72. package/.output/server/_libs/is-plain-obj.mjs +10 -0
  73. package/.output/server/_libs/isarray.mjs +14 -0
  74. package/.output/server/_libs/isbot.mjs +20 -0
  75. package/.output/server/_libs/json-schema-traverse.mjs +180 -0
  76. package/.output/server/_libs/jszip.mjs +3051 -0
  77. package/.output/server/_libs/lie.mjs +273 -0
  78. package/.output/server/_libs/lucide-react.mjs +492 -0
  79. package/.output/server/_libs/mdast-util-from-markdown.mjs +717 -0
  80. package/.output/server/_libs/mdast-util-to-hast.mjs +710 -0
  81. package/.output/server/_libs/mdast-util-to-string.mjs +38 -0
  82. package/.output/server/_libs/micromark-core-commonmark.mjs +2259 -0
  83. package/.output/server/_libs/micromark-factory-destination.mjs +94 -0
  84. package/.output/server/_libs/micromark-factory-label.mjs +63 -0
  85. package/.output/server/_libs/micromark-factory-space.mjs +24 -0
  86. package/.output/server/_libs/micromark-factory-title.mjs +65 -0
  87. package/.output/server/_libs/micromark-factory-whitespace.mjs +22 -0
  88. package/.output/server/_libs/micromark-util-character.mjs +44 -0
  89. package/.output/server/_libs/micromark-util-chunked.mjs +36 -0
  90. package/.output/server/_libs/micromark-util-classify-character+[...].mjs +12 -0
  91. package/.output/server/_libs/micromark-util-combine-extensions+[...].mjs +41 -0
  92. package/.output/server/_libs/micromark-util-decode-numeric-character-reference+[...].mjs +19 -0
  93. package/.output/server/_libs/micromark-util-decode-string.mjs +21 -0
  94. package/.output/server/_libs/micromark-util-encode.mjs +1 -0
  95. package/.output/server/_libs/micromark-util-html-tag-name.mjs +69 -0
  96. package/.output/server/_libs/micromark-util-normalize-identifier+[...].mjs +6 -0
  97. package/.output/server/_libs/micromark-util-resolve-all.mjs +15 -0
  98. package/.output/server/_libs/micromark-util-sanitize-uri.mjs +41 -0
  99. package/.output/server/_libs/micromark-util-subtokenize.mjs +346 -0
  100. package/.output/server/_libs/micromark.mjs +906 -0
  101. package/.output/server/_libs/mimic-function.mjs +47 -0
  102. package/.output/server/_libs/modelcontextprotocol__server.mjs +9738 -0
  103. package/.output/server/_libs/ocache.mjs +1 -0
  104. package/.output/server/_libs/ohash.mjs +1 -0
  105. package/.output/server/_libs/pako.mjs +4223 -0
  106. package/.output/server/_libs/process-nextick-args.mjs +48 -0
  107. package/.output/server/_libs/property-information.mjs +1209 -0
  108. package/.output/server/_libs/radix-ui.mjs +1 -0
  109. package/.output/server/_libs/radix-ui__number.mjs +6 -0
  110. package/.output/server/_libs/radix-ui__primitive.mjs +11 -0
  111. package/.output/server/_libs/radix-ui__react-accordion.mjs +1 -0
  112. package/.output/server/_libs/radix-ui__react-alert-dialog.mjs +1 -0
  113. package/.output/server/_libs/radix-ui__react-arrow.mjs +23 -0
  114. package/.output/server/_libs/radix-ui__react-aspect-ratio.mjs +1 -0
  115. package/.output/server/_libs/radix-ui__react-avatar.mjs +1 -0
  116. package/.output/server/_libs/radix-ui__react-checkbox.mjs +1 -0
  117. package/.output/server/_libs/radix-ui__react-collapsible.mjs +144 -0
  118. package/.output/server/_libs/radix-ui__react-collection.mjs +69 -0
  119. package/.output/server/_libs/radix-ui__react-compose-refs.mjs +39 -0
  120. package/.output/server/_libs/radix-ui__react-context-menu.mjs +1 -0
  121. package/.output/server/_libs/radix-ui__react-context.mjs +78 -0
  122. package/.output/server/_libs/radix-ui__react-dialog.mjs +325 -0
  123. package/.output/server/_libs/radix-ui__react-direction.mjs +9 -0
  124. package/.output/server/_libs/radix-ui__react-dropdown-menu.mjs +1 -0
  125. package/.output/server/_libs/radix-ui__react-focus-guards.mjs +29 -0
  126. package/.output/server/_libs/radix-ui__react-focus-scope.mjs +206 -0
  127. package/.output/server/_libs/radix-ui__react-form.mjs +1 -0
  128. package/.output/server/_libs/radix-ui__react-hover-card.mjs +1 -0
  129. package/.output/server/_libs/radix-ui__react-id.mjs +14 -0
  130. package/.output/server/_libs/radix-ui__react-label.mjs +1 -0
  131. package/.output/server/_libs/radix-ui__react-menu.mjs +1 -0
  132. package/.output/server/_libs/radix-ui__react-menubar.mjs +1 -0
  133. package/.output/server/_libs/radix-ui__react-popover.mjs +1 -0
  134. package/.output/server/_libs/radix-ui__react-popper.mjs +286 -0
  135. package/.output/server/_libs/radix-ui__react-portal.mjs +16 -0
  136. package/.output/server/_libs/radix-ui__react-presence.mjs +128 -0
  137. package/.output/server/_libs/radix-ui__react-primitive.mjs +42 -0
  138. package/.output/server/_libs/radix-ui__react-progress.mjs +1 -0
  139. package/.output/server/_libs/radix-ui__react-radio-group.mjs +1 -0
  140. package/.output/server/_libs/radix-ui__react-roving-focus.mjs +224 -0
  141. package/.output/server/_libs/radix-ui__react-scroll-area.mjs +721 -0
  142. package/.output/server/_libs/radix-ui__react-select.mjs +1163 -0
  143. package/.output/server/_libs/radix-ui__react-separator.mjs +28 -0
  144. package/.output/server/_libs/radix-ui__react-slider.mjs +1 -0
  145. package/.output/server/_libs/radix-ui__react-slot.mjs +99 -0
  146. package/.output/server/_libs/radix-ui__react-switch.mjs +1 -0
  147. package/.output/server/_libs/radix-ui__react-tabs.mjs +189 -0
  148. package/.output/server/_libs/radix-ui__react-toast.mjs +2 -0
  149. package/.output/server/_libs/radix-ui__react-toggle-group.mjs +1 -0
  150. package/.output/server/_libs/radix-ui__react-toggle.mjs +1 -0
  151. package/.output/server/_libs/radix-ui__react-toolbar.mjs +1 -0
  152. package/.output/server/_libs/radix-ui__react-tooltip.mjs +495 -0
  153. package/.output/server/_libs/radix-ui__react-use-previous.mjs +14 -0
  154. package/.output/server/_libs/radix-ui__react-use-size.mjs +39 -0
  155. package/.output/server/_libs/react-dom.mjs +10781 -0
  156. package/.output/server/_libs/react-markdown.mjs +147 -0
  157. package/.output/server/_libs/react-remove-scroll-bar.mjs +82 -0
  158. package/.output/server/_libs/react-remove-scroll.mjs +328 -0
  159. package/.output/server/_libs/react-style-singleton.mjs +69 -0
  160. package/.output/server/_libs/react.mjs +515 -0
  161. package/.output/server/_libs/readable-stream.mjs +1518 -0
  162. package/.output/server/_libs/remark-parse.mjs +19 -0
  163. package/.output/server/_libs/remark-rehype.mjs +21 -0
  164. package/.output/server/_libs/rou3.mjs +14 -0
  165. package/.output/server/_libs/safe-buffer.mjs +64 -0
  166. package/.output/server/_libs/semver.mjs +1938 -0
  167. package/.output/server/_libs/seroval-plugins.mjs +58 -0
  168. package/.output/server/_libs/seroval.mjs +1765 -0
  169. package/.output/server/_libs/setimmediate.mjs +152 -0
  170. package/.output/server/_libs/space-separated-tokens.mjs +6 -0
  171. package/.output/server/_libs/srvx.mjs +1029 -0
  172. package/.output/server/_libs/stubborn-fs.mjs +91 -0
  173. package/.output/server/_libs/stubborn-utils.mjs +66 -0
  174. package/.output/server/_libs/style-to-js.mjs +72 -0
  175. package/.output/server/_libs/style-to-object.mjs +38 -0
  176. package/.output/server/_libs/swr.mjs +939 -0
  177. package/.output/server/_libs/tailwind-merge.mjs +3010 -0
  178. package/.output/server/_libs/tanstack__history.mjs +217 -0
  179. package/.output/server/_libs/tanstack__react-router.mjs +1480 -0
  180. package/.output/server/_libs/tanstack__react-store.mjs +1 -0
  181. package/.output/server/_libs/tanstack__react-virtual.mjs +44 -0
  182. package/.output/server/_libs/tanstack__router-core.mjs +4827 -0
  183. package/.output/server/_libs/tanstack__store.mjs +1 -0
  184. package/.output/server/_libs/tanstack__virtual-core.mjs +1225 -0
  185. package/.output/server/_libs/tiny-invariant.mjs +12 -0
  186. package/.output/server/_libs/tiny-warning.mjs +5 -0
  187. package/.output/server/_libs/trim-lines.mjs +41 -0
  188. package/.output/server/_libs/trough.mjs +85 -0
  189. package/.output/server/_libs/tslib.mjs +1 -0
  190. package/.output/server/_libs/ufo.mjs +54 -0
  191. package/.output/server/_libs/uint8array-extras.mjs +69 -0
  192. package/.output/server/_libs/ungap__structured-clone.mjs +212 -0
  193. package/.output/server/_libs/unified.mjs +661 -0
  194. package/.output/server/_libs/unist-util-is.mjs +100 -0
  195. package/.output/server/_libs/unist-util-position.mjs +27 -0
  196. package/.output/server/_libs/unist-util-stringify-position.mjs +27 -0
  197. package/.output/server/_libs/unist-util-visit-parents.mjs +82 -0
  198. package/.output/server/_libs/unist-util-visit.mjs +24 -0
  199. package/.output/server/_libs/unstorage.mjs +1 -0
  200. package/.output/server/_libs/use-callback-ref.mjs +66 -0
  201. package/.output/server/_libs/use-sidecar.mjs +106 -0
  202. package/.output/server/_libs/use-sync-external-store.mjs +64 -0
  203. package/.output/server/_libs/util-deprecate.mjs +12 -0
  204. package/.output/server/_libs/vfile-message.mjs +138 -0
  205. package/.output/server/_libs/vfile.mjs +467 -0
  206. package/.output/server/_libs/when-exit.mjs +53 -0
  207. package/.output/server/_libs/zod.mjs +4524 -0
  208. package/.output/server/_sessionId-wMLPvC5g.mjs +123 -0
  209. package/.output/server/_ssr/CompareDrawer-BU4V0uVf.mjs +1041 -0
  210. package/.output/server/_ssr/ProxyViewerContainer-BnRwFEnn.mjs +5972 -0
  211. package/.output/server/_ssr/ReplayDialog-C7dn9pd_.mjs +322 -0
  212. package/.output/server/_ssr/RequestAnatomy-C1rWpe9-.mjs +353 -0
  213. package/.output/server/_ssr/ResponseView-hGpPaYsf.mjs +602 -0
  214. package/.output/server/_ssr/StreamingChunkSequence-BRWI1r_G.mjs +302 -0
  215. package/.output/server/_ssr/index-BKURLVPz.mjs +118 -0
  216. package/.output/server/_ssr/index.mjs +1184 -0
  217. package/.output/server/_ssr/json-viewer-BBd2DtQP.mjs +515 -0
  218. package/.output/server/_ssr/router-BcZ0D6AB.mjs +6317 -0
  219. package/.output/server/_ssr/start-HYkvq4Ni.mjs +4 -0
  220. package/.output/server/_tanstack-start-manifest_v-1y8ZVxRI.mjs +4 -0
  221. package/.output/server/index.mjs +436 -0
  222. package/.output/server/node_modules/tslib/modules/index.js +70 -0
  223. package/.output/server/node_modules/tslib/modules/package.json +3 -0
  224. package/.output/server/node_modules/tslib/package.json +47 -0
  225. package/.output/server/node_modules/tslib/tslib.js +484 -0
  226. package/.output/server/package.json +9 -0
  227. package/LICENSE +21 -0
  228. package/README.md +52 -0
  229. package/package.json +110 -0
  230. package/src/assets/favicon.svg +31 -0
  231. package/src/assets/logos/alibaba.svg +1 -0
  232. package/src/assets/logos/anthropic.svg +1 -0
  233. package/src/assets/logos/claude-code.svg +4 -0
  234. package/src/assets/logos/deepseek.svg +1 -0
  235. package/src/assets/logos/mcp.png +0 -0
  236. package/src/assets/logos/minimax.jpeg +0 -0
  237. package/src/assets/logos/openai.svg +1 -0
  238. package/src/assets/logos/opencode.svg +4 -0
  239. package/src/assets/logos/qwen.png +0 -0
  240. package/src/assets/logos/zhipuai.svg +219 -0
  241. package/src/cli/detect-tools.ts +147 -0
  242. package/src/cli/doctor.ts +521 -0
  243. package/src/cli/onboard.ts +224 -0
  244. package/src/cli/templates/command-onboard.ts +17 -0
  245. package/src/cli/templates/skill-onboard.ts +547 -0
  246. package/src/cli.ts +345 -0
  247. package/src/components/OnboardingBanner.tsx +67 -0
  248. package/src/components/ProxyViewer.tsx +545 -0
  249. package/src/components/ProxyViewerContainer.tsx +363 -0
  250. package/src/components/providers/ImportWizardDialog.tsx +349 -0
  251. package/src/components/providers/ProviderCard.tsx +474 -0
  252. package/src/components/providers/ProviderForm.tsx +494 -0
  253. package/src/components/providers/ProviderLogo.tsx +117 -0
  254. package/src/components/providers/ProvidersPanel.tsx +619 -0
  255. package/src/components/providers/SettingsDialog.tsx +202 -0
  256. package/src/components/proxy-viewer/CompareDrawer.tsx +893 -0
  257. package/src/components/proxy-viewer/ConversationGroup.tsx +107 -0
  258. package/src/components/proxy-viewer/ConversationHeader.tsx +300 -0
  259. package/src/components/proxy-viewer/LogEntry.tsx +543 -0
  260. package/src/components/proxy-viewer/LogEntryHeader.tsx +501 -0
  261. package/src/components/proxy-viewer/ReplayDialog.tsx +218 -0
  262. package/src/components/proxy-viewer/ResponseView.tsx +171 -0
  263. package/src/components/proxy-viewer/StreamingChunkSequence.tsx +188 -0
  264. package/src/components/proxy-viewer/ThreadConnector.tsx +136 -0
  265. package/src/components/proxy-viewer/TurnGroup.tsx +337 -0
  266. package/src/components/proxy-viewer/anatomy/RequestAnatomy.tsx +98 -0
  267. package/src/components/proxy-viewer/anatomy/SegmentBar.tsx +196 -0
  268. package/src/components/proxy-viewer/anatomy/tokenEstimate.ts +53 -0
  269. package/src/components/proxy-viewer/anatomy/types.ts +39 -0
  270. package/src/components/proxy-viewer/anatomy/useAnatomyJump.ts +114 -0
  271. package/src/components/proxy-viewer/cacheTrend.ts +50 -0
  272. package/src/components/proxy-viewer/diff/DiffView.tsx +321 -0
  273. package/src/components/proxy-viewer/diff/computeDiff.ts +178 -0
  274. package/src/components/proxy-viewer/diff/index.ts +3 -0
  275. package/src/components/proxy-viewer/formats/anthropic/ContentBlocks.tsx +157 -0
  276. package/src/components/proxy-viewer/formats/anthropic/ResponseView.tsx +66 -0
  277. package/src/components/proxy-viewer/formats/anthropic/thinkingExtract.ts +21 -0
  278. package/src/components/proxy-viewer/formats/index.tsx +33 -0
  279. package/src/components/proxy-viewer/formats/openai/ResponseView.tsx +170 -0
  280. package/src/components/proxy-viewer/index.ts +9 -0
  281. package/src/components/proxy-viewer/lazy.ts +37 -0
  282. package/src/components/proxy-viewer/log-formats/anthropic.ts +194 -0
  283. package/src/components/proxy-viewer/log-formats/index.ts +23 -0
  284. package/src/components/proxy-viewer/log-formats/openai.ts +167 -0
  285. package/src/components/proxy-viewer/log-formats/types.ts +40 -0
  286. package/src/components/proxy-viewer/log-formats/unknown.ts +18 -0
  287. package/src/components/proxy-viewer/logEntryVisibility.ts +39 -0
  288. package/src/components/proxy-viewer/requestDiff.ts +277 -0
  289. package/src/components/proxy-viewer/useCopyFeedback.ts +36 -0
  290. package/src/components/proxy-viewer/useKeyboardNavigation.ts +190 -0
  291. package/src/components/proxy-viewer/viewerState.ts +66 -0
  292. package/src/components/ui/badge.tsx +47 -0
  293. package/src/components/ui/button.tsx +47 -0
  294. package/src/components/ui/collapsible.tsx +21 -0
  295. package/src/components/ui/confirm-dialog.tsx +51 -0
  296. package/src/components/ui/crab-logo.tsx +95 -0
  297. package/src/components/ui/crab-variants.tsx +467 -0
  298. package/src/components/ui/dialog.tsx +129 -0
  299. package/src/components/ui/json-expansion-button.tsx +56 -0
  300. package/src/components/ui/json-viewer-bulk.ts +97 -0
  301. package/src/components/ui/json-viewer.tsx +494 -0
  302. package/src/components/ui/mcp-logo.tsx +20 -0
  303. package/src/components/ui/scroll-area.tsx +54 -0
  304. package/src/components/ui/select.tsx +178 -0
  305. package/src/components/ui/separator.tsx +28 -0
  306. package/src/components/ui/tabs.tsx +88 -0
  307. package/src/components/ui/tooltip.tsx +51 -0
  308. package/src/index.css +11 -0
  309. package/src/knowledge/candidateStore.ts +63 -0
  310. package/src/knowledge/distiller.ts +98 -0
  311. package/src/knowledge/openclawClient.ts +118 -0
  312. package/src/knowledge/redactor.ts +80 -0
  313. package/src/knowledge/types.ts +84 -0
  314. package/src/lib/apiClient.ts +49 -0
  315. package/src/lib/export-logs.ts +51 -0
  316. package/src/lib/mask.ts +4 -0
  317. package/src/lib/objectUtils.ts +22 -0
  318. package/src/lib/providerContract.ts +26 -0
  319. package/src/lib/providerTestContract.ts +107 -0
  320. package/src/lib/runtimeConfig.ts +25 -0
  321. package/src/lib/serverPort.ts +41 -0
  322. package/src/lib/sessionUrl.ts +44 -0
  323. package/src/lib/stopReason.ts +58 -0
  324. package/src/lib/useOnboarding.ts +80 -0
  325. package/src/lib/useProviders.ts +30 -0
  326. package/src/lib/useStripConfig.ts +108 -0
  327. package/src/lib/utils.ts +21 -0
  328. package/src/mcp/loopback.ts +76 -0
  329. package/src/mcp/previewExtractor.ts +166 -0
  330. package/src/mcp/server.ts +396 -0
  331. package/src/mcp/toolHandlers.ts +341 -0
  332. package/src/proxy/chunkStorage.ts +112 -0
  333. package/src/proxy/claudeCodeStrip.ts +99 -0
  334. package/src/proxy/config.ts +172 -0
  335. package/src/proxy/constants.ts +47 -0
  336. package/src/proxy/dataDir.ts +86 -0
  337. package/src/proxy/formats/anthropic/anthropicProvider.ts +75 -0
  338. package/src/proxy/formats/anthropic/handler.ts +71 -0
  339. package/src/proxy/formats/anthropic/index.ts +14 -0
  340. package/src/proxy/formats/anthropic/register.ts +4 -0
  341. package/src/proxy/formats/anthropic/schemas.ts +237 -0
  342. package/src/proxy/formats/anthropic/stream.ts +205 -0
  343. package/src/proxy/formats/handler.ts +46 -0
  344. package/src/proxy/formats/index.ts +12 -0
  345. package/src/proxy/formats/jsonSchema.ts +36 -0
  346. package/src/proxy/formats/openai/alibabaProvider.ts +38 -0
  347. package/src/proxy/formats/openai/handler.ts +96 -0
  348. package/src/proxy/formats/openai/index.ts +25 -0
  349. package/src/proxy/formats/openai/provider.ts +50 -0
  350. package/src/proxy/formats/openai/register.ts +4 -0
  351. package/src/proxy/formats/openai/schemas.ts +187 -0
  352. package/src/proxy/formats/openai/stream.ts +206 -0
  353. package/src/proxy/formats/protocol.ts +50 -0
  354. package/src/proxy/formats/providerRegistry.ts +51 -0
  355. package/src/proxy/formats/providers/index.ts +3 -0
  356. package/src/proxy/formats/registry.ts +66 -0
  357. package/src/proxy/handler.ts +334 -0
  358. package/src/proxy/logFinalizer.ts +305 -0
  359. package/src/proxy/logFinalizer.worker.ts +24 -0
  360. package/src/proxy/logIndex.ts +268 -0
  361. package/src/proxy/logger.ts +179 -0
  362. package/src/proxy/openaiOrphanToolStrip.ts +142 -0
  363. package/src/proxy/providerImporters.ts +491 -0
  364. package/src/proxy/providers.ts +613 -0
  365. package/src/proxy/schemas.ts +209 -0
  366. package/src/proxy/sessionProcess.ts +140 -0
  367. package/src/proxy/sessionRuntime.ts +85 -0
  368. package/src/proxy/sessionSupervisor.ts +283 -0
  369. package/src/proxy/sessionWorkerEntry.ts +26 -0
  370. package/src/proxy/socketTracker.ts +255 -0
  371. package/src/proxy/store.ts +412 -0
  372. package/src/proxy/upstream.ts +90 -0
  373. package/src/router.tsx +16 -0
  374. package/src/routes/__root.tsx +45 -0
  375. package/src/routes/api/config.paths.ts +14 -0
  376. package/src/routes/api/config.ts +53 -0
  377. package/src/routes/api/health.ts +15 -0
  378. package/src/routes/api/knowledge.candidates.$candidateId.promote.ts +32 -0
  379. package/src/routes/api/knowledge.candidates.ts +10 -0
  380. package/src/routes/api/knowledge.project-context.ts +18 -0
  381. package/src/routes/api/knowledge.search.ts +31 -0
  382. package/src/routes/api/knowledge.sessions.$sessionId.candidates.ts +16 -0
  383. package/src/routes/api/logs.$id.chunks.ts +36 -0
  384. package/src/routes/api/logs.$id.replay.ts +191 -0
  385. package/src/routes/api/logs.$id.ts +22 -0
  386. package/src/routes/api/logs.stream.ts +74 -0
  387. package/src/routes/api/logs.ts +59 -0
  388. package/src/routes/api/mcp.ts +25 -0
  389. package/src/routes/api/models.ts +10 -0
  390. package/src/routes/api/providers.$providerId.test.log.ts +293 -0
  391. package/src/routes/api/providers.$providerId.ts +50 -0
  392. package/src/routes/api/providers.export.ts +26 -0
  393. package/src/routes/api/providers.import.ts +47 -0
  394. package/src/routes/api/providers.scan.ts +23 -0
  395. package/src/routes/api/providers.ts +45 -0
  396. package/src/routes/api/sessions.ts +17 -0
  397. package/src/routes/index.tsx +6 -0
  398. package/src/routes/proxy/$.ts +15 -0
  399. package/src/routes/session/$sessionId.tsx +23 -0
  400. 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,3 @@
1
+ export { computeHeadersDiff, computeRequestDiff } from "./computeDiff";
2
+ export type { DiffLine, DiffResult } from "./computeDiff";
3
+ export { DiffView } from "./DiffView";
@@ -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
+ });