@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.
- package/ai-agents/.claude/skills/bootstrap-docs/SKILL.md +710 -79
- package/ai-agents/.claude/skills/check-docs/SKILL.md +83 -8
- package/ai-agents/.claude/skills/create-doc/SKILL.md +267 -55
- package/ai-agents/.claude/skills/update-doc/SKILL.md +162 -63
- package/ai-agents/.cursor/rules/devdoc-bootstrap.mdc +145 -15
- package/ai-agents/.cursor/rules/devdoc-create.mdc +108 -57
- package/ai-agents/.cursor/rules/devdoc-update.mdc +93 -70
- package/ai-agents/.cursor/rules/devdoc.mdc +21 -0
- package/ai-agents/schemas/docs.schema.json +332 -0
- package/ai-agents/schemas/theme.schema.json +243 -0
- package/dist/cli/commands/create.js +4 -9
- package/dist/cli/commands/deploy.js +50 -25
- package/dist/cli/commands/dev.js +19 -10
- package/package.json +3 -2
- package/renderer/app/api/assets/[...path]/route.js +108 -0
- package/renderer/app/api/assets/route.js +114 -0
- package/renderer/app/api/assets/upload/route.js +163 -0
- package/renderer/app/api/auth-schemes/route.js +58 -0
- package/renderer/app/api/chat/route.js +759 -0
- package/renderer/app/api/codegen/route.js +52 -0
- package/renderer/app/api/collections/route.js +675 -0
- package/renderer/app/api/debug/route.js +47 -0
- package/renderer/app/api/deploy/route.js +199 -0
- package/renderer/app/api/device/route.js +36 -0
- package/renderer/app/api/docs/route.js +205 -0
- package/renderer/app/api/domains/add/route.js +121 -0
- package/renderer/app/api/domains/lookup/route.js +43 -0
- package/renderer/app/api/domains/remove/route.js +89 -0
- package/renderer/app/api/domains/status/route.js +140 -0
- package/renderer/app/api/domains/verify/route.js +168 -0
- package/renderer/app/api/keys/regenerate/route.js +71 -0
- package/renderer/app/api/local-assets/[...path]/route.js +108 -0
- package/renderer/app/api/openapi-spec/route.js +73 -0
- package/renderer/app/api/projects/[slug]/route.js +129 -0
- package/renderer/app/api/projects/[slug]/stats/route.js +80 -0
- package/renderer/app/api/projects/register/route.js +176 -0
- package/renderer/app/api/proxy/route.js +139 -0
- package/renderer/app/api/proxy-stream/route.js +156 -0
- package/renderer/app/api/redirects/route.js +35 -0
- package/renderer/app/api/schema/route.js +85 -0
- package/renderer/app/api/subdomains/check/route.js +158 -0
- package/renderer/app/api/suggestions/route.js +175 -0
- package/renderer/app/layout.js +47 -0
- package/renderer/app/llms-full.txt/route.js +257 -0
- package/renderer/app/llms.txt/route.js +219 -0
- package/renderer/app/page.js +12 -0
- package/renderer/app/robots.txt/route.js +66 -0
- package/renderer/app/sitemap.xml/route.js +145 -0
- package/renderer/components/docs/index.js +8 -0
- package/renderer/components/docs/mdx/accordion.js +113 -0
- package/renderer/components/docs/mdx/badge.js +72 -0
- package/renderer/components/docs/mdx/callouts.js +137 -0
- package/renderer/components/docs/mdx/cards.js +175 -0
- package/renderer/components/docs/mdx/changelog.js +100 -0
- package/renderer/components/docs/mdx/code-block.js +147 -0
- package/renderer/components/docs/mdx/code-group.js +287 -0
- package/renderer/components/docs/mdx/file-embeds.js +82 -0
- package/renderer/components/docs/mdx/frame.js +59 -0
- package/renderer/components/docs/mdx/highlight.js +90 -0
- package/renderer/components/docs/mdx/iframe.js +69 -0
- package/renderer/components/docs/mdx/image.js +135 -0
- package/renderer/components/docs/mdx/index.js +134 -0
- package/renderer/components/docs/mdx/landing.js +315 -0
- package/renderer/components/docs/mdx/mermaid.js +212 -0
- package/renderer/components/docs/mdx/param-field.js +112 -0
- package/renderer/components/docs/mdx/steps.js +74 -0
- package/renderer/components/docs/mdx/tabs.js +50 -0
- package/renderer/components/docs/mdx-renderer.js +77 -0
- package/renderer/components/docs/navigation/breadcrumbs.js +64 -0
- package/renderer/components/docs/navigation/index.js +6 -0
- package/renderer/components/docs/navigation/page-nav.js +57 -0
- package/renderer/components/docs/navigation/sidebar.js +375 -0
- package/renderer/components/docs/navigation/toc.js +89 -0
- package/renderer/components/docs/notice.js +77 -0
- package/renderer/components/docs-header.js +202 -0
- package/renderer/components/docs-viewer/agent/agent-chat.js +1930 -0
- package/renderer/components/docs-viewer/agent/cards/debug-context-card.js +107 -0
- package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.js +57 -0
- package/renderer/components/docs-viewer/agent/cards/index.js +45 -0
- package/renderer/components/docs-viewer/agent/cards/response-options-card.js +154 -0
- package/renderer/components/docs-viewer/agent/cards/types.js +22 -0
- package/renderer/components/docs-viewer/agent/chat-message.js +2 -0
- package/renderer/components/docs-viewer/agent/index.js +4 -0
- package/renderer/components/docs-viewer/agent/messages/assistant-message.js +108 -0
- package/renderer/components/docs-viewer/agent/messages/chat-message.js +34 -0
- package/renderer/components/docs-viewer/agent/messages/index.js +6 -0
- package/renderer/components/docs-viewer/agent/messages/tool-call-display.js +1065 -0
- package/renderer/components/docs-viewer/agent/messages/types.js +2 -0
- package/renderer/components/docs-viewer/agent/messages/typing-indicator.js +26 -0
- package/renderer/components/docs-viewer/agent/messages/user-message.js +37 -0
- package/renderer/components/docs-viewer/code-editor/{index.tsx → index.js} +1 -1
- package/renderer/components/docs-viewer/code-editor/notes-mode.js +1338 -0
- package/renderer/components/docs-viewer/content/changelog-page.js +297 -0
- package/renderer/components/docs-viewer/content/doc-page.js +264 -0
- package/renderer/components/docs-viewer/content/documentation-viewer.js +14 -0
- package/renderer/components/docs-viewer/content/index.js +29 -0
- package/renderer/components/docs-viewer/content/not-found-page.js +300 -0
- package/renderer/components/docs-viewer/content/request-details.js +528 -0
- package/renderer/components/docs-viewer/content/sections/auth.js +108 -0
- package/renderer/components/docs-viewer/content/sections/body.js +80 -0
- package/renderer/components/docs-viewer/content/sections/headers.js +64 -0
- package/renderer/components/docs-viewer/content/sections/overview.js +56 -0
- package/renderer/components/docs-viewer/content/sections/parameters.js +64 -0
- package/renderer/components/docs-viewer/content/sections/responses.js +91 -0
- package/renderer/components/docs-viewer/global-auth-modal.js +427 -0
- package/renderer/components/docs-viewer/index.js +1552 -0
- package/renderer/components/docs-viewer/playground/auth-editor.js +418 -0
- package/renderer/components/docs-viewer/playground/body-editor.js +240 -0
- package/renderer/components/docs-viewer/playground/code-editor.js +135 -0
- package/renderer/components/docs-viewer/playground/code-snippet.js +393 -0
- package/renderer/components/docs-viewer/playground/graphql-playground.js +734 -0
- package/renderer/components/docs-viewer/playground/index.js +682 -0
- package/renderer/components/docs-viewer/playground/key-value-editor.js +317 -0
- package/renderer/components/docs-viewer/playground/method-selector.js +65 -0
- package/renderer/components/docs-viewer/playground/request-builder.js +181 -0
- package/renderer/components/docs-viewer/playground/request-tabs.js +240 -0
- package/renderer/components/docs-viewer/playground/response-cards/idle-card.js +42 -0
- package/renderer/components/docs-viewer/playground/response-cards/index.js +72 -0
- package/renderer/components/docs-viewer/playground/response-cards/loading-card.js +24 -0
- package/renderer/components/docs-viewer/playground/response-cards/network-error-card.js +28 -0
- package/renderer/components/docs-viewer/playground/response-cards/response-body-card.js +308 -0
- package/renderer/components/docs-viewer/playground/response-cards/types.js +9 -0
- package/renderer/components/docs-viewer/playground/response-viewer.js +18 -0
- package/renderer/components/docs-viewer/search/index.js +2 -0
- package/renderer/components/docs-viewer/search/search-dialog.js +367 -0
- package/renderer/components/docs-viewer/search/use-search.js +89 -0
- package/renderer/components/docs-viewer/shared/markdown-renderer.js +423 -0
- package/renderer/components/docs-viewer/shared/method-badge.js +23 -0
- package/renderer/components/docs-viewer/shared/schema-viewer.js +321 -0
- package/renderer/components/docs-viewer/sidebar/collection-tree.js +222 -0
- package/renderer/components/docs-viewer/sidebar/endpoint-options.js +512 -0
- package/renderer/components/docs-viewer/sidebar/index.js +196 -0
- package/renderer/components/docs-viewer/sidebar/right-sidebar.js +163 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-group.js +87 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-item.js +172 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-section.js +31 -0
- package/renderer/components/theme-provider.js +10 -0
- package/renderer/components/theme-toggle.js +106 -0
- package/renderer/components/ui/badge.js +29 -0
- package/renderer/components/ui/button.js +40 -0
- package/renderer/components/ui/dialog.js +50 -0
- package/renderer/components/ui/dropdown-menu.js +143 -0
- package/renderer/components/ui/input.js +12 -0
- package/renderer/components/ui/label.js +13 -0
- package/renderer/components/ui/navigation-menu.js +83 -0
- package/renderer/components/ui/select.js +116 -0
- package/renderer/components/ui/spinner.js +92 -0
- package/renderer/components/ui/tabs.js +34 -0
- package/renderer/components/ui/tooltip.js +43 -0
- package/renderer/hooks/use-code-copy.js +76 -0
- package/renderer/hooks/use-openapi-title.js +33 -0
- package/renderer/lib/api-docs/agent/index.js +4 -0
- package/renderer/lib/api-docs/agent/indexer.js +254 -0
- package/renderer/lib/api-docs/agent/spec-summary.js +227 -0
- package/renderer/lib/api-docs/agent/types.js +5 -0
- package/renderer/lib/api-docs/auth/auth-context.js +157 -0
- package/renderer/lib/api-docs/auth/auth-storage.js +66 -0
- package/renderer/lib/api-docs/auth/crypto.js +64 -0
- package/renderer/lib/api-docs/auth/index.js +3 -0
- package/renderer/lib/api-docs/code-editor/db.js +145 -0
- package/renderer/lib/api-docs/code-editor/hooks.js +254 -0
- package/renderer/lib/api-docs/code-editor/{index.ts → index.js} +3 -4
- package/renderer/lib/api-docs/code-editor/mode-context.js +154 -0
- package/renderer/lib/api-docs/code-editor/types.js +53 -0
- package/renderer/lib/api-docs/codegen/definitions.js +258 -0
- package/renderer/lib/api-docs/codegen/har.js +171 -0
- package/renderer/lib/api-docs/codegen/index.js +118 -0
- package/renderer/lib/api-docs/factories.js +136 -0
- package/renderer/lib/api-docs/{index.ts → index.js} +5 -10
- package/renderer/lib/api-docs/mobile-context.js +79 -0
- package/renderer/lib/api-docs/navigation-context.js +62 -0
- package/renderer/lib/api-docs/parsers/graphql/index.js +50 -0
- package/renderer/lib/api-docs/parsers/graphql/parser.js +350 -0
- package/renderer/lib/api-docs/parsers/graphql/transformer.js +215 -0
- package/renderer/lib/api-docs/parsers/graphql/types.js +46 -0
- package/renderer/lib/api-docs/parsers/openapi/dereferencer.js +43 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/auth.js +486 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/body.js +295 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/index.js +132 -0
- package/renderer/lib/api-docs/parsers/openapi/index.js +127 -0
- package/renderer/lib/api-docs/parsers/openapi/transformer.js +192 -0
- package/renderer/lib/api-docs/parsers/openapi/validator.js +24 -0
- package/renderer/lib/api-docs/playground/context.js +65 -0
- package/renderer/lib/api-docs/playground/navigation-context.js +74 -0
- package/renderer/lib/api-docs/playground/request-builder.js +163 -0
- package/renderer/lib/api-docs/playground/request-runner.js +224 -0
- package/renderer/lib/api-docs/playground/types.js +5 -0
- package/renderer/lib/api-docs/types.js +23 -0
- package/renderer/lib/api-docs/utils.js +212 -0
- package/renderer/lib/cache.js +157 -0
- package/renderer/lib/docs/config/domain-schema.js +161 -0
- package/renderer/lib/docs/config/index.js +5 -0
- package/renderer/lib/docs/config/loader.js +113 -0
- package/renderer/lib/docs/config/schema.js +269 -0
- package/renderer/lib/docs/index.js +8 -0
- package/renderer/lib/docs/mdx/compiler.js +128 -0
- package/renderer/lib/docs/mdx/frontmatter.js +73 -0
- package/renderer/lib/docs/mdx/index.js +8 -0
- package/renderer/lib/docs/navigation/generator.js +269 -0
- package/renderer/lib/docs/navigation/index.js +4 -0
- package/renderer/lib/docs/navigation/types.js +9 -0
- package/renderer/lib/docs-navigation-context.js +40 -0
- package/renderer/lib/multi-tenant/context.js +80 -0
- package/renderer/lib/storage/blob.js +767 -0
- package/renderer/lib/utils/icons.js +30 -0
- package/renderer/lib/utils.js +5 -0
- package/renderer/next.config.js +62 -0
- package/renderer/tsconfig.json +23 -5
- package/renderer/app/api/assets/[...path]/route.ts +0 -123
- package/renderer/app/api/assets/route.ts +0 -124
- package/renderer/app/api/assets/upload/route.ts +0 -177
- package/renderer/app/api/auth-schemes/route.ts +0 -77
- package/renderer/app/api/chat/route.ts +0 -858
- package/renderer/app/api/codegen/route.ts +0 -72
- package/renderer/app/api/collections/route.ts +0 -1002
- package/renderer/app/api/debug/route.ts +0 -53
- package/renderer/app/api/deploy/route.ts +0 -234
- package/renderer/app/api/device/route.ts +0 -42
- package/renderer/app/api/docs/route.ts +0 -201
- package/renderer/app/api/domains/add/route.ts +0 -132
- package/renderer/app/api/domains/lookup/route.ts +0 -43
- package/renderer/app/api/domains/remove/route.ts +0 -100
- package/renderer/app/api/domains/status/route.ts +0 -158
- package/renderer/app/api/domains/verify/route.ts +0 -181
- package/renderer/app/api/keys/regenerate/route.ts +0 -80
- package/renderer/app/api/local-assets/[...path]/route.ts +0 -122
- package/renderer/app/api/openapi-spec/route.ts +0 -151
- package/renderer/app/api/projects/[slug]/route.ts +0 -153
- package/renderer/app/api/projects/[slug]/stats/route.ts +0 -96
- package/renderer/app/api/projects/register/route.ts +0 -152
- package/renderer/app/api/proxy/route.ts +0 -149
- package/renderer/app/api/proxy-stream/route.ts +0 -168
- package/renderer/app/api/redirects/route.ts +0 -47
- package/renderer/app/api/schema/route.ts +0 -73
- package/renderer/app/api/subdomains/check/route.ts +0 -172
- package/renderer/app/api/suggestions/route.ts +0 -144
- package/renderer/app/layout.tsx +0 -54
- package/renderer/app/llms-full.txt/route.ts +0 -346
- package/renderer/app/llms.txt/route.ts +0 -279
- package/renderer/app/page.tsx +0 -14
- package/renderer/app/robots.txt/route.ts +0 -84
- package/renderer/app/sitemap.xml/route.ts +0 -199
- package/renderer/components/docs/index.ts +0 -12
- package/renderer/components/docs/mdx/accordion.tsx +0 -169
- package/renderer/components/docs/mdx/badge.tsx +0 -132
- package/renderer/components/docs/mdx/callouts.tsx +0 -154
- package/renderer/components/docs/mdx/cards.tsx +0 -241
- package/renderer/components/docs/mdx/changelog.tsx +0 -120
- package/renderer/components/docs/mdx/code-block.tsx +0 -186
- package/renderer/components/docs/mdx/code-group.tsx +0 -421
- package/renderer/components/docs/mdx/file-embeds.tsx +0 -105
- package/renderer/components/docs/mdx/frame.tsx +0 -112
- package/renderer/components/docs/mdx/highlight.tsx +0 -151
- package/renderer/components/docs/mdx/iframe.tsx +0 -134
- package/renderer/components/docs/mdx/image.tsx +0 -235
- package/renderer/components/docs/mdx/index.ts +0 -237
- package/renderer/components/docs/mdx/landing.tsx +0 -684
- package/renderer/components/docs/mdx/mermaid.tsx +0 -240
- package/renderer/components/docs/mdx/param-field.tsx +0 -200
- package/renderer/components/docs/mdx/steps.tsx +0 -113
- package/renderer/components/docs/mdx/tabs.tsx +0 -86
- package/renderer/components/docs/mdx-renderer.tsx +0 -100
- package/renderer/components/docs/navigation/breadcrumbs.tsx +0 -76
- package/renderer/components/docs/navigation/index.ts +0 -8
- package/renderer/components/docs/navigation/page-nav.tsx +0 -64
- package/renderer/components/docs/navigation/sidebar.tsx +0 -515
- package/renderer/components/docs/navigation/toc.tsx +0 -113
- package/renderer/components/docs/notice.tsx +0 -105
- package/renderer/components/docs-header.tsx +0 -278
- package/renderer/components/docs-viewer/agent/agent-chat.tsx +0 -2076
- package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +0 -90
- package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +0 -49
- package/renderer/components/docs-viewer/agent/cards/index.tsx +0 -50
- package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +0 -212
- package/renderer/components/docs-viewer/agent/cards/types.ts +0 -84
- package/renderer/components/docs-viewer/agent/chat-message.tsx +0 -17
- package/renderer/components/docs-viewer/agent/index.tsx +0 -6
- package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +0 -119
- package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +0 -46
- package/renderer/components/docs-viewer/agent/messages/index.ts +0 -17
- package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +0 -721
- package/renderer/components/docs-viewer/agent/messages/types.ts +0 -61
- package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +0 -24
- package/renderer/components/docs-viewer/agent/messages/user-message.tsx +0 -51
- package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +0 -1283
- package/renderer/components/docs-viewer/content/changelog-page.tsx +0 -331
- package/renderer/components/docs-viewer/content/doc-page.tsx +0 -367
- package/renderer/components/docs-viewer/content/documentation-viewer.tsx +0 -17
- package/renderer/components/docs-viewer/content/index.tsx +0 -29
- package/renderer/components/docs-viewer/content/not-found-page.tsx +0 -330
- package/renderer/components/docs-viewer/content/request-details.tsx +0 -330
- package/renderer/components/docs-viewer/content/sections/auth.tsx +0 -69
- package/renderer/components/docs-viewer/content/sections/body.tsx +0 -66
- package/renderer/components/docs-viewer/content/sections/headers.tsx +0 -43
- package/renderer/components/docs-viewer/content/sections/overview.tsx +0 -40
- package/renderer/components/docs-viewer/content/sections/parameters.tsx +0 -43
- package/renderer/components/docs-viewer/content/sections/responses.tsx +0 -87
- package/renderer/components/docs-viewer/global-auth-modal.tsx +0 -352
- package/renderer/components/docs-viewer/index.tsx +0 -1662
- package/renderer/components/docs-viewer/playground/auth-editor.tsx +0 -280
- package/renderer/components/docs-viewer/playground/body-editor.tsx +0 -221
- package/renderer/components/docs-viewer/playground/code-editor.tsx +0 -224
- package/renderer/components/docs-viewer/playground/code-snippet.tsx +0 -387
- package/renderer/components/docs-viewer/playground/graphql-playground.tsx +0 -745
- package/renderer/components/docs-viewer/playground/index.tsx +0 -671
- package/renderer/components/docs-viewer/playground/key-value-editor.tsx +0 -261
- package/renderer/components/docs-viewer/playground/method-selector.tsx +0 -60
- package/renderer/components/docs-viewer/playground/request-builder.tsx +0 -179
- package/renderer/components/docs-viewer/playground/request-tabs.tsx +0 -237
- package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +0 -21
- package/renderer/components/docs-viewer/playground/response-cards/index.tsx +0 -93
- package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +0 -16
- package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +0 -23
- package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +0 -268
- package/renderer/components/docs-viewer/playground/response-cards/types.ts +0 -82
- package/renderer/components/docs-viewer/playground/response-viewer.tsx +0 -43
- package/renderer/components/docs-viewer/search/index.ts +0 -2
- package/renderer/components/docs-viewer/search/search-dialog.tsx +0 -331
- package/renderer/components/docs-viewer/search/use-search.ts +0 -117
- package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +0 -431
- package/renderer/components/docs-viewer/shared/method-badge.tsx +0 -41
- package/renderer/components/docs-viewer/shared/schema-viewer.tsx +0 -349
- package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +0 -259
- package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +0 -316
- package/renderer/components/docs-viewer/sidebar/index.tsx +0 -282
- package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +0 -202
- package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +0 -118
- package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +0 -212
- package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +0 -38
- package/renderer/components/theme-provider.tsx +0 -11
- package/renderer/components/theme-toggle.tsx +0 -76
- package/renderer/components/ui/badge.tsx +0 -46
- package/renderer/components/ui/button.tsx +0 -59
- package/renderer/components/ui/dialog.tsx +0 -118
- package/renderer/components/ui/dropdown-menu.tsx +0 -257
- package/renderer/components/ui/input.tsx +0 -21
- package/renderer/components/ui/label.tsx +0 -24
- package/renderer/components/ui/navigation-menu.tsx +0 -168
- package/renderer/components/ui/select.tsx +0 -190
- package/renderer/components/ui/spinner.tsx +0 -114
- package/renderer/components/ui/tabs.tsx +0 -66
- package/renderer/components/ui/tooltip.tsx +0 -61
- package/renderer/hooks/use-code-copy.ts +0 -88
- package/renderer/hooks/use-openapi-title.ts +0 -44
- package/renderer/lib/api-docs/agent/index.ts +0 -6
- package/renderer/lib/api-docs/agent/indexer.ts +0 -323
- package/renderer/lib/api-docs/agent/spec-summary.ts +0 -335
- package/renderer/lib/api-docs/agent/types.ts +0 -116
- package/renderer/lib/api-docs/auth/auth-context.tsx +0 -225
- package/renderer/lib/api-docs/auth/auth-storage.ts +0 -87
- package/renderer/lib/api-docs/auth/crypto.ts +0 -89
- package/renderer/lib/api-docs/auth/index.ts +0 -4
- package/renderer/lib/api-docs/code-editor/db.ts +0 -164
- package/renderer/lib/api-docs/code-editor/hooks.ts +0 -266
- package/renderer/lib/api-docs/code-editor/mode-context.tsx +0 -207
- package/renderer/lib/api-docs/code-editor/types.ts +0 -105
- package/renderer/lib/api-docs/codegen/definitions.ts +0 -297
- package/renderer/lib/api-docs/codegen/har.ts +0 -251
- package/renderer/lib/api-docs/codegen/index.ts +0 -159
- package/renderer/lib/api-docs/factories.ts +0 -170
- package/renderer/lib/api-docs/mobile-context.tsx +0 -112
- package/renderer/lib/api-docs/navigation-context.tsx +0 -88
- package/renderer/lib/api-docs/parsers/graphql/README.md +0 -129
- package/renderer/lib/api-docs/parsers/graphql/index.ts +0 -91
- package/renderer/lib/api-docs/parsers/graphql/parser.ts +0 -491
- package/renderer/lib/api-docs/parsers/graphql/transformer.ts +0 -246
- package/renderer/lib/api-docs/parsers/graphql/types.ts +0 -283
- package/renderer/lib/api-docs/parsers/openapi/README.md +0 -32
- package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +0 -60
- package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +0 -574
- package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +0 -403
- package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +0 -232
- package/renderer/lib/api-docs/parsers/openapi/index.ts +0 -171
- package/renderer/lib/api-docs/parsers/openapi/transformer.ts +0 -278
- package/renderer/lib/api-docs/parsers/openapi/validator.ts +0 -31
- package/renderer/lib/api-docs/playground/context.tsx +0 -107
- package/renderer/lib/api-docs/playground/navigation-context.tsx +0 -124
- package/renderer/lib/api-docs/playground/request-builder.ts +0 -223
- package/renderer/lib/api-docs/playground/request-runner.ts +0 -282
- package/renderer/lib/api-docs/playground/types.ts +0 -35
- package/renderer/lib/api-docs/types.ts +0 -269
- package/renderer/lib/api-docs/utils.ts +0 -311
- package/renderer/lib/cache.ts +0 -193
- package/renderer/lib/docs/config/domain-schema.ts +0 -260
- package/renderer/lib/docs/config/index.ts +0 -43
- package/renderer/lib/docs/config/loader.ts +0 -142
- package/renderer/lib/docs/config/schema.ts +0 -308
- package/renderer/lib/docs/index.ts +0 -12
- package/renderer/lib/docs/mdx/compiler.ts +0 -176
- package/renderer/lib/docs/mdx/frontmatter.ts +0 -91
- package/renderer/lib/docs/mdx/index.ts +0 -26
- package/renderer/lib/docs/navigation/generator.ts +0 -348
- package/renderer/lib/docs/navigation/index.ts +0 -12
- package/renderer/lib/docs/navigation/types.ts +0 -123
- package/renderer/lib/docs-navigation-context.tsx +0 -80
- package/renderer/lib/multi-tenant/context.ts +0 -105
- package/renderer/lib/storage/blob.ts +0 -1083
- package/renderer/lib/utils/icons.ts +0 -48
- package/renderer/lib/utils.ts +0 -6
- package/renderer/next.config.ts +0 -76
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import * as PhosphorIcons from '@phosphor-icons/react';
|
|
2
|
+
/**
|
|
3
|
+
* Get a Phosphor icon component by its kebab-case name
|
|
4
|
+
*
|
|
5
|
+
* @param iconName - The icon name in kebab-case (e.g., "arrow-right", "github-logo")
|
|
6
|
+
* @returns The Phosphor icon component or null if not found
|
|
7
|
+
*
|
|
8
|
+
* @example
|
|
9
|
+
* const Icon = getPhosphorIcon('arrow-right')
|
|
10
|
+
* if (Icon) {
|
|
11
|
+
* return <Icon className="h-5 w-5" weight="bold" />
|
|
12
|
+
* }
|
|
13
|
+
*/ export function getPhosphorIcon(iconName) {
|
|
14
|
+
if (!iconName) return null;
|
|
15
|
+
// Convert kebab-case to PascalCase for Phosphor icons
|
|
16
|
+
const pascalCase = iconName.split('-').map((part)=>part.charAt(0).toUpperCase() + part.slice(1)).join('');
|
|
17
|
+
const IconComponent = PhosphorIcons[pascalCase];
|
|
18
|
+
if (typeof IconComponent === 'function') {
|
|
19
|
+
return IconComponent;
|
|
20
|
+
}
|
|
21
|
+
return null;
|
|
22
|
+
}
|
|
23
|
+
/**
|
|
24
|
+
* Check if a Phosphor icon exists by name
|
|
25
|
+
*
|
|
26
|
+
* @param iconName - The icon name in kebab-case
|
|
27
|
+
* @returns true if the icon exists
|
|
28
|
+
*/ export function hasPhosphorIcon(iconName) {
|
|
29
|
+
return getPhosphorIcon(iconName) !== null;
|
|
30
|
+
}
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
const nextConfig = {
|
|
2
|
+
/* config options here */ turbopack: {
|
|
3
|
+
root: __dirname,
|
|
4
|
+
resolveAlias: {
|
|
5
|
+
'@': __dirname
|
|
6
|
+
}
|
|
7
|
+
},
|
|
8
|
+
// Disable compression to allow SSE streaming to work properly
|
|
9
|
+
// Without this, Next.js buffers responses and sends them all at once
|
|
10
|
+
compress: false,
|
|
11
|
+
// Optimize package imports for better tree shaking
|
|
12
|
+
// Note: modularizeImports for @phosphor-icons/react was removed because
|
|
13
|
+
// it conflicts with optimizePackageImports and the SSR files use named exports
|
|
14
|
+
experimental: {
|
|
15
|
+
optimizePackageImports: [
|
|
16
|
+
'@phosphor-icons/react',
|
|
17
|
+
'@radix-ui/react-dialog',
|
|
18
|
+
'@radix-ui/react-dropdown-menu',
|
|
19
|
+
'@radix-ui/react-select',
|
|
20
|
+
'@radix-ui/react-tabs',
|
|
21
|
+
'@radix-ui/react-tooltip',
|
|
22
|
+
'motion',
|
|
23
|
+
'ai',
|
|
24
|
+
'@ai-sdk/react'
|
|
25
|
+
]
|
|
26
|
+
},
|
|
27
|
+
// Externalize heavy packages to reduce serverless function size
|
|
28
|
+
// These packages won't be bundled into each function, reducing the ~250MB limit issue
|
|
29
|
+
serverExternalPackages: [
|
|
30
|
+
// Syntax highlighting - shiki bundles ALL language grammars (~30MB+)
|
|
31
|
+
'shiki',
|
|
32
|
+
'rehype-pretty-code',
|
|
33
|
+
// Monaco editor - heavy code editor
|
|
34
|
+
'monaco-editor',
|
|
35
|
+
'@monaco-editor/react',
|
|
36
|
+
// Mermaid - diagram generation library
|
|
37
|
+
'mermaid',
|
|
38
|
+
// HTTP snippet generation
|
|
39
|
+
'@hoppscotch/httpsnippet'
|
|
40
|
+
],
|
|
41
|
+
// Exclude cache and large directories from output file tracing
|
|
42
|
+
// This prevents webpack cache and other large files from being included in serverless functions
|
|
43
|
+
outputFileTracingExcludes: {
|
|
44
|
+
'*': [
|
|
45
|
+
// Next.js cache directories
|
|
46
|
+
'.next/cache/**',
|
|
47
|
+
'.next/cache/webpack/**',
|
|
48
|
+
// Large node_modules that shouldn't be traced
|
|
49
|
+
'node_modules/@swc/**',
|
|
50
|
+
'node_modules/esbuild/**',
|
|
51
|
+
'node_modules/terser/**',
|
|
52
|
+
'node_modules/webpack/**',
|
|
53
|
+
// Note: shiki is NOT excluded because it's needed by /api/docs for syntax highlighting
|
|
54
|
+
// The dynamic import with fallback in route.ts handles cases where it may be unavailable
|
|
55
|
+
// Monaco editor assets
|
|
56
|
+
'node_modules/monaco-editor/**',
|
|
57
|
+
// TypeScript compiler (not needed at runtime)
|
|
58
|
+
'node_modules/typescript/**'
|
|
59
|
+
]
|
|
60
|
+
}
|
|
61
|
+
};
|
|
62
|
+
module.exports = nextConfig;
|
package/renderer/tsconfig.json
CHANGED
|
@@ -1,7 +1,11 @@
|
|
|
1
1
|
{
|
|
2
2
|
"compilerOptions": {
|
|
3
3
|
"target": "ES2017",
|
|
4
|
-
"lib": [
|
|
4
|
+
"lib": [
|
|
5
|
+
"dom",
|
|
6
|
+
"dom.iterable",
|
|
7
|
+
"esnext"
|
|
8
|
+
],
|
|
5
9
|
"allowJs": true,
|
|
6
10
|
"skipLibCheck": true,
|
|
7
11
|
"strict": true,
|
|
@@ -20,9 +24,23 @@
|
|
|
20
24
|
],
|
|
21
25
|
"baseUrl": ".",
|
|
22
26
|
"paths": {
|
|
23
|
-
"@/*": [
|
|
24
|
-
|
|
27
|
+
"@/*": [
|
|
28
|
+
"./*"
|
|
29
|
+
]
|
|
30
|
+
},
|
|
31
|
+
"checkJs": false
|
|
25
32
|
},
|
|
26
|
-
"include": [
|
|
27
|
-
|
|
33
|
+
"include": [
|
|
34
|
+
"next-env.d.ts",
|
|
35
|
+
"**/*.ts",
|
|
36
|
+
"**/*.tsx",
|
|
37
|
+
".next/types/**/*.ts"
|
|
38
|
+
],
|
|
39
|
+
"exclude": [
|
|
40
|
+
"node_modules",
|
|
41
|
+
"hoppscotch",
|
|
42
|
+
"scalar",
|
|
43
|
+
"assistant-ui",
|
|
44
|
+
"packages"
|
|
45
|
+
]
|
|
28
46
|
}
|
|
@@ -1,123 +0,0 @@
|
|
|
1
|
-
import { NextRequest, NextResponse } from 'next/server'
|
|
2
|
-
import { head } from '@vercel/blob'
|
|
3
|
-
import path from 'path'
|
|
4
|
-
import fs from 'fs'
|
|
5
|
-
|
|
6
|
-
// Check if we're in local development mode
|
|
7
|
-
const IS_LOCAL_DEV = !process.env.BLOB_READ_WRITE_TOKEN
|
|
8
|
-
const LOCAL_STORAGE_DIR = path.join(process.cwd(), '.devdoc-storage')
|
|
9
|
-
|
|
10
|
-
/**
|
|
11
|
-
* Asset proxy - serves uploaded assets for multi-tenant docs
|
|
12
|
-
*
|
|
13
|
-
* GET /api/assets/[slug]/[...path]
|
|
14
|
-
*
|
|
15
|
-
* This allows MDX files to reference assets as:
|
|
16
|
-
* 
|
|
17
|
-
*
|
|
18
|
-
* Or with the middleware rewrite, simply:
|
|
19
|
-
*  -> rewrites to /api/assets/[slug]/logo.png
|
|
20
|
-
*/
|
|
21
|
-
export async function GET(
|
|
22
|
-
request: NextRequest,
|
|
23
|
-
{ params }: { params: Promise<{ path: string[] }> }
|
|
24
|
-
) {
|
|
25
|
-
try {
|
|
26
|
-
const { path: pathSegments } = await params
|
|
27
|
-
|
|
28
|
-
if (!pathSegments || pathSegments.length < 2) {
|
|
29
|
-
return NextResponse.json(
|
|
30
|
-
{ error: 'Invalid asset path' },
|
|
31
|
-
{ status: 400 }
|
|
32
|
-
)
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
// First segment is the project slug, rest is the asset path
|
|
36
|
-
const [slug, ...assetPath] = pathSegments
|
|
37
|
-
const fileName = assetPath.join('/')
|
|
38
|
-
|
|
39
|
-
if (!slug || !fileName) {
|
|
40
|
-
return NextResponse.json(
|
|
41
|
-
{ error: 'Missing slug or file path' },
|
|
42
|
-
{ status: 400 }
|
|
43
|
-
)
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
// Build blob path
|
|
47
|
-
const blobPath = `projects/${slug}/assets/${fileName}`
|
|
48
|
-
|
|
49
|
-
if (IS_LOCAL_DEV) {
|
|
50
|
-
// Local development - serve from filesystem
|
|
51
|
-
const filePath = path.join(LOCAL_STORAGE_DIR, 'projects', slug, 'assets', fileName)
|
|
52
|
-
|
|
53
|
-
if (!fs.existsSync(filePath)) {
|
|
54
|
-
return NextResponse.json(
|
|
55
|
-
{ error: 'Asset not found' },
|
|
56
|
-
{ status: 404 }
|
|
57
|
-
)
|
|
58
|
-
}
|
|
59
|
-
|
|
60
|
-
const fileBuffer = fs.readFileSync(filePath)
|
|
61
|
-
const contentType = getContentType(fileName)
|
|
62
|
-
|
|
63
|
-
return new NextResponse(fileBuffer, {
|
|
64
|
-
status: 200,
|
|
65
|
-
headers: {
|
|
66
|
-
'Content-Type': contentType,
|
|
67
|
-
'Cache-Control': 'public, max-age=31536000, immutable',
|
|
68
|
-
},
|
|
69
|
-
})
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
// Production - fetch from Vercel Blob
|
|
73
|
-
const blobInfo = await head(blobPath).catch(() => null)
|
|
74
|
-
|
|
75
|
-
if (!blobInfo) {
|
|
76
|
-
return NextResponse.json(
|
|
77
|
-
{ error: 'Asset not found' },
|
|
78
|
-
{ status: 404 }
|
|
79
|
-
)
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
// Redirect to the blob URL (more efficient than proxying)
|
|
83
|
-
return NextResponse.redirect(blobInfo.url, {
|
|
84
|
-
status: 302,
|
|
85
|
-
headers: {
|
|
86
|
-
'Cache-Control': 'public, max-age=31536000, immutable',
|
|
87
|
-
},
|
|
88
|
-
})
|
|
89
|
-
|
|
90
|
-
} catch (error) {
|
|
91
|
-
console.error('[Asset Proxy] Error:', error)
|
|
92
|
-
return NextResponse.json(
|
|
93
|
-
{ error: 'Failed to serve asset' },
|
|
94
|
-
{ status: 500 }
|
|
95
|
-
)
|
|
96
|
-
}
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
/**
|
|
100
|
-
* Get content type from file extension
|
|
101
|
-
*/
|
|
102
|
-
function getContentType(fileName: string): string {
|
|
103
|
-
const ext = fileName.split('.').pop()?.toLowerCase()
|
|
104
|
-
const types: Record<string, string> = {
|
|
105
|
-
'jpg': 'image/jpeg',
|
|
106
|
-
'jpeg': 'image/jpeg',
|
|
107
|
-
'png': 'image/png',
|
|
108
|
-
'gif': 'image/gif',
|
|
109
|
-
'webp': 'image/webp',
|
|
110
|
-
'svg': 'image/svg+xml',
|
|
111
|
-
'ico': 'image/x-icon',
|
|
112
|
-
'pdf': 'application/pdf',
|
|
113
|
-
'mp4': 'video/mp4',
|
|
114
|
-
'webm': 'video/webm',
|
|
115
|
-
'mp3': 'audio/mpeg',
|
|
116
|
-
'wav': 'audio/wav',
|
|
117
|
-
'woff': 'font/woff',
|
|
118
|
-
'woff2': 'font/woff2',
|
|
119
|
-
'ttf': 'font/ttf',
|
|
120
|
-
'otf': 'font/otf',
|
|
121
|
-
}
|
|
122
|
-
return types[ext || ''] || 'application/octet-stream'
|
|
123
|
-
}
|
|
@@ -1,124 +0,0 @@
|
|
|
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
|
-
}
|
|
@@ -1,177 +0,0 @@
|
|
|
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
|
-
}
|
|
@@ -1,77 +0,0 @@
|
|
|
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
|
-
}
|