@brainfish-ai/devdoc 0.1.41 → 0.1.43

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (400) hide show
  1. package/ai-agents/.claude/skills/bootstrap-docs/SKILL.md +710 -79
  2. package/ai-agents/.claude/skills/check-docs/SKILL.md +83 -8
  3. package/ai-agents/.claude/skills/create-doc/SKILL.md +267 -55
  4. package/ai-agents/.claude/skills/update-doc/SKILL.md +162 -63
  5. package/ai-agents/.cursor/rules/devdoc-bootstrap.mdc +145 -15
  6. package/ai-agents/.cursor/rules/devdoc-create.mdc +108 -57
  7. package/ai-agents/.cursor/rules/devdoc-update.mdc +93 -70
  8. package/ai-agents/.cursor/rules/devdoc.mdc +21 -0
  9. package/ai-agents/schemas/docs.schema.json +332 -0
  10. package/ai-agents/schemas/theme.schema.json +243 -0
  11. package/dist/cli/commands/create.js +4 -9
  12. package/dist/cli/commands/deploy.js +50 -25
  13. package/dist/cli/commands/dev.js +19 -10
  14. package/package.json +3 -2
  15. package/renderer/app/api/assets/[...path]/route.js +108 -0
  16. package/renderer/app/api/assets/route.js +114 -0
  17. package/renderer/app/api/assets/upload/route.js +163 -0
  18. package/renderer/app/api/auth-schemes/route.js +58 -0
  19. package/renderer/app/api/chat/route.js +759 -0
  20. package/renderer/app/api/codegen/route.js +52 -0
  21. package/renderer/app/api/collections/route.js +675 -0
  22. package/renderer/app/api/debug/route.js +47 -0
  23. package/renderer/app/api/deploy/route.js +199 -0
  24. package/renderer/app/api/device/route.js +36 -0
  25. package/renderer/app/api/docs/route.js +205 -0
  26. package/renderer/app/api/domains/add/route.js +121 -0
  27. package/renderer/app/api/domains/lookup/route.js +43 -0
  28. package/renderer/app/api/domains/remove/route.js +89 -0
  29. package/renderer/app/api/domains/status/route.js +140 -0
  30. package/renderer/app/api/domains/verify/route.js +168 -0
  31. package/renderer/app/api/keys/regenerate/route.js +71 -0
  32. package/renderer/app/api/local-assets/[...path]/route.js +108 -0
  33. package/renderer/app/api/openapi-spec/route.js +73 -0
  34. package/renderer/app/api/projects/[slug]/route.js +129 -0
  35. package/renderer/app/api/projects/[slug]/stats/route.js +80 -0
  36. package/renderer/app/api/projects/register/route.js +176 -0
  37. package/renderer/app/api/proxy/route.js +139 -0
  38. package/renderer/app/api/proxy-stream/route.js +156 -0
  39. package/renderer/app/api/redirects/route.js +35 -0
  40. package/renderer/app/api/schema/route.js +85 -0
  41. package/renderer/app/api/subdomains/check/route.js +158 -0
  42. package/renderer/app/api/suggestions/route.js +175 -0
  43. package/renderer/app/layout.js +47 -0
  44. package/renderer/app/llms-full.txt/route.js +257 -0
  45. package/renderer/app/llms.txt/route.js +219 -0
  46. package/renderer/app/page.js +12 -0
  47. package/renderer/app/robots.txt/route.js +66 -0
  48. package/renderer/app/sitemap.xml/route.js +145 -0
  49. package/renderer/components/docs/index.js +8 -0
  50. package/renderer/components/docs/mdx/accordion.js +113 -0
  51. package/renderer/components/docs/mdx/badge.js +72 -0
  52. package/renderer/components/docs/mdx/callouts.js +137 -0
  53. package/renderer/components/docs/mdx/cards.js +175 -0
  54. package/renderer/components/docs/mdx/changelog.js +100 -0
  55. package/renderer/components/docs/mdx/code-block.js +147 -0
  56. package/renderer/components/docs/mdx/code-group.js +287 -0
  57. package/renderer/components/docs/mdx/file-embeds.js +82 -0
  58. package/renderer/components/docs/mdx/frame.js +59 -0
  59. package/renderer/components/docs/mdx/highlight.js +90 -0
  60. package/renderer/components/docs/mdx/iframe.js +69 -0
  61. package/renderer/components/docs/mdx/image.js +135 -0
  62. package/renderer/components/docs/mdx/index.js +134 -0
  63. package/renderer/components/docs/mdx/landing.js +315 -0
  64. package/renderer/components/docs/mdx/mermaid.js +212 -0
  65. package/renderer/components/docs/mdx/param-field.js +112 -0
  66. package/renderer/components/docs/mdx/steps.js +74 -0
  67. package/renderer/components/docs/mdx/tabs.js +50 -0
  68. package/renderer/components/docs/mdx-renderer.js +77 -0
  69. package/renderer/components/docs/navigation/breadcrumbs.js +64 -0
  70. package/renderer/components/docs/navigation/index.js +6 -0
  71. package/renderer/components/docs/navigation/page-nav.js +57 -0
  72. package/renderer/components/docs/navigation/sidebar.js +375 -0
  73. package/renderer/components/docs/navigation/toc.js +89 -0
  74. package/renderer/components/docs/notice.js +77 -0
  75. package/renderer/components/docs-header.js +202 -0
  76. package/renderer/components/docs-viewer/agent/agent-chat.js +1930 -0
  77. package/renderer/components/docs-viewer/agent/cards/debug-context-card.js +107 -0
  78. package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.js +57 -0
  79. package/renderer/components/docs-viewer/agent/cards/index.js +45 -0
  80. package/renderer/components/docs-viewer/agent/cards/response-options-card.js +154 -0
  81. package/renderer/components/docs-viewer/agent/cards/types.js +22 -0
  82. package/renderer/components/docs-viewer/agent/chat-message.js +2 -0
  83. package/renderer/components/docs-viewer/agent/index.js +4 -0
  84. package/renderer/components/docs-viewer/agent/messages/assistant-message.js +108 -0
  85. package/renderer/components/docs-viewer/agent/messages/chat-message.js +34 -0
  86. package/renderer/components/docs-viewer/agent/messages/index.js +6 -0
  87. package/renderer/components/docs-viewer/agent/messages/tool-call-display.js +1065 -0
  88. package/renderer/components/docs-viewer/agent/messages/types.js +2 -0
  89. package/renderer/components/docs-viewer/agent/messages/typing-indicator.js +26 -0
  90. package/renderer/components/docs-viewer/agent/messages/user-message.js +37 -0
  91. package/renderer/components/docs-viewer/code-editor/{index.tsx → index.js} +1 -1
  92. package/renderer/components/docs-viewer/code-editor/notes-mode.js +1338 -0
  93. package/renderer/components/docs-viewer/content/changelog-page.js +297 -0
  94. package/renderer/components/docs-viewer/content/doc-page.js +264 -0
  95. package/renderer/components/docs-viewer/content/documentation-viewer.js +14 -0
  96. package/renderer/components/docs-viewer/content/index.js +29 -0
  97. package/renderer/components/docs-viewer/content/not-found-page.js +300 -0
  98. package/renderer/components/docs-viewer/content/request-details.js +528 -0
  99. package/renderer/components/docs-viewer/content/sections/auth.js +108 -0
  100. package/renderer/components/docs-viewer/content/sections/body.js +80 -0
  101. package/renderer/components/docs-viewer/content/sections/headers.js +64 -0
  102. package/renderer/components/docs-viewer/content/sections/overview.js +56 -0
  103. package/renderer/components/docs-viewer/content/sections/parameters.js +64 -0
  104. package/renderer/components/docs-viewer/content/sections/responses.js +91 -0
  105. package/renderer/components/docs-viewer/global-auth-modal.js +427 -0
  106. package/renderer/components/docs-viewer/index.js +1552 -0
  107. package/renderer/components/docs-viewer/playground/auth-editor.js +418 -0
  108. package/renderer/components/docs-viewer/playground/body-editor.js +240 -0
  109. package/renderer/components/docs-viewer/playground/code-editor.js +135 -0
  110. package/renderer/components/docs-viewer/playground/code-snippet.js +393 -0
  111. package/renderer/components/docs-viewer/playground/graphql-playground.js +734 -0
  112. package/renderer/components/docs-viewer/playground/index.js +682 -0
  113. package/renderer/components/docs-viewer/playground/key-value-editor.js +317 -0
  114. package/renderer/components/docs-viewer/playground/method-selector.js +65 -0
  115. package/renderer/components/docs-viewer/playground/request-builder.js +181 -0
  116. package/renderer/components/docs-viewer/playground/request-tabs.js +240 -0
  117. package/renderer/components/docs-viewer/playground/response-cards/idle-card.js +42 -0
  118. package/renderer/components/docs-viewer/playground/response-cards/index.js +72 -0
  119. package/renderer/components/docs-viewer/playground/response-cards/loading-card.js +24 -0
  120. package/renderer/components/docs-viewer/playground/response-cards/network-error-card.js +28 -0
  121. package/renderer/components/docs-viewer/playground/response-cards/response-body-card.js +308 -0
  122. package/renderer/components/docs-viewer/playground/response-cards/types.js +9 -0
  123. package/renderer/components/docs-viewer/playground/response-viewer.js +18 -0
  124. package/renderer/components/docs-viewer/search/index.js +2 -0
  125. package/renderer/components/docs-viewer/search/search-dialog.js +367 -0
  126. package/renderer/components/docs-viewer/search/use-search.js +89 -0
  127. package/renderer/components/docs-viewer/shared/markdown-renderer.js +423 -0
  128. package/renderer/components/docs-viewer/shared/method-badge.js +23 -0
  129. package/renderer/components/docs-viewer/shared/schema-viewer.js +321 -0
  130. package/renderer/components/docs-viewer/sidebar/collection-tree.js +222 -0
  131. package/renderer/components/docs-viewer/sidebar/endpoint-options.js +512 -0
  132. package/renderer/components/docs-viewer/sidebar/index.js +196 -0
  133. package/renderer/components/docs-viewer/sidebar/right-sidebar.js +163 -0
  134. package/renderer/components/docs-viewer/sidebar/sidebar-group.js +87 -0
  135. package/renderer/components/docs-viewer/sidebar/sidebar-item.js +172 -0
  136. package/renderer/components/docs-viewer/sidebar/sidebar-section.js +31 -0
  137. package/renderer/components/theme-provider.js +10 -0
  138. package/renderer/components/theme-toggle.js +106 -0
  139. package/renderer/components/ui/badge.js +29 -0
  140. package/renderer/components/ui/button.js +40 -0
  141. package/renderer/components/ui/dialog.js +50 -0
  142. package/renderer/components/ui/dropdown-menu.js +143 -0
  143. package/renderer/components/ui/input.js +12 -0
  144. package/renderer/components/ui/label.js +13 -0
  145. package/renderer/components/ui/navigation-menu.js +83 -0
  146. package/renderer/components/ui/select.js +116 -0
  147. package/renderer/components/ui/spinner.js +92 -0
  148. package/renderer/components/ui/tabs.js +34 -0
  149. package/renderer/components/ui/tooltip.js +43 -0
  150. package/renderer/hooks/use-code-copy.js +76 -0
  151. package/renderer/hooks/use-openapi-title.js +33 -0
  152. package/renderer/lib/api-docs/agent/index.js +4 -0
  153. package/renderer/lib/api-docs/agent/indexer.js +254 -0
  154. package/renderer/lib/api-docs/agent/spec-summary.js +227 -0
  155. package/renderer/lib/api-docs/agent/types.js +5 -0
  156. package/renderer/lib/api-docs/auth/auth-context.js +157 -0
  157. package/renderer/lib/api-docs/auth/auth-storage.js +66 -0
  158. package/renderer/lib/api-docs/auth/crypto.js +64 -0
  159. package/renderer/lib/api-docs/auth/index.js +3 -0
  160. package/renderer/lib/api-docs/code-editor/db.js +145 -0
  161. package/renderer/lib/api-docs/code-editor/hooks.js +254 -0
  162. package/renderer/lib/api-docs/code-editor/{index.ts → index.js} +3 -4
  163. package/renderer/lib/api-docs/code-editor/mode-context.js +154 -0
  164. package/renderer/lib/api-docs/code-editor/types.js +53 -0
  165. package/renderer/lib/api-docs/codegen/definitions.js +258 -0
  166. package/renderer/lib/api-docs/codegen/har.js +171 -0
  167. package/renderer/lib/api-docs/codegen/index.js +118 -0
  168. package/renderer/lib/api-docs/factories.js +136 -0
  169. package/renderer/lib/api-docs/{index.ts → index.js} +5 -10
  170. package/renderer/lib/api-docs/mobile-context.js +79 -0
  171. package/renderer/lib/api-docs/navigation-context.js +62 -0
  172. package/renderer/lib/api-docs/parsers/graphql/index.js +50 -0
  173. package/renderer/lib/api-docs/parsers/graphql/parser.js +350 -0
  174. package/renderer/lib/api-docs/parsers/graphql/transformer.js +215 -0
  175. package/renderer/lib/api-docs/parsers/graphql/types.js +46 -0
  176. package/renderer/lib/api-docs/parsers/openapi/dereferencer.js +43 -0
  177. package/renderer/lib/api-docs/parsers/openapi/extractors/auth.js +486 -0
  178. package/renderer/lib/api-docs/parsers/openapi/extractors/body.js +295 -0
  179. package/renderer/lib/api-docs/parsers/openapi/extractors/index.js +132 -0
  180. package/renderer/lib/api-docs/parsers/openapi/index.js +127 -0
  181. package/renderer/lib/api-docs/parsers/openapi/transformer.js +192 -0
  182. package/renderer/lib/api-docs/parsers/openapi/validator.js +24 -0
  183. package/renderer/lib/api-docs/playground/context.js +65 -0
  184. package/renderer/lib/api-docs/playground/navigation-context.js +74 -0
  185. package/renderer/lib/api-docs/playground/request-builder.js +163 -0
  186. package/renderer/lib/api-docs/playground/request-runner.js +224 -0
  187. package/renderer/lib/api-docs/playground/types.js +5 -0
  188. package/renderer/lib/api-docs/types.js +23 -0
  189. package/renderer/lib/api-docs/utils.js +212 -0
  190. package/renderer/lib/cache.js +157 -0
  191. package/renderer/lib/docs/config/domain-schema.js +161 -0
  192. package/renderer/lib/docs/config/index.js +5 -0
  193. package/renderer/lib/docs/config/loader.js +113 -0
  194. package/renderer/lib/docs/config/schema.js +269 -0
  195. package/renderer/lib/docs/index.js +8 -0
  196. package/renderer/lib/docs/mdx/compiler.js +128 -0
  197. package/renderer/lib/docs/mdx/frontmatter.js +73 -0
  198. package/renderer/lib/docs/mdx/index.js +8 -0
  199. package/renderer/lib/docs/navigation/generator.js +269 -0
  200. package/renderer/lib/docs/navigation/index.js +4 -0
  201. package/renderer/lib/docs/navigation/types.js +9 -0
  202. package/renderer/lib/docs-navigation-context.js +40 -0
  203. package/renderer/lib/multi-tenant/context.js +80 -0
  204. package/renderer/lib/storage/blob.js +767 -0
  205. package/renderer/lib/utils/icons.js +30 -0
  206. package/renderer/lib/utils.js +5 -0
  207. package/renderer/next.config.js +62 -0
  208. package/renderer/tsconfig.json +23 -5
  209. package/renderer/app/api/assets/[...path]/route.ts +0 -123
  210. package/renderer/app/api/assets/route.ts +0 -124
  211. package/renderer/app/api/assets/upload/route.ts +0 -177
  212. package/renderer/app/api/auth-schemes/route.ts +0 -77
  213. package/renderer/app/api/chat/route.ts +0 -858
  214. package/renderer/app/api/codegen/route.ts +0 -72
  215. package/renderer/app/api/collections/route.ts +0 -1002
  216. package/renderer/app/api/debug/route.ts +0 -53
  217. package/renderer/app/api/deploy/route.ts +0 -234
  218. package/renderer/app/api/device/route.ts +0 -42
  219. package/renderer/app/api/docs/route.ts +0 -201
  220. package/renderer/app/api/domains/add/route.ts +0 -132
  221. package/renderer/app/api/domains/lookup/route.ts +0 -43
  222. package/renderer/app/api/domains/remove/route.ts +0 -100
  223. package/renderer/app/api/domains/status/route.ts +0 -158
  224. package/renderer/app/api/domains/verify/route.ts +0 -181
  225. package/renderer/app/api/keys/regenerate/route.ts +0 -80
  226. package/renderer/app/api/local-assets/[...path]/route.ts +0 -122
  227. package/renderer/app/api/openapi-spec/route.ts +0 -151
  228. package/renderer/app/api/projects/[slug]/route.ts +0 -153
  229. package/renderer/app/api/projects/[slug]/stats/route.ts +0 -96
  230. package/renderer/app/api/projects/register/route.ts +0 -152
  231. package/renderer/app/api/proxy/route.ts +0 -149
  232. package/renderer/app/api/proxy-stream/route.ts +0 -168
  233. package/renderer/app/api/redirects/route.ts +0 -47
  234. package/renderer/app/api/schema/route.ts +0 -73
  235. package/renderer/app/api/subdomains/check/route.ts +0 -172
  236. package/renderer/app/api/suggestions/route.ts +0 -144
  237. package/renderer/app/layout.tsx +0 -54
  238. package/renderer/app/llms-full.txt/route.ts +0 -346
  239. package/renderer/app/llms.txt/route.ts +0 -279
  240. package/renderer/app/page.tsx +0 -14
  241. package/renderer/app/robots.txt/route.ts +0 -84
  242. package/renderer/app/sitemap.xml/route.ts +0 -199
  243. package/renderer/components/docs/index.ts +0 -12
  244. package/renderer/components/docs/mdx/accordion.tsx +0 -169
  245. package/renderer/components/docs/mdx/badge.tsx +0 -132
  246. package/renderer/components/docs/mdx/callouts.tsx +0 -154
  247. package/renderer/components/docs/mdx/cards.tsx +0 -241
  248. package/renderer/components/docs/mdx/changelog.tsx +0 -120
  249. package/renderer/components/docs/mdx/code-block.tsx +0 -186
  250. package/renderer/components/docs/mdx/code-group.tsx +0 -421
  251. package/renderer/components/docs/mdx/file-embeds.tsx +0 -105
  252. package/renderer/components/docs/mdx/frame.tsx +0 -112
  253. package/renderer/components/docs/mdx/highlight.tsx +0 -151
  254. package/renderer/components/docs/mdx/iframe.tsx +0 -134
  255. package/renderer/components/docs/mdx/image.tsx +0 -235
  256. package/renderer/components/docs/mdx/index.ts +0 -237
  257. package/renderer/components/docs/mdx/landing.tsx +0 -684
  258. package/renderer/components/docs/mdx/mermaid.tsx +0 -240
  259. package/renderer/components/docs/mdx/param-field.tsx +0 -200
  260. package/renderer/components/docs/mdx/steps.tsx +0 -113
  261. package/renderer/components/docs/mdx/tabs.tsx +0 -86
  262. package/renderer/components/docs/mdx-renderer.tsx +0 -100
  263. package/renderer/components/docs/navigation/breadcrumbs.tsx +0 -76
  264. package/renderer/components/docs/navigation/index.ts +0 -8
  265. package/renderer/components/docs/navigation/page-nav.tsx +0 -64
  266. package/renderer/components/docs/navigation/sidebar.tsx +0 -515
  267. package/renderer/components/docs/navigation/toc.tsx +0 -113
  268. package/renderer/components/docs/notice.tsx +0 -105
  269. package/renderer/components/docs-header.tsx +0 -278
  270. package/renderer/components/docs-viewer/agent/agent-chat.tsx +0 -2076
  271. package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +0 -90
  272. package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +0 -49
  273. package/renderer/components/docs-viewer/agent/cards/index.tsx +0 -50
  274. package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +0 -212
  275. package/renderer/components/docs-viewer/agent/cards/types.ts +0 -84
  276. package/renderer/components/docs-viewer/agent/chat-message.tsx +0 -17
  277. package/renderer/components/docs-viewer/agent/index.tsx +0 -6
  278. package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +0 -119
  279. package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +0 -46
  280. package/renderer/components/docs-viewer/agent/messages/index.ts +0 -17
  281. package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +0 -721
  282. package/renderer/components/docs-viewer/agent/messages/types.ts +0 -61
  283. package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +0 -24
  284. package/renderer/components/docs-viewer/agent/messages/user-message.tsx +0 -51
  285. package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +0 -1283
  286. package/renderer/components/docs-viewer/content/changelog-page.tsx +0 -331
  287. package/renderer/components/docs-viewer/content/doc-page.tsx +0 -367
  288. package/renderer/components/docs-viewer/content/documentation-viewer.tsx +0 -17
  289. package/renderer/components/docs-viewer/content/index.tsx +0 -29
  290. package/renderer/components/docs-viewer/content/not-found-page.tsx +0 -330
  291. package/renderer/components/docs-viewer/content/request-details.tsx +0 -330
  292. package/renderer/components/docs-viewer/content/sections/auth.tsx +0 -69
  293. package/renderer/components/docs-viewer/content/sections/body.tsx +0 -66
  294. package/renderer/components/docs-viewer/content/sections/headers.tsx +0 -43
  295. package/renderer/components/docs-viewer/content/sections/overview.tsx +0 -40
  296. package/renderer/components/docs-viewer/content/sections/parameters.tsx +0 -43
  297. package/renderer/components/docs-viewer/content/sections/responses.tsx +0 -87
  298. package/renderer/components/docs-viewer/global-auth-modal.tsx +0 -352
  299. package/renderer/components/docs-viewer/index.tsx +0 -1662
  300. package/renderer/components/docs-viewer/playground/auth-editor.tsx +0 -280
  301. package/renderer/components/docs-viewer/playground/body-editor.tsx +0 -221
  302. package/renderer/components/docs-viewer/playground/code-editor.tsx +0 -224
  303. package/renderer/components/docs-viewer/playground/code-snippet.tsx +0 -387
  304. package/renderer/components/docs-viewer/playground/graphql-playground.tsx +0 -745
  305. package/renderer/components/docs-viewer/playground/index.tsx +0 -671
  306. package/renderer/components/docs-viewer/playground/key-value-editor.tsx +0 -261
  307. package/renderer/components/docs-viewer/playground/method-selector.tsx +0 -60
  308. package/renderer/components/docs-viewer/playground/request-builder.tsx +0 -179
  309. package/renderer/components/docs-viewer/playground/request-tabs.tsx +0 -237
  310. package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +0 -21
  311. package/renderer/components/docs-viewer/playground/response-cards/index.tsx +0 -93
  312. package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +0 -16
  313. package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +0 -23
  314. package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +0 -268
  315. package/renderer/components/docs-viewer/playground/response-cards/types.ts +0 -82
  316. package/renderer/components/docs-viewer/playground/response-viewer.tsx +0 -43
  317. package/renderer/components/docs-viewer/search/index.ts +0 -2
  318. package/renderer/components/docs-viewer/search/search-dialog.tsx +0 -331
  319. package/renderer/components/docs-viewer/search/use-search.ts +0 -117
  320. package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +0 -431
  321. package/renderer/components/docs-viewer/shared/method-badge.tsx +0 -41
  322. package/renderer/components/docs-viewer/shared/schema-viewer.tsx +0 -349
  323. package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +0 -259
  324. package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +0 -316
  325. package/renderer/components/docs-viewer/sidebar/index.tsx +0 -282
  326. package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +0 -202
  327. package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +0 -118
  328. package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +0 -212
  329. package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +0 -38
  330. package/renderer/components/theme-provider.tsx +0 -11
  331. package/renderer/components/theme-toggle.tsx +0 -76
  332. package/renderer/components/ui/badge.tsx +0 -46
  333. package/renderer/components/ui/button.tsx +0 -59
  334. package/renderer/components/ui/dialog.tsx +0 -118
  335. package/renderer/components/ui/dropdown-menu.tsx +0 -257
  336. package/renderer/components/ui/input.tsx +0 -21
  337. package/renderer/components/ui/label.tsx +0 -24
  338. package/renderer/components/ui/navigation-menu.tsx +0 -168
  339. package/renderer/components/ui/select.tsx +0 -190
  340. package/renderer/components/ui/spinner.tsx +0 -114
  341. package/renderer/components/ui/tabs.tsx +0 -66
  342. package/renderer/components/ui/tooltip.tsx +0 -61
  343. package/renderer/hooks/use-code-copy.ts +0 -88
  344. package/renderer/hooks/use-openapi-title.ts +0 -44
  345. package/renderer/lib/api-docs/agent/index.ts +0 -6
  346. package/renderer/lib/api-docs/agent/indexer.ts +0 -323
  347. package/renderer/lib/api-docs/agent/spec-summary.ts +0 -335
  348. package/renderer/lib/api-docs/agent/types.ts +0 -116
  349. package/renderer/lib/api-docs/auth/auth-context.tsx +0 -225
  350. package/renderer/lib/api-docs/auth/auth-storage.ts +0 -87
  351. package/renderer/lib/api-docs/auth/crypto.ts +0 -89
  352. package/renderer/lib/api-docs/auth/index.ts +0 -4
  353. package/renderer/lib/api-docs/code-editor/db.ts +0 -164
  354. package/renderer/lib/api-docs/code-editor/hooks.ts +0 -266
  355. package/renderer/lib/api-docs/code-editor/mode-context.tsx +0 -207
  356. package/renderer/lib/api-docs/code-editor/types.ts +0 -105
  357. package/renderer/lib/api-docs/codegen/definitions.ts +0 -297
  358. package/renderer/lib/api-docs/codegen/har.ts +0 -251
  359. package/renderer/lib/api-docs/codegen/index.ts +0 -159
  360. package/renderer/lib/api-docs/factories.ts +0 -170
  361. package/renderer/lib/api-docs/mobile-context.tsx +0 -112
  362. package/renderer/lib/api-docs/navigation-context.tsx +0 -88
  363. package/renderer/lib/api-docs/parsers/graphql/README.md +0 -129
  364. package/renderer/lib/api-docs/parsers/graphql/index.ts +0 -91
  365. package/renderer/lib/api-docs/parsers/graphql/parser.ts +0 -491
  366. package/renderer/lib/api-docs/parsers/graphql/transformer.ts +0 -246
  367. package/renderer/lib/api-docs/parsers/graphql/types.ts +0 -283
  368. package/renderer/lib/api-docs/parsers/openapi/README.md +0 -32
  369. package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +0 -60
  370. package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +0 -574
  371. package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +0 -403
  372. package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +0 -232
  373. package/renderer/lib/api-docs/parsers/openapi/index.ts +0 -171
  374. package/renderer/lib/api-docs/parsers/openapi/transformer.ts +0 -278
  375. package/renderer/lib/api-docs/parsers/openapi/validator.ts +0 -31
  376. package/renderer/lib/api-docs/playground/context.tsx +0 -107
  377. package/renderer/lib/api-docs/playground/navigation-context.tsx +0 -124
  378. package/renderer/lib/api-docs/playground/request-builder.ts +0 -223
  379. package/renderer/lib/api-docs/playground/request-runner.ts +0 -282
  380. package/renderer/lib/api-docs/playground/types.ts +0 -35
  381. package/renderer/lib/api-docs/types.ts +0 -269
  382. package/renderer/lib/api-docs/utils.ts +0 -311
  383. package/renderer/lib/cache.ts +0 -193
  384. package/renderer/lib/docs/config/domain-schema.ts +0 -260
  385. package/renderer/lib/docs/config/index.ts +0 -43
  386. package/renderer/lib/docs/config/loader.ts +0 -142
  387. package/renderer/lib/docs/config/schema.ts +0 -308
  388. package/renderer/lib/docs/index.ts +0 -12
  389. package/renderer/lib/docs/mdx/compiler.ts +0 -176
  390. package/renderer/lib/docs/mdx/frontmatter.ts +0 -91
  391. package/renderer/lib/docs/mdx/index.ts +0 -26
  392. package/renderer/lib/docs/navigation/generator.ts +0 -348
  393. package/renderer/lib/docs/navigation/index.ts +0 -12
  394. package/renderer/lib/docs/navigation/types.ts +0 -123
  395. package/renderer/lib/docs-navigation-context.tsx +0 -80
  396. package/renderer/lib/multi-tenant/context.ts +0 -105
  397. package/renderer/lib/storage/blob.ts +0 -1083
  398. package/renderer/lib/utils/icons.ts +0 -48
  399. package/renderer/lib/utils.ts +0 -6
  400. package/renderer/next.config.ts +0 -76
@@ -1,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
- }