@brainfish-ai/devdoc 0.1.41 → 0.1.43

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/ai-agents/.claude/skills/bootstrap-docs/SKILL.md +710 -79
  2. package/ai-agents/.claude/skills/check-docs/SKILL.md +83 -8
  3. package/ai-agents/.claude/skills/create-doc/SKILL.md +267 -55
  4. package/ai-agents/.claude/skills/update-doc/SKILL.md +162 -63
  5. package/ai-agents/.cursor/rules/devdoc-bootstrap.mdc +145 -15
  6. package/ai-agents/.cursor/rules/devdoc-create.mdc +108 -57
  7. package/ai-agents/.cursor/rules/devdoc-update.mdc +93 -70
  8. package/ai-agents/.cursor/rules/devdoc.mdc +21 -0
  9. package/ai-agents/schemas/docs.schema.json +332 -0
  10. package/ai-agents/schemas/theme.schema.json +243 -0
  11. package/dist/cli/commands/create.js +4 -9
  12. package/dist/cli/commands/deploy.js +50 -25
  13. package/dist/cli/commands/dev.js +19 -10
  14. package/package.json +3 -2
  15. package/renderer/app/api/assets/[...path]/route.js +108 -0
  16. package/renderer/app/api/assets/route.js +114 -0
  17. package/renderer/app/api/assets/upload/route.js +163 -0
  18. package/renderer/app/api/auth-schemes/route.js +58 -0
  19. package/renderer/app/api/chat/route.js +759 -0
  20. package/renderer/app/api/codegen/route.js +52 -0
  21. package/renderer/app/api/collections/route.js +675 -0
  22. package/renderer/app/api/debug/route.js +47 -0
  23. package/renderer/app/api/deploy/route.js +199 -0
  24. package/renderer/app/api/device/route.js +36 -0
  25. package/renderer/app/api/docs/route.js +205 -0
  26. package/renderer/app/api/domains/add/route.js +121 -0
  27. package/renderer/app/api/domains/lookup/route.js +43 -0
  28. package/renderer/app/api/domains/remove/route.js +89 -0
  29. package/renderer/app/api/domains/status/route.js +140 -0
  30. package/renderer/app/api/domains/verify/route.js +168 -0
  31. package/renderer/app/api/keys/regenerate/route.js +71 -0
  32. package/renderer/app/api/local-assets/[...path]/route.js +108 -0
  33. package/renderer/app/api/openapi-spec/route.js +73 -0
  34. package/renderer/app/api/projects/[slug]/route.js +129 -0
  35. package/renderer/app/api/projects/[slug]/stats/route.js +80 -0
  36. package/renderer/app/api/projects/register/route.js +176 -0
  37. package/renderer/app/api/proxy/route.js +139 -0
  38. package/renderer/app/api/proxy-stream/route.js +156 -0
  39. package/renderer/app/api/redirects/route.js +35 -0
  40. package/renderer/app/api/schema/route.js +85 -0
  41. package/renderer/app/api/subdomains/check/route.js +158 -0
  42. package/renderer/app/api/suggestions/route.js +175 -0
  43. package/renderer/app/layout.js +47 -0
  44. package/renderer/app/llms-full.txt/route.js +257 -0
  45. package/renderer/app/llms.txt/route.js +219 -0
  46. package/renderer/app/page.js +12 -0
  47. package/renderer/app/robots.txt/route.js +66 -0
  48. package/renderer/app/sitemap.xml/route.js +145 -0
  49. package/renderer/components/docs/index.js +8 -0
  50. package/renderer/components/docs/mdx/accordion.js +113 -0
  51. package/renderer/components/docs/mdx/badge.js +72 -0
  52. package/renderer/components/docs/mdx/callouts.js +137 -0
  53. package/renderer/components/docs/mdx/cards.js +175 -0
  54. package/renderer/components/docs/mdx/changelog.js +100 -0
  55. package/renderer/components/docs/mdx/code-block.js +147 -0
  56. package/renderer/components/docs/mdx/code-group.js +287 -0
  57. package/renderer/components/docs/mdx/file-embeds.js +82 -0
  58. package/renderer/components/docs/mdx/frame.js +59 -0
  59. package/renderer/components/docs/mdx/highlight.js +90 -0
  60. package/renderer/components/docs/mdx/iframe.js +69 -0
  61. package/renderer/components/docs/mdx/image.js +135 -0
  62. package/renderer/components/docs/mdx/index.js +134 -0
  63. package/renderer/components/docs/mdx/landing.js +315 -0
  64. package/renderer/components/docs/mdx/mermaid.js +212 -0
  65. package/renderer/components/docs/mdx/param-field.js +112 -0
  66. package/renderer/components/docs/mdx/steps.js +74 -0
  67. package/renderer/components/docs/mdx/tabs.js +50 -0
  68. package/renderer/components/docs/mdx-renderer.js +77 -0
  69. package/renderer/components/docs/navigation/breadcrumbs.js +64 -0
  70. package/renderer/components/docs/navigation/index.js +6 -0
  71. package/renderer/components/docs/navigation/page-nav.js +57 -0
  72. package/renderer/components/docs/navigation/sidebar.js +375 -0
  73. package/renderer/components/docs/navigation/toc.js +89 -0
  74. package/renderer/components/docs/notice.js +77 -0
  75. package/renderer/components/docs-header.js +202 -0
  76. package/renderer/components/docs-viewer/agent/agent-chat.js +1930 -0
  77. package/renderer/components/docs-viewer/agent/cards/debug-context-card.js +107 -0
  78. package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.js +57 -0
  79. package/renderer/components/docs-viewer/agent/cards/index.js +45 -0
  80. package/renderer/components/docs-viewer/agent/cards/response-options-card.js +154 -0
  81. package/renderer/components/docs-viewer/agent/cards/types.js +22 -0
  82. package/renderer/components/docs-viewer/agent/chat-message.js +2 -0
  83. package/renderer/components/docs-viewer/agent/index.js +4 -0
  84. package/renderer/components/docs-viewer/agent/messages/assistant-message.js +108 -0
  85. package/renderer/components/docs-viewer/agent/messages/chat-message.js +34 -0
  86. package/renderer/components/docs-viewer/agent/messages/index.js +6 -0
  87. package/renderer/components/docs-viewer/agent/messages/tool-call-display.js +1065 -0
  88. package/renderer/components/docs-viewer/agent/messages/types.js +2 -0
  89. package/renderer/components/docs-viewer/agent/messages/typing-indicator.js +26 -0
  90. package/renderer/components/docs-viewer/agent/messages/user-message.js +37 -0
  91. package/renderer/components/docs-viewer/code-editor/{index.tsx → index.js} +1 -1
  92. package/renderer/components/docs-viewer/code-editor/notes-mode.js +1338 -0
  93. package/renderer/components/docs-viewer/content/changelog-page.js +297 -0
  94. package/renderer/components/docs-viewer/content/doc-page.js +264 -0
  95. package/renderer/components/docs-viewer/content/documentation-viewer.js +14 -0
  96. package/renderer/components/docs-viewer/content/index.js +29 -0
  97. package/renderer/components/docs-viewer/content/not-found-page.js +300 -0
  98. package/renderer/components/docs-viewer/content/request-details.js +528 -0
  99. package/renderer/components/docs-viewer/content/sections/auth.js +108 -0
  100. package/renderer/components/docs-viewer/content/sections/body.js +80 -0
  101. package/renderer/components/docs-viewer/content/sections/headers.js +64 -0
  102. package/renderer/components/docs-viewer/content/sections/overview.js +56 -0
  103. package/renderer/components/docs-viewer/content/sections/parameters.js +64 -0
  104. package/renderer/components/docs-viewer/content/sections/responses.js +91 -0
  105. package/renderer/components/docs-viewer/global-auth-modal.js +427 -0
  106. package/renderer/components/docs-viewer/index.js +1552 -0
  107. package/renderer/components/docs-viewer/playground/auth-editor.js +418 -0
  108. package/renderer/components/docs-viewer/playground/body-editor.js +240 -0
  109. package/renderer/components/docs-viewer/playground/code-editor.js +135 -0
  110. package/renderer/components/docs-viewer/playground/code-snippet.js +393 -0
  111. package/renderer/components/docs-viewer/playground/graphql-playground.js +734 -0
  112. package/renderer/components/docs-viewer/playground/index.js +682 -0
  113. package/renderer/components/docs-viewer/playground/key-value-editor.js +317 -0
  114. package/renderer/components/docs-viewer/playground/method-selector.js +65 -0
  115. package/renderer/components/docs-viewer/playground/request-builder.js +181 -0
  116. package/renderer/components/docs-viewer/playground/request-tabs.js +240 -0
  117. package/renderer/components/docs-viewer/playground/response-cards/idle-card.js +42 -0
  118. package/renderer/components/docs-viewer/playground/response-cards/index.js +72 -0
  119. package/renderer/components/docs-viewer/playground/response-cards/loading-card.js +24 -0
  120. package/renderer/components/docs-viewer/playground/response-cards/network-error-card.js +28 -0
  121. package/renderer/components/docs-viewer/playground/response-cards/response-body-card.js +308 -0
  122. package/renderer/components/docs-viewer/playground/response-cards/types.js +9 -0
  123. package/renderer/components/docs-viewer/playground/response-viewer.js +18 -0
  124. package/renderer/components/docs-viewer/search/index.js +2 -0
  125. package/renderer/components/docs-viewer/search/search-dialog.js +367 -0
  126. package/renderer/components/docs-viewer/search/use-search.js +89 -0
  127. package/renderer/components/docs-viewer/shared/markdown-renderer.js +423 -0
  128. package/renderer/components/docs-viewer/shared/method-badge.js +23 -0
  129. package/renderer/components/docs-viewer/shared/schema-viewer.js +321 -0
  130. package/renderer/components/docs-viewer/sidebar/collection-tree.js +222 -0
  131. package/renderer/components/docs-viewer/sidebar/endpoint-options.js +512 -0
  132. package/renderer/components/docs-viewer/sidebar/index.js +196 -0
  133. package/renderer/components/docs-viewer/sidebar/right-sidebar.js +163 -0
  134. package/renderer/components/docs-viewer/sidebar/sidebar-group.js +87 -0
  135. package/renderer/components/docs-viewer/sidebar/sidebar-item.js +172 -0
  136. package/renderer/components/docs-viewer/sidebar/sidebar-section.js +31 -0
  137. package/renderer/components/theme-provider.js +10 -0
  138. package/renderer/components/theme-toggle.js +106 -0
  139. package/renderer/components/ui/badge.js +29 -0
  140. package/renderer/components/ui/button.js +40 -0
  141. package/renderer/components/ui/dialog.js +50 -0
  142. package/renderer/components/ui/dropdown-menu.js +143 -0
  143. package/renderer/components/ui/input.js +12 -0
  144. package/renderer/components/ui/label.js +13 -0
  145. package/renderer/components/ui/navigation-menu.js +83 -0
  146. package/renderer/components/ui/select.js +116 -0
  147. package/renderer/components/ui/spinner.js +92 -0
  148. package/renderer/components/ui/tabs.js +34 -0
  149. package/renderer/components/ui/tooltip.js +43 -0
  150. package/renderer/hooks/use-code-copy.js +76 -0
  151. package/renderer/hooks/use-openapi-title.js +33 -0
  152. package/renderer/lib/api-docs/agent/index.js +4 -0
  153. package/renderer/lib/api-docs/agent/indexer.js +254 -0
  154. package/renderer/lib/api-docs/agent/spec-summary.js +227 -0
  155. package/renderer/lib/api-docs/agent/types.js +5 -0
  156. package/renderer/lib/api-docs/auth/auth-context.js +157 -0
  157. package/renderer/lib/api-docs/auth/auth-storage.js +66 -0
  158. package/renderer/lib/api-docs/auth/crypto.js +64 -0
  159. package/renderer/lib/api-docs/auth/index.js +3 -0
  160. package/renderer/lib/api-docs/code-editor/db.js +145 -0
  161. package/renderer/lib/api-docs/code-editor/hooks.js +254 -0
  162. package/renderer/lib/api-docs/code-editor/{index.ts → index.js} +3 -4
  163. package/renderer/lib/api-docs/code-editor/mode-context.js +154 -0
  164. package/renderer/lib/api-docs/code-editor/types.js +53 -0
  165. package/renderer/lib/api-docs/codegen/definitions.js +258 -0
  166. package/renderer/lib/api-docs/codegen/har.js +171 -0
  167. package/renderer/lib/api-docs/codegen/index.js +118 -0
  168. package/renderer/lib/api-docs/factories.js +136 -0
  169. package/renderer/lib/api-docs/{index.ts → index.js} +5 -10
  170. package/renderer/lib/api-docs/mobile-context.js +79 -0
  171. package/renderer/lib/api-docs/navigation-context.js +62 -0
  172. package/renderer/lib/api-docs/parsers/graphql/index.js +50 -0
  173. package/renderer/lib/api-docs/parsers/graphql/parser.js +350 -0
  174. package/renderer/lib/api-docs/parsers/graphql/transformer.js +215 -0
  175. package/renderer/lib/api-docs/parsers/graphql/types.js +46 -0
  176. package/renderer/lib/api-docs/parsers/openapi/dereferencer.js +43 -0
  177. package/renderer/lib/api-docs/parsers/openapi/extractors/auth.js +486 -0
  178. package/renderer/lib/api-docs/parsers/openapi/extractors/body.js +295 -0
  179. package/renderer/lib/api-docs/parsers/openapi/extractors/index.js +132 -0
  180. package/renderer/lib/api-docs/parsers/openapi/index.js +127 -0
  181. package/renderer/lib/api-docs/parsers/openapi/transformer.js +192 -0
  182. package/renderer/lib/api-docs/parsers/openapi/validator.js +24 -0
  183. package/renderer/lib/api-docs/playground/context.js +65 -0
  184. package/renderer/lib/api-docs/playground/navigation-context.js +74 -0
  185. package/renderer/lib/api-docs/playground/request-builder.js +163 -0
  186. package/renderer/lib/api-docs/playground/request-runner.js +224 -0
  187. package/renderer/lib/api-docs/playground/types.js +5 -0
  188. package/renderer/lib/api-docs/types.js +23 -0
  189. package/renderer/lib/api-docs/utils.js +212 -0
  190. package/renderer/lib/cache.js +157 -0
  191. package/renderer/lib/docs/config/domain-schema.js +161 -0
  192. package/renderer/lib/docs/config/index.js +5 -0
  193. package/renderer/lib/docs/config/loader.js +113 -0
  194. package/renderer/lib/docs/config/schema.js +269 -0
  195. package/renderer/lib/docs/index.js +8 -0
  196. package/renderer/lib/docs/mdx/compiler.js +128 -0
  197. package/renderer/lib/docs/mdx/frontmatter.js +73 -0
  198. package/renderer/lib/docs/mdx/index.js +8 -0
  199. package/renderer/lib/docs/navigation/generator.js +269 -0
  200. package/renderer/lib/docs/navigation/index.js +4 -0
  201. package/renderer/lib/docs/navigation/types.js +9 -0
  202. package/renderer/lib/docs-navigation-context.js +40 -0
  203. package/renderer/lib/multi-tenant/context.js +80 -0
  204. package/renderer/lib/storage/blob.js +767 -0
  205. package/renderer/lib/utils/icons.js +30 -0
  206. package/renderer/lib/utils.js +5 -0
  207. package/renderer/next.config.js +62 -0
  208. package/renderer/tsconfig.json +23 -5
  209. package/renderer/app/api/assets/[...path]/route.ts +0 -123
  210. package/renderer/app/api/assets/route.ts +0 -124
  211. package/renderer/app/api/assets/upload/route.ts +0 -177
  212. package/renderer/app/api/auth-schemes/route.ts +0 -77
  213. package/renderer/app/api/chat/route.ts +0 -858
  214. package/renderer/app/api/codegen/route.ts +0 -72
  215. package/renderer/app/api/collections/route.ts +0 -1002
  216. package/renderer/app/api/debug/route.ts +0 -53
  217. package/renderer/app/api/deploy/route.ts +0 -234
  218. package/renderer/app/api/device/route.ts +0 -42
  219. package/renderer/app/api/docs/route.ts +0 -201
  220. package/renderer/app/api/domains/add/route.ts +0 -132
  221. package/renderer/app/api/domains/lookup/route.ts +0 -43
  222. package/renderer/app/api/domains/remove/route.ts +0 -100
  223. package/renderer/app/api/domains/status/route.ts +0 -158
  224. package/renderer/app/api/domains/verify/route.ts +0 -181
  225. package/renderer/app/api/keys/regenerate/route.ts +0 -80
  226. package/renderer/app/api/local-assets/[...path]/route.ts +0 -122
  227. package/renderer/app/api/openapi-spec/route.ts +0 -151
  228. package/renderer/app/api/projects/[slug]/route.ts +0 -153
  229. package/renderer/app/api/projects/[slug]/stats/route.ts +0 -96
  230. package/renderer/app/api/projects/register/route.ts +0 -152
  231. package/renderer/app/api/proxy/route.ts +0 -149
  232. package/renderer/app/api/proxy-stream/route.ts +0 -168
  233. package/renderer/app/api/redirects/route.ts +0 -47
  234. package/renderer/app/api/schema/route.ts +0 -73
  235. package/renderer/app/api/subdomains/check/route.ts +0 -172
  236. package/renderer/app/api/suggestions/route.ts +0 -144
  237. package/renderer/app/layout.tsx +0 -54
  238. package/renderer/app/llms-full.txt/route.ts +0 -346
  239. package/renderer/app/llms.txt/route.ts +0 -279
  240. package/renderer/app/page.tsx +0 -14
  241. package/renderer/app/robots.txt/route.ts +0 -84
  242. package/renderer/app/sitemap.xml/route.ts +0 -199
  243. package/renderer/components/docs/index.ts +0 -12
  244. package/renderer/components/docs/mdx/accordion.tsx +0 -169
  245. package/renderer/components/docs/mdx/badge.tsx +0 -132
  246. package/renderer/components/docs/mdx/callouts.tsx +0 -154
  247. package/renderer/components/docs/mdx/cards.tsx +0 -241
  248. package/renderer/components/docs/mdx/changelog.tsx +0 -120
  249. package/renderer/components/docs/mdx/code-block.tsx +0 -186
  250. package/renderer/components/docs/mdx/code-group.tsx +0 -421
  251. package/renderer/components/docs/mdx/file-embeds.tsx +0 -105
  252. package/renderer/components/docs/mdx/frame.tsx +0 -112
  253. package/renderer/components/docs/mdx/highlight.tsx +0 -151
  254. package/renderer/components/docs/mdx/iframe.tsx +0 -134
  255. package/renderer/components/docs/mdx/image.tsx +0 -235
  256. package/renderer/components/docs/mdx/index.ts +0 -237
  257. package/renderer/components/docs/mdx/landing.tsx +0 -684
  258. package/renderer/components/docs/mdx/mermaid.tsx +0 -240
  259. package/renderer/components/docs/mdx/param-field.tsx +0 -200
  260. package/renderer/components/docs/mdx/steps.tsx +0 -113
  261. package/renderer/components/docs/mdx/tabs.tsx +0 -86
  262. package/renderer/components/docs/mdx-renderer.tsx +0 -100
  263. package/renderer/components/docs/navigation/breadcrumbs.tsx +0 -76
  264. package/renderer/components/docs/navigation/index.ts +0 -8
  265. package/renderer/components/docs/navigation/page-nav.tsx +0 -64
  266. package/renderer/components/docs/navigation/sidebar.tsx +0 -515
  267. package/renderer/components/docs/navigation/toc.tsx +0 -113
  268. package/renderer/components/docs/notice.tsx +0 -105
  269. package/renderer/components/docs-header.tsx +0 -278
  270. package/renderer/components/docs-viewer/agent/agent-chat.tsx +0 -2076
  271. package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +0 -90
  272. package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +0 -49
  273. package/renderer/components/docs-viewer/agent/cards/index.tsx +0 -50
  274. package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +0 -212
  275. package/renderer/components/docs-viewer/agent/cards/types.ts +0 -84
  276. package/renderer/components/docs-viewer/agent/chat-message.tsx +0 -17
  277. package/renderer/components/docs-viewer/agent/index.tsx +0 -6
  278. package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +0 -119
  279. package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +0 -46
  280. package/renderer/components/docs-viewer/agent/messages/index.ts +0 -17
  281. package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +0 -721
  282. package/renderer/components/docs-viewer/agent/messages/types.ts +0 -61
  283. package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +0 -24
  284. package/renderer/components/docs-viewer/agent/messages/user-message.tsx +0 -51
  285. package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +0 -1283
  286. package/renderer/components/docs-viewer/content/changelog-page.tsx +0 -331
  287. package/renderer/components/docs-viewer/content/doc-page.tsx +0 -367
  288. package/renderer/components/docs-viewer/content/documentation-viewer.tsx +0 -17
  289. package/renderer/components/docs-viewer/content/index.tsx +0 -29
  290. package/renderer/components/docs-viewer/content/not-found-page.tsx +0 -330
  291. package/renderer/components/docs-viewer/content/request-details.tsx +0 -330
  292. package/renderer/components/docs-viewer/content/sections/auth.tsx +0 -69
  293. package/renderer/components/docs-viewer/content/sections/body.tsx +0 -66
  294. package/renderer/components/docs-viewer/content/sections/headers.tsx +0 -43
  295. package/renderer/components/docs-viewer/content/sections/overview.tsx +0 -40
  296. package/renderer/components/docs-viewer/content/sections/parameters.tsx +0 -43
  297. package/renderer/components/docs-viewer/content/sections/responses.tsx +0 -87
  298. package/renderer/components/docs-viewer/global-auth-modal.tsx +0 -352
  299. package/renderer/components/docs-viewer/index.tsx +0 -1662
  300. package/renderer/components/docs-viewer/playground/auth-editor.tsx +0 -280
  301. package/renderer/components/docs-viewer/playground/body-editor.tsx +0 -221
  302. package/renderer/components/docs-viewer/playground/code-editor.tsx +0 -224
  303. package/renderer/components/docs-viewer/playground/code-snippet.tsx +0 -387
  304. package/renderer/components/docs-viewer/playground/graphql-playground.tsx +0 -745
  305. package/renderer/components/docs-viewer/playground/index.tsx +0 -671
  306. package/renderer/components/docs-viewer/playground/key-value-editor.tsx +0 -261
  307. package/renderer/components/docs-viewer/playground/method-selector.tsx +0 -60
  308. package/renderer/components/docs-viewer/playground/request-builder.tsx +0 -179
  309. package/renderer/components/docs-viewer/playground/request-tabs.tsx +0 -237
  310. package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +0 -21
  311. package/renderer/components/docs-viewer/playground/response-cards/index.tsx +0 -93
  312. package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +0 -16
  313. package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +0 -23
  314. package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +0 -268
  315. package/renderer/components/docs-viewer/playground/response-cards/types.ts +0 -82
  316. package/renderer/components/docs-viewer/playground/response-viewer.tsx +0 -43
  317. package/renderer/components/docs-viewer/search/index.ts +0 -2
  318. package/renderer/components/docs-viewer/search/search-dialog.tsx +0 -331
  319. package/renderer/components/docs-viewer/search/use-search.ts +0 -117
  320. package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +0 -431
  321. package/renderer/components/docs-viewer/shared/method-badge.tsx +0 -41
  322. package/renderer/components/docs-viewer/shared/schema-viewer.tsx +0 -349
  323. package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +0 -259
  324. package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +0 -316
  325. package/renderer/components/docs-viewer/sidebar/index.tsx +0 -282
  326. package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +0 -202
  327. package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +0 -118
  328. package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +0 -212
  329. package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +0 -38
  330. package/renderer/components/theme-provider.tsx +0 -11
  331. package/renderer/components/theme-toggle.tsx +0 -76
  332. package/renderer/components/ui/badge.tsx +0 -46
  333. package/renderer/components/ui/button.tsx +0 -59
  334. package/renderer/components/ui/dialog.tsx +0 -118
  335. package/renderer/components/ui/dropdown-menu.tsx +0 -257
  336. package/renderer/components/ui/input.tsx +0 -21
  337. package/renderer/components/ui/label.tsx +0 -24
  338. package/renderer/components/ui/navigation-menu.tsx +0 -168
  339. package/renderer/components/ui/select.tsx +0 -190
  340. package/renderer/components/ui/spinner.tsx +0 -114
  341. package/renderer/components/ui/tabs.tsx +0 -66
  342. package/renderer/components/ui/tooltip.tsx +0 -61
  343. package/renderer/hooks/use-code-copy.ts +0 -88
  344. package/renderer/hooks/use-openapi-title.ts +0 -44
  345. package/renderer/lib/api-docs/agent/index.ts +0 -6
  346. package/renderer/lib/api-docs/agent/indexer.ts +0 -323
  347. package/renderer/lib/api-docs/agent/spec-summary.ts +0 -335
  348. package/renderer/lib/api-docs/agent/types.ts +0 -116
  349. package/renderer/lib/api-docs/auth/auth-context.tsx +0 -225
  350. package/renderer/lib/api-docs/auth/auth-storage.ts +0 -87
  351. package/renderer/lib/api-docs/auth/crypto.ts +0 -89
  352. package/renderer/lib/api-docs/auth/index.ts +0 -4
  353. package/renderer/lib/api-docs/code-editor/db.ts +0 -164
  354. package/renderer/lib/api-docs/code-editor/hooks.ts +0 -266
  355. package/renderer/lib/api-docs/code-editor/mode-context.tsx +0 -207
  356. package/renderer/lib/api-docs/code-editor/types.ts +0 -105
  357. package/renderer/lib/api-docs/codegen/definitions.ts +0 -297
  358. package/renderer/lib/api-docs/codegen/har.ts +0 -251
  359. package/renderer/lib/api-docs/codegen/index.ts +0 -159
  360. package/renderer/lib/api-docs/factories.ts +0 -170
  361. package/renderer/lib/api-docs/mobile-context.tsx +0 -112
  362. package/renderer/lib/api-docs/navigation-context.tsx +0 -88
  363. package/renderer/lib/api-docs/parsers/graphql/README.md +0 -129
  364. package/renderer/lib/api-docs/parsers/graphql/index.ts +0 -91
  365. package/renderer/lib/api-docs/parsers/graphql/parser.ts +0 -491
  366. package/renderer/lib/api-docs/parsers/graphql/transformer.ts +0 -246
  367. package/renderer/lib/api-docs/parsers/graphql/types.ts +0 -283
  368. package/renderer/lib/api-docs/parsers/openapi/README.md +0 -32
  369. package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +0 -60
  370. package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +0 -574
  371. package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +0 -403
  372. package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +0 -232
  373. package/renderer/lib/api-docs/parsers/openapi/index.ts +0 -171
  374. package/renderer/lib/api-docs/parsers/openapi/transformer.ts +0 -278
  375. package/renderer/lib/api-docs/parsers/openapi/validator.ts +0 -31
  376. package/renderer/lib/api-docs/playground/context.tsx +0 -107
  377. package/renderer/lib/api-docs/playground/navigation-context.tsx +0 -124
  378. package/renderer/lib/api-docs/playground/request-builder.ts +0 -223
  379. package/renderer/lib/api-docs/playground/request-runner.ts +0 -282
  380. package/renderer/lib/api-docs/playground/types.ts +0 -35
  381. package/renderer/lib/api-docs/types.ts +0 -269
  382. package/renderer/lib/api-docs/utils.ts +0 -311
  383. package/renderer/lib/cache.ts +0 -193
  384. package/renderer/lib/docs/config/domain-schema.ts +0 -260
  385. package/renderer/lib/docs/config/index.ts +0 -43
  386. package/renderer/lib/docs/config/loader.ts +0 -142
  387. package/renderer/lib/docs/config/schema.ts +0 -308
  388. package/renderer/lib/docs/index.ts +0 -12
  389. package/renderer/lib/docs/mdx/compiler.ts +0 -176
  390. package/renderer/lib/docs/mdx/frontmatter.ts +0 -91
  391. package/renderer/lib/docs/mdx/index.ts +0 -26
  392. package/renderer/lib/docs/navigation/generator.ts +0 -348
  393. package/renderer/lib/docs/navigation/index.ts +0 -12
  394. package/renderer/lib/docs/navigation/types.ts +0 -123
  395. package/renderer/lib/docs-navigation-context.tsx +0 -80
  396. package/renderer/lib/multi-tenant/context.ts +0 -105
  397. package/renderer/lib/storage/blob.ts +0 -1083
  398. package/renderer/lib/utils/icons.ts +0 -48
  399. package/renderer/lib/utils.ts +0 -6
  400. package/renderer/next.config.ts +0 -76
