@brainfish-ai/devdoc 0.1.42 → 0.1.43

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (388) hide show
  1. package/dist/cli/commands/dev.js +19 -10
  2. package/package.json +3 -2
  3. package/renderer/app/api/assets/[...path]/route.js +108 -0
  4. package/renderer/app/api/assets/route.js +114 -0
  5. package/renderer/app/api/assets/upload/route.js +163 -0
  6. package/renderer/app/api/auth-schemes/route.js +58 -0
  7. package/renderer/app/api/chat/route.js +759 -0
  8. package/renderer/app/api/codegen/route.js +52 -0
  9. package/renderer/app/api/collections/route.js +675 -0
  10. package/renderer/app/api/debug/route.js +47 -0
  11. package/renderer/app/api/deploy/route.js +199 -0
  12. package/renderer/app/api/device/route.js +36 -0
  13. package/renderer/app/api/docs/route.js +205 -0
  14. package/renderer/app/api/domains/add/route.js +121 -0
  15. package/renderer/app/api/domains/lookup/route.js +43 -0
  16. package/renderer/app/api/domains/remove/route.js +89 -0
  17. package/renderer/app/api/domains/status/route.js +140 -0
  18. package/renderer/app/api/domains/verify/route.js +168 -0
  19. package/renderer/app/api/keys/regenerate/route.js +71 -0
  20. package/renderer/app/api/local-assets/[...path]/route.js +108 -0
  21. package/renderer/app/api/openapi-spec/route.js +73 -0
  22. package/renderer/app/api/projects/[slug]/route.js +129 -0
  23. package/renderer/app/api/projects/[slug]/stats/route.js +80 -0
  24. package/renderer/app/api/projects/register/route.js +176 -0
  25. package/renderer/app/api/proxy/route.js +139 -0
  26. package/renderer/app/api/proxy-stream/route.js +156 -0
  27. package/renderer/app/api/redirects/route.js +35 -0
  28. package/renderer/app/api/schema/route.js +85 -0
  29. package/renderer/app/api/subdomains/check/route.js +158 -0
  30. package/renderer/app/api/suggestions/route.js +175 -0
  31. package/renderer/app/layout.js +47 -0
  32. package/renderer/app/llms-full.txt/route.js +257 -0
  33. package/renderer/app/llms.txt/route.js +219 -0
  34. package/renderer/app/page.js +12 -0
  35. package/renderer/app/robots.txt/route.js +66 -0
  36. package/renderer/app/sitemap.xml/route.js +145 -0
  37. package/renderer/components/docs/index.js +8 -0
  38. package/renderer/components/docs/mdx/accordion.js +113 -0
  39. package/renderer/components/docs/mdx/badge.js +72 -0
  40. package/renderer/components/docs/mdx/callouts.js +137 -0
  41. package/renderer/components/docs/mdx/cards.js +175 -0
  42. package/renderer/components/docs/mdx/changelog.js +100 -0
  43. package/renderer/components/docs/mdx/code-block.js +147 -0
  44. package/renderer/components/docs/mdx/code-group.js +287 -0
  45. package/renderer/components/docs/mdx/file-embeds.js +82 -0
  46. package/renderer/components/docs/mdx/frame.js +59 -0
  47. package/renderer/components/docs/mdx/highlight.js +90 -0
  48. package/renderer/components/docs/mdx/iframe.js +69 -0
  49. package/renderer/components/docs/mdx/image.js +135 -0
  50. package/renderer/components/docs/mdx/index.js +134 -0
  51. package/renderer/components/docs/mdx/landing.js +315 -0
  52. package/renderer/components/docs/mdx/mermaid.js +212 -0
  53. package/renderer/components/docs/mdx/param-field.js +112 -0
  54. package/renderer/components/docs/mdx/steps.js +74 -0
  55. package/renderer/components/docs/mdx/tabs.js +50 -0
  56. package/renderer/components/docs/mdx-renderer.js +77 -0
  57. package/renderer/components/docs/navigation/breadcrumbs.js +64 -0
  58. package/renderer/components/docs/navigation/index.js +6 -0
  59. package/renderer/components/docs/navigation/page-nav.js +57 -0
  60. package/renderer/components/docs/navigation/sidebar.js +375 -0
  61. package/renderer/components/docs/navigation/toc.js +89 -0
  62. package/renderer/components/docs/notice.js +77 -0
  63. package/renderer/components/docs-header.js +202 -0
  64. package/renderer/components/docs-viewer/agent/agent-chat.js +1930 -0
  65. package/renderer/components/docs-viewer/agent/cards/debug-context-card.js +107 -0
  66. package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.js +57 -0
  67. package/renderer/components/docs-viewer/agent/cards/index.js +45 -0
  68. package/renderer/components/docs-viewer/agent/cards/response-options-card.js +154 -0
  69. package/renderer/components/docs-viewer/agent/cards/types.js +22 -0
  70. package/renderer/components/docs-viewer/agent/chat-message.js +2 -0
  71. package/renderer/components/docs-viewer/agent/index.js +4 -0
  72. package/renderer/components/docs-viewer/agent/messages/assistant-message.js +108 -0
  73. package/renderer/components/docs-viewer/agent/messages/chat-message.js +34 -0
  74. package/renderer/components/docs-viewer/agent/messages/index.js +6 -0
  75. package/renderer/components/docs-viewer/agent/messages/tool-call-display.js +1065 -0
  76. package/renderer/components/docs-viewer/agent/messages/types.js +2 -0
  77. package/renderer/components/docs-viewer/agent/messages/typing-indicator.js +26 -0
  78. package/renderer/components/docs-viewer/agent/messages/user-message.js +37 -0
  79. package/renderer/components/docs-viewer/code-editor/{index.tsx → index.js} +1 -1
  80. package/renderer/components/docs-viewer/code-editor/notes-mode.js +1338 -0
  81. package/renderer/components/docs-viewer/content/changelog-page.js +297 -0
  82. package/renderer/components/docs-viewer/content/doc-page.js +264 -0
  83. package/renderer/components/docs-viewer/content/documentation-viewer.js +14 -0
  84. package/renderer/components/docs-viewer/content/index.js +29 -0
  85. package/renderer/components/docs-viewer/content/not-found-page.js +300 -0
  86. package/renderer/components/docs-viewer/content/request-details.js +528 -0
  87. package/renderer/components/docs-viewer/content/sections/auth.js +108 -0
  88. package/renderer/components/docs-viewer/content/sections/body.js +80 -0
  89. package/renderer/components/docs-viewer/content/sections/headers.js +64 -0
  90. package/renderer/components/docs-viewer/content/sections/overview.js +56 -0
  91. package/renderer/components/docs-viewer/content/sections/parameters.js +64 -0
  92. package/renderer/components/docs-viewer/content/sections/responses.js +91 -0
  93. package/renderer/components/docs-viewer/global-auth-modal.js +427 -0
  94. package/renderer/components/docs-viewer/index.js +1552 -0
  95. package/renderer/components/docs-viewer/playground/auth-editor.js +418 -0
  96. package/renderer/components/docs-viewer/playground/body-editor.js +240 -0
  97. package/renderer/components/docs-viewer/playground/code-editor.js +135 -0
  98. package/renderer/components/docs-viewer/playground/code-snippet.js +393 -0
  99. package/renderer/components/docs-viewer/playground/graphql-playground.js +734 -0
  100. package/renderer/components/docs-viewer/playground/index.js +682 -0
  101. package/renderer/components/docs-viewer/playground/key-value-editor.js +317 -0
  102. package/renderer/components/docs-viewer/playground/method-selector.js +65 -0
  103. package/renderer/components/docs-viewer/playground/request-builder.js +181 -0
  104. package/renderer/components/docs-viewer/playground/request-tabs.js +240 -0
  105. package/renderer/components/docs-viewer/playground/response-cards/idle-card.js +42 -0
  106. package/renderer/components/docs-viewer/playground/response-cards/index.js +72 -0
  107. package/renderer/components/docs-viewer/playground/response-cards/loading-card.js +24 -0
  108. package/renderer/components/docs-viewer/playground/response-cards/network-error-card.js +28 -0
  109. package/renderer/components/docs-viewer/playground/response-cards/response-body-card.js +308 -0
  110. package/renderer/components/docs-viewer/playground/response-cards/types.js +9 -0
  111. package/renderer/components/docs-viewer/playground/response-viewer.js +18 -0
  112. package/renderer/components/docs-viewer/search/index.js +2 -0
  113. package/renderer/components/docs-viewer/search/search-dialog.js +367 -0
  114. package/renderer/components/docs-viewer/search/use-search.js +89 -0
  115. package/renderer/components/docs-viewer/shared/markdown-renderer.js +423 -0
  116. package/renderer/components/docs-viewer/shared/method-badge.js +23 -0
  117. package/renderer/components/docs-viewer/shared/schema-viewer.js +321 -0
  118. package/renderer/components/docs-viewer/sidebar/collection-tree.js +222 -0
  119. package/renderer/components/docs-viewer/sidebar/endpoint-options.js +512 -0
  120. package/renderer/components/docs-viewer/sidebar/index.js +196 -0
  121. package/renderer/components/docs-viewer/sidebar/right-sidebar.js +163 -0
  122. package/renderer/components/docs-viewer/sidebar/sidebar-group.js +87 -0
  123. package/renderer/components/docs-viewer/sidebar/sidebar-item.js +172 -0
  124. package/renderer/components/docs-viewer/sidebar/sidebar-section.js +31 -0
  125. package/renderer/components/theme-provider.js +10 -0
  126. package/renderer/components/theme-toggle.js +106 -0
  127. package/renderer/components/ui/badge.js +29 -0
  128. package/renderer/components/ui/button.js +40 -0
  129. package/renderer/components/ui/dialog.js +50 -0
  130. package/renderer/components/ui/dropdown-menu.js +143 -0
  131. package/renderer/components/ui/input.js +12 -0
  132. package/renderer/components/ui/label.js +13 -0
  133. package/renderer/components/ui/navigation-menu.js +83 -0
  134. package/renderer/components/ui/select.js +116 -0
  135. package/renderer/components/ui/spinner.js +92 -0
  136. package/renderer/components/ui/tabs.js +34 -0
  137. package/renderer/components/ui/tooltip.js +43 -0
  138. package/renderer/hooks/use-code-copy.js +76 -0
  139. package/renderer/hooks/use-openapi-title.js +33 -0
  140. package/renderer/lib/api-docs/agent/index.js +4 -0
  141. package/renderer/lib/api-docs/agent/indexer.js +254 -0
  142. package/renderer/lib/api-docs/agent/spec-summary.js +227 -0
  143. package/renderer/lib/api-docs/agent/types.js +5 -0
  144. package/renderer/lib/api-docs/auth/auth-context.js +157 -0
  145. package/renderer/lib/api-docs/auth/auth-storage.js +66 -0
  146. package/renderer/lib/api-docs/auth/crypto.js +64 -0
  147. package/renderer/lib/api-docs/auth/index.js +3 -0
  148. package/renderer/lib/api-docs/code-editor/db.js +145 -0
  149. package/renderer/lib/api-docs/code-editor/hooks.js +254 -0
  150. package/renderer/lib/api-docs/code-editor/{index.ts → index.js} +3 -4
  151. package/renderer/lib/api-docs/code-editor/mode-context.js +154 -0
  152. package/renderer/lib/api-docs/code-editor/types.js +53 -0
  153. package/renderer/lib/api-docs/codegen/definitions.js +258 -0
  154. package/renderer/lib/api-docs/codegen/har.js +171 -0
  155. package/renderer/lib/api-docs/codegen/index.js +118 -0
  156. package/renderer/lib/api-docs/factories.js +136 -0
  157. package/renderer/lib/api-docs/{index.ts → index.js} +5 -10
  158. package/renderer/lib/api-docs/mobile-context.js +79 -0
  159. package/renderer/lib/api-docs/navigation-context.js +62 -0
  160. package/renderer/lib/api-docs/parsers/graphql/index.js +50 -0
  161. package/renderer/lib/api-docs/parsers/graphql/parser.js +350 -0
  162. package/renderer/lib/api-docs/parsers/graphql/transformer.js +215 -0
  163. package/renderer/lib/api-docs/parsers/graphql/types.js +46 -0
  164. package/renderer/lib/api-docs/parsers/openapi/dereferencer.js +43 -0
  165. package/renderer/lib/api-docs/parsers/openapi/extractors/auth.js +486 -0
  166. package/renderer/lib/api-docs/parsers/openapi/extractors/body.js +295 -0
  167. package/renderer/lib/api-docs/parsers/openapi/extractors/index.js +132 -0
  168. package/renderer/lib/api-docs/parsers/openapi/index.js +127 -0
  169. package/renderer/lib/api-docs/parsers/openapi/transformer.js +192 -0
  170. package/renderer/lib/api-docs/parsers/openapi/validator.js +24 -0
  171. package/renderer/lib/api-docs/playground/context.js +65 -0
  172. package/renderer/lib/api-docs/playground/navigation-context.js +74 -0
  173. package/renderer/lib/api-docs/playground/request-builder.js +163 -0
  174. package/renderer/lib/api-docs/playground/request-runner.js +224 -0
  175. package/renderer/lib/api-docs/playground/types.js +5 -0
  176. package/renderer/lib/api-docs/types.js +23 -0
  177. package/renderer/lib/api-docs/utils.js +212 -0
  178. package/renderer/lib/cache.js +157 -0
  179. package/renderer/lib/docs/config/domain-schema.js +161 -0
  180. package/renderer/lib/docs/config/index.js +5 -0
  181. package/renderer/lib/docs/config/loader.js +113 -0
  182. package/renderer/lib/docs/config/schema.js +269 -0
  183. package/renderer/lib/docs/index.js +8 -0
  184. package/renderer/lib/docs/mdx/compiler.js +128 -0
  185. package/renderer/lib/docs/mdx/frontmatter.js +73 -0
  186. package/renderer/lib/docs/mdx/index.js +8 -0
  187. package/renderer/lib/docs/navigation/generator.js +269 -0
  188. package/renderer/lib/docs/navigation/index.js +4 -0
  189. package/renderer/lib/docs/navigation/types.js +9 -0
  190. package/renderer/lib/docs-navigation-context.js +40 -0
  191. package/renderer/lib/multi-tenant/context.js +80 -0
  192. package/renderer/lib/storage/blob.js +767 -0
  193. package/renderer/lib/utils/icons.js +30 -0
  194. package/renderer/lib/utils.js +5 -0
  195. package/renderer/next.config.js +62 -0
  196. package/renderer/tsconfig.json +23 -5
  197. package/renderer/app/api/assets/[...path]/route.ts +0 -123
  198. package/renderer/app/api/assets/route.ts +0 -124
  199. package/renderer/app/api/assets/upload/route.ts +0 -177
  200. package/renderer/app/api/auth-schemes/route.ts +0 -77
  201. package/renderer/app/api/chat/route.ts +0 -858
  202. package/renderer/app/api/codegen/route.ts +0 -72
  203. package/renderer/app/api/collections/route.ts +0 -1002
  204. package/renderer/app/api/debug/route.ts +0 -53
  205. package/renderer/app/api/deploy/route.ts +0 -234
  206. package/renderer/app/api/device/route.ts +0 -42
  207. package/renderer/app/api/docs/route.ts +0 -201
  208. package/renderer/app/api/domains/add/route.ts +0 -132
  209. package/renderer/app/api/domains/lookup/route.ts +0 -43
  210. package/renderer/app/api/domains/remove/route.ts +0 -100
  211. package/renderer/app/api/domains/status/route.ts +0 -158
  212. package/renderer/app/api/domains/verify/route.ts +0 -181
  213. package/renderer/app/api/keys/regenerate/route.ts +0 -80
  214. package/renderer/app/api/local-assets/[...path]/route.ts +0 -122
  215. package/renderer/app/api/openapi-spec/route.ts +0 -151
  216. package/renderer/app/api/projects/[slug]/route.ts +0 -153
  217. package/renderer/app/api/projects/[slug]/stats/route.ts +0 -96
  218. package/renderer/app/api/projects/register/route.ts +0 -152
  219. package/renderer/app/api/proxy/route.ts +0 -149
  220. package/renderer/app/api/proxy-stream/route.ts +0 -168
  221. package/renderer/app/api/redirects/route.ts +0 -47
  222. package/renderer/app/api/schema/route.ts +0 -73
  223. package/renderer/app/api/subdomains/check/route.ts +0 -172
  224. package/renderer/app/api/suggestions/route.ts +0 -144
  225. package/renderer/app/layout.tsx +0 -54
  226. package/renderer/app/llms-full.txt/route.ts +0 -346
  227. package/renderer/app/llms.txt/route.ts +0 -279
  228. package/renderer/app/page.tsx +0 -14
  229. package/renderer/app/robots.txt/route.ts +0 -84
  230. package/renderer/app/sitemap.xml/route.ts +0 -199
  231. package/renderer/components/docs/index.ts +0 -12
  232. package/renderer/components/docs/mdx/accordion.tsx +0 -169
  233. package/renderer/components/docs/mdx/badge.tsx +0 -132
  234. package/renderer/components/docs/mdx/callouts.tsx +0 -154
  235. package/renderer/components/docs/mdx/cards.tsx +0 -241
  236. package/renderer/components/docs/mdx/changelog.tsx +0 -120
  237. package/renderer/components/docs/mdx/code-block.tsx +0 -186
  238. package/renderer/components/docs/mdx/code-group.tsx +0 -421
  239. package/renderer/components/docs/mdx/file-embeds.tsx +0 -105
  240. package/renderer/components/docs/mdx/frame.tsx +0 -112
  241. package/renderer/components/docs/mdx/highlight.tsx +0 -151
  242. package/renderer/components/docs/mdx/iframe.tsx +0 -134
  243. package/renderer/components/docs/mdx/image.tsx +0 -235
  244. package/renderer/components/docs/mdx/index.ts +0 -237
  245. package/renderer/components/docs/mdx/landing.tsx +0 -684
  246. package/renderer/components/docs/mdx/mermaid.tsx +0 -240
  247. package/renderer/components/docs/mdx/param-field.tsx +0 -200
  248. package/renderer/components/docs/mdx/steps.tsx +0 -113
  249. package/renderer/components/docs/mdx/tabs.tsx +0 -86
  250. package/renderer/components/docs/mdx-renderer.tsx +0 -100
  251. package/renderer/components/docs/navigation/breadcrumbs.tsx +0 -76
  252. package/renderer/components/docs/navigation/index.ts +0 -8
  253. package/renderer/components/docs/navigation/page-nav.tsx +0 -64
  254. package/renderer/components/docs/navigation/sidebar.tsx +0 -515
  255. package/renderer/components/docs/navigation/toc.tsx +0 -113
  256. package/renderer/components/docs/notice.tsx +0 -105
  257. package/renderer/components/docs-header.tsx +0 -278
  258. package/renderer/components/docs-viewer/agent/agent-chat.tsx +0 -2076
  259. package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +0 -90
  260. package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +0 -49
  261. package/renderer/components/docs-viewer/agent/cards/index.tsx +0 -50
  262. package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +0 -212
  263. package/renderer/components/docs-viewer/agent/cards/types.ts +0 -84
  264. package/renderer/components/docs-viewer/agent/chat-message.tsx +0 -17
  265. package/renderer/components/docs-viewer/agent/index.tsx +0 -6
  266. package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +0 -119
  267. package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +0 -46
  268. package/renderer/components/docs-viewer/agent/messages/index.ts +0 -17
  269. package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +0 -721
  270. package/renderer/components/docs-viewer/agent/messages/types.ts +0 -61
  271. package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +0 -24
  272. package/renderer/components/docs-viewer/agent/messages/user-message.tsx +0 -51
  273. package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +0 -1283
  274. package/renderer/components/docs-viewer/content/changelog-page.tsx +0 -331
  275. package/renderer/components/docs-viewer/content/doc-page.tsx +0 -367
  276. package/renderer/components/docs-viewer/content/documentation-viewer.tsx +0 -17
  277. package/renderer/components/docs-viewer/content/index.tsx +0 -29
  278. package/renderer/components/docs-viewer/content/not-found-page.tsx +0 -330
  279. package/renderer/components/docs-viewer/content/request-details.tsx +0 -330
  280. package/renderer/components/docs-viewer/content/sections/auth.tsx +0 -69
  281. package/renderer/components/docs-viewer/content/sections/body.tsx +0 -66
  282. package/renderer/components/docs-viewer/content/sections/headers.tsx +0 -43
  283. package/renderer/components/docs-viewer/content/sections/overview.tsx +0 -40
  284. package/renderer/components/docs-viewer/content/sections/parameters.tsx +0 -43
  285. package/renderer/components/docs-viewer/content/sections/responses.tsx +0 -87
  286. package/renderer/components/docs-viewer/global-auth-modal.tsx +0 -352
  287. package/renderer/components/docs-viewer/index.tsx +0 -1670
  288. package/renderer/components/docs-viewer/playground/auth-editor.tsx +0 -280
  289. package/renderer/components/docs-viewer/playground/body-editor.tsx +0 -221
  290. package/renderer/components/docs-viewer/playground/code-editor.tsx +0 -224
  291. package/renderer/components/docs-viewer/playground/code-snippet.tsx +0 -387
  292. package/renderer/components/docs-viewer/playground/graphql-playground.tsx +0 -745
  293. package/renderer/components/docs-viewer/playground/index.tsx +0 -671
  294. package/renderer/components/docs-viewer/playground/key-value-editor.tsx +0 -261
  295. package/renderer/components/docs-viewer/playground/method-selector.tsx +0 -60
  296. package/renderer/components/docs-viewer/playground/request-builder.tsx +0 -179
  297. package/renderer/components/docs-viewer/playground/request-tabs.tsx +0 -237
  298. package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +0 -21
  299. package/renderer/components/docs-viewer/playground/response-cards/index.tsx +0 -93
  300. package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +0 -16
  301. package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +0 -23
  302. package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +0 -268
  303. package/renderer/components/docs-viewer/playground/response-cards/types.ts +0 -82
  304. package/renderer/components/docs-viewer/playground/response-viewer.tsx +0 -43
  305. package/renderer/components/docs-viewer/search/index.ts +0 -2
  306. package/renderer/components/docs-viewer/search/search-dialog.tsx +0 -331
  307. package/renderer/components/docs-viewer/search/use-search.ts +0 -117
  308. package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +0 -431
  309. package/renderer/components/docs-viewer/shared/method-badge.tsx +0 -41
  310. package/renderer/components/docs-viewer/shared/schema-viewer.tsx +0 -349
  311. package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +0 -259
  312. package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +0 -316
  313. package/renderer/components/docs-viewer/sidebar/index.tsx +0 -282
  314. package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +0 -202
  315. package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +0 -118
  316. package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +0 -212
  317. package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +0 -38
  318. package/renderer/components/theme-provider.tsx +0 -11
  319. package/renderer/components/theme-toggle.tsx +0 -76
  320. package/renderer/components/ui/badge.tsx +0 -46
  321. package/renderer/components/ui/button.tsx +0 -59
  322. package/renderer/components/ui/dialog.tsx +0 -118
  323. package/renderer/components/ui/dropdown-menu.tsx +0 -257
  324. package/renderer/components/ui/input.tsx +0 -21
  325. package/renderer/components/ui/label.tsx +0 -24
  326. package/renderer/components/ui/navigation-menu.tsx +0 -168
  327. package/renderer/components/ui/select.tsx +0 -190
  328. package/renderer/components/ui/spinner.tsx +0 -114
  329. package/renderer/components/ui/tabs.tsx +0 -66
  330. package/renderer/components/ui/tooltip.tsx +0 -61
  331. package/renderer/hooks/use-code-copy.ts +0 -88
  332. package/renderer/hooks/use-openapi-title.ts +0 -44
  333. package/renderer/lib/api-docs/agent/index.ts +0 -6
  334. package/renderer/lib/api-docs/agent/indexer.ts +0 -323
  335. package/renderer/lib/api-docs/agent/spec-summary.ts +0 -335
  336. package/renderer/lib/api-docs/agent/types.ts +0 -116
  337. package/renderer/lib/api-docs/auth/auth-context.tsx +0 -225
  338. package/renderer/lib/api-docs/auth/auth-storage.ts +0 -87
  339. package/renderer/lib/api-docs/auth/crypto.ts +0 -89
  340. package/renderer/lib/api-docs/auth/index.ts +0 -4
  341. package/renderer/lib/api-docs/code-editor/db.ts +0 -164
  342. package/renderer/lib/api-docs/code-editor/hooks.ts +0 -266
  343. package/renderer/lib/api-docs/code-editor/mode-context.tsx +0 -207
  344. package/renderer/lib/api-docs/code-editor/types.ts +0 -105
  345. package/renderer/lib/api-docs/codegen/definitions.ts +0 -297
  346. package/renderer/lib/api-docs/codegen/har.ts +0 -251
  347. package/renderer/lib/api-docs/codegen/index.ts +0 -159
  348. package/renderer/lib/api-docs/factories.ts +0 -170
  349. package/renderer/lib/api-docs/mobile-context.tsx +0 -112
  350. package/renderer/lib/api-docs/navigation-context.tsx +0 -88
  351. package/renderer/lib/api-docs/parsers/graphql/README.md +0 -129
  352. package/renderer/lib/api-docs/parsers/graphql/index.ts +0 -91
  353. package/renderer/lib/api-docs/parsers/graphql/parser.ts +0 -491
  354. package/renderer/lib/api-docs/parsers/graphql/transformer.ts +0 -246
  355. package/renderer/lib/api-docs/parsers/graphql/types.ts +0 -283
  356. package/renderer/lib/api-docs/parsers/openapi/README.md +0 -32
  357. package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +0 -60
  358. package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +0 -574
  359. package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +0 -403
  360. package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +0 -232
  361. package/renderer/lib/api-docs/parsers/openapi/index.ts +0 -171
  362. package/renderer/lib/api-docs/parsers/openapi/transformer.ts +0 -278
  363. package/renderer/lib/api-docs/parsers/openapi/validator.ts +0 -31
  364. package/renderer/lib/api-docs/playground/context.tsx +0 -107
  365. package/renderer/lib/api-docs/playground/navigation-context.tsx +0 -124
  366. package/renderer/lib/api-docs/playground/request-builder.ts +0 -223
  367. package/renderer/lib/api-docs/playground/request-runner.ts +0 -282
  368. package/renderer/lib/api-docs/playground/types.ts +0 -35
  369. package/renderer/lib/api-docs/types.ts +0 -269
  370. package/renderer/lib/api-docs/utils.ts +0 -311
  371. package/renderer/lib/cache.ts +0 -193
  372. package/renderer/lib/docs/config/domain-schema.ts +0 -260
  373. package/renderer/lib/docs/config/index.ts +0 -43
  374. package/renderer/lib/docs/config/loader.ts +0 -142
  375. package/renderer/lib/docs/config/schema.ts +0 -308
  376. package/renderer/lib/docs/index.ts +0 -12
  377. package/renderer/lib/docs/mdx/compiler.ts +0 -176
  378. package/renderer/lib/docs/mdx/frontmatter.ts +0 -91
  379. package/renderer/lib/docs/mdx/index.ts +0 -26
  380. package/renderer/lib/docs/navigation/generator.ts +0 -348
  381. package/renderer/lib/docs/navigation/index.ts +0 -12
  382. package/renderer/lib/docs/navigation/types.ts +0 -123
  383. package/renderer/lib/docs-navigation-context.tsx +0 -80
  384. package/renderer/lib/multi-tenant/context.ts +0 -105
  385. package/renderer/lib/storage/blob.ts +0 -1083
  386. package/renderer/lib/utils/icons.ts +0 -48
  387. package/renderer/lib/utils.ts +0 -6
  388. package/renderer/next.config.ts +0 -76
