@brainfish-ai/devdoc 0.1.42 → 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 (388) hide show
  1. package/dist/cli/commands/dev.js +19 -10
  2. package/package.json +3 -2
  3. package/renderer/app/api/assets/[...path]/route.js +108 -0
  4. package/renderer/app/api/assets/route.js +114 -0
  5. package/renderer/app/api/assets/upload/route.js +163 -0
  6. package/renderer/app/api/auth-schemes/route.js +58 -0
  7. package/renderer/app/api/chat/route.js +759 -0
  8. package/renderer/app/api/codegen/route.js +52 -0
  9. package/renderer/app/api/collections/route.js +675 -0
  10. package/renderer/app/api/debug/route.js +47 -0
  11. package/renderer/app/api/deploy/route.js +199 -0
  12. package/renderer/app/api/device/route.js +36 -0
  13. package/renderer/app/api/docs/route.js +205 -0
  14. package/renderer/app/api/domains/add/route.js +121 -0
  15. package/renderer/app/api/domains/lookup/route.js +43 -0
  16. package/renderer/app/api/domains/remove/route.js +89 -0
  17. package/renderer/app/api/domains/status/route.js +140 -0
  18. package/renderer/app/api/domains/verify/route.js +168 -0
  19. package/renderer/app/api/keys/regenerate/route.js +71 -0
  20. package/renderer/app/api/local-assets/[...path]/route.js +108 -0
  21. package/renderer/app/api/openapi-spec/route.js +73 -0
  22. package/renderer/app/api/projects/[slug]/route.js +129 -0
  23. package/renderer/app/api/projects/[slug]/stats/route.js +80 -0
  24. package/renderer/app/api/projects/register/route.js +176 -0
  25. package/renderer/app/api/proxy/route.js +139 -0
  26. package/renderer/app/api/proxy-stream/route.js +156 -0
  27. package/renderer/app/api/redirects/route.js +35 -0
  28. package/renderer/app/api/schema/route.js +85 -0
  29. package/renderer/app/api/subdomains/check/route.js +158 -0
  30. package/renderer/app/api/suggestions/route.js +175 -0
  31. package/renderer/app/layout.js +47 -0
  32. package/renderer/app/llms-full.txt/route.js +257 -0
  33. package/renderer/app/llms.txt/route.js +219 -0
  34. package/renderer/app/page.js +12 -0
  35. package/renderer/app/robots.txt/route.js +66 -0
  36. package/renderer/app/sitemap.xml/route.js +145 -0
  37. package/renderer/components/docs/index.js +8 -0
  38. package/renderer/components/docs/mdx/accordion.js +113 -0
  39. package/renderer/components/docs/mdx/badge.js +72 -0
  40. package/renderer/components/docs/mdx/callouts.js +137 -0
  41. package/renderer/components/docs/mdx/cards.js +175 -0
  42. package/renderer/components/docs/mdx/changelog.js +100 -0
  43. package/renderer/components/docs/mdx/code-block.js +147 -0
  44. package/renderer/components/docs/mdx/code-group.js +287 -0
  45. package/renderer/components/docs/mdx/file-embeds.js +82 -0
  46. package/renderer/components/docs/mdx/frame.js +59 -0
  47. package/renderer/components/docs/mdx/highlight.js +90 -0
  48. package/renderer/components/docs/mdx/iframe.js +69 -0
  49. package/renderer/components/docs/mdx/image.js +135 -0
  50. package/renderer/components/docs/mdx/index.js +134 -0
  51. package/renderer/components/docs/mdx/landing.js +315 -0
  52. package/renderer/components/docs/mdx/mermaid.js +212 -0
  53. package/renderer/components/docs/mdx/param-field.js +112 -0
  54. package/renderer/components/docs/mdx/steps.js +74 -0
  55. package/renderer/components/docs/mdx/tabs.js +50 -0
  56. package/renderer/components/docs/mdx-renderer.js +77 -0
  57. package/renderer/components/docs/navigation/breadcrumbs.js +64 -0
  58. package/renderer/components/docs/navigation/index.js +6 -0
  59. package/renderer/components/docs/navigation/page-nav.js +57 -0
  60. package/renderer/components/docs/navigation/sidebar.js +375 -0
  61. package/renderer/components/docs/navigation/toc.js +89 -0
  62. package/renderer/components/docs/notice.js +77 -0
  63. package/renderer/components/docs-header.js +202 -0
  64. package/renderer/components/docs-viewer/agent/agent-chat.js +1930 -0
  65. package/renderer/components/docs-viewer/agent/cards/debug-context-card.js +107 -0
  66. package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.js +57 -0
  67. package/renderer/components/docs-viewer/agent/cards/index.js +45 -0
  68. package/renderer/components/docs-viewer/agent/cards/response-options-card.js +154 -0
  69. package/renderer/components/docs-viewer/agent/cards/types.js +22 -0
  70. package/renderer/components/docs-viewer/agent/chat-message.js +2 -0
  71. package/renderer/components/docs-viewer/agent/index.js +4 -0
  72. package/renderer/components/docs-viewer/agent/messages/assistant-message.js +108 -0
  73. package/renderer/components/docs-viewer/agent/messages/chat-message.js +34 -0
  74. package/renderer/components/docs-viewer/agent/messages/index.js +6 -0
  75. package/renderer/components/docs-viewer/agent/messages/tool-call-display.js +1065 -0
  76. package/renderer/components/docs-viewer/agent/messages/types.js +2 -0
  77. package/renderer/components/docs-viewer/agent/messages/typing-indicator.js +26 -0
  78. package/renderer/components/docs-viewer/agent/messages/user-message.js +37 -0
  79. package/renderer/components/docs-viewer/code-editor/{index.tsx → index.js} +1 -1
  80. package/renderer/components/docs-viewer/code-editor/notes-mode.js +1338 -0
  81. package/renderer/components/docs-viewer/content/changelog-page.js +297 -0
  82. package/renderer/components/docs-viewer/content/doc-page.js +264 -0
  83. package/renderer/components/docs-viewer/content/documentation-viewer.js +14 -0
  84. package/renderer/components/docs-viewer/content/index.js +29 -0
  85. package/renderer/components/docs-viewer/content/not-found-page.js +300 -0
  86. package/renderer/components/docs-viewer/content/request-details.js +528 -0
  87. package/renderer/components/docs-viewer/content/sections/auth.js +108 -0
  88. package/renderer/components/docs-viewer/content/sections/body.js +80 -0
  89. package/renderer/components/docs-viewer/content/sections/headers.js +64 -0
  90. package/renderer/components/docs-viewer/content/sections/overview.js +56 -0
  91. package/renderer/components/docs-viewer/content/sections/parameters.js +64 -0
  92. package/renderer/components/docs-viewer/content/sections/responses.js +91 -0
  93. package/renderer/components/docs-viewer/global-auth-modal.js +427 -0
  94. package/renderer/components/docs-viewer/index.js +1552 -0
  95. package/renderer/components/docs-viewer/playground/auth-editor.js +418 -0
  96. package/renderer/components/docs-viewer/playground/body-editor.js +240 -0
  97. package/renderer/components/docs-viewer/playground/code-editor.js +135 -0
  98. package/renderer/components/docs-viewer/playground/code-snippet.js +393 -0
  99. package/renderer/components/docs-viewer/playground/graphql-playground.js +734 -0
  100. package/renderer/components/docs-viewer/playground/index.js +682 -0
  101. package/renderer/components/docs-viewer/playground/key-value-editor.js +317 -0
  102. package/renderer/components/docs-viewer/playground/method-selector.js +65 -0
  103. package/renderer/components/docs-viewer/playground/request-builder.js +181 -0
  104. package/renderer/components/docs-viewer/playground/request-tabs.js +240 -0
  105. package/renderer/components/docs-viewer/playground/response-cards/idle-card.js +42 -0
  106. package/renderer/components/docs-viewer/playground/response-cards/index.js +72 -0
  107. package/renderer/components/docs-viewer/playground/response-cards/loading-card.js +24 -0
  108. package/renderer/components/docs-viewer/playground/response-cards/network-error-card.js +28 -0
  109. package/renderer/components/docs-viewer/playground/response-cards/response-body-card.js +308 -0
  110. package/renderer/components/docs-viewer/playground/response-cards/types.js +9 -0
  111. package/renderer/components/docs-viewer/playground/response-viewer.js +18 -0
  112. package/renderer/components/docs-viewer/search/index.js +2 -0
  113. package/renderer/components/docs-viewer/search/search-dialog.js +367 -0
  114. package/renderer/components/docs-viewer/search/use-search.js +89 -0
  115. package/renderer/components/docs-viewer/shared/markdown-renderer.js +423 -0
  116. package/renderer/components/docs-viewer/shared/method-badge.js +23 -0
  117. package/renderer/components/docs-viewer/shared/schema-viewer.js +321 -0
  118. package/renderer/components/docs-viewer/sidebar/collection-tree.js +222 -0
  119. package/renderer/components/docs-viewer/sidebar/endpoint-options.js +512 -0
  120. package/renderer/components/docs-viewer/sidebar/index.js +196 -0
  121. package/renderer/components/docs-viewer/sidebar/right-sidebar.js +163 -0
  122. package/renderer/components/docs-viewer/sidebar/sidebar-group.js +87 -0
  123. package/renderer/components/docs-viewer/sidebar/sidebar-item.js +172 -0
  124. package/renderer/components/docs-viewer/sidebar/sidebar-section.js +31 -0
  125. package/renderer/components/theme-provider.js +10 -0
  126. package/renderer/components/theme-toggle.js +106 -0
  127. package/renderer/components/ui/badge.js +29 -0
  128. package/renderer/components/ui/button.js +40 -0
  129. package/renderer/components/ui/dialog.js +50 -0
  130. package/renderer/components/ui/dropdown-menu.js +143 -0
  131. package/renderer/components/ui/input.js +12 -0
  132. package/renderer/components/ui/label.js +13 -0
  133. package/renderer/components/ui/navigation-menu.js +83 -0
  134. package/renderer/components/ui/select.js +116 -0
  135. package/renderer/components/ui/spinner.js +92 -0
  136. package/renderer/components/ui/tabs.js +34 -0
  137. package/renderer/components/ui/tooltip.js +43 -0
  138. package/renderer/hooks/use-code-copy.js +76 -0
  139. package/renderer/hooks/use-openapi-title.js +33 -0
  140. package/renderer/lib/api-docs/agent/index.js +4 -0
  141. package/renderer/lib/api-docs/agent/indexer.js +254 -0
  142. package/renderer/lib/api-docs/agent/spec-summary.js +227 -0
  143. package/renderer/lib/api-docs/agent/types.js +5 -0
  144. package/renderer/lib/api-docs/auth/auth-context.js +157 -0
  145. package/renderer/lib/api-docs/auth/auth-storage.js +66 -0
  146. package/renderer/lib/api-docs/auth/crypto.js +64 -0
  147. package/renderer/lib/api-docs/auth/index.js +3 -0
  148. package/renderer/lib/api-docs/code-editor/db.js +145 -0
  149. package/renderer/lib/api-docs/code-editor/hooks.js +254 -0
  150. package/renderer/lib/api-docs/code-editor/{index.ts → index.js} +3 -4
  151. package/renderer/lib/api-docs/code-editor/mode-context.js +154 -0
  152. package/renderer/lib/api-docs/code-editor/types.js +53 -0
  153. package/renderer/lib/api-docs/codegen/definitions.js +258 -0
  154. package/renderer/lib/api-docs/codegen/har.js +171 -0
  155. package/renderer/lib/api-docs/codegen/index.js +118 -0
  156. package/renderer/lib/api-docs/factories.js +136 -0
  157. package/renderer/lib/api-docs/{index.ts → index.js} +5 -10
  158. package/renderer/lib/api-docs/mobile-context.js +79 -0
  159. package/renderer/lib/api-docs/navigation-context.js +62 -0
  160. package/renderer/lib/api-docs/parsers/graphql/index.js +50 -0
  161. package/renderer/lib/api-docs/parsers/graphql/parser.js +350 -0
  162. package/renderer/lib/api-docs/parsers/graphql/transformer.js +215 -0
  163. package/renderer/lib/api-docs/parsers/graphql/types.js +46 -0
  164. package/renderer/lib/api-docs/parsers/openapi/dereferencer.js +43 -0
  165. package/renderer/lib/api-docs/parsers/openapi/extractors/auth.js +486 -0
  166. package/renderer/lib/api-docs/parsers/openapi/extractors/body.js +295 -0
  167. package/renderer/lib/api-docs/parsers/openapi/extractors/index.js +132 -0
  168. package/renderer/lib/api-docs/parsers/openapi/index.js +127 -0
  169. package/renderer/lib/api-docs/parsers/openapi/transformer.js +192 -0
  170. package/renderer/lib/api-docs/parsers/openapi/validator.js +24 -0
  171. package/renderer/lib/api-docs/playground/context.js +65 -0
  172. package/renderer/lib/api-docs/playground/navigation-context.js +74 -0
  173. package/renderer/lib/api-docs/playground/request-builder.js +163 -0
  174. package/renderer/lib/api-docs/playground/request-runner.js +224 -0
  175. package/renderer/lib/api-docs/playground/types.js +5 -0
  176. package/renderer/lib/api-docs/types.js +23 -0
  177. package/renderer/lib/api-docs/utils.js +212 -0
  178. package/renderer/lib/cache.js +157 -0
  179. package/renderer/lib/docs/config/domain-schema.js +161 -0
  180. package/renderer/lib/docs/config/index.js +5 -0
  181. package/renderer/lib/docs/config/loader.js +113 -0
  182. package/renderer/lib/docs/config/schema.js +269 -0
  183. package/renderer/lib/docs/index.js +8 -0
  184. package/renderer/lib/docs/mdx/compiler.js +128 -0
  185. package/renderer/lib/docs/mdx/frontmatter.js +73 -0
  186. package/renderer/lib/docs/mdx/index.js +8 -0
  187. package/renderer/lib/docs/navigation/generator.js +269 -0
  188. package/renderer/lib/docs/navigation/index.js +4 -0
  189. package/renderer/lib/docs/navigation/types.js +9 -0
  190. package/renderer/lib/docs-navigation-context.js +40 -0
  191. package/renderer/lib/multi-tenant/context.js +80 -0
  192. package/renderer/lib/storage/blob.js +767 -0
  193. package/renderer/lib/utils/icons.js +30 -0
  194. package/renderer/lib/utils.js +5 -0
  195. package/renderer/next.config.js +62 -0
  196. package/renderer/tsconfig.json +23 -5
  197. package/renderer/app/api/assets/[...path]/route.ts +0 -123
  198. package/renderer/app/api/assets/route.ts +0 -124
  199. package/renderer/app/api/assets/upload/route.ts +0 -177
  200. package/renderer/app/api/auth-schemes/route.ts +0 -77
  201. package/renderer/app/api/chat/route.ts +0 -858
  202. package/renderer/app/api/codegen/route.ts +0 -72
  203. package/renderer/app/api/collections/route.ts +0 -1002
  204. package/renderer/app/api/debug/route.ts +0 -53
  205. package/renderer/app/api/deploy/route.ts +0 -234
  206. package/renderer/app/api/device/route.ts +0 -42
  207. package/renderer/app/api/docs/route.ts +0 -201
  208. package/renderer/app/api/domains/add/route.ts +0 -132
  209. package/renderer/app/api/domains/lookup/route.ts +0 -43
  210. package/renderer/app/api/domains/remove/route.ts +0 -100
  211. package/renderer/app/api/domains/status/route.ts +0 -158
  212. package/renderer/app/api/domains/verify/route.ts +0 -181
  213. package/renderer/app/api/keys/regenerate/route.ts +0 -80
  214. package/renderer/app/api/local-assets/[...path]/route.ts +0 -122
  215. package/renderer/app/api/openapi-spec/route.ts +0 -151
  216. package/renderer/app/api/projects/[slug]/route.ts +0 -153
  217. package/renderer/app/api/projects/[slug]/stats/route.ts +0 -96
  218. package/renderer/app/api/projects/register/route.ts +0 -152
  219. package/renderer/app/api/proxy/route.ts +0 -149
  220. package/renderer/app/api/proxy-stream/route.ts +0 -168
  221. package/renderer/app/api/redirects/route.ts +0 -47
  222. package/renderer/app/api/schema/route.ts +0 -73
  223. package/renderer/app/api/subdomains/check/route.ts +0 -172
  224. package/renderer/app/api/suggestions/route.ts +0 -144
  225. package/renderer/app/layout.tsx +0 -54
  226. package/renderer/app/llms-full.txt/route.ts +0 -346
  227. package/renderer/app/llms.txt/route.ts +0 -279
  228. package/renderer/app/page.tsx +0 -14
  229. package/renderer/app/robots.txt/route.ts +0 -84
  230. package/renderer/app/sitemap.xml/route.ts +0 -199
  231. package/renderer/components/docs/index.ts +0 -12
  232. package/renderer/components/docs/mdx/accordion.tsx +0 -169
  233. package/renderer/components/docs/mdx/badge.tsx +0 -132
  234. package/renderer/components/docs/mdx/callouts.tsx +0 -154
  235. package/renderer/components/docs/mdx/cards.tsx +0 -241
  236. package/renderer/components/docs/mdx/changelog.tsx +0 -120
  237. package/renderer/components/docs/mdx/code-block.tsx +0 -186
  238. package/renderer/components/docs/mdx/code-group.tsx +0 -421
  239. package/renderer/components/docs/mdx/file-embeds.tsx +0 -105
  240. package/renderer/components/docs/mdx/frame.tsx +0 -112
  241. package/renderer/components/docs/mdx/highlight.tsx +0 -151
  242. package/renderer/components/docs/mdx/iframe.tsx +0 -134
  243. package/renderer/components/docs/mdx/image.tsx +0 -235
  244. package/renderer/components/docs/mdx/index.ts +0 -237
  245. package/renderer/components/docs/mdx/landing.tsx +0 -684
  246. package/renderer/components/docs/mdx/mermaid.tsx +0 -240
  247. package/renderer/components/docs/mdx/param-field.tsx +0 -200
  248. package/renderer/components/docs/mdx/steps.tsx +0 -113
  249. package/renderer/components/docs/mdx/tabs.tsx +0 -86
  250. package/renderer/components/docs/mdx-renderer.tsx +0 -100
  251. package/renderer/components/docs/navigation/breadcrumbs.tsx +0 -76
  252. package/renderer/components/docs/navigation/index.ts +0 -8
  253. package/renderer/components/docs/navigation/page-nav.tsx +0 -64
  254. package/renderer/components/docs/navigation/sidebar.tsx +0 -515
  255. package/renderer/components/docs/navigation/toc.tsx +0 -113
  256. package/renderer/components/docs/notice.tsx +0 -105
  257. package/renderer/components/docs-header.tsx +0 -278
  258. package/renderer/components/docs-viewer/agent/agent-chat.tsx +0 -2076
  259. package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +0 -90
  260. package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +0 -49
  261. package/renderer/components/docs-viewer/agent/cards/index.tsx +0 -50
  262. package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +0 -212
  263. package/renderer/components/docs-viewer/agent/cards/types.ts +0 -84
  264. package/renderer/components/docs-viewer/agent/chat-message.tsx +0 -17
  265. package/renderer/components/docs-viewer/agent/index.tsx +0 -6
  266. package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +0 -119
  267. package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +0 -46
  268. package/renderer/components/docs-viewer/agent/messages/index.ts +0 -17
  269. package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +0 -721
  270. package/renderer/components/docs-viewer/agent/messages/types.ts +0 -61
  271. package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +0 -24
  272. package/renderer/components/docs-viewer/agent/messages/user-message.tsx +0 -51
  273. package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +0 -1283
  274. package/renderer/components/docs-viewer/content/changelog-page.tsx +0 -331
  275. package/renderer/components/docs-viewer/content/doc-page.tsx +0 -367
  276. package/renderer/components/docs-viewer/content/documentation-viewer.tsx +0 -17
  277. package/renderer/components/docs-viewer/content/index.tsx +0 -29
  278. package/renderer/components/docs-viewer/content/not-found-page.tsx +0 -330
  279. package/renderer/components/docs-viewer/content/request-details.tsx +0 -330
  280. package/renderer/components/docs-viewer/content/sections/auth.tsx +0 -69
  281. package/renderer/components/docs-viewer/content/sections/body.tsx +0 -66
  282. package/renderer/components/docs-viewer/content/sections/headers.tsx +0 -43
  283. package/renderer/components/docs-viewer/content/sections/overview.tsx +0 -40
  284. package/renderer/components/docs-viewer/content/sections/parameters.tsx +0 -43
  285. package/renderer/components/docs-viewer/content/sections/responses.tsx +0 -87
  286. package/renderer/components/docs-viewer/global-auth-modal.tsx +0 -352
  287. package/renderer/components/docs-viewer/index.tsx +0 -1670
  288. package/renderer/components/docs-viewer/playground/auth-editor.tsx +0 -280
  289. package/renderer/components/docs-viewer/playground/body-editor.tsx +0 -221
  290. package/renderer/components/docs-viewer/playground/code-editor.tsx +0 -224
  291. package/renderer/components/docs-viewer/playground/code-snippet.tsx +0 -387
  292. package/renderer/components/docs-viewer/playground/graphql-playground.tsx +0 -745
  293. package/renderer/components/docs-viewer/playground/index.tsx +0 -671
  294. package/renderer/components/docs-viewer/playground/key-value-editor.tsx +0 -261
  295. package/renderer/components/docs-viewer/playground/method-selector.tsx +0 -60
  296. package/renderer/components/docs-viewer/playground/request-builder.tsx +0 -179
  297. package/renderer/components/docs-viewer/playground/request-tabs.tsx +0 -237
  298. package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +0 -21
  299. package/renderer/components/docs-viewer/playground/response-cards/index.tsx +0 -93
  300. package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +0 -16
  301. package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +0 -23
  302. package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +0 -268
  303. package/renderer/components/docs-viewer/playground/response-cards/types.ts +0 -82
  304. package/renderer/components/docs-viewer/playground/response-viewer.tsx +0 -43
  305. package/renderer/components/docs-viewer/search/index.ts +0 -2
  306. package/renderer/components/docs-viewer/search/search-dialog.tsx +0 -331
  307. package/renderer/components/docs-viewer/search/use-search.ts +0 -117
  308. package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +0 -431
  309. package/renderer/components/docs-viewer/shared/method-badge.tsx +0 -41
  310. package/renderer/components/docs-viewer/shared/schema-viewer.tsx +0 -349
  311. package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +0 -259
  312. package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +0 -316
  313. package/renderer/components/docs-viewer/sidebar/index.tsx +0 -282
  314. package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +0 -202
  315. package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +0 -118
  316. package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +0 -212
  317. package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +0 -38
  318. package/renderer/components/theme-provider.tsx +0 -11
  319. package/renderer/components/theme-toggle.tsx +0 -76
  320. package/renderer/components/ui/badge.tsx +0 -46
  321. package/renderer/components/ui/button.tsx +0 -59
  322. package/renderer/components/ui/dialog.tsx +0 -118
  323. package/renderer/components/ui/dropdown-menu.tsx +0 -257
  324. package/renderer/components/ui/input.tsx +0 -21
  325. package/renderer/components/ui/label.tsx +0 -24
  326. package/renderer/components/ui/navigation-menu.tsx +0 -168
  327. package/renderer/components/ui/select.tsx +0 -190
  328. package/renderer/components/ui/spinner.tsx +0 -114
  329. package/renderer/components/ui/tabs.tsx +0 -66
  330. package/renderer/components/ui/tooltip.tsx +0 -61
  331. package/renderer/hooks/use-code-copy.ts +0 -88
  332. package/renderer/hooks/use-openapi-title.ts +0 -44
  333. package/renderer/lib/api-docs/agent/index.ts +0 -6
  334. package/renderer/lib/api-docs/agent/indexer.ts +0 -323
  335. package/renderer/lib/api-docs/agent/spec-summary.ts +0 -335
  336. package/renderer/lib/api-docs/agent/types.ts +0 -116
  337. package/renderer/lib/api-docs/auth/auth-context.tsx +0 -225
  338. package/renderer/lib/api-docs/auth/auth-storage.ts +0 -87
  339. package/renderer/lib/api-docs/auth/crypto.ts +0 -89
  340. package/renderer/lib/api-docs/auth/index.ts +0 -4
  341. package/renderer/lib/api-docs/code-editor/db.ts +0 -164
  342. package/renderer/lib/api-docs/code-editor/hooks.ts +0 -266
  343. package/renderer/lib/api-docs/code-editor/mode-context.tsx +0 -207
  344. package/renderer/lib/api-docs/code-editor/types.ts +0 -105
  345. package/renderer/lib/api-docs/codegen/definitions.ts +0 -297
  346. package/renderer/lib/api-docs/codegen/har.ts +0 -251
  347. package/renderer/lib/api-docs/codegen/index.ts +0 -159
  348. package/renderer/lib/api-docs/factories.ts +0 -170
  349. package/renderer/lib/api-docs/mobile-context.tsx +0 -112
  350. package/renderer/lib/api-docs/navigation-context.tsx +0 -88
  351. package/renderer/lib/api-docs/parsers/graphql/README.md +0 -129
  352. package/renderer/lib/api-docs/parsers/graphql/index.ts +0 -91
  353. package/renderer/lib/api-docs/parsers/graphql/parser.ts +0 -491
  354. package/renderer/lib/api-docs/parsers/graphql/transformer.ts +0 -246
  355. package/renderer/lib/api-docs/parsers/graphql/types.ts +0 -283
  356. package/renderer/lib/api-docs/parsers/openapi/README.md +0 -32
  357. package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +0 -60
  358. package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +0 -574
  359. package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +0 -403
  360. package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +0 -232
  361. package/renderer/lib/api-docs/parsers/openapi/index.ts +0 -171
  362. package/renderer/lib/api-docs/parsers/openapi/transformer.ts +0 -278
  363. package/renderer/lib/api-docs/parsers/openapi/validator.ts +0 -31
  364. package/renderer/lib/api-docs/playground/context.tsx +0 -107
  365. package/renderer/lib/api-docs/playground/navigation-context.tsx +0 -124
  366. package/renderer/lib/api-docs/playground/request-builder.ts +0 -223
  367. package/renderer/lib/api-docs/playground/request-runner.ts +0 -282
  368. package/renderer/lib/api-docs/playground/types.ts +0 -35
  369. package/renderer/lib/api-docs/types.ts +0 -269
  370. package/renderer/lib/api-docs/utils.ts +0 -311
  371. package/renderer/lib/cache.ts +0 -193
  372. package/renderer/lib/docs/config/domain-schema.ts +0 -260
  373. package/renderer/lib/docs/config/index.ts +0 -43
  374. package/renderer/lib/docs/config/loader.ts +0 -142
  375. package/renderer/lib/docs/config/schema.ts +0 -308
  376. package/renderer/lib/docs/index.ts +0 -12
  377. package/renderer/lib/docs/mdx/compiler.ts +0 -176
  378. package/renderer/lib/docs/mdx/frontmatter.ts +0 -91
  379. package/renderer/lib/docs/mdx/index.ts +0 -26
  380. package/renderer/lib/docs/navigation/generator.ts +0 -348
  381. package/renderer/lib/docs/navigation/index.ts +0 -12
  382. package/renderer/lib/docs/navigation/types.ts +0 -123
  383. package/renderer/lib/docs-navigation-context.tsx +0 -80
  384. package/renderer/lib/multi-tenant/context.ts +0 -105
  385. package/renderer/lib/storage/blob.ts +0 -1083
  386. package/renderer/lib/utils/icons.ts +0 -48
  387. package/renderer/lib/utils.ts +0 -6
  388. package/renderer/next.config.ts +0 -76