@@ -1,237 +0,0 @@
1
- 'use client'
2
-
3
- import { useState, useEffect } from 'react'
4
- import type { BrainfishRESTReqBody, BrainfishRESTAuth } from '@/lib/api-docs/types'
5
- import { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs'
6
- import { Badge } from '@/components/ui/badge'
7
- // import { Button } from '@/components/ui/button'
8
- import { ShieldCheck, Link as LinkIcon, WarningCircle } from '@phosphor-icons/react'
9
- import { KeyValueEditor, KeyValueItem } from './key-value-editor'
10
- import { BodyEditor } from './body-editor'
11
- import { AuthEditor } from './auth-editor'
12
- import { getAuthTypeLabel } from '@/lib/api-docs/auth'
13
- import { usePlaygroundNavigation, type PlaygroundTab } from '@/lib/api-docs/playground/navigation-context'
14
- import { cn } from '@/lib/utils'
15
-
16
- interface RequestTabsProps {
17
- params: KeyValueItem[]
18
- headers: KeyValueItem[]
19
- body: string | null
20
- bodyContentType: BrainfishRESTReqBody['contentType']
21
- auth: BrainfishRESTAuth
22
- usingGlobalAuth?: boolean
23
- globalAuth?: BrainfishRESTAuth | null
24
- onParamsChange: (params: KeyValueItem[]) => void
25
- onHeadersChange: (headers: KeyValueItem[]) => void
26
- onBodyChange: (body: string | null) => void
27
- onBodyContentTypeChange: (contentType: BrainfishRESTReqBody['contentType']) => void
28
- onAuthChange: (auth: BrainfishRESTAuth) => void
29
- onUseGlobalAuth?: () => void
30
- }
31
-
32
- export function RequestTabs({
33
- params,
34
- headers,
35
- body,
36
- bodyContentType,
37
- auth,
38
- usingGlobalAuth = false,
39
- globalAuth,
40
- onParamsChange,
41
- onHeadersChange,
42
- onBodyChange,
43
- onBodyContentTypeChange,
44
- onAuthChange,
45
- onUseGlobalAuth,
46
- }: RequestTabsProps) {
47
- const paramsCount = params.filter((p) => p.active && p.key).length
48
- const headersCount = headers.filter((h) => h.active && h.key).length
49
- const hasBody = body !== null && body.length > 0
50
- const hasAuth = auth.authType !== 'none'
51
-
52
- // Always default to Parameters tab for consistency
53
- const [activeTab, setActiveTab] = useState<PlaygroundTab>('params')
54
-
55
- // Get navigation context
56
- const { state: navState, setActiveTab: syncActiveTab, clearHighlight } = usePlaygroundNavigation()
57
-
58
- // Sync with navigation context
59
- useEffect(() => {
60
- if (navState.activeTab && navState.activeTab !== activeTab) {
61
- setActiveTab(navState.activeTab)
62
- }
63
- }, [navState.activeTab, activeTab])
64
-
65
- // Update context when tab changes locally
66
- const handleTabChange = (tab: string) => {
67
- setActiveTab(tab as PlaygroundTab)
68
- syncActiveTab(tab as PlaygroundTab)
69
- clearHighlight()
70
- }
71
-
72
- // Check if a tab should show error indicator
73
- const shouldShowTabError = (tab: PlaygroundTab) => {
74
- if (!navState.showError || !navState.highlightField) return false
75
-
76
- switch (tab) {
77
- case 'params':
78
- return navState.highlightField.type === 'param'
79
- case 'headers':
80
- return navState.highlightField.type === 'header'
81
- case 'body':
82
- return navState.highlightField.type === 'body'
83
- case 'auth':
84
- return navState.highlightField.type === 'auth'
85
- default:
86
- return false
87
- }
88
- }
89
-
90
- return (
91
- <Tabs value={activeTab} onValueChange={handleTabChange} className="flex flex-col h-full gap-0">
92
- <TabsList className="w-full justify-start rounded-none border-b bg-muted/30 h-auto p-0 overflow-x-auto overflow-y-hidden flex-shrink-0">
93
- <TabsTrigger
94
- value="params"
95
- className={cn(
96
- "rounded-none border-b-2 border-transparent data-[state=active]:border-primary data-[state=active]:bg-background px-2 sm:px-4 py-2 sm:py-3 text-xs sm:text-sm whitespace-nowrap",
97
- shouldShowTabError('params') && "border-red-500 data-[state=active]:border-red-500"
98
- )}
99
- >
100
- <span className="hidden sm:inline">Parameters</span>
101
- <span className="sm:hidden">Params</span>
102
- {shouldShowTabError('params') ? (
103
- <WarningCircle className="ml-1 sm:ml-2 h-4 w-4 text-red-500 animate-pulse" weight="fill" />
104
- ) : paramsCount > 0 ? (
105
- <Badge variant="secondary" className="ml-1 sm:ml-2 h-4 sm:h-5 px-1 sm:px-1.5 text-xs">
106
- {paramsCount}
107
- </Badge>
108
- ) : null}
109
- </TabsTrigger>
110
- <TabsTrigger
111
- value="body"
112
- className={cn(
113
- "rounded-none border-b-2 border-transparent data-[state=active]:border-primary data-[state=active]:bg-background px-2 sm:px-4 py-2 sm:py-3 text-xs sm:text-sm whitespace-nowrap",
114
- shouldShowTabError('body') && "border-red-500 data-[state=active]:border-red-500"
115
- )}
116
- >
117
- Body
118
- {shouldShowTabError('body') ? (
119
- <WarningCircle className="ml-1 sm:ml-2 h-4 w-4 text-red-500 animate-pulse" weight="fill" />
120
- ) : hasBody ? (
121
- <span className="ml-1 sm:ml-2 w-2 h-2 bg-primary rounded-full" />
122
- ) : null}
123
- </TabsTrigger>
124
- <TabsTrigger
125
- value="headers"
126
- className={cn(
127
- "rounded-none border-b-2 border-transparent data-[state=active]:border-primary data-[state=active]:bg-background px-2 sm:px-4 py-2 sm:py-3 text-xs sm:text-sm whitespace-nowrap",
128
- shouldShowTabError('headers') && "border-red-500 data-[state=active]:border-red-500"
129
- )}
130
- >
131
- Headers
132
- {shouldShowTabError('headers') ? (
133
- <WarningCircle className="ml-1 sm:ml-2 h-4 w-4 text-red-500 animate-pulse" weight="fill" />
134
- ) : headersCount > 0 ? (
135
- <Badge variant="secondary" className="ml-1 sm:ml-2 h-4 sm:h-5 px-1 sm:px-1.5 text-xs">
136
- {headersCount}
137
- </Badge>
138
- ) : null}
139
- </TabsTrigger>
140
- <TabsTrigger
141
- value="auth"
142
- className={cn(
143
- "rounded-none border-b-2 border-transparent data-[state=active]:border-primary data-[state=active]:bg-background px-2 sm:px-4 py-2 sm:py-3 text-xs sm:text-sm whitespace-nowrap",
144
- shouldShowTabError('auth') && "border-red-500 data-[state=active]:border-red-500"
145
- )}
146
- >
147
- <span className="hidden sm:inline">Authorization</span>
148
- <span className="sm:hidden">Auth</span>
149
- {shouldShowTabError('auth') ? (
150
- <WarningCircle className="ml-1 sm:ml-2 h-4 w-4 text-red-500 animate-pulse" weight="fill" />
151
- ) : usingGlobalAuth ? (
152
- <LinkIcon className="ml-1 sm:ml-2 h-3.5 w-3.5 text-green-600 dark:text-green-400" weight="bold" />
153
- ) : hasAuth ? (
154
- <span className="ml-1 sm:ml-2 w-2 h-2 bg-primary rounded-full" />
155
- ) : null}
156
- </TabsTrigger>
157
- </TabsList>
158
-
159
- <TabsContent value="params" className="flex-1 m-0 overflow-auto">
160
- <KeyValueEditor
161
- items={params}
162
- onChange={onParamsChange}
163
- title="Query Parameters"
164
- keyLabel="Name"
165
- valueLabel="Value"
166
- showError={navState.highlightField?.type === 'param' && navState.showError}
167
- />
168
- </TabsContent>
169
-
170
- <TabsContent value="body" className="flex-1 m-0 overflow-auto">
171
- <BodyEditor
172
- body={body}
173
- contentType={bodyContentType}
174
- onBodyChange={onBodyChange}
175
- onContentTypeChange={onBodyContentTypeChange}
176
- showError={navState.highlightField?.type === 'body' && navState.showError}
177
- />
178
- </TabsContent>
179
-
180
- <TabsContent value="headers" className="flex-1 m-0 overflow-auto">
181
- <KeyValueEditor
182
- items={headers}
183
- onChange={onHeadersChange}
184
- title="Request Headers"
185
- keyLabel="Name"
186
- valueLabel="Value"
187
- showError={navState.highlightField?.type === 'header' && navState.showError}
188
- />
189
- </TabsContent>
190
-
191
- <TabsContent value="auth" className="flex-1 m-0 overflow-hidden flex flex-col">
192
- {/* Subtle Auth Status Bar */}
193
- {globalAuth && globalAuth.authType !== 'none' && (
194
- <div className="flex items-center justify-between px-4 py-1.5 bg-muted/40 border-b border-border text-xs">
195
- {usingGlobalAuth ? (
196
- <>
197
- <span className="flex items-center gap-1.5 text-muted-foreground">
198
- <LinkIcon className="h-3 w-3 text-green-600 dark:text-green-400" weight="bold" />
199
- <span>Global: <span className="text-foreground font-medium">{getAuthTypeLabel(globalAuth)}</span></span>
200
- </span>
201
- <button
202
- onClick={() => onAuthChange({ ...globalAuth })}
203
- className="text-muted-foreground hover:text-foreground transition-colors"
204
- >
205
- Customize
206
- </button>
207
- </>
208
- ) : (
209
- <>
210
- <span className="flex items-center gap-1.5 text-muted-foreground">
211
- <ShieldCheck className="h-3 w-3" weight="bold" />
212
- <span>Custom auth</span>
213
- </span>
214
- {onUseGlobalAuth && (
215
- <button
216
- onClick={onUseGlobalAuth}
217
- className="text-muted-foreground hover:text-foreground transition-colors"
218
- >
219
- Use global
220
- </button>
221
- )}
222
- </>
223
- )}
224
- </div>
225
- )}
226
-
227
- <div className="flex-1 overflow-auto">
228
- <AuthEditor
229
- auth={auth}
230
- onChange={onAuthChange}
231
- showError={navState.highlightField?.type === 'auth' && navState.showError}
232
- />
233
- </div>
234
- </TabsContent>
235
- </Tabs>
236
- )
237
- }
@@ -1,21 +0,0 @@
1
- 'use client'
2
-
3
- import { WifiHigh } from '@phosphor-icons/react'
4
- import { Badge } from '@/components/ui/badge'
5
-
6
- export function IdleCard() {
7
- return (
8
- <div className="flex flex-col h-full">
9
- <div className="flex items-center justify-center flex-1 p-8">
10
- <div className="text-center">
11
- <WifiHigh className="h-16 w-16 text-muted-foreground/30 mx-auto mb-4" weight="thin" />
12
- <p className="text-muted-foreground mb-2">Send a request to see the response</p>
13
- <p className="text-sm text-muted-foreground/70">
14
- Press <kbd className="px-2 py-1 bg-muted rounded text-xs font-mono">Enter</kbd> or click{' '}
15
- <Badge variant="default" className="text-xs">Send</Badge>
16
- </p>
17
- </div>
18
- </div>
19
- </div>
20
- )
21
- }
@@ -1,93 +0,0 @@
1
- 'use client'
2
-
3
- // Export types
4
- export * from './types'
5
-
6
- // Export individual cards
7
- export { IdleCard } from './idle-card'
8
- export { LoadingCard } from './loading-card'
9
- export { NetworkErrorCard } from './network-error-card'
10
- export { ResponseBodyCard } from './response-body-card'
11
-
12
- import { IdleCard } from './idle-card'
13
- import { LoadingCard } from './loading-card'
14
- import { NetworkErrorCard } from './network-error-card'
15
- import { ResponseBodyCard } from './response-body-card'
16
- import type { PlaygroundResponseState } from '@/lib/api-docs/playground/types'
17
- import type { DebugContext } from '../response-viewer'
18
-
19
- interface ResponseCardRendererProps {
20
- responseState: PlaygroundResponseState
21
- onDebugRequest?: (context: DebugContext) => void
22
- onExplainRequest?: (context: DebugContext) => void
23
- }
24
-
25
- /**
26
- * Renders the appropriate response card based on the response state.
27
- * This is the main entry point for response visualization.
28
- */
29
- export function ResponseCardRenderer({ responseState, onDebugRequest, onExplainRequest }: ResponseCardRendererProps) {
30
- // Idle state
31
- if (responseState.type === 'idle') {
32
- return <IdleCard />
33
- }
34
-
35
- // Loading state (but not streaming)
36
- if (responseState.type === 'loading') {
37
- return <LoadingCard />
38
- }
39
-
40
- // Error state with no response (network error)
41
- if (responseState.type === 'error' && !responseState.response) {
42
- return <NetworkErrorCard errorMessage={responseState.error} />
43
- }
44
-
45
- // Get response for success, streaming, or error with response
46
- const response = responseState.response
47
- if (!response) {
48
- return null
49
- }
50
-
51
- const isStreaming = responseState.type === 'streaming'
52
- const isError = response.status >= 400
53
-
54
- // Get content type
55
- const contentType = response.headers['content-type'] || response.headers['Content-Type'] || null
56
-
57
- // Build context for AI
58
- const buildContext = () => ({
59
- status: response.status,
60
- statusText: response.statusText,
61
- responseBody: typeof response.body === 'string' ? response.body : '[Binary content]',
62
- errorMessage: response.error,
63
- })
64
-
65
- // Handle debug request (for errors)
66
- const handleDebug = onDebugRequest ? () => {
67
- onDebugRequest(buildContext())
68
- } : undefined
69
-
70
- // Handle explain request (for success)
71
- const handleExplain = onExplainRequest ? () => {
72
- onExplainRequest(buildContext())
73
- } : undefined
74
-
75
- return (
76
- <ResponseBodyCard
77
- meta={{
78
- status: response.status,
79
- statusText: response.statusText,
80
- responseTime: response.responseTime,
81
- size: response.size,
82
- isStreaming: isStreaming || response.isStreaming,
83
- }}
84
- body={typeof response.body === 'string' ? response.body : '[Binary content]'}
85
- headers={response.headers}
86
- contentType={contentType}
87
- isError={isError}
88
- errorMessage={response.error}
89
- onDebug={isError ? handleDebug : undefined}
90
- onExplain={!isError ? handleExplain : undefined}
91
- />
92
- )
93
- }
@@ -1,16 +0,0 @@
1
- 'use client'
2
-
3
- import { Spinner } from '@phosphor-icons/react'
4
-
5
- export function LoadingCard() {
6
- return (
7
- <div className="flex flex-col h-full">
8
- <div className="flex items-center justify-center flex-1 p-8">
9
- <div className="text-center">
10
- <Spinner size={32} className="text-muted-foreground animate-spin mx-auto" />
11
- <p className="text-sm text-muted-foreground mt-3">Sending request...</p>
12
- </div>
13
- </div>
14
- </div>
15
- )
16
- }
@@ -1,23 +0,0 @@
1
- 'use client'
2
-
3
- import { WifiSlash } from '@phosphor-icons/react'
4
-
5
- interface NetworkErrorCardProps {
6
- errorMessage?: string
7
- }
8
-
9
- export function NetworkErrorCard({ errorMessage }: NetworkErrorCardProps) {
10
- return (
11
- <div className="flex flex-col h-full">
12
- <div className="flex items-center justify-center flex-1 p-8">
13
- <div className="text-center">
14
- <WifiSlash className="h-16 w-16 text-red-500/50 mx-auto mb-4" weight="thin" />
15
- <p className="text-red-500 font-medium mb-2">Request Failed</p>
16
- <p className="text-sm text-muted-foreground max-w-md">
17
- {errorMessage || 'An error occurred while sending the request'}
18
- </p>
19
- </div>
20
- </div>
21
- </div>
22
- )
23
- }
@@ -1,268 +0,0 @@
1
- 'use client'
2
-
3
- import { useState, useMemo, useCallback } from 'react'
4
- import { Copy, Check, Clock, Database, WarningCircle, Lightning, Bug, Lightbulb } from '@phosphor-icons/react'
5
- import { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs'
6
- import { Badge } from '@/components/ui/badge'
7
- import { Button } from '@/components/ui/button'
8
- import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip'
9
- import { CodeViewer, SupportedLanguage } from '../code-editor'
10
- import type { ResponseMeta } from './types'
11
-
12
- interface ResponseBodyCardProps {
13
- meta: ResponseMeta
14
- body: string
15
- headers: Record<string, string>
16
- contentType: string | null
17
- isError?: boolean
18
- errorMessage?: string
19
- onDebug?: () => void
20
- onExplain?: () => void
21
- }
22
-
23
- // Detect language from content type
24
- function detectLanguage(contentType: string | null, body?: string): SupportedLanguage {
25
- if (!contentType) return 'plaintext'
26
-
27
- if (contentType.includes('json')) return 'json'
28
- if (contentType.includes('xml')) return 'xml'
29
- if (contentType.includes('html')) return 'html'
30
- if (contentType.includes('javascript')) return 'javascript'
31
- if (contentType.includes('typescript')) return 'typescript'
32
- if (contentType.includes('event-stream')) return 'json'
33
- if (body && body.trim().startsWith('{')) return 'json'
34
-
35
- return 'plaintext'
36
- }
37
-
38
- function formatSize(bytes: number): string {
39
- if (bytes < 1024) return `${bytes} B`
40
- if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(2)} KB`
41
- return `${(bytes / (1024 * 1024)).toFixed(2)} MB`
42
- }
43
-
44
- // Format response body - handles both SSE and regular JSON
45
- function formatBody(body: string, contentType: string | null, isSSE: boolean): string {
46
- if (!body) return ''
47
-
48
- // Check if this is SSE format
49
- if (isSSE || body.includes('data: ')) {
50
- const lines = body.split(/\r?\n/).filter(line => line.trim())
51
- const events: string[] = []
52
-
53
- for (const line of lines) {
54
- if (line.startsWith('data: ')) {
55
- const data = line.slice(6)
56
- try {
57
- const parsed = JSON.parse(data)
58
- events.push(JSON.stringify(parsed, null, 2))
59
- } catch {
60
- events.push(data)
61
- }
62
- } else {
63
- events.push(line)
64
- }
65
- }
66
-
67
- return events.join('\n')
68
- }
69
-
70
- // Regular JSON - try to parse and format
71
- if (contentType?.includes('json')) {
72
- try {
73
- const parsed = JSON.parse(body)
74
- return JSON.stringify(parsed, null, 2)
75
- } catch {
76
- return body
77
- }
78
- }
79
-
80
- return body
81
- }
82
-
83
- export function ResponseBodyCard({
84
- meta,
85
- body,
86
- headers,
87
- contentType,
88
- isError = false,
89
- errorMessage,
90
- onDebug,
91
- onExplain
92
- }: ResponseBodyCardProps) {
93
- const [copied, setCopied] = useState(false)
94
-
95
- const isSSE = contentType?.includes('event-stream')
96
- const detectedLanguage = useMemo(() => detectLanguage(contentType, body), [contentType, body])
97
- const formattedBody = useMemo(() => formatBody(body, contentType, !!isSSE), [body, contentType, isSSE])
98
- const headersCount = Object.keys(headers).length
99
-
100
- const copyToClipboard = useCallback((text: string) => {
101
- navigator.clipboard.writeText(text)
102
- setCopied(true)
103
- setTimeout(() => setCopied(false), 2000)
104
- }, [])
105
-
106
- const statusVariant = isError ? 'destructive' : 'default'
107
-
108
- return (
109
- <TooltipProvider>
110
- <div className="flex flex-col h-full">
111
- {/* Response Meta */}
112
- <div className="sticky top-0 z-10 flex items-center justify-between bg-muted/50 px-4 py-3 border-b border-border">
113
- <div className="flex items-center gap-4 text-sm">
114
- {/* Status */}
115
- <Badge variant={statusVariant} className="font-semibold">
116
- {meta.status} {meta.statusText}
117
- </Badge>
118
-
119
- {/* Time */}
120
- <span className="flex items-center gap-1.5 text-muted-foreground">
121
- <Clock className="h-4 w-4" weight="bold" />
122
- <span>{meta.responseTime}ms</span>
123
- </span>
124
-
125
- {/* Size */}
126
- <span className="flex items-center gap-1.5 text-muted-foreground">
127
- <Database className="h-4 w-4" weight="bold" />
128
- <span>{formatSize(meta.size)}</span>
129
- </span>
130
-
131
- {/* Streaming indicator */}
132
- {meta.isStreaming && (
133
- <span className="flex items-center gap-1.5 text-amber-600 dark:text-amber-400">
134
- <Lightning className="h-4 w-4 animate-pulse" weight="fill" />
135
- <span>SSE</span>
136
- </span>
137
- )}
138
- </div>
139
-
140
- <div className="flex items-center gap-2">
141
- {errorMessage && (
142
- <span className="flex items-center gap-1.5 text-red-500 text-sm">
143
- <WarningCircle className="h-4 w-4" weight="fill" />
144
- Error
145
- </span>
146
- )}
147
-
148
- {/* Copy button */}
149
- {formattedBody && (
150
- <Tooltip>
151
- <TooltipTrigger asChild>
152
- <Button
153
- variant="ghost"
154
- size="icon"
155
- className="h-8 w-8"
156
- onClick={() => copyToClipboard(formattedBody)}
157
- >
158
- {copied ? (
159
- <Check className="h-4 w-4 text-green-500" weight="bold" />
160
- ) : (
161
- <Copy className="h-4 w-4" weight="bold" />
162
- )}
163
- </Button>
164
- </TooltipTrigger>
165
- <TooltipContent>{copied ? 'Copied!' : 'Copy response'}</TooltipContent>
166
- </Tooltip>
167
- )}
168
- </div>
169
- </div>
170
-
171
- {/* Tabs */}
172
- <Tabs defaultValue="body" className="flex flex-col flex-1 gap-0">
173
- <TabsList className="w-full justify-start rounded-none border-b bg-background h-auto p-0">
174
- <TabsTrigger
175
- value="body"
176
- className="rounded-none border-b-2 border-transparent data-[state=active]:border-primary px-4 py-3"
177
- >
178
- Body
179
- </TabsTrigger>
180
- <TabsTrigger
181
- value="headers"
182
- className="rounded-none border-b-2 border-transparent data-[state=active]:border-primary px-4 py-3"
183
- >
184
- Headers
185
- <Badge variant="secondary" className="ml-2 h-5 px-1.5">
186
- {headersCount}
187
- </Badge>
188
- </TabsTrigger>
189
- </TabsList>
190
-
191
- <TabsContent value="body" className="flex-1 m-0 overflow-hidden relative">
192
- {formattedBody ? (
193
- <CodeViewer
194
- value={formattedBody}
195
- language={detectedLanguage}
196
- height="100%"
197
- minHeight={200}
198
- theme="dark"
199
- showLineNumbers={false}
200
- showBorder={false}
201
- rounded={false}
202
- />
203
- ) : (
204
- <div className="flex items-center justify-center h-full text-muted-foreground">
205
- No response body
206
- </div>
207
- )}
208
-
209
- {/* Action buttons - overlay on code viewer */}
210
- <div className="absolute top-2.5 right-3 z-20 flex items-center gap-2">
211
- {/* Explain button - for success responses */}
212
- {!isError && onExplain && (
213
- <Tooltip>
214
- <TooltipTrigger asChild>
215
- <button
216
- className="flex items-center gap-1.5 px-2.5 py-1.5 text-xs font-medium text-white/70 bg-white/10 hover:bg-white/20 hover:text-white transition-colors rounded-md backdrop-blur-sm"
217
- onClick={onExplain}
218
- >
219
- <Lightbulb className="h-3.5 w-3.5" weight="bold" />
220
- <span>Explain</span>
221
- </button>
222
- </TooltipTrigger>
223
- <TooltipContent side="left">Ask AI to explain this response</TooltipContent>
224
- </Tooltip>
225
- )}
226
-
227
- {/* Debug button - for error responses */}
228
- {isError && onDebug && (
229
- <Tooltip>
230
- <TooltipTrigger asChild>
231
- <button
232
- className="flex items-center gap-1.5 px-2.5 py-1.5 text-xs font-medium text-white/70 bg-white/10 hover:bg-white/20 hover:text-white transition-colors rounded-md backdrop-blur-sm"
233
- onClick={onDebug}
234
- >
235
- <Bug className="h-3.5 w-3.5" weight="bold" />
236
- <span>Debug</span>
237
- </button>
238
- </TooltipTrigger>
239
- <TooltipContent side="left">Ask AI to help debug this error</TooltipContent>
240
- </Tooltip>
241
- )}
242
- </div>
243
- </TabsContent>
244
-
245
- <TabsContent value="headers" className="flex-1 m-0 overflow-auto">
246
- <div className="divide-y divide-border">
247
- {Object.entries(headers).map(([key, value]) => (
248
- <div key={key} className="flex items-start gap-4 px-4 py-3">
249
- <span className="font-mono text-sm font-semibold text-foreground min-w-[180px]">
250
- {key}
251
- </span>
252
- <span className="font-mono text-sm text-muted-foreground break-all">
253
- {value}
254
- </span>
255
- </div>
256
- ))}
257
- {headersCount === 0 && (
258
- <div className="flex items-center justify-center p-8 text-muted-foreground">
259
- No response headers
260
- </div>
261
- )}
262
- </div>
263
- </TabsContent>
264
- </Tabs>
265
- </div>
266
- </TooltipProvider>
267
- )
268
- }