@brainfish-ai/devdoc 0.1.41 → 0.1.43

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