@@ -0,0 +1,89 @@
1
+ import { NextResponse } from 'next/server';
2
+ import { validateApiKey, getProjectCustomDomain, removeCustomDomain } from '@/lib/storage/blob';
3
+ import { normalizeDomain } from '@/lib/docs/config';
4
+ /**
5
+ * DELETE /api/domains/remove
6
+ *
7
+ * Remove a custom domain from a project.
8
+ *
9
+ * Headers:
10
+ * Authorization: Bearer <api_key>
11
+ *
12
+ * Body:
13
+ * { customDomain: "docs.example.com" } // Optional, removes project's domain if not specified
14
+ *
15
+ * Response:
16
+ * {
17
+ * success: true,
18
+ * message: "Domain docs.example.com removed successfully"
19
+ * }
20
+ */ export async function DELETE(request) {
21
+ try {
22
+ // Validate API key
23
+ const authHeader = request.headers.get('Authorization');
24
+ const apiKey = authHeader?.replace('Bearer ', '');
25
+ if (!apiKey) {
26
+ return NextResponse.json({
27
+ error: 'API key required. Provide via Authorization header.'
28
+ }, {
29
+ status: 401
30
+ });
31
+ }
32
+ const projectSlug = await validateApiKey(apiKey);
33
+ if (!projectSlug) {
34
+ return NextResponse.json({
35
+ error: 'Invalid API key'
36
+ }, {
37
+ status: 403
38
+ });
39
+ }
40
+ // Parse request body
41
+ const body = await request.json().catch(()=>({}));
42
+ let customDomain = body.customDomain ? normalizeDomain(body.customDomain) : null;
43
+ // If no domain specified, get the project's custom domain
44
+ if (!customDomain) {
45
+ const projectDomain = await getProjectCustomDomain(projectSlug);
46
+ if (!projectDomain) {
47
+ return NextResponse.json({
48
+ error: 'No custom domain configured for this project'
49
+ }, {
50
+ status: 404
51
+ });
52
+ }
53
+ customDomain = projectDomain.customDomain;
54
+ }
55
+ // Remove the domain
56
+ const result = await removeCustomDomain(customDomain, projectSlug);
57
+ if (!result.success) {
58
+ return NextResponse.json({
59
+ error: result.error
60
+ }, {
61
+ status: 400
62
+ });
63
+ }
64
+ // TODO: In production, also remove from Vercel via API
65
+ // await vercelApi.removeDomain(customDomain)
66
+ return NextResponse.json({
67
+ success: true,
68
+ message: `Domain ${customDomain} removed successfully`,
69
+ domain: customDomain,
70
+ projectSlug
71
+ });
72
+ } catch (error) {
73
+ console.error('[Domains API] Error removing domain:', error);
74
+ const message = error instanceof Error ? error.message : String(error);
75
+ return NextResponse.json({
76
+ error: 'Failed to remove domain',
77
+ details: message
78
+ }, {
79
+ status: 500
80
+ });
81
+ }
82
+ }
83
+ /**
84
+ * POST /api/domains/remove
85
+ *
86
+ * Alternative method for removing domain (some clients don't support DELETE with body)
87
+ */ export async function POST(request) {
88
+ return DELETE(request);
89
+ }
@@ -0,0 +1,140 @@
1
+ import { NextResponse } from 'next/server';
2
+ import { validateApiKey, getProjectCustomDomain, getCustomDomainEntry } from '@/lib/storage/blob';
3
+ import { normalizeDomain, getDnsInstructions } from '@/lib/docs/config';
4
+ import { getProjectUrl } from '@/lib/multi-tenant/context';
5
+ /**
6
+ * GET /api/domains/status
7
+ *
8
+ * Get the status of a custom domain.
9
+ *
10
+ * Headers:
11
+ * Authorization: Bearer <api_key>
12
+ *
13
+ * Query:
14
+ * ?domain=docs.example.com // Optional, uses project's domain if not specified
15
+ *
16
+ * Response:
17
+ * {
18
+ * domain: "docs.example.com",
19
+ * status: "active",
20
+ * projectSlug: "my-project",
21
+ * projectUrl: "https://my-project.devdoc.sh",
22
+ * customUrl: "https://docs.example.com",
23
+ * createdAt: "2026-01-24T...",
24
+ * verifiedAt: "2026-01-24T..."
25
+ * }
26
+ */ export async function GET(request) {
27
+ try {
28
+ // Validate API key
29
+ const authHeader = request.headers.get('Authorization');
30
+ const apiKey = authHeader?.replace('Bearer ', '');
31
+ if (!apiKey) {
32
+ return NextResponse.json({
33
+ error: 'API key required. Provide via Authorization header.'
34
+ }, {
35
+ status: 401
36
+ });
37
+ }
38
+ const projectSlug = await validateApiKey(apiKey);
39
+ if (!projectSlug) {
40
+ return NextResponse.json({
41
+ error: 'Invalid API key'
42
+ }, {
43
+ status: 403
44
+ });
45
+ }
46
+ // Get domain from query or use project's domain
47
+ const { searchParams } = new URL(request.url);
48
+ let customDomain = searchParams.get('domain');
49
+ if (customDomain) {
50
+ customDomain = normalizeDomain(customDomain);
51
+ }
52
+ let domainEntry;
53
+ if (customDomain) {
54
+ // Get specific domain
55
+ domainEntry = await getCustomDomainEntry(customDomain);
56
+ if (!domainEntry) {
57
+ return NextResponse.json({
58
+ error: `Domain ${customDomain} not found`
59
+ }, {
60
+ status: 404
61
+ });
62
+ }
63
+ // Verify domain belongs to this project
64
+ if (domainEntry.projectSlug !== projectSlug) {
65
+ return NextResponse.json({
66
+ error: `Domain ${customDomain} does not belong to this project`
67
+ }, {
68
+ status: 403
69
+ });
70
+ }
71
+ } else {
72
+ // Get project's custom domain
73
+ domainEntry = await getProjectCustomDomain(projectSlug);
74
+ if (!domainEntry) {
75
+ return NextResponse.json({
76
+ hasCustomDomain: false,
77
+ projectSlug,
78
+ projectUrl: getProjectUrl(projectSlug),
79
+ message: 'No custom domain configured. Use /api/domains/add to add one.'
80
+ });
81
+ }
82
+ }
83
+ // Build response based on status
84
+ const projectUrl = getProjectUrl(projectSlug);
85
+ const response = {
86
+ hasCustomDomain: true,
87
+ domain: domainEntry.customDomain,
88
+ status: domainEntry.status,
89
+ projectSlug,
90
+ projectUrl,
91
+ createdAt: domainEntry.createdAt
92
+ };
93
+ // Add status-specific information
94
+ switch(domainEntry.status){
95
+ case 'pending':
96
+ {
97
+ const dnsInstructions = getDnsInstructions(domainEntry.customDomain);
98
+ dnsInstructions.txt.value = domainEntry.verificationToken || '';
99
+ response.message = 'Waiting for DNS configuration';
100
+ response.dnsRecords = {
101
+ cname: dnsInstructions.cname,
102
+ txt: dnsInstructions.txt
103
+ };
104
+ response.nextStep = 'Add the DNS records above, then run "devdoc domain verify"';
105
+ break;
106
+ }
107
+ case 'dns_verified':
108
+ response.message = 'DNS verified, SSL certificate provisioning in progress';
109
+ response.verifiedAt = domainEntry.verifiedAt;
110
+ response.nextStep = 'SSL certificate should be ready within 1-24 hours';
111
+ break;
112
+ case 'ssl_provisioning':
113
+ response.message = 'SSL certificate being provisioned';
114
+ response.verifiedAt = domainEntry.verifiedAt;
115
+ response.nextStep = 'Almost ready! Check back in a few minutes';
116
+ break;
117
+ case 'active':
118
+ response.message = 'Domain is active and working';
119
+ response.customUrl = `https://${domainEntry.customDomain}`;
120
+ response.verifiedAt = domainEntry.verifiedAt;
121
+ break;
122
+ case 'error':
123
+ response.message = 'Domain configuration error';
124
+ response.error = domainEntry.errorMessage;
125
+ response.nextStep = 'Check DNS configuration and try verifying again';
126
+ break;
127
+ }
128
+ response.lastCheckedAt = domainEntry.lastCheckedAt;
129
+ return NextResponse.json(response);
130
+ } catch (error) {
131
+ console.error('[Domains API] Error getting status:', error);
132
+ const message = error instanceof Error ? error.message : String(error);
133
+ return NextResponse.json({
134
+ error: 'Failed to get domain status',
135
+ details: message
136
+ }, {
137
+ status: 500
138
+ });
139
+ }
140
+ }
@@ -0,0 +1,168 @@
1
+ import { NextResponse } from 'next/server';
2
+ import { validateApiKey, getProjectCustomDomain, getCustomDomainEntry, updateCustomDomainStatus } from '@/lib/storage/blob';
3
+ import { normalizeDomain } from '@/lib/docs/config';
4
+ import dns from 'dns';
5
+ import { promisify } from 'util';
6
+ const resolveCname = promisify(dns.resolveCname);
7
+ const resolveTxt = promisify(dns.resolveTxt);
8
+ /**
9
+ * POST /api/domains/verify
10
+ *
11
+ * Verify DNS configuration for a custom domain.
12
+ * Checks CNAME and TXT records, then triggers SSL provisioning.
13
+ *
14
+ * Headers:
15
+ * Authorization: Bearer <api_key>
16
+ *
17
+ * Body:
18
+ * { customDomain: "docs.example.com" } // Optional, uses project's domain if not specified
19
+ *
20
+ * Response:
21
+ * {
22
+ * success: true,
23
+ * domain: "docs.example.com",
24
+ * status: "dns_verified",
25
+ * checks: {
26
+ * cname: { found: true, value: "cname.devdoc-dns.com" },
27
+ * txt: { found: true, verified: true }
28
+ * }
29
+ * }
30
+ */ export async function POST(request) {
31
+ try {
32
+ // Validate API key
33
+ const authHeader = request.headers.get('Authorization');
34
+ const apiKey = authHeader?.replace('Bearer ', '');
35
+ if (!apiKey) {
36
+ return NextResponse.json({
37
+ error: 'API key required. Provide via Authorization header.'
38
+ }, {
39
+ status: 401
40
+ });
41
+ }
42
+ const projectSlug = await validateApiKey(apiKey);
43
+ if (!projectSlug) {
44
+ return NextResponse.json({
45
+ error: 'Invalid API key'
46
+ }, {
47
+ status: 403
48
+ });
49
+ }
50
+ // Parse request body
51
+ const body = await request.json().catch(()=>({}));
52
+ let customDomain = body.customDomain ? normalizeDomain(body.customDomain) : null;
53
+ // If no domain specified, get the project's custom domain
54
+ if (!customDomain) {
55
+ const projectDomain = await getProjectCustomDomain(projectSlug);
56
+ if (!projectDomain) {
57
+ return NextResponse.json({
58
+ error: 'No custom domain configured for this project. Add one first with /api/domains/add'
59
+ }, {
60
+ status: 404
61
+ });
62
+ }
63
+ customDomain = projectDomain.customDomain;
64
+ }
65
+ // Get domain entry
66
+ const domainEntry = await getCustomDomainEntry(customDomain);
67
+ if (!domainEntry) {
68
+ return NextResponse.json({
69
+ error: `Domain ${customDomain} not found`
70
+ }, {
71
+ status: 404
72
+ });
73
+ }
74
+ // Verify domain belongs to this project
75
+ if (domainEntry.projectSlug !== projectSlug) {
76
+ return NextResponse.json({
77
+ error: `Domain ${customDomain} does not belong to this project`
78
+ }, {
79
+ status: 403
80
+ });
81
+ }
82
+ // Check DNS records
83
+ const checks = {
84
+ cname: {
85
+ found: false,
86
+ value: null,
87
+ expected: 'cname.devdoc-dns.com'
88
+ },
89
+ txt: {
90
+ found: false,
91
+ verified: false,
92
+ expected: domainEntry.verificationToken
93
+ }
94
+ };
95
+ // Check CNAME record
96
+ try {
97
+ const cnameRecords = await resolveCname(customDomain);
98
+ if (cnameRecords && cnameRecords.length > 0) {
99
+ checks.cname.found = true;
100
+ checks.cname.value = cnameRecords[0];
101
+ }
102
+ } catch {
103
+ // CNAME not found or DNS error
104
+ }
105
+ // Check TXT record for verification
106
+ const txtRecordName = `_devdoc-verify.${customDomain}`;
107
+ try {
108
+ const txtRecords = await resolveTxt(txtRecordName);
109
+ if (txtRecords && txtRecords.length > 0) {
110
+ checks.txt.found = true;
111
+ // TXT records can be arrays, flatten and check
112
+ const allTxtValues = txtRecords.flat();
113
+ checks.txt.verified = allTxtValues.some((val)=>val === domainEntry.verificationToken);
114
+ }
115
+ } catch {
116
+ // TXT not found or DNS error
117
+ }
118
+ // Determine overall status
119
+ const cnameValid = checks.cname.found && checks.cname.value?.toLowerCase().includes('devdoc');
120
+ const txtValid = checks.txt.found && checks.txt.verified;
121
+ let newStatus = domainEntry.status;
122
+ let message = '';
123
+ if (cnameValid && txtValid) {
124
+ // DNS is verified, update status
125
+ newStatus = 'dns_verified';
126
+ message = 'DNS verified! SSL certificate will be provisioned automatically (1-24 hours).';
127
+ await updateCustomDomainStatus(customDomain, 'dns_verified');
128
+ // In production, this would trigger Vercel API to add domain
129
+ // For now, we simulate SSL provisioning by updating status
130
+ // TODO: Integrate with Vercel Domains API
131
+ } else if (cnameValid && !txtValid) {
132
+ message = 'CNAME record found, but TXT verification record is missing or incorrect.';
133
+ } else if (!cnameValid && txtValid) {
134
+ message = 'TXT verification found, but CNAME record is missing or incorrect.';
135
+ } else {
136
+ message = 'DNS records not found. Please add the required CNAME and TXT records.';
137
+ }
138
+ return NextResponse.json({
139
+ success: cnameValid && txtValid,
140
+ domain: customDomain,
141
+ projectSlug,
142
+ status: newStatus,
143
+ message,
144
+ checks: {
145
+ cname: {
146
+ found: checks.cname.found,
147
+ value: checks.cname.value,
148
+ valid: cnameValid,
149
+ expected: checks.cname.expected
150
+ },
151
+ txt: {
152
+ found: checks.txt.found,
153
+ verified: checks.txt.verified,
154
+ expected: checks.txt.expected
155
+ }
156
+ }
157
+ });
158
+ } catch (error) {
159
+ console.error('[Domains API] Error verifying domain:', error);
160
+ const message = error instanceof Error ? error.message : String(error);
161
+ return NextResponse.json({
162
+ error: 'Failed to verify domain',
163
+ details: message
164
+ }, {
165
+ status: 500
166
+ });
167
+ }
168
+ }
@@ -0,0 +1,71 @@
1
+ import { NextResponse } from 'next/server';
2
+ import { validateApiKey, regenerateApiKey, projectExists } from '@/lib/storage/blob';
3
+ /**
4
+ * POST /api/keys/regenerate
5
+ * Regenerate API key for a project
6
+ *
7
+ * Headers:
8
+ * Authorization: Bearer <current_api_key>
9
+ * Body:
10
+ * { slug: string }
11
+ */ export async function POST(request) {
12
+ try {
13
+ const body = await request.json();
14
+ const { slug } = body;
15
+ if (!slug || typeof slug !== 'string') {
16
+ return NextResponse.json({
17
+ error: 'Missing or invalid project slug'
18
+ }, {
19
+ status: 400
20
+ });
21
+ }
22
+ // Validate current API key
23
+ const authHeader = request.headers.get('Authorization');
24
+ const currentKey = authHeader?.replace('Bearer ', '');
25
+ if (!currentKey) {
26
+ return NextResponse.json({
27
+ error: 'API key required'
28
+ }, {
29
+ status: 401
30
+ });
31
+ }
32
+ // Check project exists
33
+ const exists = await projectExists(slug);
34
+ if (!exists) {
35
+ return NextResponse.json({
36
+ error: 'Project not found'
37
+ }, {
38
+ status: 404
39
+ });
40
+ }
41
+ // Validate the provided key belongs to this project
42
+ const validatedSlug = await validateApiKey(currentKey);
43
+ if (validatedSlug !== slug) {
44
+ return NextResponse.json({
45
+ error: 'Invalid API key for this project'
46
+ }, {
47
+ status: 403
48
+ });
49
+ }
50
+ // Regenerate the key
51
+ const newKey = await regenerateApiKey(slug, currentKey);
52
+ if (!newKey) {
53
+ return NextResponse.json({
54
+ error: 'Failed to regenerate API key'
55
+ }, {
56
+ status: 500
57
+ });
58
+ }
59
+ return NextResponse.json({
60
+ success: true,
61
+ apiKey: newKey
62
+ });
63
+ } catch (error) {
64
+ console.error('[Keys API] Error:', error);
65
+ return NextResponse.json({
66
+ error: 'Internal server error'
67
+ }, {
68
+ status: 500
69
+ });
70
+ }
71
+ }
@@ -0,0 +1,108 @@
1
+ import { NextResponse } from 'next/server';
2
+ import path from 'path';
3
+ import fs from 'fs';
4
+ /**
5
+ * Local asset server for devdoc dev mode
6
+ *
7
+ * Serves assets from the user's project folder (STARTER_PATH/assets/)
8
+ * This allows /assets/images/logo.png to work in local development
9
+ */ export async function GET(request, { params }) {
10
+ try {
11
+ const { path: pathSegments } = await params;
12
+ if (!pathSegments || pathSegments.length === 0) {
13
+ return NextResponse.json({
14
+ error: 'Invalid asset path'
15
+ }, {
16
+ status: 400
17
+ });
18
+ }
19
+ // Get the project path from environment
20
+ const projectPath = process.env.STARTER_PATH;
21
+ if (!projectPath) {
22
+ return NextResponse.json({
23
+ error: 'Not in local development mode'
24
+ }, {
25
+ status: 404
26
+ });
27
+ }
28
+ // Build file path - assets are in projectPath/assets/
29
+ const assetPath = pathSegments.join('/');
30
+ const filePath = path.join(projectPath, 'assets', assetPath);
31
+ // Security: ensure we're not escaping the assets directory
32
+ const resolvedPath = path.resolve(filePath);
33
+ const assetsDir = path.resolve(path.join(projectPath, 'assets'));
34
+ if (!resolvedPath.startsWith(assetsDir)) {
35
+ return NextResponse.json({
36
+ error: 'Invalid path'
37
+ }, {
38
+ status: 403
39
+ });
40
+ }
41
+ // Check if file exists
42
+ if (!fs.existsSync(resolvedPath)) {
43
+ // Also try without 'assets' prefix (for /assets/images/x.png -> images/x.png)
44
+ const altPath = path.join(projectPath, assetPath);
45
+ const resolvedAltPath = path.resolve(altPath);
46
+ if (resolvedAltPath.startsWith(path.resolve(projectPath)) && fs.existsSync(resolvedAltPath)) {
47
+ const fileBuffer = fs.readFileSync(resolvedAltPath);
48
+ const contentType = getContentType(assetPath);
49
+ return new NextResponse(fileBuffer, {
50
+ status: 200,
51
+ headers: {
52
+ 'Content-Type': contentType,
53
+ 'Cache-Control': 'no-cache'
54
+ }
55
+ });
56
+ }
57
+ return NextResponse.json({
58
+ error: 'Asset not found',
59
+ path: assetPath
60
+ }, {
61
+ status: 404
62
+ });
63
+ }
64
+ const fileBuffer = fs.readFileSync(resolvedPath);
65
+ const contentType = getContentType(assetPath);
66
+ return new NextResponse(fileBuffer, {
67
+ status: 200,
68
+ headers: {
69
+ 'Content-Type': contentType,
70
+ 'Cache-Control': 'no-cache'
71
+ }
72
+ });
73
+ } catch (error) {
74
+ console.error('[Local Assets] Error:', error);
75
+ return NextResponse.json({
76
+ error: 'Failed to serve asset'
77
+ }, {
78
+ status: 500
79
+ });
80
+ }
81
+ }
82
+ /**
83
+ * Get content type from file extension
84
+ */ function getContentType(fileName) {
85
+ const ext = fileName.split('.').pop()?.toLowerCase();
86
+ const types = {
87
+ 'jpg': 'image/jpeg',
88
+ 'jpeg': 'image/jpeg',
89
+ 'png': 'image/png',
90
+ 'gif': 'image/gif',
91
+ 'webp': 'image/webp',
92
+ 'svg': 'image/svg+xml',
93
+ 'ico': 'image/x-icon',
94
+ 'pdf': 'application/pdf',
95
+ 'mp4': 'video/mp4',
96
+ 'webm': 'video/webm',
97
+ 'mp3': 'audio/mpeg',
98
+ 'wav': 'audio/wav',
99
+ 'woff': 'font/woff',
100
+ 'woff2': 'font/woff2',
101
+ 'ttf': 'font/ttf',
102
+ 'otf': 'font/otf',
103
+ 'json': 'application/json',
104
+ 'css': 'text/css',
105
+ 'js': 'application/javascript'
106
+ };
107
+ return types[ext || ''] || 'application/octet-stream';
108
+ }
@@ -0,0 +1,73 @@
1
+ /**
2
+ * OpenAPI Spec API Route
3
+ *
4
+ * Returns the OpenAPI specification from local file
5
+ */ import { NextResponse } from 'next/server';
6
+ import { readFileSync, existsSync } from 'fs';
7
+ import { join, isAbsolute } from 'path';
8
+ const STARTER_PATH = process.env.STARTER_PATH || 'devdoc-docs';
9
+ const LOCAL_SPEC_PATH = process.env.LOCAL_SPEC_PATH || `${STARTER_PATH}/api-reference/openapi.json`;
10
+ // Helper to resolve paths
11
+ function resolvePath(...paths) {
12
+ if (isAbsolute(STARTER_PATH)) {
13
+ return join(STARTER_PATH, ...paths);
14
+ }
15
+ return join(process.cwd(), STARTER_PATH, ...paths);
16
+ }
17
+ // Load OpenAPI spec from local file
18
+ function loadLocalSpec() {
19
+ try {
20
+ const relativePath = LOCAL_SPEC_PATH;
21
+ const fullPath = relativePath.startsWith(STARTER_PATH) || isAbsolute(relativePath) ? isAbsolute(relativePath) ? relativePath : resolvePath(relativePath.replace(STARTER_PATH + '/', '')) : resolvePath(relativePath);
22
+ if (!existsSync(fullPath)) {
23
+ console.warn('[OpenAPI] Spec file not found:', fullPath);
24
+ return null;
25
+ }
26
+ const content = readFileSync(fullPath, 'utf-8');
27
+ return JSON.parse(content);
28
+ } catch (error) {
29
+ console.error('[OpenAPI] Error loading local spec:', error);
30
+ return null;
31
+ }
32
+ }
33
+ export async function GET() {
34
+ try {
35
+ const spec = loadLocalSpec();
36
+ if (!spec) {
37
+ return NextResponse.json({
38
+ openapi: '3.0.0',
39
+ info: {
40
+ title: 'API Documentation',
41
+ version: '1.0.0',
42
+ description: 'OpenAPI specification not found.'
43
+ },
44
+ paths: {}
45
+ }, {
46
+ headers: {
47
+ 'Content-Type': 'application/json',
48
+ 'Cache-Control': 'no-cache, no-store, must-revalidate'
49
+ }
50
+ });
51
+ }
52
+ return NextResponse.json(spec, {
53
+ headers: {
54
+ 'Content-Type': 'application/json',
55
+ 'Cache-Control': 'no-cache, no-store, must-revalidate'
56
+ }
57
+ });
58
+ } catch (error) {
59
+ console.error('[OpenAPI] Unexpected error:', error);
60
+ return NextResponse.json({
61
+ openapi: '3.0.0',
62
+ info: {
63
+ title: 'API Documentation',
64
+ version: '1.0.0',
65
+ description: 'Error loading specification.'
66
+ },
67
+ paths: {},
68
+ error: error instanceof Error ? error.message : 'Unknown error'
69
+ }, {
70
+ status: 500
71
+ });
72
+ }
73
+ }