@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.
- package/LICENSE +33 -0
- package/README.md +415 -0
- package/bin/devdoc.js +13 -0
- package/dist/cli/commands/build.d.ts +5 -0
- package/dist/cli/commands/build.js +87 -0
- package/dist/cli/commands/check.d.ts +1 -0
- package/dist/cli/commands/check.js +143 -0
- package/dist/cli/commands/create.d.ts +24 -0
- package/dist/cli/commands/create.js +387 -0
- package/dist/cli/commands/deploy.d.ts +9 -0
- package/dist/cli/commands/deploy.js +433 -0
- package/dist/cli/commands/dev.d.ts +6 -0
- package/dist/cli/commands/dev.js +139 -0
- package/dist/cli/commands/init.d.ts +11 -0
- package/dist/cli/commands/init.js +238 -0
- package/dist/cli/commands/keys.d.ts +12 -0
- package/dist/cli/commands/keys.js +165 -0
- package/dist/cli/commands/start.d.ts +5 -0
- package/dist/cli/commands/start.js +56 -0
- package/dist/cli/commands/upload.d.ts +13 -0
- package/dist/cli/commands/upload.js +238 -0
- package/dist/cli/commands/whoami.d.ts +8 -0
- package/dist/cli/commands/whoami.js +91 -0
- package/dist/cli/index.d.ts +1 -0
- package/dist/cli/index.js +106 -0
- package/dist/config/index.d.ts +80 -0
- package/dist/config/index.js +133 -0
- package/dist/constants.d.ts +9 -0
- package/dist/constants.js +13 -0
- package/dist/index.d.ts +7 -0
- package/dist/index.js +12 -0
- package/dist/utils/logger.d.ts +16 -0
- package/dist/utils/logger.js +61 -0
- package/dist/utils/paths.d.ts +16 -0
- package/dist/utils/paths.js +50 -0
- package/package.json +51 -0
- package/renderer/app/api/assets/[...path]/route.ts +123 -0
- package/renderer/app/api/assets/route.ts +124 -0
- package/renderer/app/api/assets/upload/route.ts +177 -0
- package/renderer/app/api/auth-schemes/route.ts +77 -0
- package/renderer/app/api/chat/route.ts +858 -0
- package/renderer/app/api/codegen/route.ts +72 -0
- package/renderer/app/api/collections/route.ts +1016 -0
- package/renderer/app/api/debug/route.ts +53 -0
- package/renderer/app/api/deploy/route.ts +234 -0
- package/renderer/app/api/device/route.ts +42 -0
- package/renderer/app/api/docs/route.ts +187 -0
- package/renderer/app/api/keys/regenerate/route.ts +80 -0
- package/renderer/app/api/openapi-spec/route.ts +151 -0
- package/renderer/app/api/projects/[slug]/route.ts +153 -0
- package/renderer/app/api/projects/[slug]/stats/route.ts +96 -0
- package/renderer/app/api/projects/register/route.ts +152 -0
- package/renderer/app/api/proxy/route.ts +149 -0
- package/renderer/app/api/proxy-stream/route.ts +168 -0
- package/renderer/app/api/redirects/route.ts +47 -0
- package/renderer/app/api/schema/route.ts +65 -0
- package/renderer/app/api/subdomains/check/route.ts +172 -0
- package/renderer/app/api/suggestions/route.ts +144 -0
- package/renderer/app/favicon.ico +0 -0
- package/renderer/app/globals.css +1103 -0
- package/renderer/app/layout.tsx +47 -0
- package/renderer/app/llms-full.txt/route.ts +346 -0
- package/renderer/app/llms.txt/route.ts +279 -0
- package/renderer/app/page.tsx +14 -0
- package/renderer/app/robots.txt/route.ts +84 -0
- package/renderer/app/sitemap.xml/route.ts +199 -0
- package/renderer/components/docs/index.ts +12 -0
- package/renderer/components/docs/mdx/accordion.tsx +169 -0
- package/renderer/components/docs/mdx/badge.tsx +132 -0
- package/renderer/components/docs/mdx/callouts.tsx +154 -0
- package/renderer/components/docs/mdx/cards.tsx +213 -0
- package/renderer/components/docs/mdx/changelog.tsx +120 -0
- package/renderer/components/docs/mdx/code-block.tsx +186 -0
- package/renderer/components/docs/mdx/code-group.tsx +421 -0
- package/renderer/components/docs/mdx/file-embeds.tsx +105 -0
- package/renderer/components/docs/mdx/frame.tsx +112 -0
- package/renderer/components/docs/mdx/highlight.tsx +151 -0
- package/renderer/components/docs/mdx/iframe.tsx +134 -0
- package/renderer/components/docs/mdx/image.tsx +235 -0
- package/renderer/components/docs/mdx/index.ts +204 -0
- package/renderer/components/docs/mdx/mermaid.tsx +240 -0
- package/renderer/components/docs/mdx/param-field.tsx +200 -0
- package/renderer/components/docs/mdx/steps.tsx +113 -0
- package/renderer/components/docs/mdx/tabs.tsx +86 -0
- package/renderer/components/docs/mdx-renderer.tsx +100 -0
- package/renderer/components/docs/navigation/breadcrumbs.tsx +76 -0
- package/renderer/components/docs/navigation/index.ts +8 -0
- package/renderer/components/docs/navigation/page-nav.tsx +64 -0
- package/renderer/components/docs/navigation/sidebar.tsx +515 -0
- package/renderer/components/docs/navigation/toc.tsx +113 -0
- package/renderer/components/docs/notice.tsx +105 -0
- package/renderer/components/docs-header.tsx +274 -0
- package/renderer/components/docs-viewer/agent/agent-chat.tsx +2076 -0
- package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +90 -0
- package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +49 -0
- package/renderer/components/docs-viewer/agent/cards/index.tsx +50 -0
- package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +212 -0
- package/renderer/components/docs-viewer/agent/cards/types.ts +84 -0
- package/renderer/components/docs-viewer/agent/chat-message.tsx +17 -0
- package/renderer/components/docs-viewer/agent/index.tsx +6 -0
- package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +119 -0
- package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +46 -0
- package/renderer/components/docs-viewer/agent/messages/index.ts +17 -0
- package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +721 -0
- package/renderer/components/docs-viewer/agent/messages/types.ts +61 -0
- package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +24 -0
- package/renderer/components/docs-viewer/agent/messages/user-message.tsx +51 -0
- package/renderer/components/docs-viewer/code-editor/index.tsx +2 -0
- package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +1283 -0
- package/renderer/components/docs-viewer/content/changelog-page.tsx +331 -0
- package/renderer/components/docs-viewer/content/doc-page.tsx +285 -0
- package/renderer/components/docs-viewer/content/documentation-viewer.tsx +17 -0
- package/renderer/components/docs-viewer/content/index.tsx +29 -0
- package/renderer/components/docs-viewer/content/introduction.tsx +21 -0
- package/renderer/components/docs-viewer/content/request-details.tsx +330 -0
- package/renderer/components/docs-viewer/content/sections/auth.tsx +69 -0
- package/renderer/components/docs-viewer/content/sections/body.tsx +66 -0
- package/renderer/components/docs-viewer/content/sections/headers.tsx +43 -0
- package/renderer/components/docs-viewer/content/sections/overview.tsx +40 -0
- package/renderer/components/docs-viewer/content/sections/parameters.tsx +43 -0
- package/renderer/components/docs-viewer/content/sections/responses.tsx +87 -0
- package/renderer/components/docs-viewer/global-auth-modal.tsx +352 -0
- package/renderer/components/docs-viewer/index.tsx +1466 -0
- package/renderer/components/docs-viewer/playground/auth-editor.tsx +280 -0
- package/renderer/components/docs-viewer/playground/body-editor.tsx +221 -0
- package/renderer/components/docs-viewer/playground/code-editor.tsx +224 -0
- package/renderer/components/docs-viewer/playground/code-snippet.tsx +387 -0
- package/renderer/components/docs-viewer/playground/graphql-playground.tsx +745 -0
- package/renderer/components/docs-viewer/playground/index.tsx +671 -0
- package/renderer/components/docs-viewer/playground/key-value-editor.tsx +261 -0
- package/renderer/components/docs-viewer/playground/method-selector.tsx +60 -0
- package/renderer/components/docs-viewer/playground/request-builder.tsx +179 -0
- package/renderer/components/docs-viewer/playground/request-tabs.tsx +237 -0
- package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +21 -0
- package/renderer/components/docs-viewer/playground/response-cards/index.tsx +93 -0
- package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +16 -0
- package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +23 -0
- package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +268 -0
- package/renderer/components/docs-viewer/playground/response-cards/types.ts +82 -0
- package/renderer/components/docs-viewer/playground/response-viewer.tsx +43 -0
- package/renderer/components/docs-viewer/search/index.ts +2 -0
- package/renderer/components/docs-viewer/search/search-dialog.tsx +331 -0
- package/renderer/components/docs-viewer/search/use-search.ts +117 -0
- package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +431 -0
- package/renderer/components/docs-viewer/shared/method-badge.tsx +41 -0
- package/renderer/components/docs-viewer/shared/schema-viewer.tsx +349 -0
- package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +239 -0
- package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +316 -0
- package/renderer/components/docs-viewer/sidebar/index.tsx +343 -0
- package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +202 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +118 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +226 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +52 -0
- package/renderer/components/theme-provider.tsx +11 -0
- package/renderer/components/theme-toggle.tsx +76 -0
- package/renderer/components/ui/badge.tsx +46 -0
- package/renderer/components/ui/button.tsx +59 -0
- package/renderer/components/ui/dialog.tsx +118 -0
- package/renderer/components/ui/dropdown-menu.tsx +257 -0
- package/renderer/components/ui/input.tsx +21 -0
- package/renderer/components/ui/label.tsx +24 -0
- package/renderer/components/ui/navigation-menu.tsx +168 -0
- package/renderer/components/ui/select.tsx +190 -0
- package/renderer/components/ui/spinner.tsx +114 -0
- package/renderer/components/ui/tabs.tsx +66 -0
- package/renderer/components/ui/tooltip.tsx +61 -0
- package/renderer/hooks/use-code-copy.ts +88 -0
- package/renderer/hooks/use-openapi-title.ts +44 -0
- package/renderer/lib/api-docs/agent/index.ts +6 -0
- package/renderer/lib/api-docs/agent/indexer.ts +323 -0
- package/renderer/lib/api-docs/agent/spec-summary.ts +335 -0
- package/renderer/lib/api-docs/agent/types.ts +116 -0
- package/renderer/lib/api-docs/auth/auth-context.tsx +225 -0
- package/renderer/lib/api-docs/auth/auth-storage.ts +87 -0
- package/renderer/lib/api-docs/auth/crypto.ts +89 -0
- package/renderer/lib/api-docs/auth/index.ts +4 -0
- package/renderer/lib/api-docs/code-editor/db.ts +164 -0
- package/renderer/lib/api-docs/code-editor/hooks.ts +266 -0
- package/renderer/lib/api-docs/code-editor/index.ts +6 -0
- package/renderer/lib/api-docs/code-editor/mode-context.tsx +207 -0
- package/renderer/lib/api-docs/code-editor/types.ts +105 -0
- package/renderer/lib/api-docs/codegen/definitions.ts +297 -0
- package/renderer/lib/api-docs/codegen/har.ts +251 -0
- package/renderer/lib/api-docs/codegen/index.ts +159 -0
- package/renderer/lib/api-docs/factories.ts +151 -0
- package/renderer/lib/api-docs/index.ts +17 -0
- package/renderer/lib/api-docs/mobile-context.tsx +112 -0
- package/renderer/lib/api-docs/navigation-context.tsx +88 -0
- package/renderer/lib/api-docs/parsers/graphql/README.md +129 -0
- package/renderer/lib/api-docs/parsers/graphql/index.ts +91 -0
- package/renderer/lib/api-docs/parsers/graphql/parser.ts +491 -0
- package/renderer/lib/api-docs/parsers/graphql/transformer.ts +246 -0
- package/renderer/lib/api-docs/parsers/graphql/types.ts +283 -0
- package/renderer/lib/api-docs/parsers/openapi/README.md +32 -0
- package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +60 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +574 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +403 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +232 -0
- package/renderer/lib/api-docs/parsers/openapi/index.ts +171 -0
- package/renderer/lib/api-docs/parsers/openapi/transformer.ts +277 -0
- package/renderer/lib/api-docs/parsers/openapi/validator.ts +31 -0
- package/renderer/lib/api-docs/playground/context.tsx +107 -0
- package/renderer/lib/api-docs/playground/navigation-context.tsx +124 -0
- package/renderer/lib/api-docs/playground/request-builder.ts +223 -0
- package/renderer/lib/api-docs/playground/request-runner.ts +282 -0
- package/renderer/lib/api-docs/playground/types.ts +35 -0
- package/renderer/lib/api-docs/types.ts +269 -0
- package/renderer/lib/api-docs/utils.ts +311 -0
- package/renderer/lib/cache.ts +193 -0
- package/renderer/lib/docs/config/index.ts +29 -0
- package/renderer/lib/docs/config/loader.ts +142 -0
- package/renderer/lib/docs/config/schema.ts +298 -0
- package/renderer/lib/docs/index.ts +12 -0
- package/renderer/lib/docs/mdx/compiler.ts +176 -0
- package/renderer/lib/docs/mdx/frontmatter.ts +80 -0
- package/renderer/lib/docs/mdx/index.ts +26 -0
- package/renderer/lib/docs/navigation/generator.ts +348 -0
- package/renderer/lib/docs/navigation/index.ts +12 -0
- package/renderer/lib/docs/navigation/types.ts +123 -0
- package/renderer/lib/docs-navigation-context.tsx +80 -0
- package/renderer/lib/multi-tenant/context.ts +105 -0
- package/renderer/lib/storage/blob.ts +845 -0
- package/renderer/lib/utils.ts +6 -0
- package/renderer/next.config.ts +76 -0
- package/renderer/package.json +66 -0
- package/renderer/postcss.config.mjs +5 -0
- package/renderer/public/assets/images/screenshot.png +0 -0
- package/renderer/public/assets/logo/dark.svg +9 -0
- package/renderer/public/assets/logo/light.svg +9 -0
- package/renderer/public/assets/logo.svg +9 -0
- package/renderer/public/file.svg +1 -0
- package/renderer/public/globe.svg +1 -0
- package/renderer/public/icon.png +0 -0
- package/renderer/public/logo.svg +9 -0
- package/renderer/public/window.svg +1 -0
- package/renderer/tsconfig.json +28 -0
- package/templates/basic/README.md +139 -0
- package/templates/basic/assets/favicon.svg +4 -0
- package/templates/basic/assets/logo.svg +9 -0
- package/templates/basic/docs.json +47 -0
- package/templates/basic/guides/configuration.mdx +149 -0
- package/templates/basic/guides/overview.mdx +96 -0
- package/templates/basic/index.mdx +39 -0
- package/templates/basic/package.json +14 -0
- package/templates/basic/quickstart.mdx +92 -0
- package/templates/basic/vercel.json +6 -0
- package/templates/graphql/README.md +139 -0
- package/templates/graphql/api-reference/schema.graphql +305 -0
- package/templates/graphql/assets/favicon.svg +4 -0
- package/templates/graphql/assets/logo.svg +9 -0
- package/templates/graphql/docs.json +54 -0
- package/templates/graphql/guides/configuration.mdx +149 -0
- package/templates/graphql/guides/overview.mdx +96 -0
- package/templates/graphql/index.mdx +39 -0
- package/templates/graphql/package.json +14 -0
- package/templates/graphql/quickstart.mdx +92 -0
- package/templates/graphql/vercel.json +6 -0
- package/templates/openapi/README.md +139 -0
- package/templates/openapi/api-reference/openapi.json +419 -0
- package/templates/openapi/assets/favicon.svg +4 -0
- package/templates/openapi/assets/logo.svg +9 -0
- package/templates/openapi/docs.json +61 -0
- package/templates/openapi/guides/configuration.mdx +149 -0
- package/templates/openapi/guides/overview.mdx +96 -0
- package/templates/openapi/index.mdx +39 -0
- package/templates/openapi/package.json +14 -0
- package/templates/openapi/quickstart.mdx +92 -0
- 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
|
+
}
|