@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
@@ -0,0 +1,129 @@
1
+ import { NextResponse } from 'next/server';
2
+ import { validateApiKey, getProjectContent, getProjectApiKey, deleteProject, projectExists } from '@/lib/storage/blob';
3
+ /**
4
+ * GET /api/projects/[slug]
5
+ * Get detailed project information
6
+ *
7
+ * Headers:
8
+ * Authorization: Bearer <api_key>
9
+ */ export async function GET(request, { params }) {
10
+ try {
11
+ const { slug } = await params;
12
+ // Validate API key
13
+ const authHeader = request.headers.get('Authorization');
14
+ const apiKey = authHeader?.replace('Bearer ', '');
15
+ if (!apiKey) {
16
+ return NextResponse.json({
17
+ error: 'API key required'
18
+ }, {
19
+ status: 401
20
+ });
21
+ }
22
+ // Validate the API key belongs to this project
23
+ const validatedSlug = await validateApiKey(apiKey);
24
+ if (validatedSlug !== slug) {
25
+ return NextResponse.json({
26
+ error: 'API key does not match this project'
27
+ }, {
28
+ status: 403
29
+ });
30
+ }
31
+ // Get project details
32
+ const content = await getProjectContent(slug);
33
+ const keyData = await getProjectApiKey(slug);
34
+ if (!content) {
35
+ return NextResponse.json({
36
+ error: 'Project not found'
37
+ }, {
38
+ status: 404
39
+ });
40
+ }
41
+ // Parse docs.json
42
+ let docsConfig = {};
43
+ try {
44
+ docsConfig = JSON.parse(content.docsJson);
45
+ } catch {
46
+ // Ignore parse errors
47
+ }
48
+ // Build file list (without content for security/size)
49
+ const files = content.files.map((f)=>({
50
+ path: f.path,
51
+ size: f.content.length
52
+ }));
53
+ return NextResponse.json({
54
+ slug: content.slug,
55
+ name: content.name,
56
+ url: `https://${slug}.devdoc.sh`,
57
+ createdAt: content.createdAt,
58
+ updatedAt: content.updatedAt,
59
+ lastDeployedAt: keyData?.lastUsedAt || content.updatedAt,
60
+ filesCount: content.files.length,
61
+ totalSize: content.files.reduce((sum, f)=>sum + f.content.length, 0),
62
+ files,
63
+ config: {
64
+ name: docsConfig.name,
65
+ favicon: docsConfig.favicon,
66
+ navigation: docsConfig.navigation ? 'configured' : 'default'
67
+ }
68
+ });
69
+ } catch (error) {
70
+ console.error('[Projects API] GET Error:', error);
71
+ return NextResponse.json({
72
+ error: 'Internal server error'
73
+ }, {
74
+ status: 500
75
+ });
76
+ }
77
+ }
78
+ /**
79
+ * DELETE /api/projects/[slug]
80
+ * Delete a project and all its content
81
+ *
82
+ * Headers:
83
+ * Authorization: Bearer <api_key>
84
+ */ export async function DELETE(request, { params }) {
85
+ try {
86
+ const { slug } = await params;
87
+ // Validate API key
88
+ const authHeader = request.headers.get('Authorization');
89
+ const apiKey = authHeader?.replace('Bearer ', '');
90
+ if (!apiKey) {
91
+ return NextResponse.json({
92
+ error: 'API key required'
93
+ }, {
94
+ status: 401
95
+ });
96
+ }
97
+ // Validate the API key belongs to this project
98
+ const validatedSlug = await validateApiKey(apiKey);
99
+ if (validatedSlug !== slug) {
100
+ return NextResponse.json({
101
+ error: 'API key does not match this project'
102
+ }, {
103
+ status: 403
104
+ });
105
+ }
106
+ // Check project exists
107
+ const exists = await projectExists(slug);
108
+ if (!exists) {
109
+ return NextResponse.json({
110
+ error: 'Project not found'
111
+ }, {
112
+ status: 404
113
+ });
114
+ }
115
+ // Delete the project
116
+ await deleteProject(slug);
117
+ return NextResponse.json({
118
+ success: true,
119
+ message: `Project ${slug} has been deleted`
120
+ });
121
+ } catch (error) {
122
+ console.error('[Projects API] DELETE Error:', error);
123
+ return NextResponse.json({
124
+ error: 'Internal server error'
125
+ }, {
126
+ status: 500
127
+ });
128
+ }
129
+ }
@@ -0,0 +1,80 @@
1
+ import { NextResponse } from 'next/server';
2
+ import { validateApiKey, getProjectContent, getProjectApiKey } from '@/lib/storage/blob';
3
+ /**
4
+ * GET /api/projects/[slug]/stats
5
+ * Get project statistics
6
+ *
7
+ * Headers:
8
+ * Authorization: Bearer <api_key>
9
+ */ export async function GET(request, { params }) {
10
+ try {
11
+ const { slug } = await params;
12
+ // Validate API key
13
+ const authHeader = request.headers.get('Authorization');
14
+ const apiKey = authHeader?.replace('Bearer ', '');
15
+ if (!apiKey) {
16
+ return NextResponse.json({
17
+ error: 'API key required'
18
+ }, {
19
+ status: 401
20
+ });
21
+ }
22
+ // Validate the API key belongs to this project
23
+ const validatedSlug = await validateApiKey(apiKey);
24
+ if (validatedSlug !== slug) {
25
+ return NextResponse.json({
26
+ error: 'API key does not match this project'
27
+ }, {
28
+ status: 403
29
+ });
30
+ }
31
+ // Get project details
32
+ const content = await getProjectContent(slug);
33
+ const keyData = await getProjectApiKey(slug);
34
+ if (!content) {
35
+ return NextResponse.json({
36
+ error: 'Project not found'
37
+ }, {
38
+ status: 404
39
+ });
40
+ }
41
+ // Calculate stats
42
+ const totalSize = content.files.reduce((sum, f)=>sum + f.content.length, 0);
43
+ const mdxFiles = content.files.filter((f)=>f.path.endsWith('.mdx') || f.path.endsWith('.md'));
44
+ const jsonFiles = content.files.filter((f)=>f.path.endsWith('.json'));
45
+ const otherFiles = content.files.filter((f)=>!f.path.endsWith('.mdx') && !f.path.endsWith('.md') && !f.path.endsWith('.json'));
46
+ // Calculate days since creation
47
+ const createdDate = new Date(content.createdAt);
48
+ const now = new Date();
49
+ const daysSinceCreation = Math.floor((now.getTime() - createdDate.getTime()) / (1000 * 60 * 60 * 24));
50
+ return NextResponse.json({
51
+ slug,
52
+ stats: {
53
+ totalFiles: content.files.length,
54
+ mdxPages: mdxFiles.length,
55
+ configFiles: jsonFiles.length,
56
+ otherFiles: otherFiles.length,
57
+ totalSizeBytes: totalSize,
58
+ totalSizeKB: Math.round(totalSize / 1024 * 10) / 10
59
+ },
60
+ deployment: {
61
+ createdAt: content.createdAt,
62
+ updatedAt: content.updatedAt,
63
+ lastDeployedAt: keyData?.lastUsedAt || content.updatedAt,
64
+ daysSinceCreation
65
+ },
66
+ // Placeholder for future analytics
67
+ analytics: {
68
+ pageViews: 'coming soon',
69
+ uniqueVisitors: 'coming soon'
70
+ }
71
+ });
72
+ } catch (error) {
73
+ console.error('[Projects Stats API] Error:', error);
74
+ return NextResponse.json({
75
+ error: 'Internal server error'
76
+ }, {
77
+ status: 500
78
+ });
79
+ }
80
+ }
@@ -0,0 +1,176 @@
1
+ import { NextResponse } from 'next/server';
2
+ import { isSubdomainRegistered, registerSubdomain, generateApiKey, storeProjectApiKey } from '@/lib/storage/blob';
3
+ // Reserved/blacklisted subdomains (same as in subdomains/check)
4
+ const BLACKLISTED_SUBDOMAINS = new Set([
5
+ 'www',
6
+ 'api',
7
+ 'app',
8
+ 'admin',
9
+ 'dashboard',
10
+ 'console',
11
+ 'panel',
12
+ 'manage',
13
+ 'login',
14
+ 'signin',
15
+ 'signup',
16
+ 'register',
17
+ 'auth',
18
+ 'oauth',
19
+ 'sso',
20
+ 'devdoc',
21
+ 'brainfish',
22
+ 'docs',
23
+ 'documentation',
24
+ 'help',
25
+ 'support',
26
+ 'status',
27
+ 'blog',
28
+ 'news',
29
+ 'mail',
30
+ 'email',
31
+ 'smtp',
32
+ 'ftp',
33
+ 'cdn',
34
+ 'static',
35
+ 'assets',
36
+ 'images',
37
+ 'files',
38
+ 'media',
39
+ 'download',
40
+ 'downloads',
41
+ 'test',
42
+ 'testing',
43
+ 'dev',
44
+ 'development',
45
+ 'staging',
46
+ 'prod',
47
+ 'production',
48
+ 'demo',
49
+ 'example',
50
+ 'sandbox',
51
+ 'preview',
52
+ 'secure',
53
+ 'ssl',
54
+ 'security',
55
+ 'abuse',
56
+ 'spam',
57
+ 'postmaster',
58
+ 'hostmaster',
59
+ 'webmaster',
60
+ 'null',
61
+ 'undefined',
62
+ 'true',
63
+ 'false',
64
+ 'root',
65
+ 'system',
66
+ 'localhost'
67
+ ]);
68
+ /**
69
+ * POST /api/projects/register
70
+ * Register a new project and generate an API key
71
+ *
72
+ * Body:
73
+ * name: string - Project display name
74
+ * slug: string - Project slug (URL-safe identifier)
75
+ * subdomain: string - Desired subdomain for <subdomain>.devdoc.sh
76
+ *
77
+ * Returns:
78
+ * success: boolean
79
+ * projectId: string
80
+ * slug: string
81
+ * subdomain: string
82
+ * apiKey: string
83
+ * url: string
84
+ */ export async function POST(request) {
85
+ try {
86
+ const body = await request.json();
87
+ const { name, slug, subdomain } = body;
88
+ // Validate required fields
89
+ if (!name || !slug || !subdomain) {
90
+ return NextResponse.json({
91
+ error: 'name, slug, and subdomain are required'
92
+ }, {
93
+ status: 400
94
+ });
95
+ }
96
+ // Normalize subdomain
97
+ const normalizedSubdomain = subdomain.toLowerCase().trim();
98
+ // Validate subdomain format
99
+ if (normalizedSubdomain.length < 3) {
100
+ return NextResponse.json({
101
+ error: 'Subdomain must be at least 3 characters'
102
+ }, {
103
+ status: 400
104
+ });
105
+ }
106
+ if (normalizedSubdomain.length > 63) {
107
+ return NextResponse.json({
108
+ error: 'Subdomain must be 63 characters or less'
109
+ }, {
110
+ status: 400
111
+ });
112
+ }
113
+ if (!/^[a-z0-9]([a-z0-9-]*[a-z0-9])?$/.test(normalizedSubdomain)) {
114
+ return NextResponse.json({
115
+ error: 'Subdomain must start and end with alphanumeric characters'
116
+ }, {
117
+ status: 400
118
+ });
119
+ }
120
+ // Check blacklist
121
+ if (BLACKLISTED_SUBDOMAINS.has(normalizedSubdomain)) {
122
+ return NextResponse.json({
123
+ error: `"${normalizedSubdomain}" is a reserved subdomain`
124
+ }, {
125
+ status: 400
126
+ });
127
+ }
128
+ // Use subdomain as the project identifier (slug)
129
+ // This ensures subdomain uniqueness = project uniqueness
130
+ const projectSlug = normalizedSubdomain;
131
+ // Check if subdomain already registered (O(1) lookup from registry)
132
+ const exists = await isSubdomainRegistered(projectSlug);
133
+ if (exists) {
134
+ return NextResponse.json({
135
+ error: `Project with subdomain "${normalizedSubdomain}" already exists`
136
+ }, {
137
+ status: 409
138
+ });
139
+ }
140
+ // Generate API key and project ID
141
+ const apiKey = generateApiKey();
142
+ const projectId = `proj_${projectSlug}_${Math.random().toString(36).substring(2, 8)}`;
143
+ // Register in domain registry (O(1) lookup for future checks)
144
+ await registerSubdomain(projectSlug, projectId, name, apiKey);
145
+ // Also store API key in project folder (for backwards compatibility)
146
+ await storeProjectApiKey(projectSlug, apiKey);
147
+ const response = {
148
+ success: true,
149
+ projectId,
150
+ slug: projectSlug,
151
+ subdomain: normalizedSubdomain,
152
+ apiKey,
153
+ url: `https://${normalizedSubdomain}.devdoc.sh`
154
+ };
155
+ console.log(`[Projects API] Registered new project: ${projectSlug}`);
156
+ return NextResponse.json(response, {
157
+ status: 201
158
+ });
159
+ } catch (error) {
160
+ console.error('[Projects API] Register Error:', error);
161
+ const message = error instanceof Error ? error.message : String(error);
162
+ // Provide clearer error messages for common issues
163
+ let userMessage = 'Failed to register project';
164
+ if (message.includes('blob already exists')) {
165
+ userMessage = 'Project storage conflict. Please try again or use a different subdomain.';
166
+ } else if (message.includes('network') || message.includes('fetch')) {
167
+ userMessage = 'Network error connecting to storage service';
168
+ }
169
+ return NextResponse.json({
170
+ error: userMessage,
171
+ details: message
172
+ }, {
173
+ status: 500
174
+ });
175
+ }
176
+ }
@@ -0,0 +1,139 @@
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
+ */ import { NextResponse } from 'next/server';
7
+ // Headers that should not be forwarded
8
+ const EXCLUDED_REQUEST_HEADERS = [
9
+ 'host',
10
+ 'connection',
11
+ 'content-length',
12
+ 'transfer-encoding',
13
+ 'keep-alive',
14
+ 'upgrade',
15
+ 'proxy-connection',
16
+ 'proxy-authorization'
17
+ ];
18
+ const EXCLUDED_RESPONSE_HEADERS = [
19
+ 'transfer-encoding',
20
+ 'connection',
21
+ 'keep-alive',
22
+ 'content-encoding'
23
+ ];
24
+ export async function POST(request) {
25
+ try {
26
+ const body = await request.json();
27
+ const { url, method, headers, requestBody } = body;
28
+ if (!url) {
29
+ return NextResponse.json({
30
+ error: 'URL is required'
31
+ }, {
32
+ status: 400
33
+ });
34
+ }
35
+ // Validate URL to prevent SSRF attacks
36
+ const parsedUrl = new URL(url);
37
+ const allowedProtocols = [
38
+ 'http:',
39
+ 'https:'
40
+ ];
41
+ if (!allowedProtocols.includes(parsedUrl.protocol)) {
42
+ return NextResponse.json({
43
+ error: 'Invalid URL protocol'
44
+ }, {
45
+ status: 400
46
+ });
47
+ }
48
+ // Build headers for the proxied request
49
+ const proxyHeaders = {};
50
+ for (const [key, value] of Object.entries(headers || {})){
51
+ if (!EXCLUDED_REQUEST_HEADERS.includes(key.toLowerCase())) {
52
+ proxyHeaders[key] = value;
53
+ }
54
+ }
55
+ // Make the proxied request
56
+ const startTime = Date.now();
57
+ const response = await fetch(url, {
58
+ method: method || 'GET',
59
+ headers: proxyHeaders,
60
+ body: requestBody || undefined
61
+ });
62
+ const responseTime = Date.now() - startTime;
63
+ // Read response body
64
+ const contentType = response.headers.get('content-type') || '';
65
+ let responseBody = null;
66
+ if (contentType.includes('application/json') || contentType.includes('text/')) {
67
+ responseBody = await response.text();
68
+ } else {
69
+ // For binary content, convert to base64
70
+ const buffer = await response.arrayBuffer();
71
+ responseBody = Buffer.from(buffer).toString('base64');
72
+ }
73
+ // Build response headers
74
+ const responseHeaders = {};
75
+ response.headers.forEach((value, key)=>{
76
+ if (!EXCLUDED_RESPONSE_HEADERS.includes(key.toLowerCase())) {
77
+ responseHeaders[key] = value;
78
+ }
79
+ });
80
+ return NextResponse.json({
81
+ status: response.status,
82
+ statusText: response.statusText,
83
+ headers: responseHeaders,
84
+ body: responseBody,
85
+ responseTime,
86
+ size: responseBody ? new Blob([
87
+ responseBody
88
+ ]).size : 0
89
+ });
90
+ } catch (error) {
91
+ return NextResponse.json({
92
+ status: 0,
93
+ statusText: 'Proxy Error',
94
+ headers: {},
95
+ body: null,
96
+ responseTime: 0,
97
+ size: 0,
98
+ error: error instanceof Error ? error.message : 'Proxy request failed'
99
+ });
100
+ }
101
+ }
102
+ // Also support GET for simple requests
103
+ export async function GET(request) {
104
+ const url = request.nextUrl.searchParams.get('url');
105
+ if (!url) {
106
+ return NextResponse.json({
107
+ error: 'URL query parameter is required'
108
+ }, {
109
+ status: 400
110
+ });
111
+ }
112
+ try {
113
+ const parsedUrl = new URL(url);
114
+ const allowedProtocols = [
115
+ 'http:',
116
+ 'https:'
117
+ ];
118
+ if (!allowedProtocols.includes(parsedUrl.protocol)) {
119
+ return NextResponse.json({
120
+ error: 'Invalid URL protocol'
121
+ }, {
122
+ status: 400
123
+ });
124
+ }
125
+ const response = await fetch(url);
126
+ const body = await response.text();
127
+ return NextResponse.json({
128
+ status: response.status,
129
+ statusText: response.statusText,
130
+ body
131
+ });
132
+ } catch (error) {
133
+ return NextResponse.json({
134
+ error: error instanceof Error ? error.message : 'Request failed'
135
+ }, {
136
+ status: 500
137
+ });
138
+ }
139
+ }
@@ -0,0 +1,156 @@
1
+ /**
2
+ * Streaming Proxy Route
3
+ *
4
+ * Forwards requests and streams responses back for SSE endpoints.
5
+ * Auto-detects SSE based on response content-type.
6
+ */ // Headers that should not be forwarded
7
+ const EXCLUDED_REQUEST_HEADERS = [
8
+ 'host',
9
+ 'connection',
10
+ 'content-length',
11
+ 'transfer-encoding',
12
+ 'keep-alive',
13
+ 'upgrade',
14
+ 'proxy-connection',
15
+ 'proxy-authorization'
16
+ ];
17
+ export async function POST(request) {
18
+ const startTime = Date.now();
19
+ try {
20
+ const body = await request.json();
21
+ const { url, method, headers, requestBody } = body;
22
+ if (!url) {
23
+ return new Response(JSON.stringify({
24
+ error: 'URL is required'
25
+ }), {
26
+ status: 400,
27
+ headers: {
28
+ 'Content-Type': 'application/json'
29
+ }
30
+ });
31
+ }
32
+ // Validate URL
33
+ const parsedUrl = new URL(url);
34
+ const allowedProtocols = [
35
+ 'http:',
36
+ 'https:'
37
+ ];
38
+ if (!allowedProtocols.includes(parsedUrl.protocol)) {
39
+ return new Response(JSON.stringify({
40
+ error: 'Invalid URL protocol'
41
+ }), {
42
+ status: 400,
43
+ headers: {
44
+ 'Content-Type': 'application/json'
45
+ }
46
+ });
47
+ }
48
+ // Build headers for the proxied request
49
+ const proxyHeaders = {};
50
+ for (const [key, value] of Object.entries(headers || {})){
51
+ if (!EXCLUDED_REQUEST_HEADERS.includes(key.toLowerCase())) {
52
+ proxyHeaders[key] = value;
53
+ }
54
+ }
55
+ // Make the proxied request
56
+ const response = await fetch(url, {
57
+ method: method || 'GET',
58
+ headers: proxyHeaders,
59
+ body: requestBody || undefined
60
+ });
61
+ const contentType = response.headers.get('content-type') || '';
62
+ // Check if this is a streaming response (SSE or similar)
63
+ const isStreaming = contentType.includes('text/event-stream') || contentType.includes('application/x-ndjson') || contentType.includes('application/stream+json');
64
+ if (isStreaming && response.body) {
65
+ // For streaming responses, wrap in our SSE format
66
+ const encoder = new TextEncoder();
67
+ // Send metadata first, then stream chunks
68
+ const stream = new ReadableStream({
69
+ async start (controller) {
70
+ // Send metadata
71
+ const metadata = {
72
+ type: 'metadata',
73
+ status: response.status,
74
+ statusText: response.statusText,
75
+ headers: Object.fromEntries(response.headers.entries()),
76
+ contentType
77
+ };
78
+ controller.enqueue(encoder.encode(`data: ${JSON.stringify(metadata)}\n\n`));
79
+ const reader = response.body.getReader();
80
+ const decoder = new TextDecoder();
81
+ let totalSize = 0;
82
+ try {
83
+ while(true){
84
+ const { done, value } = await reader.read();
85
+ if (done) {
86
+ controller.enqueue(encoder.encode(`data: ${JSON.stringify({
87
+ type: 'done',
88
+ totalSize
89
+ })}\n\n`));
90
+ controller.close();
91
+ break;
92
+ }
93
+ const chunk = decoder.decode(value, {
94
+ stream: true
95
+ });
96
+ totalSize += value.length;
97
+ // Send chunk
98
+ const chunkEvent = {
99
+ type: 'chunk',
100
+ data: chunk,
101
+ size: value.length
102
+ };
103
+ controller.enqueue(encoder.encode(`data: ${JSON.stringify(chunkEvent)}\n\n`));
104
+ }
105
+ } catch (error) {
106
+ const errorEvent = {
107
+ type: 'error',
108
+ error: error instanceof Error ? error.message : 'Stream error'
109
+ };
110
+ controller.enqueue(encoder.encode(`data: ${JSON.stringify(errorEvent)}\n\n`));
111
+ controller.close();
112
+ }
113
+ }
114
+ });
115
+ return new Response(stream, {
116
+ headers: {
117
+ 'Content-Type': 'text/event-stream',
118
+ 'Cache-Control': 'no-cache',
119
+ 'Connection': 'keep-alive'
120
+ }
121
+ });
122
+ }
123
+ // For non-streaming responses, return as JSON
124
+ const responseBody = await response.text();
125
+ const responseHeaders = {};
126
+ response.headers.forEach((value, key)=>{
127
+ responseHeaders[key] = value;
128
+ });
129
+ return new Response(JSON.stringify({
130
+ status: response.status,
131
+ statusText: response.statusText,
132
+ headers: responseHeaders,
133
+ body: responseBody,
134
+ size: new Blob([
135
+ responseBody
136
+ ]).size,
137
+ responseTime: Date.now() - startTime,
138
+ isStreaming: false
139
+ }), {
140
+ headers: {
141
+ 'Content-Type': 'application/json'
142
+ }
143
+ });
144
+ } catch (error) {
145
+ return new Response(JSON.stringify({
146
+ status: 0,
147
+ statusText: 'Proxy Error',
148
+ error: error instanceof Error ? error.message : 'Proxy request failed'
149
+ }), {
150
+ status: 500,
151
+ headers: {
152
+ 'Content-Type': 'application/json'
153
+ }
154
+ });
155
+ }
156
+ }