@brainfish-ai/devdoc 0.1.42 → 0.1.44

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 (398) hide show
  1. package/dist/cli/commands/create.js +2 -2
  2. package/dist/cli/commands/dev.js +19 -10
  3. package/package.json +3 -2
  4. package/renderer/app/api/assets/[...path]/route.js +108 -0
  5. package/renderer/app/api/assets/route.js +114 -0
  6. package/renderer/app/api/assets/upload/route.js +163 -0
  7. package/renderer/app/api/auth-schemes/route.js +58 -0
  8. package/renderer/app/api/chat/route.js +759 -0
  9. package/renderer/app/api/codegen/route.js +52 -0
  10. package/renderer/app/api/collections/route.js +706 -0
  11. package/renderer/app/api/debug/route.js +47 -0
  12. package/renderer/app/api/deploy/route.js +199 -0
  13. package/renderer/app/api/device/route.js +36 -0
  14. package/renderer/app/api/docs/route.js +205 -0
  15. package/renderer/app/api/domains/add/route.js +121 -0
  16. package/renderer/app/api/domains/lookup/route.js +43 -0
  17. package/renderer/app/api/domains/remove/route.js +89 -0
  18. package/renderer/app/api/domains/status/route.js +140 -0
  19. package/renderer/app/api/domains/verify/route.js +168 -0
  20. package/renderer/app/api/keys/regenerate/route.js +71 -0
  21. package/renderer/app/api/local-assets/[...path]/route.js +108 -0
  22. package/renderer/app/api/openapi-spec/route.js +73 -0
  23. package/renderer/app/api/projects/[slug]/route.js +129 -0
  24. package/renderer/app/api/projects/[slug]/stats/route.js +80 -0
  25. package/renderer/app/api/projects/register/route.js +176 -0
  26. package/renderer/app/api/proxy/route.js +139 -0
  27. package/renderer/app/api/proxy-stream/route.js +156 -0
  28. package/renderer/app/api/redirects/route.js +35 -0
  29. package/renderer/app/api/schema/route.js +85 -0
  30. package/renderer/app/api/subdomains/check/route.js +158 -0
  31. package/renderer/app/api/suggestions/route.js +195 -0
  32. package/renderer/app/globals.css +69 -0
  33. package/renderer/app/layout.js +47 -0
  34. package/renderer/app/llms-full.txt/route.js +266 -0
  35. package/renderer/app/llms.txt/route.js +228 -0
  36. package/renderer/app/page.js +12 -0
  37. package/renderer/app/robots.txt/route.js +66 -0
  38. package/renderer/app/sitemap.xml/route.js +155 -0
  39. package/renderer/components/docs/index.js +8 -0
  40. package/renderer/components/docs/mdx/accordion.js +113 -0
  41. package/renderer/components/docs/mdx/badge.js +72 -0
  42. package/renderer/components/docs/mdx/callouts.js +137 -0
  43. package/renderer/components/docs/mdx/cards.js +175 -0
  44. package/renderer/components/docs/mdx/changelog.js +100 -0
  45. package/renderer/components/docs/mdx/code-block.js +147 -0
  46. package/renderer/components/docs/mdx/code-group.js +287 -0
  47. package/renderer/components/docs/mdx/file-embeds.js +82 -0
  48. package/renderer/components/docs/mdx/frame.js +59 -0
  49. package/renderer/components/docs/mdx/highlight.js +90 -0
  50. package/renderer/components/docs/mdx/iframe.js +69 -0
  51. package/renderer/components/docs/mdx/image.js +135 -0
  52. package/renderer/components/docs/mdx/index.js +134 -0
  53. package/renderer/components/docs/mdx/landing.js +317 -0
  54. package/renderer/components/docs/mdx/mermaid.js +212 -0
  55. package/renderer/components/docs/mdx/param-field.js +112 -0
  56. package/renderer/components/docs/mdx/steps.js +74 -0
  57. package/renderer/components/docs/mdx/tabs.js +50 -0
  58. package/renderer/components/docs/mdx-renderer.js +77 -0
  59. package/renderer/components/docs/navigation/breadcrumbs.js +64 -0
  60. package/renderer/components/docs/navigation/index.js +6 -0
  61. package/renderer/components/docs/navigation/page-nav.js +57 -0
  62. package/renderer/components/docs/navigation/sidebar.js +375 -0
  63. package/renderer/components/docs/navigation/toc.js +89 -0
  64. package/renderer/components/docs/notice.js +77 -0
  65. package/renderer/components/docs-header.js +202 -0
  66. package/renderer/components/docs-viewer/agent/agent-chat.js +1831 -0
  67. package/renderer/components/docs-viewer/agent/agent-popup-button.js +99 -0
  68. package/renderer/components/docs-viewer/agent/cards/debug-context-card.js +107 -0
  69. package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.js +57 -0
  70. package/renderer/components/docs-viewer/agent/cards/index.js +45 -0
  71. package/renderer/components/docs-viewer/agent/cards/response-options-card.js +154 -0
  72. package/renderer/components/docs-viewer/agent/cards/types.js +22 -0
  73. package/renderer/components/docs-viewer/agent/chat-message.js +2 -0
  74. package/renderer/components/docs-viewer/agent/index.js +7 -0
  75. package/renderer/components/docs-viewer/agent/messages/assistant-message.js +108 -0
  76. package/renderer/components/docs-viewer/agent/messages/chat-message.js +34 -0
  77. package/renderer/components/docs-viewer/agent/messages/index.js +6 -0
  78. package/renderer/components/docs-viewer/agent/messages/tool-call-display.js +1065 -0
  79. package/renderer/components/docs-viewer/agent/messages/types.js +2 -0
  80. package/renderer/components/docs-viewer/agent/messages/typing-indicator.js +26 -0
  81. package/renderer/components/docs-viewer/agent/messages/user-message.js +37 -0
  82. package/renderer/components/docs-viewer/code-editor/{index.tsx → index.js} +1 -1
  83. package/renderer/components/docs-viewer/code-editor/notes-mode.js +1338 -0
  84. package/renderer/components/docs-viewer/content/changelog-page.js +297 -0
  85. package/renderer/components/docs-viewer/content/content-router.js +182 -0
  86. package/renderer/components/docs-viewer/content/doc-page.js +290 -0
  87. package/renderer/components/docs-viewer/content/documentation-viewer.js +14 -0
  88. package/renderer/components/docs-viewer/content/index.js +31 -0
  89. package/renderer/components/docs-viewer/content/not-found-page.js +300 -0
  90. package/renderer/components/docs-viewer/content/request-details.js +528 -0
  91. package/renderer/components/docs-viewer/content/sections/auth.js +108 -0
  92. package/renderer/components/docs-viewer/content/sections/body.js +80 -0
  93. package/renderer/components/docs-viewer/content/sections/headers.js +64 -0
  94. package/renderer/components/docs-viewer/content/sections/overview.js +56 -0
  95. package/renderer/components/docs-viewer/content/sections/parameters.js +64 -0
  96. package/renderer/components/docs-viewer/content/sections/responses.js +91 -0
  97. package/renderer/components/docs-viewer/global-auth-modal.js +427 -0
  98. package/renderer/components/docs-viewer/index.js +1448 -0
  99. package/renderer/components/docs-viewer/playground/auth-editor.js +418 -0
  100. package/renderer/components/docs-viewer/playground/body-editor.js +240 -0
  101. package/renderer/components/docs-viewer/playground/code-editor.js +135 -0
  102. package/renderer/components/docs-viewer/playground/code-snippet.js +393 -0
  103. package/renderer/components/docs-viewer/playground/graphql-playground.js +936 -0
  104. package/renderer/components/docs-viewer/playground/index.js +682 -0
  105. package/renderer/components/docs-viewer/playground/key-value-editor.js +317 -0
  106. package/renderer/components/docs-viewer/playground/method-selector.js +65 -0
  107. package/renderer/components/docs-viewer/playground/request-builder.js +181 -0
  108. package/renderer/components/docs-viewer/playground/request-tabs.js +240 -0
  109. package/renderer/components/docs-viewer/playground/response-cards/idle-card.js +42 -0
  110. package/renderer/components/docs-viewer/playground/response-cards/index.js +72 -0
  111. package/renderer/components/docs-viewer/playground/response-cards/loading-card.js +24 -0
  112. package/renderer/components/docs-viewer/playground/response-cards/network-error-card.js +28 -0
  113. package/renderer/components/docs-viewer/playground/response-cards/response-body-card.js +308 -0
  114. package/renderer/components/docs-viewer/playground/response-cards/types.js +9 -0
  115. package/renderer/components/docs-viewer/playground/response-viewer.js +18 -0
  116. package/renderer/components/docs-viewer/search/index.js +2 -0
  117. package/renderer/components/docs-viewer/search/search-dialog.js +367 -0
  118. package/renderer/components/docs-viewer/search/use-search.js +89 -0
  119. package/renderer/components/docs-viewer/shared/markdown-renderer.js +423 -0
  120. package/renderer/components/docs-viewer/shared/method-badge.js +23 -0
  121. package/renderer/components/docs-viewer/shared/schema-viewer.js +321 -0
  122. package/renderer/components/docs-viewer/sidebar/collection-tree.js +222 -0
  123. package/renderer/components/docs-viewer/sidebar/endpoint-options.js +512 -0
  124. package/renderer/components/docs-viewer/sidebar/index.js +196 -0
  125. package/renderer/components/docs-viewer/sidebar/right-sidebar.js +159 -0
  126. package/renderer/components/docs-viewer/sidebar/sidebar-group.js +87 -0
  127. package/renderer/components/docs-viewer/sidebar/sidebar-item.js +172 -0
  128. package/renderer/components/docs-viewer/sidebar/sidebar-section.js +31 -0
  129. package/renderer/components/theme-provider.js +10 -0
  130. package/renderer/components/theme-toggle.js +86 -0
  131. package/renderer/components/ui/badge.js +29 -0
  132. package/renderer/components/ui/button.js +40 -0
  133. package/renderer/components/ui/dialog.js +50 -0
  134. package/renderer/components/ui/dropdown-menu.js +143 -0
  135. package/renderer/components/ui/input.js +12 -0
  136. package/renderer/components/ui/label.js +13 -0
  137. package/renderer/components/ui/navigation-menu.js +83 -0
  138. package/renderer/components/ui/select.js +116 -0
  139. package/renderer/components/ui/spinner.js +92 -0
  140. package/renderer/components/ui/tabs.js +34 -0
  141. package/renderer/components/ui/tooltip.js +43 -0
  142. package/renderer/hooks/use-code-copy.js +76 -0
  143. package/renderer/hooks/use-openapi-title.js +33 -0
  144. package/renderer/hooks/use-route-state.js +159 -0
  145. package/renderer/lib/api-docs/agent/index.js +4 -0
  146. package/renderer/lib/api-docs/agent/indexer.js +254 -0
  147. package/renderer/lib/api-docs/agent/spec-summary.js +227 -0
  148. package/renderer/lib/api-docs/agent/types.js +5 -0
  149. package/renderer/lib/api-docs/agent/use-suggestions.js +97 -0
  150. package/renderer/lib/api-docs/auth/auth-context.js +157 -0
  151. package/renderer/lib/api-docs/auth/auth-storage.js +66 -0
  152. package/renderer/lib/api-docs/auth/crypto.js +64 -0
  153. package/renderer/lib/api-docs/auth/index.js +3 -0
  154. package/renderer/lib/api-docs/code-editor/db.js +145 -0
  155. package/renderer/lib/api-docs/code-editor/hooks.js +254 -0
  156. package/renderer/lib/api-docs/code-editor/{index.ts → index.js} +3 -4
  157. package/renderer/lib/api-docs/code-editor/mode-context.js +126 -0
  158. package/renderer/lib/api-docs/code-editor/types.js +53 -0
  159. package/renderer/lib/api-docs/codegen/definitions.js +258 -0
  160. package/renderer/lib/api-docs/codegen/har.js +171 -0
  161. package/renderer/lib/api-docs/codegen/index.js +118 -0
  162. package/renderer/lib/api-docs/factories.js +136 -0
  163. package/renderer/lib/api-docs/{index.ts → index.js} +5 -10
  164. package/renderer/lib/api-docs/mobile-context.js +116 -0
  165. package/renderer/lib/api-docs/navigation-context.js +62 -0
  166. package/renderer/lib/api-docs/parsers/graphql/index.js +50 -0
  167. package/renderer/lib/api-docs/parsers/graphql/parser.js +350 -0
  168. package/renderer/lib/api-docs/parsers/graphql/transformer.js +215 -0
  169. package/renderer/lib/api-docs/parsers/graphql/types.js +46 -0
  170. package/renderer/lib/api-docs/parsers/openapi/dereferencer.js +43 -0
  171. package/renderer/lib/api-docs/parsers/openapi/extractors/auth.js +486 -0
  172. package/renderer/lib/api-docs/parsers/openapi/extractors/body.js +295 -0
  173. package/renderer/lib/api-docs/parsers/openapi/extractors/index.js +132 -0
  174. package/renderer/lib/api-docs/parsers/openapi/index.js +127 -0
  175. package/renderer/lib/api-docs/parsers/openapi/transformer.js +192 -0
  176. package/renderer/lib/api-docs/parsers/openapi/validator.js +24 -0
  177. package/renderer/lib/api-docs/playground/context.js +65 -0
  178. package/renderer/lib/api-docs/playground/navigation-context.js +74 -0
  179. package/renderer/lib/api-docs/playground/request-builder.js +163 -0
  180. package/renderer/lib/api-docs/playground/request-runner.js +224 -0
  181. package/renderer/lib/api-docs/playground/types.js +5 -0
  182. package/renderer/lib/api-docs/types.js +23 -0
  183. package/renderer/lib/api-docs/utils.js +212 -0
  184. package/renderer/lib/cache.js +157 -0
  185. package/renderer/lib/docs/config/domain-schema.js +161 -0
  186. package/renderer/lib/docs/config/environment.js +38 -0
  187. package/renderer/lib/docs/config/index.js +6 -0
  188. package/renderer/lib/docs/config/loader.js +113 -0
  189. package/renderer/lib/docs/config/schema.js +281 -0
  190. package/renderer/lib/docs/index.js +8 -0
  191. package/renderer/lib/docs/mdx/compiler.js +131 -0
  192. package/renderer/lib/docs/mdx/frontmatter.js +73 -0
  193. package/renderer/lib/docs/mdx/index.js +10 -0
  194. package/renderer/lib/docs/mdx/remark-mermaid.js +63 -0
  195. package/renderer/lib/docs/navigation/generator.js +269 -0
  196. package/renderer/lib/docs/navigation/index.js +3 -0
  197. package/renderer/lib/docs/navigation/types.js +11 -0
  198. package/renderer/lib/docs-navigation-context.js +40 -0
  199. package/renderer/lib/docs-navigation.js +140 -0
  200. package/renderer/lib/multi-tenant/context.js +80 -0
  201. package/renderer/lib/storage/blob.js +767 -0
  202. package/renderer/lib/utils/icons.js +30 -0
  203. package/renderer/lib/utils.js +5 -0
  204. package/renderer/next.config.js +62 -0
  205. package/renderer/package.json +1 -0
  206. package/renderer/tsconfig.json +23 -5
  207. package/renderer/app/api/assets/[...path]/route.ts +0 -123
  208. package/renderer/app/api/assets/route.ts +0 -124
  209. package/renderer/app/api/assets/upload/route.ts +0 -177
  210. package/renderer/app/api/auth-schemes/route.ts +0 -77
  211. package/renderer/app/api/chat/route.ts +0 -858
  212. package/renderer/app/api/codegen/route.ts +0 -72
  213. package/renderer/app/api/collections/route.ts +0 -1002
  214. package/renderer/app/api/debug/route.ts +0 -53
  215. package/renderer/app/api/deploy/route.ts +0 -234
  216. package/renderer/app/api/device/route.ts +0 -42
  217. package/renderer/app/api/docs/route.ts +0 -201
  218. package/renderer/app/api/domains/add/route.ts +0 -132
  219. package/renderer/app/api/domains/lookup/route.ts +0 -43
  220. package/renderer/app/api/domains/remove/route.ts +0 -100
  221. package/renderer/app/api/domains/status/route.ts +0 -158
  222. package/renderer/app/api/domains/verify/route.ts +0 -181
  223. package/renderer/app/api/keys/regenerate/route.ts +0 -80
  224. package/renderer/app/api/local-assets/[...path]/route.ts +0 -122
  225. package/renderer/app/api/openapi-spec/route.ts +0 -151
  226. package/renderer/app/api/projects/[slug]/route.ts +0 -153
  227. package/renderer/app/api/projects/[slug]/stats/route.ts +0 -96
  228. package/renderer/app/api/projects/register/route.ts +0 -152
  229. package/renderer/app/api/proxy/route.ts +0 -149
  230. package/renderer/app/api/proxy-stream/route.ts +0 -168
  231. package/renderer/app/api/redirects/route.ts +0 -47
  232. package/renderer/app/api/schema/route.ts +0 -73
  233. package/renderer/app/api/subdomains/check/route.ts +0 -172
  234. package/renderer/app/api/suggestions/route.ts +0 -144
  235. package/renderer/app/layout.tsx +0 -54
  236. package/renderer/app/llms-full.txt/route.ts +0 -346
  237. package/renderer/app/llms.txt/route.ts +0 -279
  238. package/renderer/app/page.tsx +0 -14
  239. package/renderer/app/robots.txt/route.ts +0 -84
  240. package/renderer/app/sitemap.xml/route.ts +0 -199
  241. package/renderer/components/docs/index.ts +0 -12
  242. package/renderer/components/docs/mdx/accordion.tsx +0 -169
  243. package/renderer/components/docs/mdx/badge.tsx +0 -132
  244. package/renderer/components/docs/mdx/callouts.tsx +0 -154
  245. package/renderer/components/docs/mdx/cards.tsx +0 -241
  246. package/renderer/components/docs/mdx/changelog.tsx +0 -120
  247. package/renderer/components/docs/mdx/code-block.tsx +0 -186
  248. package/renderer/components/docs/mdx/code-group.tsx +0 -421
  249. package/renderer/components/docs/mdx/file-embeds.tsx +0 -105
  250. package/renderer/components/docs/mdx/frame.tsx +0 -112
  251. package/renderer/components/docs/mdx/highlight.tsx +0 -151
  252. package/renderer/components/docs/mdx/iframe.tsx +0 -134
  253. package/renderer/components/docs/mdx/image.tsx +0 -235
  254. package/renderer/components/docs/mdx/index.ts +0 -237
  255. package/renderer/components/docs/mdx/landing.tsx +0 -684
  256. package/renderer/components/docs/mdx/mermaid.tsx +0 -240
  257. package/renderer/components/docs/mdx/param-field.tsx +0 -200
  258. package/renderer/components/docs/mdx/steps.tsx +0 -113
  259. package/renderer/components/docs/mdx/tabs.tsx +0 -86
  260. package/renderer/components/docs/mdx-renderer.tsx +0 -100
  261. package/renderer/components/docs/navigation/breadcrumbs.tsx +0 -76
  262. package/renderer/components/docs/navigation/index.ts +0 -8
  263. package/renderer/components/docs/navigation/page-nav.tsx +0 -64
  264. package/renderer/components/docs/navigation/sidebar.tsx +0 -515
  265. package/renderer/components/docs/navigation/toc.tsx +0 -113
  266. package/renderer/components/docs/notice.tsx +0 -105
  267. package/renderer/components/docs-header.tsx +0 -278
  268. package/renderer/components/docs-viewer/agent/agent-chat.tsx +0 -2076
  269. package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +0 -90
  270. package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +0 -49
  271. package/renderer/components/docs-viewer/agent/cards/index.tsx +0 -50
  272. package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +0 -212
  273. package/renderer/components/docs-viewer/agent/cards/types.ts +0 -84
  274. package/renderer/components/docs-viewer/agent/chat-message.tsx +0 -17
  275. package/renderer/components/docs-viewer/agent/index.tsx +0 -6
  276. package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +0 -119
  277. package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +0 -46
  278. package/renderer/components/docs-viewer/agent/messages/index.ts +0 -17
  279. package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +0 -721
  280. package/renderer/components/docs-viewer/agent/messages/types.ts +0 -61
  281. package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +0 -24
  282. package/renderer/components/docs-viewer/agent/messages/user-message.tsx +0 -51
  283. package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +0 -1283
  284. package/renderer/components/docs-viewer/content/changelog-page.tsx +0 -331
  285. package/renderer/components/docs-viewer/content/doc-page.tsx +0 -367
  286. package/renderer/components/docs-viewer/content/documentation-viewer.tsx +0 -17
  287. package/renderer/components/docs-viewer/content/index.tsx +0 -29
  288. package/renderer/components/docs-viewer/content/not-found-page.tsx +0 -330
  289. package/renderer/components/docs-viewer/content/request-details.tsx +0 -330
  290. package/renderer/components/docs-viewer/content/sections/auth.tsx +0 -69
  291. package/renderer/components/docs-viewer/content/sections/body.tsx +0 -66
  292. package/renderer/components/docs-viewer/content/sections/headers.tsx +0 -43
  293. package/renderer/components/docs-viewer/content/sections/overview.tsx +0 -40
  294. package/renderer/components/docs-viewer/content/sections/parameters.tsx +0 -43
  295. package/renderer/components/docs-viewer/content/sections/responses.tsx +0 -87
  296. package/renderer/components/docs-viewer/global-auth-modal.tsx +0 -352
  297. package/renderer/components/docs-viewer/index.tsx +0 -1670
  298. package/renderer/components/docs-viewer/playground/auth-editor.tsx +0 -280
  299. package/renderer/components/docs-viewer/playground/body-editor.tsx +0 -221
  300. package/renderer/components/docs-viewer/playground/code-editor.tsx +0 -224
  301. package/renderer/components/docs-viewer/playground/code-snippet.tsx +0 -387
  302. package/renderer/components/docs-viewer/playground/graphql-playground.tsx +0 -745
  303. package/renderer/components/docs-viewer/playground/index.tsx +0 -671
  304. package/renderer/components/docs-viewer/playground/key-value-editor.tsx +0 -261
  305. package/renderer/components/docs-viewer/playground/method-selector.tsx +0 -60
  306. package/renderer/components/docs-viewer/playground/request-builder.tsx +0 -179
  307. package/renderer/components/docs-viewer/playground/request-tabs.tsx +0 -237
  308. package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +0 -21
  309. package/renderer/components/docs-viewer/playground/response-cards/index.tsx +0 -93
  310. package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +0 -16
  311. package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +0 -23
  312. package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +0 -268
  313. package/renderer/components/docs-viewer/playground/response-cards/types.ts +0 -82
  314. package/renderer/components/docs-viewer/playground/response-viewer.tsx +0 -43
  315. package/renderer/components/docs-viewer/search/index.ts +0 -2
  316. package/renderer/components/docs-viewer/search/search-dialog.tsx +0 -331
  317. package/renderer/components/docs-viewer/search/use-search.ts +0 -117
  318. package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +0 -431
  319. package/renderer/components/docs-viewer/shared/method-badge.tsx +0 -41
  320. package/renderer/components/docs-viewer/shared/schema-viewer.tsx +0 -349
  321. package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +0 -259
  322. package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +0 -316
  323. package/renderer/components/docs-viewer/sidebar/index.tsx +0 -282
  324. package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +0 -202
  325. package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +0 -118
  326. package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +0 -212
  327. package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +0 -38
  328. package/renderer/components/theme-provider.tsx +0 -11
  329. package/renderer/components/theme-toggle.tsx +0 -76
  330. package/renderer/components/ui/badge.tsx +0 -46
  331. package/renderer/components/ui/button.tsx +0 -59
  332. package/renderer/components/ui/dialog.tsx +0 -118
  333. package/renderer/components/ui/dropdown-menu.tsx +0 -257
  334. package/renderer/components/ui/input.tsx +0 -21
  335. package/renderer/components/ui/label.tsx +0 -24
  336. package/renderer/components/ui/navigation-menu.tsx +0 -168
  337. package/renderer/components/ui/select.tsx +0 -190
  338. package/renderer/components/ui/spinner.tsx +0 -114
  339. package/renderer/components/ui/tabs.tsx +0 -66
  340. package/renderer/components/ui/tooltip.tsx +0 -61
  341. package/renderer/hooks/use-code-copy.ts +0 -88
  342. package/renderer/hooks/use-openapi-title.ts +0 -44
  343. package/renderer/lib/api-docs/agent/index.ts +0 -6
  344. package/renderer/lib/api-docs/agent/indexer.ts +0 -323
  345. package/renderer/lib/api-docs/agent/spec-summary.ts +0 -335
  346. package/renderer/lib/api-docs/agent/types.ts +0 -116
  347. package/renderer/lib/api-docs/auth/auth-context.tsx +0 -225
  348. package/renderer/lib/api-docs/auth/auth-storage.ts +0 -87
  349. package/renderer/lib/api-docs/auth/crypto.ts +0 -89
  350. package/renderer/lib/api-docs/auth/index.ts +0 -4
  351. package/renderer/lib/api-docs/code-editor/db.ts +0 -164
  352. package/renderer/lib/api-docs/code-editor/hooks.ts +0 -266
  353. package/renderer/lib/api-docs/code-editor/mode-context.tsx +0 -207
  354. package/renderer/lib/api-docs/code-editor/types.ts +0 -105
  355. package/renderer/lib/api-docs/codegen/definitions.ts +0 -297
  356. package/renderer/lib/api-docs/codegen/har.ts +0 -251
  357. package/renderer/lib/api-docs/codegen/index.ts +0 -159
  358. package/renderer/lib/api-docs/factories.ts +0 -170
  359. package/renderer/lib/api-docs/mobile-context.tsx +0 -112
  360. package/renderer/lib/api-docs/navigation-context.tsx +0 -88
  361. package/renderer/lib/api-docs/parsers/graphql/README.md +0 -129
  362. package/renderer/lib/api-docs/parsers/graphql/index.ts +0 -91
  363. package/renderer/lib/api-docs/parsers/graphql/parser.ts +0 -491
  364. package/renderer/lib/api-docs/parsers/graphql/transformer.ts +0 -246
  365. package/renderer/lib/api-docs/parsers/graphql/types.ts +0 -283
  366. package/renderer/lib/api-docs/parsers/openapi/README.md +0 -32
  367. package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +0 -60
  368. package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +0 -574
  369. package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +0 -403
  370. package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +0 -232
  371. package/renderer/lib/api-docs/parsers/openapi/index.ts +0 -171
  372. package/renderer/lib/api-docs/parsers/openapi/transformer.ts +0 -278
  373. package/renderer/lib/api-docs/parsers/openapi/validator.ts +0 -31
  374. package/renderer/lib/api-docs/playground/context.tsx +0 -107
  375. package/renderer/lib/api-docs/playground/navigation-context.tsx +0 -124
  376. package/renderer/lib/api-docs/playground/request-builder.ts +0 -223
  377. package/renderer/lib/api-docs/playground/request-runner.ts +0 -282
  378. package/renderer/lib/api-docs/playground/types.ts +0 -35
  379. package/renderer/lib/api-docs/types.ts +0 -269
  380. package/renderer/lib/api-docs/utils.ts +0 -311
  381. package/renderer/lib/cache.ts +0 -193
  382. package/renderer/lib/docs/config/domain-schema.ts +0 -260
  383. package/renderer/lib/docs/config/index.ts +0 -43
  384. package/renderer/lib/docs/config/loader.ts +0 -142
  385. package/renderer/lib/docs/config/schema.ts +0 -308
  386. package/renderer/lib/docs/index.ts +0 -12
  387. package/renderer/lib/docs/mdx/compiler.ts +0 -176
  388. package/renderer/lib/docs/mdx/frontmatter.ts +0 -91
  389. package/renderer/lib/docs/mdx/index.ts +0 -26
  390. package/renderer/lib/docs/navigation/generator.ts +0 -348
  391. package/renderer/lib/docs/navigation/index.ts +0 -12
  392. package/renderer/lib/docs/navigation/types.ts +0 -123
  393. package/renderer/lib/docs-navigation-context.tsx +0 -80
  394. package/renderer/lib/multi-tenant/context.ts +0 -105
  395. package/renderer/lib/storage/blob.ts +0 -1083
  396. package/renderer/lib/utils/icons.ts +0 -48
  397. package/renderer/lib/utils.ts +0 -6
  398. package/renderer/next.config.ts +0 -76
