@brainfish-ai/devdoc 0.1.21

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 (268) hide show
  1. package/LICENSE +33 -0
  2. package/README.md +415 -0
  3. package/bin/devdoc.js +13 -0
  4. package/dist/cli/commands/build.d.ts +5 -0
  5. package/dist/cli/commands/build.js +87 -0
  6. package/dist/cli/commands/check.d.ts +1 -0
  7. package/dist/cli/commands/check.js +143 -0
  8. package/dist/cli/commands/create.d.ts +24 -0
  9. package/dist/cli/commands/create.js +387 -0
  10. package/dist/cli/commands/deploy.d.ts +9 -0
  11. package/dist/cli/commands/deploy.js +433 -0
  12. package/dist/cli/commands/dev.d.ts +6 -0
  13. package/dist/cli/commands/dev.js +139 -0
  14. package/dist/cli/commands/init.d.ts +11 -0
  15. package/dist/cli/commands/init.js +238 -0
  16. package/dist/cli/commands/keys.d.ts +12 -0
  17. package/dist/cli/commands/keys.js +165 -0
  18. package/dist/cli/commands/start.d.ts +5 -0
  19. package/dist/cli/commands/start.js +56 -0
  20. package/dist/cli/commands/upload.d.ts +13 -0
  21. package/dist/cli/commands/upload.js +238 -0
  22. package/dist/cli/commands/whoami.d.ts +8 -0
  23. package/dist/cli/commands/whoami.js +91 -0
  24. package/dist/cli/index.d.ts +1 -0
  25. package/dist/cli/index.js +106 -0
  26. package/dist/config/index.d.ts +80 -0
  27. package/dist/config/index.js +133 -0
  28. package/dist/constants.d.ts +9 -0
  29. package/dist/constants.js +13 -0
  30. package/dist/index.d.ts +7 -0
  31. package/dist/index.js +12 -0
  32. package/dist/utils/logger.d.ts +16 -0
  33. package/dist/utils/logger.js +61 -0
  34. package/dist/utils/paths.d.ts +16 -0
  35. package/dist/utils/paths.js +50 -0
  36. package/package.json +51 -0
  37. package/renderer/app/api/assets/[...path]/route.ts +123 -0
  38. package/renderer/app/api/assets/route.ts +124 -0
  39. package/renderer/app/api/assets/upload/route.ts +177 -0
  40. package/renderer/app/api/auth-schemes/route.ts +77 -0
  41. package/renderer/app/api/chat/route.ts +858 -0
  42. package/renderer/app/api/codegen/route.ts +72 -0
  43. package/renderer/app/api/collections/route.ts +1016 -0
  44. package/renderer/app/api/debug/route.ts +53 -0
  45. package/renderer/app/api/deploy/route.ts +234 -0
  46. package/renderer/app/api/device/route.ts +42 -0
  47. package/renderer/app/api/docs/route.ts +187 -0
  48. package/renderer/app/api/keys/regenerate/route.ts +80 -0
  49. package/renderer/app/api/openapi-spec/route.ts +151 -0
  50. package/renderer/app/api/projects/[slug]/route.ts +153 -0
  51. package/renderer/app/api/projects/[slug]/stats/route.ts +96 -0
  52. package/renderer/app/api/projects/register/route.ts +152 -0
  53. package/renderer/app/api/proxy/route.ts +149 -0
  54. package/renderer/app/api/proxy-stream/route.ts +168 -0
  55. package/renderer/app/api/redirects/route.ts +47 -0
  56. package/renderer/app/api/schema/route.ts +65 -0
  57. package/renderer/app/api/subdomains/check/route.ts +172 -0
  58. package/renderer/app/api/suggestions/route.ts +144 -0
  59. package/renderer/app/favicon.ico +0 -0
  60. package/renderer/app/globals.css +1103 -0
  61. package/renderer/app/layout.tsx +47 -0
  62. package/renderer/app/llms-full.txt/route.ts +346 -0
  63. package/renderer/app/llms.txt/route.ts +279 -0
  64. package/renderer/app/page.tsx +14 -0
  65. package/renderer/app/robots.txt/route.ts +84 -0
  66. package/renderer/app/sitemap.xml/route.ts +199 -0
  67. package/renderer/components/docs/index.ts +12 -0
  68. package/renderer/components/docs/mdx/accordion.tsx +169 -0
  69. package/renderer/components/docs/mdx/badge.tsx +132 -0
  70. package/renderer/components/docs/mdx/callouts.tsx +154 -0
  71. package/renderer/components/docs/mdx/cards.tsx +213 -0
  72. package/renderer/components/docs/mdx/changelog.tsx +120 -0
  73. package/renderer/components/docs/mdx/code-block.tsx +186 -0
  74. package/renderer/components/docs/mdx/code-group.tsx +421 -0
  75. package/renderer/components/docs/mdx/file-embeds.tsx +105 -0
  76. package/renderer/components/docs/mdx/frame.tsx +112 -0
  77. package/renderer/components/docs/mdx/highlight.tsx +151 -0
  78. package/renderer/components/docs/mdx/iframe.tsx +134 -0
  79. package/renderer/components/docs/mdx/image.tsx +235 -0
  80. package/renderer/components/docs/mdx/index.ts +204 -0
  81. package/renderer/components/docs/mdx/mermaid.tsx +240 -0
  82. package/renderer/components/docs/mdx/param-field.tsx +200 -0
  83. package/renderer/components/docs/mdx/steps.tsx +113 -0
  84. package/renderer/components/docs/mdx/tabs.tsx +86 -0
  85. package/renderer/components/docs/mdx-renderer.tsx +100 -0
  86. package/renderer/components/docs/navigation/breadcrumbs.tsx +76 -0
  87. package/renderer/components/docs/navigation/index.ts +8 -0
  88. package/renderer/components/docs/navigation/page-nav.tsx +64 -0
  89. package/renderer/components/docs/navigation/sidebar.tsx +515 -0
  90. package/renderer/components/docs/navigation/toc.tsx +113 -0
  91. package/renderer/components/docs/notice.tsx +105 -0
  92. package/renderer/components/docs-header.tsx +274 -0
  93. package/renderer/components/docs-viewer/agent/agent-chat.tsx +2076 -0
  94. package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +90 -0
  95. package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +49 -0
  96. package/renderer/components/docs-viewer/agent/cards/index.tsx +50 -0
  97. package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +212 -0
  98. package/renderer/components/docs-viewer/agent/cards/types.ts +84 -0
  99. package/renderer/components/docs-viewer/agent/chat-message.tsx +17 -0
  100. package/renderer/components/docs-viewer/agent/index.tsx +6 -0
  101. package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +119 -0
  102. package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +46 -0
  103. package/renderer/components/docs-viewer/agent/messages/index.ts +17 -0
  104. package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +721 -0
  105. package/renderer/components/docs-viewer/agent/messages/types.ts +61 -0
  106. package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +24 -0
  107. package/renderer/components/docs-viewer/agent/messages/user-message.tsx +51 -0
  108. package/renderer/components/docs-viewer/code-editor/index.tsx +2 -0
  109. package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +1283 -0
  110. package/renderer/components/docs-viewer/content/changelog-page.tsx +331 -0
  111. package/renderer/components/docs-viewer/content/doc-page.tsx +285 -0
  112. package/renderer/components/docs-viewer/content/documentation-viewer.tsx +17 -0
  113. package/renderer/components/docs-viewer/content/index.tsx +29 -0
  114. package/renderer/components/docs-viewer/content/introduction.tsx +21 -0
  115. package/renderer/components/docs-viewer/content/request-details.tsx +330 -0
  116. package/renderer/components/docs-viewer/content/sections/auth.tsx +69 -0
  117. package/renderer/components/docs-viewer/content/sections/body.tsx +66 -0
  118. package/renderer/components/docs-viewer/content/sections/headers.tsx +43 -0
  119. package/renderer/components/docs-viewer/content/sections/overview.tsx +40 -0
  120. package/renderer/components/docs-viewer/content/sections/parameters.tsx +43 -0
  121. package/renderer/components/docs-viewer/content/sections/responses.tsx +87 -0
  122. package/renderer/components/docs-viewer/global-auth-modal.tsx +352 -0
  123. package/renderer/components/docs-viewer/index.tsx +1466 -0
  124. package/renderer/components/docs-viewer/playground/auth-editor.tsx +280 -0
  125. package/renderer/components/docs-viewer/playground/body-editor.tsx +221 -0
  126. package/renderer/components/docs-viewer/playground/code-editor.tsx +224 -0
  127. package/renderer/components/docs-viewer/playground/code-snippet.tsx +387 -0
  128. package/renderer/components/docs-viewer/playground/graphql-playground.tsx +745 -0
  129. package/renderer/components/docs-viewer/playground/index.tsx +671 -0
  130. package/renderer/components/docs-viewer/playground/key-value-editor.tsx +261 -0
  131. package/renderer/components/docs-viewer/playground/method-selector.tsx +60 -0
  132. package/renderer/components/docs-viewer/playground/request-builder.tsx +179 -0
  133. package/renderer/components/docs-viewer/playground/request-tabs.tsx +237 -0
  134. package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +21 -0
  135. package/renderer/components/docs-viewer/playground/response-cards/index.tsx +93 -0
  136. package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +16 -0
  137. package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +23 -0
  138. package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +268 -0
  139. package/renderer/components/docs-viewer/playground/response-cards/types.ts +82 -0
  140. package/renderer/components/docs-viewer/playground/response-viewer.tsx +43 -0
  141. package/renderer/components/docs-viewer/search/index.ts +2 -0
  142. package/renderer/components/docs-viewer/search/search-dialog.tsx +331 -0
  143. package/renderer/components/docs-viewer/search/use-search.ts +117 -0
  144. package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +431 -0
  145. package/renderer/components/docs-viewer/shared/method-badge.tsx +41 -0
  146. package/renderer/components/docs-viewer/shared/schema-viewer.tsx +349 -0
  147. package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +239 -0
  148. package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +316 -0
  149. package/renderer/components/docs-viewer/sidebar/index.tsx +343 -0
  150. package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +202 -0
  151. package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +118 -0
  152. package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +226 -0
  153. package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +52 -0
  154. package/renderer/components/theme-provider.tsx +11 -0
  155. package/renderer/components/theme-toggle.tsx +76 -0
  156. package/renderer/components/ui/badge.tsx +46 -0
  157. package/renderer/components/ui/button.tsx +59 -0
  158. package/renderer/components/ui/dialog.tsx +118 -0
  159. package/renderer/components/ui/dropdown-menu.tsx +257 -0
  160. package/renderer/components/ui/input.tsx +21 -0
  161. package/renderer/components/ui/label.tsx +24 -0
  162. package/renderer/components/ui/navigation-menu.tsx +168 -0
  163. package/renderer/components/ui/select.tsx +190 -0
  164. package/renderer/components/ui/spinner.tsx +114 -0
  165. package/renderer/components/ui/tabs.tsx +66 -0
  166. package/renderer/components/ui/tooltip.tsx +61 -0
  167. package/renderer/hooks/use-code-copy.ts +88 -0
  168. package/renderer/hooks/use-openapi-title.ts +44 -0
  169. package/renderer/lib/api-docs/agent/index.ts +6 -0
  170. package/renderer/lib/api-docs/agent/indexer.ts +323 -0
  171. package/renderer/lib/api-docs/agent/spec-summary.ts +335 -0
  172. package/renderer/lib/api-docs/agent/types.ts +116 -0
  173. package/renderer/lib/api-docs/auth/auth-context.tsx +225 -0
  174. package/renderer/lib/api-docs/auth/auth-storage.ts +87 -0
  175. package/renderer/lib/api-docs/auth/crypto.ts +89 -0
  176. package/renderer/lib/api-docs/auth/index.ts +4 -0
  177. package/renderer/lib/api-docs/code-editor/db.ts +164 -0
  178. package/renderer/lib/api-docs/code-editor/hooks.ts +266 -0
  179. package/renderer/lib/api-docs/code-editor/index.ts +6 -0
  180. package/renderer/lib/api-docs/code-editor/mode-context.tsx +207 -0
  181. package/renderer/lib/api-docs/code-editor/types.ts +105 -0
  182. package/renderer/lib/api-docs/codegen/definitions.ts +297 -0
  183. package/renderer/lib/api-docs/codegen/har.ts +251 -0
  184. package/renderer/lib/api-docs/codegen/index.ts +159 -0
  185. package/renderer/lib/api-docs/factories.ts +151 -0
  186. package/renderer/lib/api-docs/index.ts +17 -0
  187. package/renderer/lib/api-docs/mobile-context.tsx +112 -0
  188. package/renderer/lib/api-docs/navigation-context.tsx +88 -0
  189. package/renderer/lib/api-docs/parsers/graphql/README.md +129 -0
  190. package/renderer/lib/api-docs/parsers/graphql/index.ts +91 -0
  191. package/renderer/lib/api-docs/parsers/graphql/parser.ts +491 -0
  192. package/renderer/lib/api-docs/parsers/graphql/transformer.ts +246 -0
  193. package/renderer/lib/api-docs/parsers/graphql/types.ts +283 -0
  194. package/renderer/lib/api-docs/parsers/openapi/README.md +32 -0
  195. package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +60 -0
  196. package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +574 -0
  197. package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +403 -0
  198. package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +232 -0
  199. package/renderer/lib/api-docs/parsers/openapi/index.ts +171 -0
  200. package/renderer/lib/api-docs/parsers/openapi/transformer.ts +277 -0
  201. package/renderer/lib/api-docs/parsers/openapi/validator.ts +31 -0
  202. package/renderer/lib/api-docs/playground/context.tsx +107 -0
  203. package/renderer/lib/api-docs/playground/navigation-context.tsx +124 -0
  204. package/renderer/lib/api-docs/playground/request-builder.ts +223 -0
  205. package/renderer/lib/api-docs/playground/request-runner.ts +282 -0
  206. package/renderer/lib/api-docs/playground/types.ts +35 -0
  207. package/renderer/lib/api-docs/types.ts +269 -0
  208. package/renderer/lib/api-docs/utils.ts +311 -0
  209. package/renderer/lib/cache.ts +193 -0
  210. package/renderer/lib/docs/config/index.ts +29 -0
  211. package/renderer/lib/docs/config/loader.ts +142 -0
  212. package/renderer/lib/docs/config/schema.ts +298 -0
  213. package/renderer/lib/docs/index.ts +12 -0
  214. package/renderer/lib/docs/mdx/compiler.ts +176 -0
  215. package/renderer/lib/docs/mdx/frontmatter.ts +80 -0
  216. package/renderer/lib/docs/mdx/index.ts +26 -0
  217. package/renderer/lib/docs/navigation/generator.ts +348 -0
  218. package/renderer/lib/docs/navigation/index.ts +12 -0
  219. package/renderer/lib/docs/navigation/types.ts +123 -0
  220. package/renderer/lib/docs-navigation-context.tsx +80 -0
  221. package/renderer/lib/multi-tenant/context.ts +105 -0
  222. package/renderer/lib/storage/blob.ts +845 -0
  223. package/renderer/lib/utils.ts +6 -0
  224. package/renderer/next.config.ts +76 -0
  225. package/renderer/package.json +66 -0
  226. package/renderer/postcss.config.mjs +5 -0
  227. package/renderer/public/assets/images/screenshot.png +0 -0
  228. package/renderer/public/assets/logo/dark.svg +9 -0
  229. package/renderer/public/assets/logo/light.svg +9 -0
  230. package/renderer/public/assets/logo.svg +9 -0
  231. package/renderer/public/file.svg +1 -0
  232. package/renderer/public/globe.svg +1 -0
  233. package/renderer/public/icon.png +0 -0
  234. package/renderer/public/logo.svg +9 -0
  235. package/renderer/public/window.svg +1 -0
  236. package/renderer/tsconfig.json +28 -0
  237. package/templates/basic/README.md +139 -0
  238. package/templates/basic/assets/favicon.svg +4 -0
  239. package/templates/basic/assets/logo.svg +9 -0
  240. package/templates/basic/docs.json +47 -0
  241. package/templates/basic/guides/configuration.mdx +149 -0
  242. package/templates/basic/guides/overview.mdx +96 -0
  243. package/templates/basic/index.mdx +39 -0
  244. package/templates/basic/package.json +14 -0
  245. package/templates/basic/quickstart.mdx +92 -0
  246. package/templates/basic/vercel.json +6 -0
  247. package/templates/graphql/README.md +139 -0
  248. package/templates/graphql/api-reference/schema.graphql +305 -0
  249. package/templates/graphql/assets/favicon.svg +4 -0
  250. package/templates/graphql/assets/logo.svg +9 -0
  251. package/templates/graphql/docs.json +54 -0
  252. package/templates/graphql/guides/configuration.mdx +149 -0
  253. package/templates/graphql/guides/overview.mdx +96 -0
  254. package/templates/graphql/index.mdx +39 -0
  255. package/templates/graphql/package.json +14 -0
  256. package/templates/graphql/quickstart.mdx +92 -0
  257. package/templates/graphql/vercel.json +6 -0
  258. package/templates/openapi/README.md +139 -0
  259. package/templates/openapi/api-reference/openapi.json +419 -0
  260. package/templates/openapi/assets/favicon.svg +4 -0
  261. package/templates/openapi/assets/logo.svg +9 -0
  262. package/templates/openapi/docs.json +61 -0
  263. package/templates/openapi/guides/configuration.mdx +149 -0
  264. package/templates/openapi/guides/overview.mdx +96 -0
  265. package/templates/openapi/index.mdx +39 -0
  266. package/templates/openapi/package.json +14 -0
  267. package/templates/openapi/quickstart.mdx +92 -0
  268. package/templates/openapi/vercel.json +6 -0
