@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,167 @@
1
+ import { OpenAIRequestSchema, parseOpenAIResponse } from "../../../proxy/formats/openai/schemas";
2
+ import type { AnatomySegment } from "../anatomy/types";
3
+ import { countCharacters, estimateTokens } from "../anatomy/tokenEstimate";
4
+ import type { LogFormatAdapter } from "./types";
5
+ import { emptyRequestAnalysis, EMPTY_RESPONSE_ANALYSIS } from "./types";
6
+
7
+ type OpenAIMessageContent = string | ReadonlyArray<Record<string, unknown>> | null | undefined;
8
+
9
+ /** Flatten an OpenAI `content` field to its text representation. */
10
+ function contentToText(content: unknown): string {
11
+ if (typeof content === "string") return content;
12
+ if (!Array.isArray(content)) return "";
13
+ const parts: string[] = [];
14
+ for (const block of content) {
15
+ if (block === null || typeof block !== "object") continue;
16
+ // eslint-disable-next-line @typescript-eslint/consistent-type-assertions
17
+ const b = block as Record<string, unknown>;
18
+ const type = b.type;
19
+ if (type === "text" && typeof b.text === "string") {
20
+ parts.push(b.text);
21
+ } else if (type === "image_url") {
22
+ parts.push("[image]");
23
+ }
24
+ }
25
+ return parts.join("\n");
26
+ }
27
+
28
+ /** Flatten an OpenAI message to text — content plus optional tool calls plus optional reasoning. */
29
+ function messageToText(message: Record<string, unknown>): string {
30
+ const parts: string[] = [];
31
+ const text = contentToText(message.content);
32
+ if (text.length > 0) parts.push(text);
33
+ const reasoning =
34
+ typeof message.reasoning_content === "string"
35
+ ? message.reasoning_content
36
+ : typeof message.thinking === "string"
37
+ ? message.thinking
38
+ : typeof message.think === "string"
39
+ ? message.think
40
+ : "";
41
+ if (reasoning.length > 0) parts.push(reasoning);
42
+ const toolCalls = message.tool_calls;
43
+ if (Array.isArray(toolCalls)) {
44
+ for (const call of toolCalls) {
45
+ if (call === null || typeof call !== "object") continue;
46
+ // eslint-disable-next-line @typescript-eslint/consistent-type-assertions
47
+ const fn = (call as { function?: { name?: unknown; arguments?: unknown } }).function;
48
+ if (fn === undefined) continue;
49
+ const name = typeof fn.name === "string" ? fn.name : "";
50
+ const args = typeof fn.arguments === "string" ? fn.arguments : "";
51
+ parts.push(`${name} ${args}`.trim());
52
+ }
53
+ }
54
+ return parts.join("\n");
55
+ }
56
+
57
+ /** Flatten an OpenAI `tools` array to its schema text. */
58
+ function toolsToText(tools: unknown): string {
59
+ if (!Array.isArray(tools)) return "";
60
+ const parts: string[] = [];
61
+ for (const tool of tools) {
62
+ if (tool === null || typeof tool !== "object") continue;
63
+ // eslint-disable-next-line @typescript-eslint/consistent-type-assertions
64
+ const t = tool as {
65
+ function?: { name?: unknown; description?: unknown; parameters?: unknown };
66
+ };
67
+ const fn = t.function;
68
+ if (fn === undefined) continue;
69
+ if (typeof fn.name === "string") parts.push(fn.name);
70
+ if (typeof fn.description === "string") parts.push(fn.description);
71
+ if (fn.parameters !== undefined) parts.push(JSON.stringify(fn.parameters));
72
+ }
73
+ return parts.join("\n");
74
+ }
75
+
76
+ function segment(
77
+ role: AnatomySegment["role"],
78
+ label: string,
79
+ text: string,
80
+ path: string,
81
+ ): AnatomySegment {
82
+ return {
83
+ role,
84
+ label,
85
+ text,
86
+ size: estimateTokens(text),
87
+ characters: countCharacters(text),
88
+ path,
89
+ };
90
+ }
91
+
92
+ export const openAILogFormatAdapter: LogFormatAdapter = {
93
+ format: "openai",
94
+
95
+ analyzeRequest(rawBody) {
96
+ if (rawBody === null) return emptyRequestAnalysis(rawBody);
97
+ try {
98
+ const result = OpenAIRequestSchema.safeParse(JSON.parse(rawBody));
99
+ if (!result.success) return emptyRequestAnalysis(rawBody);
100
+ return {
101
+ parsed: result.data,
102
+ comparisonValue: result.data,
103
+ messageCount: result.data.messages.length,
104
+ toolCount:
105
+ result.data.tools !== undefined && result.data.tools.length > 0
106
+ ? result.data.tools.length
107
+ : null,
108
+ };
109
+ } catch {
110
+ return emptyRequestAnalysis(rawBody);
111
+ }
112
+ },
113
+
114
+ analyzeResponse(responseText) {
115
+ if (responseText === null) return EMPTY_RESPONSE_ANALYSIS;
116
+ const parsed = parseOpenAIResponse(responseText);
117
+ if (parsed === null) return EMPTY_RESPONSE_ANALYSIS;
118
+ const toolNames =
119
+ parsed.choices[0]?.message?.tool_calls
120
+ ?.map((toolCall) => toolCall.function?.name ?? "")
121
+ .filter((name) => name !== "") ?? [];
122
+ return {
123
+ parsed,
124
+ toolNames: toolNames.length > 0 ? toolNames : null,
125
+ };
126
+ },
127
+
128
+ anatomySegments(parsed) {
129
+ if (parsed === null || typeof parsed !== "object") return null;
130
+ // We deliberately skip OpenAIRequestSchema validation here for the
131
+ // same reason as the Anthropic adapter: the Anatomy view should
132
+ // render even when the body shape is slightly off-schema. We only
133
+ // need the top-level `messages` and `tools` arrays.
134
+ // eslint-disable-next-line @typescript-eslint/consistent-type-assertions
135
+ const body = parsed as { messages?: unknown; tools?: unknown };
136
+ const segments: AnatomySegment[] = [];
137
+
138
+ // Promote a leading role: "system" message into a "system" segment.
139
+ // Spec: derive system role from a leading role: "system" message, then
140
+ // a segment per remaining message. We still keep the system message in
141
+ // the segment list (not skipped) when there is no separate system field
142
+ // to compare against — the leading role: "system" message IS the system
143
+ // content for OpenAI Chat Completions. If a separate `system` field
144
+ // existed and matched the message, we would dedupe, but Chat
145
+ // Completions has no such field, so we never have to skip.
146
+ if (Array.isArray(body.messages)) {
147
+ body.messages.forEach((message, index) => {
148
+ if (message === null || typeof message !== "object") return;
149
+ // eslint-disable-next-line @typescript-eslint/consistent-type-assertions
150
+ const m = message as { role?: unknown; content?: unknown };
151
+ const role =
152
+ m.role === "user" || m.role === "assistant" || m.role === "system" || m.role === "tool"
153
+ ? m.role
154
+ : "user";
155
+ const text = messageToText(m);
156
+ segments.push(segment(role, `[${index}] ${role}`, text, `/messages/${index}`));
157
+ });
158
+ }
159
+
160
+ if (Array.isArray(body.tools) && body.tools.length > 0) {
161
+ const text = toolsToText(body.tools);
162
+ segments.push(segment("tools", "tools", text, "/tools"));
163
+ }
164
+
165
+ return segments.length > 0 ? segments : null;
166
+ },
167
+ };
@@ -0,0 +1,40 @@
1
+ import type { RequestFormat } from "../../../proxy/schemas";
2
+ import type { AnatomySegment } from "../anatomy/types";
3
+
4
+ export type RequestAnalysis = {
5
+ parsed: unknown | null;
6
+ comparisonValue: unknown;
7
+ messageCount: number | null;
8
+ toolCount: number | null;
9
+ };
10
+
11
+ export type ResponseAnalysis = {
12
+ parsed: unknown | null;
13
+ toolNames: string[] | null;
14
+ };
15
+
16
+ export type LogFormatAdapter = {
17
+ readonly format: RequestFormat;
18
+ analyzeRequest(rawBody: string | null): RequestAnalysis;
19
+ analyzeResponse(responseText: string | null): ResponseAnalysis;
20
+ /**
21
+ * Derive the ordered list of segments shown in the Anatomy tab for a
22
+ * parsed request body. Returns `null` when the body is `null` or fails
23
+ * the format's schema (e.g. unknown format).
24
+ */
25
+ anatomySegments(parsed: unknown): AnatomySegment[] | null;
26
+ };
27
+
28
+ export function emptyRequestAnalysis(rawBody: string | null): RequestAnalysis {
29
+ return {
30
+ parsed: null,
31
+ comparisonValue: rawBody,
32
+ messageCount: null,
33
+ toolCount: null,
34
+ };
35
+ }
36
+
37
+ export const EMPTY_RESPONSE_ANALYSIS: ResponseAnalysis = {
38
+ parsed: null,
39
+ toolNames: null,
40
+ };
@@ -0,0 +1,18 @@
1
+ import type { LogFormatAdapter } from "./types";
2
+ import { emptyRequestAnalysis, EMPTY_RESPONSE_ANALYSIS } from "./types";
3
+
4
+ export const unknownLogFormatAdapter: LogFormatAdapter = {
5
+ format: "unknown",
6
+
7
+ analyzeRequest(rawBody) {
8
+ return emptyRequestAnalysis(rawBody);
9
+ },
10
+
11
+ analyzeResponse() {
12
+ return EMPTY_RESPONSE_ANALYSIS;
13
+ },
14
+
15
+ anatomySegments() {
16
+ return null;
17
+ },
18
+ };
@@ -0,0 +1,39 @@
1
+ /**
2
+ * Pure visibility rule for the "Raw Request" tab. Extracted so it can be
3
+ * unit-tested without rendering React. The tab appears only when all three
4
+ * are true: anthropic format, full view mode, strip toggle enabled.
5
+ */
6
+ export function shouldShowRawRequestTab(
7
+ apiFormat: string,
8
+ viewMode: "simple" | "full",
9
+ strip: boolean,
10
+ ): boolean {
11
+ return apiFormat === "anthropic" && viewMode === "full" && strip;
12
+ }
13
+
14
+ /**
15
+ * Pure visibility rule for the "Diff with Raw" button in the Headers tab.
16
+ * The button only makes sense when the user is in full mode (where the
17
+ * `Raw Headers` tab is shown) and we actually captured raw headers.
18
+ */
19
+ export function shouldShowHeadersDiffButton(
20
+ viewMode: "simple" | "full",
21
+ hasRawHeaders: boolean,
22
+ ): boolean {
23
+ return viewMode === "full" && hasRawHeaders;
24
+ }
25
+
26
+ /**
27
+ * Pure visibility rule for the "Diff with Raw" button in the Request tab.
28
+ * Mirrors the conditions for the `Raw Request` tab itself: full mode plus
29
+ * the strip toggle being on for an anthropic-format request. We also need
30
+ * an actual raw request body to diff against.
31
+ */
32
+ export function shouldShowRequestDiffButton(
33
+ apiFormat: string,
34
+ viewMode: "simple" | "full",
35
+ strip: boolean,
36
+ hasRawRequest: boolean,
37
+ ): boolean {
38
+ return apiFormat === "anthropic" && viewMode === "full" && strip && hasRawRequest;
39
+ }
@@ -0,0 +1,277 @@
1
+ /**
2
+ * Path-aligned JSON-tree diff helpers for the "compare two log requests" feature.
3
+ *
4
+ * Given two captured log Request payloads, we:
5
+ * 1. Normalize each payload (deep-clone, sort object keys, keep array order)
6
+ * so that non-semantic differences (key order, string whitespace) do not
7
+ * generate false positives.
8
+ * 2. Walk the two trees in lockstep, emitting a list of path-anchored
9
+ * `DiffOp`s. Equal subtrees collapse into a single op at the subtree
10
+ * root, so the renderer can choose to expand or hide them.
11
+ * 3. Emit ops in path-sorted order (depth-first, sibling order = object-key
12
+ * ascending then array-index ascending) so the renderer can lay them
13
+ * out linearly.
14
+ *
15
+ * No runtime dependencies. Mirrors the `cacheTrend.ts` pure-helper pattern.
16
+ */
17
+
18
+ export type JsonPrimitive = string | number | boolean | null;
19
+ export type JsonValue = JsonPrimitive | JsonValue[] | { [key: string]: JsonValue };
20
+
21
+ export type JsonNode =
22
+ | { kind: "object"; value: Record<string, JsonNode> }
23
+ | { kind: "array"; value: JsonNode[] }
24
+ | { kind: "primitive"; value: JsonPrimitive };
25
+
26
+ export type DiffOp =
27
+ | { kind: "equal"; path: string; value: JsonNode }
28
+ | { kind: "added"; path: string; value: JsonNode }
29
+ | { kind: "removed"; path: string; value: JsonNode }
30
+ | {
31
+ kind: "changed";
32
+ path: string;
33
+ left: JsonNode;
34
+ right: JsonNode;
35
+ };
36
+
37
+ /** A single segment of a JSON path: an object key or an array index. */
38
+ export type PathSegment = string | number;
39
+
40
+ const ROOT_PATH = "";
41
+
42
+ /** Render a path segment list as a human-readable string.
43
+ *
44
+ * - Root (empty list) → `""` (no label; the renderer hides the gutter)
45
+ * - Object key → `.foo`
46
+ * - Array index → `[3]`
47
+ * - Top-level object key → `messages` (no leading dot)
48
+ *
49
+ * Examples:
50
+ * [] → ""
51
+ * ["messages"] → "messages"
52
+ * ["messages", 3] → "messages[3]"
53
+ * ["messages", 3, "content"] → "messages[3].content"
54
+ * ["messages", 3, "content", 0] → "messages[3].content[0]"
55
+ */
56
+ export function formatPath(segments: PathSegment[]): string {
57
+ if (segments.length === 0) return ROOT_PATH;
58
+ let out = "";
59
+ for (let i = 0; i < segments.length; i++) {
60
+ const seg = segments[i];
61
+ if (seg === undefined) continue;
62
+ if (typeof seg === "number") {
63
+ out += `[${seg}]`;
64
+ } else if (i === 0) {
65
+ out += seg;
66
+ } else {
67
+ out += `.${seg}`;
68
+ }
69
+ }
70
+ return out;
71
+ }
72
+
73
+ function isPlainObject(value: unknown): value is Record<string, unknown> {
74
+ return typeof value === "object" && value !== null && !Array.isArray(value);
75
+ }
76
+
77
+ /**
78
+ * Normalize a captured Request body for comparison.
79
+ *
80
+ * Captured Request bodies are sometimes already-parsed objects and sometimes
81
+ * JSON-encoded strings (depending on which capture path produced the log).
82
+ * We:
83
+ * 1. Try `JSON.parse` if the input is a string.
84
+ * 2. Deep-walk the value, building a fresh `JsonNode` tree.
85
+ * 3. Sort object keys lexicographically (key order is not semantically
86
+ * meaningful for most LLM SDKs but varies between SDK versions, so we
87
+ * canonicalize it).
88
+ * 4. Keep array order (semantically meaningful: `messages[]` order is
89
+ * conversation order; reordering it would silently destroy the diff).
90
+ * 5. Leave primitives alone (no whitespace trim — those can be meaningful
91
+ * in user-message text).
92
+ *
93
+ * Idempotent at the data level: re-normalizing the same shape (in any key
94
+ * order) yields the same tree.
95
+ */
96
+ export function normalizeRequest(raw: unknown): JsonNode {
97
+ if (typeof raw === "string") {
98
+ try {
99
+ return toNode(JSON.parse(raw));
100
+ } catch {
101
+ // Unparseable string: wrap as a primitive string node so the diff can
102
+ // still operate on it (changed vs. another string, or added/removed
103
+ // vs. an object).
104
+ return { kind: "primitive", value: raw };
105
+ }
106
+ }
107
+ return toNode(raw);
108
+ }
109
+
110
+ function toNode(value: unknown): JsonNode {
111
+ if (value === null) return { kind: "primitive", value: null };
112
+ if (typeof value === "string") return { kind: "primitive", value };
113
+ if (typeof value === "number") return { kind: "primitive", value };
114
+ if (typeof value === "boolean") return { kind: "primitive", value };
115
+ if (Array.isArray(value)) {
116
+ return { kind: "array", value: value.map((v) => toNode(v)) };
117
+ }
118
+ if (isPlainObject(value)) {
119
+ const out: Record<string, JsonNode> = {};
120
+ for (const k of Object.keys(value).sort()) {
121
+ out[k] = toNode(value[k]);
122
+ }
123
+ return { kind: "object", value: out };
124
+ }
125
+ // Functions, symbols, bigints, undefined — treat as null in the diff model.
126
+ return { kind: "primitive", value: null };
127
+ }
128
+
129
+ /**
130
+ * Compute the path-aligned diff of two normalized trees.
131
+ *
132
+ * Emits `DiffOp[]` in path-sorted order. For each path:
133
+ * - if both trees have the same value, emit a single `equal` op at the
134
+ * deepest common ancestor (so equal subtrees collapse into one op);
135
+ * - if only the right has it, emit `added`;
136
+ * - if only the left has it, emit `removed`;
137
+ * - if both have it but it differs, emit `changed` (and recurse into
138
+ * objects/arrays so the user sees *which* field changed).
139
+ *
140
+ * Pure: does not mutate `left` or `right`.
141
+ */
142
+ export function diffTrees(left: JsonNode, right: JsonNode): DiffOp[] {
143
+ const ops: DiffOp[] = [];
144
+ walk([], left, right, ops);
145
+ return ops;
146
+ }
147
+
148
+ function walk(segments: PathSegment[], left: JsonNode, right: JsonNode, out: DiffOp[]): void {
149
+ const path = formatPath(segments);
150
+
151
+ if (nodeEqual(left, right)) {
152
+ out.push({ kind: "equal", path, value: left });
153
+ return;
154
+ }
155
+
156
+ // Type mismatch or primitive change.
157
+ if (left.kind !== right.kind) {
158
+ out.push({ kind: "changed", path, left, right });
159
+ return;
160
+ }
161
+
162
+ if (left.kind === "primitive" && right.kind === "primitive") {
163
+ out.push({ kind: "changed", path, left, right });
164
+ return;
165
+ }
166
+
167
+ if (left.kind === "object" && right.kind === "object") {
168
+ const leftKeys = Object.keys(left.value);
169
+ const rightKeys = Object.keys(right.value);
170
+ const rightKeySet = new Set(rightKeys);
171
+
172
+ for (const k of leftKeys) {
173
+ const lChild = left.value[k];
174
+ if (lChild === undefined) continue;
175
+ if (!rightKeySet.has(k)) {
176
+ out.push({
177
+ kind: "removed",
178
+ path: formatPath([...segments, k]),
179
+ value: lChild,
180
+ });
181
+ } else {
182
+ const rChild = right.value[k];
183
+ if (rChild === undefined) continue;
184
+ walk([...segments, k], lChild, rChild, out);
185
+ }
186
+ }
187
+ for (const k of rightKeys) {
188
+ if (leftKeys.includes(k)) continue;
189
+ const rChild = right.value[k];
190
+ if (rChild === undefined) continue;
191
+ out.push({
192
+ kind: "added",
193
+ path: formatPath([...segments, k]),
194
+ value: rChild,
195
+ });
196
+ }
197
+ return;
198
+ }
199
+
200
+ if (left.kind === "array" && right.kind === "array") {
201
+ const minLen = Math.min(left.value.length, right.value.length);
202
+ for (let i = 0; i < minLen; i++) {
203
+ const lChild = left.value[i];
204
+ const rChild = right.value[i];
205
+ if (lChild === undefined || rChild === undefined) continue;
206
+ walk([...segments, i], lChild, rChild, out);
207
+ }
208
+ for (let i = minLen; i < right.value.length; i++) {
209
+ const rChild = right.value[i];
210
+ if (rChild === undefined) continue;
211
+ out.push({
212
+ kind: "added",
213
+ path: formatPath([...segments, i]),
214
+ value: rChild,
215
+ });
216
+ }
217
+ for (let i = minLen; i < left.value.length; i++) {
218
+ const lChild = left.value[i];
219
+ if (lChild === undefined) continue;
220
+ out.push({
221
+ kind: "removed",
222
+ path: formatPath([...segments, i]),
223
+ value: lChild,
224
+ });
225
+ }
226
+ }
227
+ }
228
+
229
+ function nodeEqual(a: JsonNode, b: JsonNode): boolean {
230
+ if (a.kind !== b.kind) return false;
231
+ if (a.kind === "primitive" && b.kind === "primitive") {
232
+ return a.value === b.value;
233
+ }
234
+ if (a.kind === "array" && b.kind === "array") {
235
+ if (a.value.length !== b.value.length) return false;
236
+ for (let i = 0; i < a.value.length; i++) {
237
+ const ai = a.value[i];
238
+ const bi = b.value[i];
239
+ if (ai === undefined || bi === undefined) return false;
240
+ if (!nodeEqual(ai, bi)) return false;
241
+ }
242
+ return true;
243
+ }
244
+ if (a.kind === "object" && b.kind === "object") {
245
+ const aKeys = Object.keys(a.value);
246
+ const bKeys = Object.keys(b.value);
247
+ if (aKeys.length !== bKeys.length) return false;
248
+ for (const k of aKeys) {
249
+ const av = a.value[k];
250
+ const bv = b.value[k];
251
+ if (av === undefined || bv === undefined) return false;
252
+ if (!nodeEqual(av, bv)) return false;
253
+ }
254
+ return true;
255
+ }
256
+ return false;
257
+ }
258
+
259
+ /** Render a JsonNode to a compact human-readable preview. Used by the
260
+ * diff view's gutter or change rows where we want a one-line summary
261
+ * without pretty-printing the whole subtree. */
262
+ export function previewNode(node: JsonNode, maxLen = 80): string {
263
+ let s: string;
264
+ switch (node.kind) {
265
+ case "primitive":
266
+ s = node.value === null ? "null" : JSON.stringify(node.value);
267
+ break;
268
+ case "array":
269
+ s = `[… ${node.value.length} items]`;
270
+ break;
271
+ case "object":
272
+ s = `{… ${Object.keys(node.value).length} keys}`;
273
+ break;
274
+ }
275
+ if (s.length > maxLen) s = `${s.slice(0, maxLen - 1)}…`;
276
+ return s;
277
+ }
@@ -0,0 +1,36 @@
1
+ import { useCallback, useEffect, useRef, useState, type MouseEvent } from "react";
2
+
3
+ /**
4
+ * Clipboard write with a 2s "Copied!" indicator. Returns null-safe
5
+ * callbacks so callers can pass `text === null` through unconditionally
6
+ * (the hook still runs; the click handler short-circuits).
7
+ */
8
+ export function useCopyFeedback(text: string | null): {
9
+ copied: boolean;
10
+ copy: (event: MouseEvent) => void;
11
+ } {
12
+ const [copied, setCopied] = useState(false);
13
+ const timerRef = useRef<ReturnType<typeof setTimeout> | null>(null);
14
+
15
+ useEffect(
16
+ () => () => {
17
+ if (timerRef.current !== null) clearTimeout(timerRef.current);
18
+ },
19
+ [],
20
+ );
21
+
22
+ const copy = useCallback(
23
+ (event: MouseEvent) => {
24
+ event.stopPropagation();
25
+ if (text === null) return;
26
+ void window.navigator.clipboard.writeText(text).then(() => {
27
+ setCopied(true);
28
+ if (timerRef.current !== null) clearTimeout(timerRef.current);
29
+ timerRef.current = setTimeout(() => setCopied(false), 2000);
30
+ });
31
+ },
32
+ [text],
33
+ );
34
+
35
+ return { copied, copy };
36
+ }