@@ -0,0 +1,936 @@
1
+ 'use client';
2
+ import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
3
+ import { useState, useCallback, useEffect, useRef } from 'react';
4
+ import { cn } from '@/lib/utils';
5
+ import { Play, ArrowClockwise, Copy, Check, Spinner, X, FileText, BracketsCurly, Key, CaretRight, CaretDown, MagnifyingGlass, Lightning, ArrowsClockwise } from '@phosphor-icons/react';
6
+ import { Button } from '@/components/ui/button';
7
+ import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip';
8
+ import Editor, { loader } from '@monaco-editor/react';
9
+ import { buildSchema } from 'graphql';
10
+ import { getAutocompleteSuggestions, CompletionItemKind } from 'graphql-language-service';
11
+ // Configure Monaco to load from CDN
12
+ loader.config({
13
+ paths: {
14
+ vs: 'https://cdn.jsdelivr.net/npm/monaco-editor@0.45.0/min/vs'
15
+ }
16
+ });
17
+ // Store parsed schema and completion provider disposable
18
+ let parsedSchema = null;
19
+ let completionProviderDisposable = null;
20
+ // Map graphql-language-service completion kinds to Monaco kinds
21
+ function toMonacoCompletionKind(kind, monaco) {
22
+ const mKinds = monaco.languages.CompletionItemKind;
23
+ switch(kind){
24
+ case CompletionItemKind.Field:
25
+ return mKinds.Field;
26
+ case CompletionItemKind.Variable:
27
+ return mKinds.Variable;
28
+ case CompletionItemKind.Property:
29
+ return mKinds.Property;
30
+ case CompletionItemKind.EnumMember:
31
+ return mKinds.EnumMember;
32
+ case CompletionItemKind.Class:
33
+ return mKinds.Class;
34
+ case CompletionItemKind.Interface:
35
+ return mKinds.Interface;
36
+ case CompletionItemKind.Function:
37
+ return mKinds.Function;
38
+ case CompletionItemKind.Snippet:
39
+ return mKinds.Snippet;
40
+ case CompletionItemKind.Keyword:
41
+ return mKinds.Keyword;
42
+ case CompletionItemKind.Constant:
43
+ return mKinds.Constant;
44
+ case CompletionItemKind.Enum:
45
+ return mKinds.Enum;
46
+ default:
47
+ return mKinds.Text;
48
+ }
49
+ }
50
+ // Find fields already selected in the current selection set
51
+ function getExistingFieldsInSelectionSet(documentText, cursorOffset) {
52
+ const existingFields = new Set();
53
+ // Find the opening brace of the current selection set
54
+ let braceCount = 0;
55
+ let selectionSetStart = -1;
56
+ for(let i = cursorOffset - 1; i >= 0; i--){
57
+ const char = documentText[i];
58
+ if (char === '}') braceCount++;
59
+ if (char === '{') {
60
+ if (braceCount === 0) {
61
+ selectionSetStart = i;
62
+ break;
63
+ }
64
+ braceCount--;
65
+ }
66
+ }
67
+ if (selectionSetStart === -1) return existingFields;
68
+ // Find the closing brace
69
+ braceCount = 1;
70
+ let selectionSetEnd = documentText.length;
71
+ for(let i = selectionSetStart + 1; i < documentText.length; i++){
72
+ const char = documentText[i];
73
+ if (char === '{') braceCount++;
74
+ if (char === '}') {
75
+ braceCount--;
76
+ if (braceCount === 0) {
77
+ selectionSetEnd = i;
78
+ break;
79
+ }
80
+ }
81
+ }
82
+ // Extract the selection set content
83
+ const selectionSetContent = documentText.slice(selectionSetStart + 1, selectionSetEnd);
84
+ // Parse field names - match field names that appear on their own or with arguments/aliases
85
+ // Matches: fieldName, fieldName(args), alias: fieldName, ...fieldName
86
+ const lines = selectionSetContent.split('\n');
87
+ for (const line of lines){
88
+ const trimmed = line.trim();
89
+ if (!trimmed || trimmed.startsWith('#')) continue; // Skip empty lines and comments
90
+ // Match field name at start of line (possibly with alias)
91
+ // Pattern: optionalAlias: fieldName or just fieldName
92
+ const fieldMatch = trimmed.match(/^(?:\.\.\.)?(?:[a-zA-Z_][a-zA-Z0-9_]*\s*:\s*)?([a-zA-Z_][a-zA-Z0-9_]*)/);
93
+ if (fieldMatch) {
94
+ const fieldName = fieldMatch[1];
95
+ // Skip GraphQL keywords but include __typename
96
+ if (![
97
+ 'query',
98
+ 'mutation',
99
+ 'subscription',
100
+ 'fragment',
101
+ 'on'
102
+ ].includes(fieldName)) {
103
+ existingFields.add(fieldName);
104
+ }
105
+ }
106
+ }
107
+ return existingFields;
108
+ }
109
+ // Register GraphQL completion provider using graphql-language-service
110
+ function registerGraphQLCompletionProvider(monaco, schema) {
111
+ // Dispose previous provider if exists
112
+ if (completionProviderDisposable) {
113
+ completionProviderDisposable.dispose();
114
+ }
115
+ completionProviderDisposable = monaco.languages.registerCompletionItemProvider('graphql', {
116
+ triggerCharacters: [
117
+ '{',
118
+ '(',
119
+ ' ',
120
+ ':',
121
+ '$',
122
+ '@',
123
+ '.',
124
+ '\n'
125
+ ],
126
+ provideCompletionItems: (model, position)=>{
127
+ const documentText = model.getValue();
128
+ const cursorOffset = model.getOffsetAt(position);
129
+ // graphql-language-service Position interface
130
+ const gqlPosition = {
131
+ line: position.lineNumber - 1,
132
+ character: position.column - 1,
133
+ setLine: (line)=>{
134
+ gqlPosition.line = line;
135
+ },
136
+ setCharacter: (char)=>{
137
+ gqlPosition.character = char;
138
+ },
139
+ lessThanOrEqualTo: (other)=>{
140
+ return gqlPosition.line < other.line || gqlPosition.line === other.line && gqlPosition.character <= other.character;
141
+ }
142
+ };
143
+ try {
144
+ // Get completions from graphql-language-service
145
+ const completions = getAutocompleteSuggestions(schema, documentText, gqlPosition);
146
+ // Get existing fields in current selection set to filter them out
147
+ const existingFields = getExistingFieldsInSelectionSet(documentText, cursorOffset);
148
+ // Filter out already-selected fields (only for Field kind suggestions)
149
+ const filteredCompletions = completions.filter((item)=>{
150
+ // Only filter out Field suggestions (kind = 5), keep keywords, types, etc.
151
+ if (item.kind === CompletionItemKind.Field) {
152
+ return !existingFields.has(item.label);
153
+ }
154
+ return true;
155
+ });
156
+ const word = model.getWordUntilPosition(position);
157
+ const range = {
158
+ startLineNumber: position.lineNumber,
159
+ endLineNumber: position.lineNumber,
160
+ startColumn: word.startColumn,
161
+ endColumn: word.endColumn
162
+ };
163
+ const suggestions = filteredCompletions.map((item, index)=>({
164
+ label: item.label,
165
+ kind: toMonacoCompletionKind(item.kind, monaco),
166
+ insertText: item.insertText || item.label,
167
+ insertTextRules: item.insertText?.includes('$') ? monaco.languages.CompletionItemInsertTextRule.InsertAsSnippet : undefined,
168
+ detail: item.detail || undefined,
169
+ documentation: item.documentation || undefined,
170
+ sortText: String(index).padStart(5, '0'),
171
+ range
172
+ }));
173
+ return {
174
+ suggestions
175
+ };
176
+ } catch (err) {
177
+ console.error('[GraphQL Playground] Completion error:', err);
178
+ return {
179
+ suggestions: []
180
+ };
181
+ }
182
+ }
183
+ });
184
+ }
185
+ const STORAGE_KEY = 'brainfish-graphql-playground';
186
+ // ============================================================================
187
+ // Helpers
188
+ // ============================================================================
189
+ function loadStoredState() {
190
+ if (typeof window === 'undefined') return null;
191
+ try {
192
+ const stored = localStorage.getItem(STORAGE_KEY);
193
+ if (stored) return JSON.parse(stored);
194
+ } catch {}
195
+ return null;
196
+ }
197
+ function saveStoredState(state) {
198
+ if (typeof window === 'undefined') return;
199
+ try {
200
+ localStorage.setItem(STORAGE_KEY, JSON.stringify(state));
201
+ } catch {}
202
+ }
203
+ function SchemaExplorer({ operations, onSelectOperation, selectedOperationId }) {
204
+ const [searchQuery, setSearchQuery] = useState('');
205
+ const [expandedSections, setExpandedSections] = useState({
206
+ query: true,
207
+ mutation: true,
208
+ subscription: true
209
+ });
210
+ // Group operations by type
211
+ const queries = operations.filter((op)=>op.operationType === 'query');
212
+ const mutations = operations.filter((op)=>op.operationType === 'mutation');
213
+ const subscriptions = operations.filter((op)=>op.operationType === 'subscription');
214
+ // Filter by search
215
+ const filterOps = (ops)=>{
216
+ if (!searchQuery.trim()) return ops;
217
+ const query = searchQuery.toLowerCase();
218
+ return ops.filter((op)=>op.name.toLowerCase().includes(query) || op.description?.toLowerCase().includes(query));
219
+ };
220
+ const filteredQueries = filterOps(queries);
221
+ const filteredMutations = filterOps(mutations);
222
+ const filteredSubscriptions = filterOps(subscriptions);
223
+ const toggleSection = (section)=>{
224
+ setExpandedSections((prev)=>({
225
+ ...prev,
226
+ [section]: !prev[section]
227
+ }));
228
+ };
229
+ const renderSection = (title, icon, ops, sectionKey, iconColor)=>{
230
+ if (ops.length === 0) return null;
231
+ const isExpanded = expandedSections[sectionKey];
232
+ return /*#__PURE__*/ _jsxs("div", {
233
+ className: "mb-2",
234
+ children: [
235
+ /*#__PURE__*/ _jsxs("button", {
236
+ onClick: ()=>toggleSection(sectionKey),
237
+ className: "w-full flex items-center gap-2 px-3 py-2 text-xs font-semibold uppercase tracking-wider text-muted-foreground hover:bg-muted/50 transition-colors",
238
+ children: [
239
+ isExpanded ? /*#__PURE__*/ _jsx(CaretDown, {
240
+ className: "w-3 h-3"
241
+ }) : /*#__PURE__*/ _jsx(CaretRight, {
242
+ className: "w-3 h-3"
243
+ }),
244
+ /*#__PURE__*/ _jsx("span", {
245
+ className: iconColor,
246
+ children: icon
247
+ }),
248
+ /*#__PURE__*/ _jsx("span", {
249
+ children: title
250
+ }),
251
+ /*#__PURE__*/ _jsx("span", {
252
+ className: "ml-auto text-[10px] font-normal opacity-60",
253
+ children: ops.length
254
+ })
255
+ ]
256
+ }),
257
+ isExpanded && /*#__PURE__*/ _jsx("div", {
258
+ className: "ml-2",
259
+ children: ops.map((op)=>/*#__PURE__*/ _jsx("button", {
260
+ onClick: ()=>onSelectOperation(op),
261
+ className: cn('w-full flex items-start gap-2 px-3 py-2 text-left', 'hover:bg-muted/50 transition-colors rounded-sm', 'border-l-2', selectedOperationId === op.id ? 'bg-muted/70 border-primary' : 'border-transparent'),
262
+ children: /*#__PURE__*/ _jsxs("div", {
263
+ className: "flex-1 min-w-0",
264
+ children: [
265
+ /*#__PURE__*/ _jsx("div", {
266
+ className: "text-sm font-medium truncate",
267
+ children: op.name
268
+ }),
269
+ op.description && /*#__PURE__*/ _jsx("div", {
270
+ className: "text-xs text-muted-foreground truncate mt-0.5",
271
+ children: op.description
272
+ })
273
+ ]
274
+ })
275
+ }, op.id))
276
+ })
277
+ ]
278
+ });
279
+ };
280
+ const hasOperations = operations.length > 0;
281
+ const hasFilteredResults = filteredQueries.length > 0 || filteredMutations.length > 0 || filteredSubscriptions.length > 0;
282
+ return /*#__PURE__*/ _jsxs("div", {
283
+ className: "flex flex-col h-full border-r border-border bg-muted/20",
284
+ children: [
285
+ /*#__PURE__*/ _jsx("div", {
286
+ className: "p-2 border-b border-border",
287
+ children: /*#__PURE__*/ _jsxs("div", {
288
+ className: "relative",
289
+ children: [
290
+ /*#__PURE__*/ _jsx(MagnifyingGlass, {
291
+ className: "absolute left-2.5 top-1/2 -translate-y-1/2 w-4 h-4 text-muted-foreground"
292
+ }),
293
+ /*#__PURE__*/ _jsx("input", {
294
+ type: "text",
295
+ value: searchQuery,
296
+ onChange: (e)=>setSearchQuery(e.target.value),
297
+ placeholder: "Search operations...",
298
+ className: cn('w-full pl-8 pr-3 py-1.5 text-sm rounded-md', 'bg-background border border-border', 'focus:outline-none focus:ring-1 focus:ring-primary', 'placeholder:text-muted-foreground')
299
+ })
300
+ ]
301
+ })
302
+ }),
303
+ /*#__PURE__*/ _jsx("div", {
304
+ className: "flex-1 overflow-y-auto py-2",
305
+ children: !hasOperations ? /*#__PURE__*/ _jsxs("div", {
306
+ className: "flex flex-col items-center justify-center h-full px-4 text-center",
307
+ children: [
308
+ /*#__PURE__*/ _jsx(FileText, {
309
+ className: "w-8 h-8 text-muted-foreground/50 mb-2"
310
+ }),
311
+ /*#__PURE__*/ _jsx("p", {
312
+ className: "text-sm text-muted-foreground",
313
+ children: "No schema loaded"
314
+ }),
315
+ /*#__PURE__*/ _jsx("p", {
316
+ className: "text-xs text-muted-foreground/60 mt-1",
317
+ children: "Operations will appear here"
318
+ })
319
+ ]
320
+ }) : !hasFilteredResults ? /*#__PURE__*/ _jsx("div", {
321
+ className: "flex flex-col items-center justify-center h-32 px-4 text-center",
322
+ children: /*#__PURE__*/ _jsx("p", {
323
+ className: "text-sm text-muted-foreground",
324
+ children: "No matching operations"
325
+ })
326
+ }) : /*#__PURE__*/ _jsxs(_Fragment, {
327
+ children: [
328
+ renderSection('Queries', /*#__PURE__*/ _jsx(MagnifyingGlass, {
329
+ className: "w-3.5 h-3.5",
330
+ weight: "bold"
331
+ }), filteredQueries, 'query', 'text-blue-500'),
332
+ renderSection('Mutations', /*#__PURE__*/ _jsx(Lightning, {
333
+ className: "w-3.5 h-3.5",
334
+ weight: "bold"
335
+ }), filteredMutations, 'mutation', 'text-orange-500'),
336
+ renderSection('Subscriptions', /*#__PURE__*/ _jsx(ArrowsClockwise, {
337
+ className: "w-3.5 h-3.5",
338
+ weight: "bold"
339
+ }), filteredSubscriptions, 'subscription', 'text-purple-500')
340
+ ]
341
+ })
342
+ })
343
+ ]
344
+ });
345
+ }
346
+ // ============================================================================
347
+ // Main Component
348
+ // ============================================================================
349
+ export function GraphQLPlayground({ endpoint, defaultQuery = `# Welcome to the GraphQL Playground!
350
+ #
351
+ # Select an operation from the sidebar, or
352
+ # type your query below and press Ctrl+Enter
353
+
354
+ query {
355
+ __typename
356
+ }
357
+ `, headers: defaultHeaders = {}, operations = [], selectedOperationId: externalSelectedId, hideExplorer = false, className, theme = 'dark', schemaSDL }) {
358
+ const [query, setQuery] = useState(defaultQuery);
359
+ const [variables, setVariables] = useState('{}');
360
+ const [customHeaders, setCustomHeaders] = useState(Object.keys(defaultHeaders).length > 0 ? JSON.stringify(defaultHeaders, null, 2) : '{}');
361
+ const [response, setResponse] = useState(null);
362
+ const [isLoading, setIsLoading] = useState(false);
363
+ const [error, setError] = useState(null);
364
+ const [copied, setCopied] = useState(false);
365
+ const [activeTab, setActiveTab] = useState('variables');
366
+ const [responseTime, setResponseTime] = useState(null);
367
+ const [internalSelectedId, setInternalSelectedId] = useState();
368
+ // Use external selection if provided, otherwise use internal
369
+ const selectedOperationId = externalSelectedId ?? internalSelectedId;
370
+ const abortControllerRef = useRef(null);
371
+ const hasLoadedRef = useRef(false);
372
+ const monacoRef = useRef(null);
373
+ // Initialize Monaco GraphQL when editor mounts
374
+ const handleEditorDidMount = useCallback((editor, monaco)=>{
375
+ monacoRef.current = monaco;
376
+ // Parse schema and register completion provider
377
+ if (schemaSDL) {
378
+ try {
379
+ parsedSchema = buildSchema(schemaSDL);
380
+ registerGraphQLCompletionProvider(monaco, parsedSchema);
381
+ } catch (err) {
382
+ console.error('[GraphQL Playground] Failed to parse schema:', err);
383
+ }
384
+ }
385
+ }, [
386
+ schemaSDL
387
+ ]);
388
+ // Update completion provider when schema changes
389
+ useEffect(()=>{
390
+ if (schemaSDL && monacoRef.current) {
391
+ try {
392
+ parsedSchema = buildSchema(schemaSDL);
393
+ registerGraphQLCompletionProvider(monacoRef.current, parsedSchema);
394
+ } catch (err) {
395
+ console.error('[GraphQL Playground] Failed to update schema:', err);
396
+ }
397
+ }
398
+ }, [
399
+ schemaSDL
400
+ ]);
401
+ // Load from localStorage on mount
402
+ useEffect(()=>{
403
+ const stored = loadStoredState();
404
+ if (stored) {
405
+ if (stored.query) setQuery(stored.query);
406
+ if (stored.variables) setVariables(stored.variables);
407
+ if (stored.headers) setCustomHeaders(stored.headers);
408
+ }
409
+ hasLoadedRef.current = true;
410
+ }, []);
411
+ // Save to localStorage when values change
412
+ useEffect(()=>{
413
+ if (!hasLoadedRef.current) return;
414
+ const timeoutId = setTimeout(()=>{
415
+ saveStoredState({
416
+ query,
417
+ variables,
418
+ headers: customHeaders
419
+ });
420
+ }, 500);
421
+ return ()=>clearTimeout(timeoutId);
422
+ }, [
423
+ query,
424
+ variables,
425
+ customHeaders
426
+ ]);
427
+ // Handle operation selection from sidebar (internal)
428
+ const handleSelectOperation = useCallback((operation)=>{
429
+ setQuery(operation.query);
430
+ setInternalSelectedId(operation.id);
431
+ // Set example variables if available
432
+ if (operation.exampleVariables && Object.keys(operation.exampleVariables).length > 0) {
433
+ setVariables(JSON.stringify(operation.exampleVariables, null, 2));
434
+ } else {
435
+ setVariables('{}');
436
+ }
437
+ // Clear previous response
438
+ setResponse(null);
439
+ setError(null);
440
+ setResponseTime(null);
441
+ }, []);
442
+ // Sync with external selection (from main sidebar)
443
+ useEffect(()=>{
444
+ if (externalSelectedId && operations.length > 0) {
445
+ const op = operations.find((o)=>o.id === externalSelectedId);
446
+ if (op) {
447
+ setQuery(op.query);
448
+ if (op.exampleVariables && Object.keys(op.exampleVariables).length > 0) {
449
+ setVariables(JSON.stringify(op.exampleVariables, null, 2));
450
+ } else {
451
+ setVariables('{}');
452
+ }
453
+ setResponse(null);
454
+ setError(null);
455
+ setResponseTime(null);
456
+ }
457
+ }
458
+ }, [
459
+ externalSelectedId,
460
+ operations
461
+ ]);
462
+ // Execute GraphQL query
463
+ const executeQuery = useCallback(async ()=>{
464
+ setIsLoading(true);
465
+ setError(null);
466
+ setResponse(null);
467
+ setResponseTime(null);
468
+ if (abortControllerRef.current) {
469
+ abortControllerRef.current.abort();
470
+ }
471
+ abortControllerRef.current = new AbortController();
472
+ const startTime = performance.now();
473
+ try {
474
+ let parsedVariables = {};
475
+ try {
476
+ parsedVariables = variables.trim() && variables.trim() !== '{}' ? JSON.parse(variables) : {};
477
+ } catch {
478
+ setError('Invalid JSON in variables');
479
+ setIsLoading(false);
480
+ return;
481
+ }
482
+ let parsedHeaders = {};
483
+ try {
484
+ parsedHeaders = customHeaders.trim() && customHeaders.trim() !== '{}' ? JSON.parse(customHeaders) : {};
485
+ } catch {
486
+ setError('Invalid JSON in headers');
487
+ setIsLoading(false);
488
+ return;
489
+ }
490
+ const res = await fetch(endpoint, {
491
+ method: 'POST',
492
+ headers: {
493
+ 'Content-Type': 'application/json',
494
+ Accept: 'application/json',
495
+ ...defaultHeaders,
496
+ ...parsedHeaders
497
+ },
498
+ body: JSON.stringify({
499
+ query,
500
+ variables: parsedVariables
501
+ }),
502
+ signal: abortControllerRef.current.signal
503
+ });
504
+ const endTime = performance.now();
505
+ setResponseTime(Math.round(endTime - startTime));
506
+ const data = await res.json();
507
+ setResponse(data);
508
+ } catch (err) {
509
+ if (err instanceof Error && err.name === 'AbortError') return;
510
+ setError(err instanceof Error ? err.message : 'Request failed');
511
+ } finally{
512
+ setIsLoading(false);
513
+ }
514
+ }, [
515
+ query,
516
+ variables,
517
+ customHeaders,
518
+ endpoint,
519
+ defaultHeaders
520
+ ]);
521
+ const handleCancel = useCallback(()=>{
522
+ if (abortControllerRef.current) {
523
+ abortControllerRef.current.abort();
524
+ abortControllerRef.current = null;
525
+ }
526
+ setIsLoading(false);
527
+ }, []);
528
+ useEffect(()=>{
529
+ const handleKeyDown = (e)=>{
530
+ if ((e.ctrlKey || e.metaKey) && e.key === 'Enter') {
531
+ e.preventDefault();
532
+ if (!isLoading) executeQuery();
533
+ }
534
+ };
535
+ window.addEventListener('keydown', handleKeyDown);
536
+ return ()=>window.removeEventListener('keydown', handleKeyDown);
537
+ }, [
538
+ executeQuery,
539
+ isLoading
540
+ ]);
541
+ const copyResponse = useCallback(()=>{
542
+ if (response) {
543
+ navigator.clipboard.writeText(JSON.stringify(response, null, 2));
544
+ setCopied(true);
545
+ setTimeout(()=>setCopied(false), 2000);
546
+ }
547
+ }, [
548
+ response
549
+ ]);
550
+ const handleReset = useCallback(()=>{
551
+ setQuery(defaultQuery);
552
+ setVariables('{}');
553
+ setCustomHeaders(Object.keys(defaultHeaders).length > 0 ? JSON.stringify(defaultHeaders, null, 2) : '{}');
554
+ setResponse(null);
555
+ setError(null);
556
+ setResponseTime(null);
557
+ setInternalSelectedId(undefined);
558
+ localStorage.removeItem(STORAGE_KEY);
559
+ }, [
560
+ defaultQuery,
561
+ defaultHeaders
562
+ ]);
563
+ const monacoTheme = theme === 'dark' ? 'vs-dark' : 'vs';
564
+ const hasErrors = response?.errors && response.errors.length > 0;
565
+ const statusColor = hasErrors ? 'text-red-400' : response ? 'text-emerald-400' : 'text-zinc-500';
566
+ return /*#__PURE__*/ _jsx(TooltipProvider, {
567
+ children: /*#__PURE__*/ _jsxs("div", {
568
+ className: cn('flex flex-col h-full bg-background', className),
569
+ children: [
570
+ /*#__PURE__*/ _jsxs("div", {
571
+ className: "flex items-center justify-between gap-2 px-4 py-2 border-b border-border bg-muted/50",
572
+ children: [
573
+ /*#__PURE__*/ _jsx("div", {
574
+ className: "flex items-center gap-2",
575
+ children: /*#__PURE__*/ _jsxs("div", {
576
+ className: "hidden sm:flex items-center gap-2 px-3 py-1.5 rounded-md bg-background border border-border",
577
+ children: [
578
+ /*#__PURE__*/ _jsx("span", {
579
+ className: "text-xs font-medium text-emerald-500",
580
+ children: "POST"
581
+ }),
582
+ /*#__PURE__*/ _jsx("span", {
583
+ className: "text-xs text-muted-foreground font-mono truncate max-w-[200px] md:max-w-[400px]",
584
+ children: endpoint
585
+ })
586
+ ]
587
+ })
588
+ }),
589
+ /*#__PURE__*/ _jsxs("div", {
590
+ className: "flex items-center gap-2",
591
+ children: [
592
+ /*#__PURE__*/ _jsx(Button, {
593
+ onClick: isLoading ? handleCancel : executeQuery,
594
+ variant: isLoading ? 'destructive' : 'default',
595
+ size: "sm",
596
+ className: "px-4",
597
+ children: isLoading ? /*#__PURE__*/ _jsxs(_Fragment, {
598
+ children: [
599
+ /*#__PURE__*/ _jsx(X, {
600
+ className: "h-4 w-4 mr-2",
601
+ weight: "bold"
602
+ }),
603
+ "Cancel"
604
+ ]
605
+ }) : /*#__PURE__*/ _jsxs(_Fragment, {
606
+ children: [
607
+ /*#__PURE__*/ _jsx(Play, {
608
+ className: "h-4 w-4 mr-2",
609
+ weight: "fill"
610
+ }),
611
+ "Run"
612
+ ]
613
+ })
614
+ }),
615
+ /*#__PURE__*/ _jsxs(Tooltip, {
616
+ children: [
617
+ /*#__PURE__*/ _jsx(TooltipTrigger, {
618
+ asChild: true,
619
+ children: /*#__PURE__*/ _jsx(Button, {
620
+ variant: "outline",
621
+ size: "icon",
622
+ onClick: handleReset,
623
+ children: /*#__PURE__*/ _jsx(ArrowClockwise, {
624
+ className: "h-4 w-4",
625
+ weight: "bold"
626
+ })
627
+ })
628
+ }),
629
+ /*#__PURE__*/ _jsx(TooltipContent, {
630
+ children: "Reset to defaults"
631
+ })
632
+ ]
633
+ })
634
+ ]
635
+ })
636
+ ]
637
+ }),
638
+ /*#__PURE__*/ _jsxs("div", {
639
+ className: "flex-1 flex overflow-hidden",
640
+ children: [
641
+ !hideExplorer && operations.length > 0 && /*#__PURE__*/ _jsx("div", {
642
+ className: "w-64 flex-shrink-0 hidden md:block",
643
+ children: /*#__PURE__*/ _jsx(SchemaExplorer, {
644
+ operations: operations,
645
+ onSelectOperation: handleSelectOperation,
646
+ selectedOperationId: selectedOperationId
647
+ })
648
+ }),
649
+ /*#__PURE__*/ _jsxs("div", {
650
+ className: "flex-1 flex flex-col lg:flex-row overflow-hidden",
651
+ children: [
652
+ /*#__PURE__*/ _jsxs("div", {
653
+ className: "flex-1 flex flex-col border-b lg:border-b-0 lg:border-r border-border min-h-[200px] lg:min-h-0",
654
+ children: [
655
+ /*#__PURE__*/ _jsxs("div", {
656
+ className: "flex-1 flex flex-col min-h-0",
657
+ children: [
658
+ /*#__PURE__*/ _jsxs("div", {
659
+ className: "flex items-center px-3 py-2 border-b border-border bg-muted/30",
660
+ children: [
661
+ /*#__PURE__*/ _jsx(FileText, {
662
+ className: "h-4 w-4 mr-2 text-muted-foreground"
663
+ }),
664
+ /*#__PURE__*/ _jsx("span", {
665
+ className: "text-xs font-medium text-muted-foreground uppercase tracking-wider",
666
+ children: "Query"
667
+ }),
668
+ /*#__PURE__*/ _jsx("span", {
669
+ className: "ml-auto text-xs text-muted-foreground",
670
+ children: "Ctrl+Enter to run"
671
+ })
672
+ ]
673
+ }),
674
+ /*#__PURE__*/ _jsx("div", {
675
+ className: "flex-1 min-h-0",
676
+ children: /*#__PURE__*/ _jsx(Editor, {
677
+ height: "100%",
678
+ language: "graphql",
679
+ value: query,
680
+ onMount: handleEditorDidMount,
681
+ onChange: (value)=>{
682
+ setQuery(value || '');
683
+ setInternalSelectedId(undefined); // Clear internal selection when manually editing
684
+ },
685
+ theme: monacoTheme,
686
+ options: {
687
+ minimap: {
688
+ enabled: false
689
+ },
690
+ scrollBeyondLastLine: false,
691
+ fontSize: 13,
692
+ fontFamily: "'JetBrains Mono', 'Fira Code', 'SF Mono', Consolas, monospace",
693
+ lineNumbers: 'on',
694
+ renderLineHighlight: 'line',
695
+ scrollbar: {
696
+ vertical: 'auto',
697
+ horizontal: 'auto'
698
+ },
699
+ padding: {
700
+ top: 12,
701
+ bottom: 12
702
+ },
703
+ wordWrap: 'on',
704
+ automaticLayout: true,
705
+ tabSize: 2,
706
+ quickSuggestions: true,
707
+ suggestOnTriggerCharacters: true
708
+ },
709
+ loading: /*#__PURE__*/ _jsxs("div", {
710
+ className: "flex items-center justify-center h-full text-muted-foreground text-sm",
711
+ children: [
712
+ /*#__PURE__*/ _jsx(Spinner, {
713
+ className: "w-5 h-5 animate-spin mr-2"
714
+ }),
715
+ "Loading editor..."
716
+ ]
717
+ })
718
+ })
719
+ })
720
+ ]
721
+ }),
722
+ /*#__PURE__*/ _jsxs("div", {
723
+ className: "border-t border-border h-[180px] flex flex-col",
724
+ children: [
725
+ /*#__PURE__*/ _jsxs("div", {
726
+ className: "flex border-b border-border bg-muted/30",
727
+ children: [
728
+ /*#__PURE__*/ _jsxs("button", {
729
+ onClick: ()=>setActiveTab('variables'),
730
+ className: cn('flex items-center gap-1.5 px-4 py-2 text-xs font-medium uppercase tracking-wider transition-colors', activeTab === 'variables' ? 'text-primary border-b-2 border-primary -mb-px bg-background' : 'text-muted-foreground hover:text-foreground'),
731
+ children: [
732
+ /*#__PURE__*/ _jsx(BracketsCurly, {
733
+ className: "h-3.5 w-3.5"
734
+ }),
735
+ "Variables"
736
+ ]
737
+ }),
738
+ /*#__PURE__*/ _jsxs("button", {
739
+ onClick: ()=>setActiveTab('headers'),
740
+ className: cn('flex items-center gap-1.5 px-4 py-2 text-xs font-medium uppercase tracking-wider transition-colors', activeTab === 'headers' ? 'text-primary border-b-2 border-primary -mb-px bg-background' : 'text-muted-foreground hover:text-foreground'),
741
+ children: [
742
+ /*#__PURE__*/ _jsx(Key, {
743
+ className: "h-3.5 w-3.5"
744
+ }),
745
+ "Headers"
746
+ ]
747
+ })
748
+ ]
749
+ }),
750
+ /*#__PURE__*/ _jsx("div", {
751
+ className: "flex-1 min-h-0",
752
+ children: /*#__PURE__*/ _jsx(Editor, {
753
+ height: "100%",
754
+ language: "json",
755
+ value: activeTab === 'variables' ? variables : customHeaders,
756
+ onChange: (value)=>activeTab === 'variables' ? setVariables(value || '{}') : setCustomHeaders(value || '{}'),
757
+ theme: monacoTheme,
758
+ options: {
759
+ minimap: {
760
+ enabled: false
761
+ },
762
+ scrollBeyondLastLine: false,
763
+ fontSize: 13,
764
+ fontFamily: "'JetBrains Mono', 'Fira Code', 'SF Mono', Consolas, monospace",
765
+ lineNumbers: 'off',
766
+ renderLineHighlight: 'none',
767
+ scrollbar: {
768
+ vertical: 'auto',
769
+ horizontal: 'auto'
770
+ },
771
+ padding: {
772
+ top: 8,
773
+ bottom: 8
774
+ },
775
+ wordWrap: 'on',
776
+ automaticLayout: true,
777
+ tabSize: 2,
778
+ folding: false
779
+ },
780
+ loading: /*#__PURE__*/ _jsx("div", {
781
+ className: "flex items-center justify-center h-full text-muted-foreground text-sm",
782
+ children: "Loading..."
783
+ })
784
+ })
785
+ })
786
+ ]
787
+ })
788
+ ]
789
+ }),
790
+ /*#__PURE__*/ _jsxs("div", {
791
+ className: "flex-1 flex flex-col min-h-[200px] lg:min-h-0",
792
+ children: [
793
+ /*#__PURE__*/ _jsxs("div", {
794
+ className: "flex items-center justify-between px-3 py-2 border-b border-border bg-muted/30",
795
+ children: [
796
+ /*#__PURE__*/ _jsxs("div", {
797
+ className: "flex items-center gap-3",
798
+ children: [
799
+ /*#__PURE__*/ _jsx("span", {
800
+ className: "text-xs font-medium text-muted-foreground uppercase tracking-wider",
801
+ children: "Response"
802
+ }),
803
+ responseTime !== null && /*#__PURE__*/ _jsxs("span", {
804
+ className: "text-xs text-muted-foreground",
805
+ children: [
806
+ responseTime,
807
+ "ms"
808
+ ]
809
+ }),
810
+ response && /*#__PURE__*/ _jsx("span", {
811
+ className: cn('text-xs font-medium', statusColor),
812
+ children: hasErrors ? 'Error' : 'Success'
813
+ })
814
+ ]
815
+ }),
816
+ response && /*#__PURE__*/ _jsxs(Tooltip, {
817
+ children: [
818
+ /*#__PURE__*/ _jsx(TooltipTrigger, {
819
+ asChild: true,
820
+ children: /*#__PURE__*/ _jsx(Button, {
821
+ variant: "ghost",
822
+ size: "sm",
823
+ onClick: copyResponse,
824
+ className: "h-7 px-2",
825
+ children: copied ? /*#__PURE__*/ _jsxs(_Fragment, {
826
+ children: [
827
+ /*#__PURE__*/ _jsx(Check, {
828
+ className: "h-3.5 w-3.5 mr-1 text-emerald-500"
829
+ }),
830
+ /*#__PURE__*/ _jsx("span", {
831
+ className: "text-xs",
832
+ children: "Copied"
833
+ })
834
+ ]
835
+ }) : /*#__PURE__*/ _jsxs(_Fragment, {
836
+ children: [
837
+ /*#__PURE__*/ _jsx(Copy, {
838
+ className: "h-3.5 w-3.5 mr-1"
839
+ }),
840
+ /*#__PURE__*/ _jsx("span", {
841
+ className: "text-xs",
842
+ children: "Copy"
843
+ })
844
+ ]
845
+ })
846
+ })
847
+ }),
848
+ /*#__PURE__*/ _jsx(TooltipContent, {
849
+ children: "Copy response"
850
+ })
851
+ ]
852
+ })
853
+ ]
854
+ }),
855
+ /*#__PURE__*/ _jsx("div", {
856
+ className: "flex-1 min-h-0 overflow-hidden",
857
+ children: isLoading ? /*#__PURE__*/ _jsxs("div", {
858
+ className: "flex items-center justify-center h-full",
859
+ children: [
860
+ /*#__PURE__*/ _jsx(Spinner, {
861
+ className: "w-6 h-6 animate-spin text-primary"
862
+ }),
863
+ /*#__PURE__*/ _jsx("span", {
864
+ className: "ml-2 text-sm text-muted-foreground",
865
+ children: "Executing query..."
866
+ })
867
+ ]
868
+ }) : error ? /*#__PURE__*/ _jsx("div", {
869
+ className: "p-4",
870
+ children: /*#__PURE__*/ _jsx("div", {
871
+ className: "p-4 bg-destructive/10 border border-destructive/20 rounded-lg",
872
+ children: /*#__PURE__*/ _jsx("p", {
873
+ className: "text-destructive text-sm font-mono",
874
+ children: error
875
+ })
876
+ })
877
+ }) : response ? /*#__PURE__*/ _jsx(Editor, {
878
+ height: "100%",
879
+ language: "json",
880
+ value: JSON.stringify(response, null, 2),
881
+ theme: monacoTheme,
882
+ options: {
883
+ readOnly: true,
884
+ minimap: {
885
+ enabled: false
886
+ },
887
+ scrollBeyondLastLine: false,
888
+ fontSize: 13,
889
+ fontFamily: "'JetBrains Mono', 'Fira Code', 'SF Mono', Consolas, monospace",
890
+ lineNumbers: 'on',
891
+ renderLineHighlight: 'none',
892
+ scrollbar: {
893
+ vertical: 'auto',
894
+ horizontal: 'auto'
895
+ },
896
+ padding: {
897
+ top: 12,
898
+ bottom: 12
899
+ },
900
+ wordWrap: 'on',
901
+ automaticLayout: true,
902
+ folding: true
903
+ }
904
+ }) : /*#__PURE__*/ _jsxs("div", {
905
+ className: "flex flex-col items-center justify-center h-full text-muted-foreground",
906
+ children: [
907
+ /*#__PURE__*/ _jsx(Play, {
908
+ className: "h-12 w-12 mb-3 opacity-20"
909
+ }),
910
+ /*#__PURE__*/ _jsx("p", {
911
+ className: "text-sm",
912
+ children: "Run a query to see results"
913
+ }),
914
+ /*#__PURE__*/ _jsx("p", {
915
+ className: "text-xs mt-1 opacity-60",
916
+ children: "Press Ctrl+Enter or click Run"
917
+ })
918
+ ]
919
+ })
920
+ })
921
+ ]
922
+ })
923
+ ]
924
+ })
925
+ ]
926
+ })
927
+ ]
928
+ })
929
+ });
930
+ }
931
+ export function GraphQLPlaygroundWithExplorer(props) {
932
+ return /*#__PURE__*/ _jsx(GraphQLPlayground, {
933
+ ...props
934
+ });
935
+ }
936
+ export default GraphQLPlayground;