@@ -1,151 +0,0 @@
1
- import { NextResponse } from 'next/server'
2
- import { CacheUtils } from '@/lib/cache'
3
-
4
- // Configuration - you can set these as environment variables
5
- const BRAINFISH_API_BASE_URL = process.env.BRAINFISH_API_BASE_URL || 'https://api.brainfish.ai/api'
6
- const BRAINFISH_CATALOG_ID = process.env.BRAINFISH_CATALOG_ID || 'your_catalog_id_here'
7
- const BRAINFISH_JWT_TOKEN = process.env.BRAINFISH_JWT_TOKEN || 'your_jwt_token_here'
8
-
9
- // Cache configuration
10
- const CACHE_KEY = `openapi-spec-${BRAINFISH_CATALOG_ID}`
11
- const CACHE_TTL = 60 * 5 // 5 minutes
12
- const FALLBACK_CACHE_KEY = `${CACHE_KEY}-fallback`
13
-
14
- interface OpenApiSpec {
15
- openapi: string;
16
- info: {
17
- title: string;
18
- version: string;
19
- description: string;
20
- };
21
- paths: Record<string, unknown>;
22
- servers: Array<{
23
- url: string;
24
- description: string;
25
- }>;
26
- }
27
-
28
- async function fetchOpenApiSpec(): Promise<OpenApiSpec> {
29
- // First, try to get from cache
30
- const cachedSpec = await CacheUtils.get<OpenApiSpec>(CACHE_KEY)
31
- if (cachedSpec) {
32
- return cachedSpec
33
- }
34
-
35
- // If not in cache, try to fetch from API
36
- try {
37
- // IMPORTANT: Use dots (.) not slashes (/) - e.g., catalogs.openapi-spec NOT catalogs/openapi-spec
38
- const url = `${BRAINFISH_API_BASE_URL}/catalogs.openapi-spec`
39
-
40
- console.log('[OpenAPI] Fetching from:', url)
41
- console.log('[OpenAPI] Request body:', { catalogId: BRAINFISH_CATALOG_ID })
42
-
43
- const response = await fetch(url, {
44
- method: 'POST',
45
- headers: {
46
- 'Authorization': `Bearer ${BRAINFISH_JWT_TOKEN}`,
47
- 'Accept': 'application/json',
48
- 'Content-Type': 'application/json',
49
- 'User-Agent': 'Brainfish-API-Docs/1.0',
50
- },
51
- body: JSON.stringify({
52
- catalogId: BRAINFISH_CATALOG_ID
53
- }),
54
- // Ensure fetch doesn't cache
55
- cache: 'no-store',
56
- })
57
-
58
- console.log('[OpenAPI] Response status:', response.status, response.statusText)
59
- console.log('[OpenAPI] Response headers:', Object.fromEntries(response.headers.entries()))
60
-
61
- if (!response.ok) {
62
- // Try to get error details from response body
63
- let errorBody = ''
64
- try {
65
- errorBody = await response.text()
66
- console.log('[OpenAPI] Error response body:', errorBody)
67
- } catch {
68
- console.log('[OpenAPI] Could not read error response body')
69
- }
70
-
71
- throw new Error(`Failed to fetch OpenAPI spec: ${response.status} ${response.statusText}${errorBody ? ` - ${errorBody}` : ''}`)
72
- }
73
-
74
- const spec = await response.json() as OpenApiSpec
75
-
76
- // Cache the successful response
77
- await CacheUtils.set(CACHE_KEY, spec, CACHE_TTL)
78
- // Also store as fallback (with longer TTL)
79
- await CacheUtils.set(FALLBACK_CACHE_KEY, spec, 60 * 60 * 24 * 365)
80
- return spec
81
- } catch (error) {
82
- const errorMessage = error instanceof Error ? error.message : 'Unknown error'
83
- console.error('[OpenAPI] Error fetching spec:', errorMessage)
84
-
85
- // Try to get fallback from cache
86
- const fallbackSpec = await CacheUtils.get<OpenApiSpec>(FALLBACK_CACHE_KEY)
87
- if (fallbackSpec) {
88
- console.warn('[OpenAPI] Using fallback cache due to API error')
89
- return fallbackSpec
90
- }
91
-
92
- // If no fallback available, return basic spec
93
- console.warn('[OpenAPI] No fallback available, using basic spec')
94
- return {
95
- openapi: '3.0.0',
96
- info: {
97
- title: 'Brainfish API',
98
- version: '1.0.0',
99
- description: 'API temporarily unavailable. Please try again later.',
100
- },
101
- paths: {},
102
- servers: [
103
- {
104
- url: BRAINFISH_API_BASE_URL.replace('/api', ''),
105
- description: 'Brainfish API'
106
- }
107
- ]
108
- }
109
- }
110
- }
111
-
112
- export async function GET() {
113
- try {
114
- const spec = await fetchOpenApiSpec()
115
-
116
- return NextResponse.json(spec, {
117
- headers: {
118
- 'Content-Type': 'application/json',
119
- 'Cache-Control': 'no-cache, no-store, must-revalidate', // Disable browser caching
120
- },
121
- })
122
- } catch (error) {
123
- console.error('[OpenAPI] Unexpected error in endpoint:', error)
124
-
125
- // As a last resort, try the fallback cache
126
- const fallbackSpec = await CacheUtils.get<OpenApiSpec>(FALLBACK_CACHE_KEY)
127
- if (fallbackSpec) {
128
- console.warn('[OpenAPI] Using fallback cache in error handler')
129
- return NextResponse.json(fallbackSpec, {
130
- headers: {
131
- 'Content-Type': 'application/json',
132
- 'Cache-Control': 'no-cache, no-store, must-revalidate',
133
- },
134
- })
135
- }
136
-
137
- return NextResponse.json(
138
- {
139
- error: 'OpenAPI specification temporarily unavailable',
140
- openapi: '3.0.0',
141
- info: {
142
- title: 'Brainfish API',
143
- version: '1.0.0',
144
- description: 'Service temporarily unavailable. Please try again later.',
145
- },
146
- paths: {},
147
- },
148
- { status: 503 } // Service Unavailable
149
- )
150
- }
151
- }
@@ -1,153 +0,0 @@
1
- import { NextRequest, NextResponse } from 'next/server'
2
- import {
3
- validateApiKey,
4
- getProjectContent,
5
- getProjectApiKey,
6
- deleteProject,
7
- projectExists
8
- } from '@/lib/storage/blob'
9
-
10
- interface RouteParams {
11
- params: Promise<{ slug: string }>
12
- }
13
-
14
- /**
15
- * GET /api/projects/[slug]
16
- * Get detailed project information
17
- *
18
- * Headers:
19
- * Authorization: Bearer <api_key>
20
- */
21
- export async function GET(request: NextRequest, { params }: RouteParams) {
22
- try {
23
- const { slug } = await params
24
-
25
- // Validate API key
26
- const authHeader = request.headers.get('Authorization')
27
- const apiKey = authHeader?.replace('Bearer ', '')
28
-
29
- if (!apiKey) {
30
- return NextResponse.json(
31
- { error: 'API key required' },
32
- { status: 401 }
33
- )
34
- }
35
-
36
- // Validate the API key belongs to this project
37
- const validatedSlug = await validateApiKey(apiKey)
38
-
39
- if (validatedSlug !== slug) {
40
- return NextResponse.json(
41
- { error: 'API key does not match this project' },
42
- { status: 403 }
43
- )
44
- }
45
-
46
- // Get project details
47
- const content = await getProjectContent(slug)
48
- const keyData = await getProjectApiKey(slug)
49
-
50
- if (!content) {
51
- return NextResponse.json(
52
- { error: 'Project not found' },
53
- { status: 404 }
54
- )
55
- }
56
-
57
- // Parse docs.json
58
- let docsConfig: Record<string, unknown> = {}
59
- try {
60
- docsConfig = JSON.parse(content.docsJson)
61
- } catch {
62
- // Ignore parse errors
63
- }
64
-
65
- // Build file list (without content for security/size)
66
- const files = content.files.map(f => ({
67
- path: f.path,
68
- size: f.content.length,
69
- }))
70
-
71
- return NextResponse.json({
72
- slug: content.slug,
73
- name: content.name,
74
- url: `https://${slug}.devdoc.sh`,
75
- createdAt: content.createdAt,
76
- updatedAt: content.updatedAt,
77
- lastDeployedAt: keyData?.lastUsedAt || content.updatedAt,
78
- filesCount: content.files.length,
79
- totalSize: content.files.reduce((sum, f) => sum + f.content.length, 0),
80
- files,
81
- config: {
82
- name: docsConfig.name,
83
- favicon: docsConfig.favicon,
84
- navigation: docsConfig.navigation ? 'configured' : 'default',
85
- },
86
- })
87
-
88
- } catch (error) {
89
- console.error('[Projects API] GET Error:', error)
90
- return NextResponse.json(
91
- { error: 'Internal server error' },
92
- { status: 500 }
93
- )
94
- }
95
- }
96
-
97
- /**
98
- * DELETE /api/projects/[slug]
99
- * Delete a project and all its content
100
- *
101
- * Headers:
102
- * Authorization: Bearer <api_key>
103
- */
104
- export async function DELETE(request: NextRequest, { params }: RouteParams) {
105
- try {
106
- const { slug } = await params
107
-
108
- // Validate API key
109
- const authHeader = request.headers.get('Authorization')
110
- const apiKey = authHeader?.replace('Bearer ', '')
111
-
112
- if (!apiKey) {
113
- return NextResponse.json(
114
- { error: 'API key required' },
115
- { status: 401 }
116
- )
117
- }
118
-
119
- // Validate the API key belongs to this project
120
- const validatedSlug = await validateApiKey(apiKey)
121
-
122
- if (validatedSlug !== slug) {
123
- return NextResponse.json(
124
- { error: 'API key does not match this project' },
125
- { status: 403 }
126
- )
127
- }
128
-
129
- // Check project exists
130
- const exists = await projectExists(slug)
131
- if (!exists) {
132
- return NextResponse.json(
133
- { error: 'Project not found' },
134
- { status: 404 }
135
- )
136
- }
137
-
138
- // Delete the project
139
- await deleteProject(slug)
140
-
141
- return NextResponse.json({
142
- success: true,
143
- message: `Project ${slug} has been deleted`,
144
- })
145
-
146
- } catch (error) {
147
- console.error('[Projects API] DELETE Error:', error)
148
- return NextResponse.json(
149
- { error: 'Internal server error' },
150
- { status: 500 }
151
- )
152
- }
153
- }
@@ -1,96 +0,0 @@
1
- import { NextRequest, NextResponse } from 'next/server'
2
- import {
3
- validateApiKey,
4
- getProjectContent,
5
- getProjectApiKey,
6
- } from '@/lib/storage/blob'
7
-
8
- interface RouteParams {
9
- params: Promise<{ slug: string }>
10
- }
11
-
12
- /**
13
- * GET /api/projects/[slug]/stats
14
- * Get project statistics
15
- *
16
- * Headers:
17
- * Authorization: Bearer <api_key>
18
- */
19
- export async function GET(request: NextRequest, { params }: RouteParams) {
20
- try {
21
- const { slug } = await params
22
-
23
- // Validate API key
24
- const authHeader = request.headers.get('Authorization')
25
- const apiKey = authHeader?.replace('Bearer ', '')
26
-
27
- if (!apiKey) {
28
- return NextResponse.json(
29
- { error: 'API key required' },
30
- { status: 401 }
31
- )
32
- }
33
-
34
- // Validate the API key belongs to this project
35
- const validatedSlug = await validateApiKey(apiKey)
36
-
37
- if (validatedSlug !== slug) {
38
- return NextResponse.json(
39
- { error: 'API key does not match this project' },
40
- { status: 403 }
41
- )
42
- }
43
-
44
- // Get project details
45
- const content = await getProjectContent(slug)
46
- const keyData = await getProjectApiKey(slug)
47
-
48
- if (!content) {
49
- return NextResponse.json(
50
- { error: 'Project not found' },
51
- { status: 404 }
52
- )
53
- }
54
-
55
- // Calculate stats
56
- const totalSize = content.files.reduce((sum, f) => sum + f.content.length, 0)
57
- const mdxFiles = content.files.filter(f => f.path.endsWith('.mdx') || f.path.endsWith('.md'))
58
- const jsonFiles = content.files.filter(f => f.path.endsWith('.json'))
59
- const otherFiles = content.files.filter(f => !f.path.endsWith('.mdx') && !f.path.endsWith('.md') && !f.path.endsWith('.json'))
60
-
61
- // Calculate days since creation
62
- const createdDate = new Date(content.createdAt)
63
- const now = new Date()
64
- const daysSinceCreation = Math.floor((now.getTime() - createdDate.getTime()) / (1000 * 60 * 60 * 24))
65
-
66
- return NextResponse.json({
67
- slug,
68
- stats: {
69
- totalFiles: content.files.length,
70
- mdxPages: mdxFiles.length,
71
- configFiles: jsonFiles.length,
72
- otherFiles: otherFiles.length,
73
- totalSizeBytes: totalSize,
74
- totalSizeKB: Math.round(totalSize / 1024 * 10) / 10,
75
- },
76
- deployment: {
77
- createdAt: content.createdAt,
78
- updatedAt: content.updatedAt,
79
- lastDeployedAt: keyData?.lastUsedAt || content.updatedAt,
80
- daysSinceCreation,
81
- },
82
- // Placeholder for future analytics
83
- analytics: {
84
- pageViews: 'coming soon',
85
- uniqueVisitors: 'coming soon',
86
- },
87
- })
88
-
89
- } catch (error) {
90
- console.error('[Projects Stats API] Error:', error)
91
- return NextResponse.json(
92
- { error: 'Internal server error' },
93
- { status: 500 }
94
- )
95
- }
96
- }
@@ -1,152 +0,0 @@
1
- import { NextRequest, NextResponse } from 'next/server'
2
- import {
3
- isSubdomainRegistered,
4
- registerSubdomain,
5
- generateApiKey,
6
- storeProjectApiKey,
7
- } from '@/lib/storage/blob'
8
-
9
- // Reserved/blacklisted subdomains (same as in subdomains/check)
10
- const BLACKLISTED_SUBDOMAINS = new Set([
11
- 'www', 'api', 'app', 'admin', 'dashboard', 'console', 'panel', 'manage',
12
- 'login', 'signin', 'signup', 'register', 'auth', 'oauth', 'sso',
13
- 'devdoc', 'brainfish', 'docs', 'documentation', 'help', 'support', 'status', 'blog', 'news',
14
- 'mail', 'email', 'smtp', 'ftp', 'cdn', 'static', 'assets', 'images', 'files', 'media', 'download', 'downloads',
15
- 'test', 'testing', 'dev', 'development', 'staging', 'prod', 'production', 'demo', 'example', 'sandbox', 'preview',
16
- 'secure', 'ssl', 'security', 'abuse', 'spam', 'postmaster', 'hostmaster', 'webmaster',
17
- 'null', 'undefined', 'true', 'false', 'root', 'system', 'localhost',
18
- ])
19
-
20
- interface RegisterRequest {
21
- name: string
22
- slug: string
23
- subdomain: string
24
- }
25
-
26
- interface RegisterResponse {
27
- success: boolean
28
- projectId: string
29
- slug: string
30
- subdomain: string
31
- apiKey: string
32
- url: string
33
- }
34
-
35
- /**
36
- * POST /api/projects/register
37
- * Register a new project and generate an API key
38
- *
39
- * Body:
40
- * name: string - Project display name
41
- * slug: string - Project slug (URL-safe identifier)
42
- * subdomain: string - Desired subdomain for <subdomain>.devdoc.sh
43
- *
44
- * Returns:
45
- * success: boolean
46
- * projectId: string
47
- * slug: string
48
- * subdomain: string
49
- * apiKey: string
50
- * url: string
51
- */
52
- export async function POST(request: NextRequest) {
53
- try {
54
- const body = await request.json() as RegisterRequest
55
- const { name, slug, subdomain } = body
56
-
57
- // Validate required fields
58
- if (!name || !slug || !subdomain) {
59
- return NextResponse.json(
60
- { error: 'name, slug, and subdomain are required' },
61
- { status: 400 }
62
- )
63
- }
64
-
65
- // Normalize subdomain
66
- const normalizedSubdomain = subdomain.toLowerCase().trim()
67
-
68
- // Validate subdomain format
69
- if (normalizedSubdomain.length < 3) {
70
- return NextResponse.json(
71
- { error: 'Subdomain must be at least 3 characters' },
72
- { status: 400 }
73
- )
74
- }
75
-
76
- if (normalizedSubdomain.length > 63) {
77
- return NextResponse.json(
78
- { error: 'Subdomain must be 63 characters or less' },
79
- { status: 400 }
80
- )
81
- }
82
-
83
- if (!/^[a-z0-9]([a-z0-9-]*[a-z0-9])?$/.test(normalizedSubdomain)) {
84
- return NextResponse.json(
85
- { error: 'Subdomain must start and end with alphanumeric characters' },
86
- { status: 400 }
87
- )
88
- }
89
-
90
- // Check blacklist
91
- if (BLACKLISTED_SUBDOMAINS.has(normalizedSubdomain)) {
92
- return NextResponse.json(
93
- { error: `"${normalizedSubdomain}" is a reserved subdomain` },
94
- { status: 400 }
95
- )
96
- }
97
-
98
- // Use subdomain as the project identifier (slug)
99
- // This ensures subdomain uniqueness = project uniqueness
100
- const projectSlug = normalizedSubdomain
101
-
102
- // Check if subdomain already registered (O(1) lookup from registry)
103
- const exists = await isSubdomainRegistered(projectSlug)
104
-
105
- if (exists) {
106
- return NextResponse.json(
107
- { error: `Project with subdomain "${normalizedSubdomain}" already exists` },
108
- { status: 409 }
109
- )
110
- }
111
-
112
- // Generate API key and project ID
113
- const apiKey = generateApiKey()
114
- const projectId = `proj_${projectSlug}_${Math.random().toString(36).substring(2, 8)}`
115
-
116
- // Register in domain registry (O(1) lookup for future checks)
117
- await registerSubdomain(projectSlug, projectId, name, apiKey)
118
-
119
- // Also store API key in project folder (for backwards compatibility)
120
- await storeProjectApiKey(projectSlug, apiKey)
121
-
122
- const response: RegisterResponse = {
123
- success: true,
124
- projectId,
125
- slug: projectSlug,
126
- subdomain: normalizedSubdomain,
127
- apiKey,
128
- url: `https://${normalizedSubdomain}.devdoc.sh`,
129
- }
130
-
131
- console.log(`[Projects API] Registered new project: ${projectSlug}`)
132
-
133
- return NextResponse.json(response, { status: 201 })
134
-
135
- } catch (error) {
136
- console.error('[Projects API] Register Error:', error)
137
- const message = error instanceof Error ? error.message : String(error)
138
-
139
- // Provide clearer error messages for common issues
140
- let userMessage = 'Failed to register project'
141
- if (message.includes('blob already exists')) {
142
- userMessage = 'Project storage conflict. Please try again or use a different subdomain.'
143
- } else if (message.includes('network') || message.includes('fetch')) {
144
- userMessage = 'Network error connecting to storage service'
145
- }
146
-
147
- return NextResponse.json(
148
- { error: userMessage, details: message },
149
- { status: 500 }
150
- )
151
- }
152
- }
@@ -1,149 +0,0 @@
1
- /**
2
- * API Proxy Route
3
- *
4
- * Forwards requests to external APIs to bypass CORS restrictions
5
- * and ensure auth headers are sent properly.
6
- */
7
-
8
- import { NextRequest, NextResponse } from 'next/server'
9
-
10
- // Headers that should not be forwarded
11
- const EXCLUDED_REQUEST_HEADERS = [
12
- 'host',
13
- 'connection',
14
- 'content-length',
15
- 'transfer-encoding',
16
- 'keep-alive',
17
- 'upgrade',
18
- 'proxy-connection',
19
- 'proxy-authorization',
20
- ]
21
-
22
- const EXCLUDED_RESPONSE_HEADERS = [
23
- 'transfer-encoding',
24
- 'connection',
25
- 'keep-alive',
26
- 'content-encoding',
27
- ]
28
-
29
- export async function POST(request: NextRequest) {
30
- try {
31
- const body = await request.json()
32
-
33
- const { url, method, headers, requestBody } = body as {
34
- url: string
35
- method: string
36
- headers: Record<string, string>
37
- requestBody?: string | null
38
- }
39
-
40
- if (!url) {
41
- return NextResponse.json(
42
- { error: 'URL is required' },
43
- { status: 400 }
44
- )
45
- }
46
-
47
- // Validate URL to prevent SSRF attacks
48
- const parsedUrl = new URL(url)
49
- const allowedProtocols = ['http:', 'https:']
50
- if (!allowedProtocols.includes(parsedUrl.protocol)) {
51
- return NextResponse.json(
52
- { error: 'Invalid URL protocol' },
53
- { status: 400 }
54
- )
55
- }
56
-
57
- // Build headers for the proxied request
58
- const proxyHeaders: Record<string, string> = {}
59
- for (const [key, value] of Object.entries(headers || {})) {
60
- if (!EXCLUDED_REQUEST_HEADERS.includes(key.toLowerCase())) {
61
- proxyHeaders[key] = value
62
- }
63
- }
64
-
65
- // Make the proxied request
66
- const startTime = Date.now()
67
- const response = await fetch(url, {
68
- method: method || 'GET',
69
- headers: proxyHeaders,
70
- body: requestBody || undefined,
71
- })
72
- const responseTime = Date.now() - startTime
73
-
74
- // Read response body
75
- const contentType = response.headers.get('content-type') || ''
76
- let responseBody: string | null = null
77
-
78
- if (contentType.includes('application/json') || contentType.includes('text/')) {
79
- responseBody = await response.text()
80
- } else {
81
- // For binary content, convert to base64
82
- const buffer = await response.arrayBuffer()
83
- responseBody = Buffer.from(buffer).toString('base64')
84
- }
85
-
86
- // Build response headers
87
- const responseHeaders: Record<string, string> = {}
88
- response.headers.forEach((value, key) => {
89
- if (!EXCLUDED_RESPONSE_HEADERS.includes(key.toLowerCase())) {
90
- responseHeaders[key] = value
91
- }
92
- })
93
-
94
- return NextResponse.json({
95
- status: response.status,
96
- statusText: response.statusText,
97
- headers: responseHeaders,
98
- body: responseBody,
99
- responseTime,
100
- size: responseBody ? new Blob([responseBody]).size : 0,
101
- })
102
- } catch (error) {
103
- return NextResponse.json({
104
- status: 0,
105
- statusText: 'Proxy Error',
106
- headers: {},
107
- body: null,
108
- responseTime: 0,
109
- size: 0,
110
- error: error instanceof Error ? error.message : 'Proxy request failed',
111
- })
112
- }
113
- }
114
-
115
- // Also support GET for simple requests
116
- export async function GET(request: NextRequest) {
117
- const url = request.nextUrl.searchParams.get('url')
118
-
119
- if (!url) {
120
- return NextResponse.json(
121
- { error: 'URL query parameter is required' },
122
- { status: 400 }
123
- )
124
- }
125
-
126
- try {
127
- const parsedUrl = new URL(url)
128
- const allowedProtocols = ['http:', 'https:']
129
- if (!allowedProtocols.includes(parsedUrl.protocol)) {
130
- return NextResponse.json(
131
- { error: 'Invalid URL protocol' },
132
- { status: 400 }
133
- )
134
- }
135
-
136
- const response = await fetch(url)
137
- const body = await response.text()
138
-
139
- return NextResponse.json({
140
- status: response.status,
141
- statusText: response.statusText,
142
- body,
143
- })
144
- } catch (error) {
145
- return NextResponse.json({
146
- error: error instanceof Error ? error.message : 'Request failed',
147
- }, { status: 500 })
148
- }
149
- }