@@ -0,0 +1,124 @@
1
+ import { NextRequest, NextResponse } from 'next/server'
2
+ import { validateApiKey, listProjectAssets, deleteProjectAsset } from '@/lib/storage/blob'
3
+
4
+ /**
5
+ * List assets API
6
+ *
7
+ * GET /api/assets?slug=<project-slug>
8
+ * Headers:
9
+ * Authorization: Bearer <api_key>
10
+ */
11
+ export async function GET(request: NextRequest) {
12
+ try {
13
+ const { searchParams } = new URL(request.url)
14
+ const slug = searchParams.get('slug')
15
+
16
+ if (!slug) {
17
+ return NextResponse.json(
18
+ { error: 'Missing slug parameter' },
19
+ { status: 400 }
20
+ )
21
+ }
22
+
23
+ // Validate API key
24
+ const authHeader = request.headers.get('Authorization')
25
+ const apiKey = authHeader?.replace('Bearer ', '')
26
+
27
+ if (!apiKey) {
28
+ return NextResponse.json(
29
+ { error: 'API key required' },
30
+ { status: 401 }
31
+ )
32
+ }
33
+
34
+ const validatedSlug = await validateApiKey(apiKey)
35
+ if (validatedSlug !== slug) {
36
+ return NextResponse.json(
37
+ { error: 'Invalid API key for this project' },
38
+ { status: 403 }
39
+ )
40
+ }
41
+
42
+ const assets = await listProjectAssets(slug)
43
+
44
+ return NextResponse.json({
45
+ success: true,
46
+ slug,
47
+ assets,
48
+ count: assets.length,
49
+ })
50
+
51
+ } catch (error) {
52
+ console.error('[Assets API] Error:', error)
53
+ const message = error instanceof Error ? error.message : String(error)
54
+
55
+ return NextResponse.json(
56
+ { error: 'Failed to list assets', details: message },
57
+ { status: 500 }
58
+ )
59
+ }
60
+ }
61
+
62
+ /**
63
+ * Delete asset API
64
+ *
65
+ * DELETE /api/assets?slug=<project-slug>&fileName=<file-name>
66
+ * Headers:
67
+ * Authorization: Bearer <api_key>
68
+ */
69
+ export async function DELETE(request: NextRequest) {
70
+ try {
71
+ const { searchParams } = new URL(request.url)
72
+ const slug = searchParams.get('slug')
73
+ const fileName = searchParams.get('fileName')
74
+
75
+ if (!slug || !fileName) {
76
+ return NextResponse.json(
77
+ { error: 'Missing slug or fileName parameter' },
78
+ { status: 400 }
79
+ )
80
+ }
81
+
82
+ // Validate API key
83
+ const authHeader = request.headers.get('Authorization')
84
+ const apiKey = authHeader?.replace('Bearer ', '')
85
+
86
+ if (!apiKey) {
87
+ return NextResponse.json(
88
+ { error: 'API key required' },
89
+ { status: 401 }
90
+ )
91
+ }
92
+
93
+ const validatedSlug = await validateApiKey(apiKey)
94
+ if (validatedSlug !== slug) {
95
+ return NextResponse.json(
96
+ { error: 'Invalid API key for this project' },
97
+ { status: 403 }
98
+ )
99
+ }
100
+
101
+ const deleted = await deleteProjectAsset(slug, fileName)
102
+
103
+ if (!deleted) {
104
+ return NextResponse.json(
105
+ { error: 'Asset not found' },
106
+ { status: 404 }
107
+ )
108
+ }
109
+
110
+ return NextResponse.json({
111
+ success: true,
112
+ message: `Asset ${fileName} deleted`,
113
+ })
114
+
115
+ } catch (error) {
116
+ console.error('[Assets API] Error:', error)
117
+ const message = error instanceof Error ? error.message : String(error)
118
+
119
+ return NextResponse.json(
120
+ { error: 'Failed to delete asset', details: message },
121
+ { status: 500 }
122
+ )
123
+ }
124
+ }
@@ -0,0 +1,177 @@
1
+ import { NextRequest, NextResponse } from 'next/server'
2
+ import { put } from '@vercel/blob'
3
+ import { validateApiKey } from '@/lib/storage/blob'
4
+ import path from 'path'
5
+ import fs from 'fs'
6
+
7
+ // Max file size: 25MB
8
+ const MAX_FILE_SIZE = 25 * 1024 * 1024
9
+
10
+ // Allowed file types for assets
11
+ const ALLOWED_TYPES = [
12
+ 'image/jpeg',
13
+ 'image/png',
14
+ 'image/gif',
15
+ 'image/webp',
16
+ 'image/svg+xml',
17
+ 'image/x-icon',
18
+ 'image/vnd.microsoft.icon',
19
+ 'application/pdf',
20
+ 'video/mp4',
21
+ 'video/webm',
22
+ 'audio/mpeg',
23
+ 'audio/wav',
24
+ 'font/woff',
25
+ 'font/woff2',
26
+ 'font/ttf',
27
+ 'font/otf',
28
+ 'application/font-woff',
29
+ 'application/font-woff2',
30
+ ]
31
+
32
+ // Check if we're in local development mode
33
+ const IS_LOCAL_DEV = !process.env.BLOB_READ_WRITE_TOKEN
34
+ const LOCAL_STORAGE_DIR = path.join(process.cwd(), '.devdoc-storage')
35
+
36
+ /**
37
+ * Upload asset API - receives files from CLI and stores in Vercel Blob
38
+ *
39
+ * POST /api/assets/upload
40
+ * Headers:
41
+ * Authorization: Bearer <api_key>
42
+ * Content-Type: multipart/form-data
43
+ * Body: FormData with:
44
+ * - file: File to upload
45
+ * - slug: Project slug
46
+ * - path: Optional path/filename for the asset
47
+ */
48
+ export async function POST(request: NextRequest) {
49
+ try {
50
+ // Validate API key
51
+ const authHeader = request.headers.get('Authorization')
52
+ const apiKey = authHeader?.replace('Bearer ', '')
53
+
54
+ if (!apiKey) {
55
+ return NextResponse.json(
56
+ { error: 'API key required. Provide via Authorization header.' },
57
+ { status: 401 }
58
+ )
59
+ }
60
+
61
+ const validatedSlug = await validateApiKey(apiKey)
62
+ if (!validatedSlug) {
63
+ return NextResponse.json(
64
+ { error: 'Invalid API key' },
65
+ { status: 403 }
66
+ )
67
+ }
68
+
69
+ // Parse multipart form data
70
+ const formData = await request.formData()
71
+ const file = formData.get('file') as File | null
72
+ const slug = formData.get('slug') as string | null
73
+ const assetPath = formData.get('path') as string | null
74
+
75
+ if (!file) {
76
+ return NextResponse.json(
77
+ { error: 'No file provided' },
78
+ { status: 400 }
79
+ )
80
+ }
81
+
82
+ if (!slug) {
83
+ return NextResponse.json(
84
+ { error: 'Project slug is required' },
85
+ { status: 400 }
86
+ )
87
+ }
88
+
89
+ // Verify the API key matches the project
90
+ if (validatedSlug !== slug) {
91
+ return NextResponse.json(
92
+ { error: 'API key does not match project' },
93
+ { status: 403 }
94
+ )
95
+ }
96
+
97
+ // Validate file size
98
+ if (file.size > MAX_FILE_SIZE) {
99
+ return NextResponse.json(
100
+ { error: `File size exceeds maximum allowed size of 25MB. Current size: ${(file.size / (1024 * 1024)).toFixed(2)}MB` },
101
+ { status: 400 }
102
+ )
103
+ }
104
+
105
+ // Validate file type
106
+ const contentType = file.type
107
+ if (!ALLOWED_TYPES.includes(contentType) && !contentType.startsWith('image/')) {
108
+ return NextResponse.json(
109
+ { error: `File type "${contentType}" is not allowed. Allowed types: images, PDFs, videos, audio, and fonts.` },
110
+ { status: 400 }
111
+ )
112
+ }
113
+
114
+ // Generate asset path
115
+ const fileName = assetPath || file.name
116
+ const sanitizedFileName = fileName.replace(/[^a-zA-Z0-9._-]/g, '_')
117
+ const blobPath = `projects/${slug}/assets/${sanitizedFileName}`
118
+
119
+ // Get file buffer
120
+ const buffer = Buffer.from(await file.arrayBuffer())
121
+
122
+ // Store in local filesystem or Vercel Blob
123
+ let url: string
124
+
125
+ if (IS_LOCAL_DEV) {
126
+ // Local development - store in filesystem
127
+ const assetDir = path.join(LOCAL_STORAGE_DIR, 'projects', slug, 'assets')
128
+ fs.mkdirSync(assetDir, { recursive: true })
129
+
130
+ const filePath = path.join(assetDir, sanitizedFileName)
131
+ fs.writeFileSync(filePath, buffer)
132
+
133
+ url = `file://${filePath}`
134
+ } else {
135
+ // Production - store in Vercel Blob
136
+ const blob = await put(blobPath, buffer, {
137
+ access: 'public',
138
+ contentType,
139
+ allowOverwrite: true,
140
+ })
141
+
142
+ url = blob.url
143
+ }
144
+
145
+ return NextResponse.json({
146
+ success: true,
147
+ url,
148
+ path: blobPath,
149
+ fileName: sanitizedFileName,
150
+ size: file.size,
151
+ contentType,
152
+ })
153
+
154
+ } catch (error) {
155
+ console.error('[Asset Upload API] Error:', error)
156
+ const message = error instanceof Error ? error.message : String(error)
157
+
158
+ return NextResponse.json(
159
+ { error: 'Upload failed', details: message },
160
+ { status: 500 }
161
+ )
162
+ }
163
+ }
164
+
165
+ /**
166
+ * OPTIONS for CORS preflight
167
+ */
168
+ export async function OPTIONS() {
169
+ return new NextResponse(null, {
170
+ status: 200,
171
+ headers: {
172
+ 'Access-Control-Allow-Origin': '*',
173
+ 'Access-Control-Allow-Methods': 'POST, OPTIONS',
174
+ 'Access-Control-Allow-Headers': 'Content-Type, Authorization',
175
+ },
176
+ })
177
+ }
@@ -0,0 +1,77 @@
1
+ /**
2
+ * Auth Schemes API Route
3
+ *
4
+ * Returns the security schemes defined in the OpenAPI spec
5
+ * so the frontend can suggest the correct auth type
6
+ */
7
+
8
+ import { NextResponse } from 'next/server'
9
+ import { extractSecuritySchemes, getDefaultAuth } from '@/lib/api-docs/parsers/openapi/extractors/auth'
10
+ import type { OpenAPI } from 'openapi-types'
11
+
12
+ // Import the same fetch logic as the openapi-spec route
13
+ import { CacheUtils } from '@/lib/cache'
14
+
15
+ const BRAINFISH_API_BASE_URL = process.env.BRAINFISH_API_BASE_URL || 'https://api.brainfish.ai/api'
16
+ const BRAINFISH_CATALOG_ID = process.env.BRAINFISH_CATALOG_ID || 'your_catalog_id_here'
17
+ const BRAINFISH_JWT_TOKEN = process.env.BRAINFISH_JWT_TOKEN || 'your_jwt_token_here'
18
+ const CACHE_KEY = `openapi-spec-${BRAINFISH_CATALOG_ID}`
19
+
20
+ // Fetch the OpenAPI spec directly (same as openapi-spec route)
21
+ async function fetchOpenApiSpec(): Promise<OpenAPI.Document | null> {
22
+ try {
23
+ // First try cache
24
+ const cachedSpec = await CacheUtils.get<OpenAPI.Document>(CACHE_KEY)
25
+ if (cachedSpec) {
26
+ return cachedSpec
27
+ }
28
+
29
+ // Fetch from API
30
+ const url = `${BRAINFISH_API_BASE_URL}/catalogs.openapi-spec`
31
+ const response = await fetch(url, {
32
+ method: 'POST',
33
+ headers: {
34
+ 'Authorization': `Bearer ${BRAINFISH_JWT_TOKEN}`,
35
+ 'Accept': 'application/json',
36
+ 'Content-Type': 'application/json',
37
+ },
38
+ body: JSON.stringify({ catalogId: BRAINFISH_CATALOG_ID }),
39
+ cache: 'no-store',
40
+ })
41
+
42
+ if (!response.ok) {
43
+ return null
44
+ }
45
+
46
+ return await response.json()
47
+ } catch {
48
+ return null
49
+ }
50
+ }
51
+
52
+ export async function GET() {
53
+ try {
54
+ const spec = await fetchOpenApiSpec()
55
+
56
+ if (!spec) {
57
+ return NextResponse.json({
58
+ schemes: [],
59
+ defaultAuth: null,
60
+ })
61
+ }
62
+
63
+ const schemes = extractSecuritySchemes(spec)
64
+ const defaultAuth = getDefaultAuth(spec)
65
+
66
+ return NextResponse.json({
67
+ schemes,
68
+ defaultAuth,
69
+ })
70
+ } catch (error) {
71
+ return NextResponse.json({
72
+ schemes: [],
73
+ defaultAuth: null,
74
+ error: error instanceof Error ? error.message : 'Unknown error',
75
+ }, { status: 500 })
76
+ }
77
+ }