@brainfish-ai/devdoc 0.1.42 → 0.1.43
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- 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 -1670
- 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,129 @@
|
|
|
1
|
+
import { NextResponse } from 'next/server';
|
|
2
|
+
import { validateApiKey, getProjectContent, getProjectApiKey, deleteProject, projectExists } from '@/lib/storage/blob';
|
|
3
|
+
/**
|
|
4
|
+
* GET /api/projects/[slug]
|
|
5
|
+
* Get detailed project information
|
|
6
|
+
*
|
|
7
|
+
* Headers:
|
|
8
|
+
* Authorization: Bearer <api_key>
|
|
9
|
+
*/ export async function GET(request, { params }) {
|
|
10
|
+
try {
|
|
11
|
+
const { slug } = await params;
|
|
12
|
+
// Validate API key
|
|
13
|
+
const authHeader = request.headers.get('Authorization');
|
|
14
|
+
const apiKey = authHeader?.replace('Bearer ', '');
|
|
15
|
+
if (!apiKey) {
|
|
16
|
+
return NextResponse.json({
|
|
17
|
+
error: 'API key required'
|
|
18
|
+
}, {
|
|
19
|
+
status: 401
|
|
20
|
+
});
|
|
21
|
+
}
|
|
22
|
+
// Validate the API key belongs to this project
|
|
23
|
+
const validatedSlug = await validateApiKey(apiKey);
|
|
24
|
+
if (validatedSlug !== slug) {
|
|
25
|
+
return NextResponse.json({
|
|
26
|
+
error: 'API key does not match this project'
|
|
27
|
+
}, {
|
|
28
|
+
status: 403
|
|
29
|
+
});
|
|
30
|
+
}
|
|
31
|
+
// Get project details
|
|
32
|
+
const content = await getProjectContent(slug);
|
|
33
|
+
const keyData = await getProjectApiKey(slug);
|
|
34
|
+
if (!content) {
|
|
35
|
+
return NextResponse.json({
|
|
36
|
+
error: 'Project not found'
|
|
37
|
+
}, {
|
|
38
|
+
status: 404
|
|
39
|
+
});
|
|
40
|
+
}
|
|
41
|
+
// Parse docs.json
|
|
42
|
+
let docsConfig = {};
|
|
43
|
+
try {
|
|
44
|
+
docsConfig = JSON.parse(content.docsJson);
|
|
45
|
+
} catch {
|
|
46
|
+
// Ignore parse errors
|
|
47
|
+
}
|
|
48
|
+
// Build file list (without content for security/size)
|
|
49
|
+
const files = content.files.map((f)=>({
|
|
50
|
+
path: f.path,
|
|
51
|
+
size: f.content.length
|
|
52
|
+
}));
|
|
53
|
+
return NextResponse.json({
|
|
54
|
+
slug: content.slug,
|
|
55
|
+
name: content.name,
|
|
56
|
+
url: `https://${slug}.devdoc.sh`,
|
|
57
|
+
createdAt: content.createdAt,
|
|
58
|
+
updatedAt: content.updatedAt,
|
|
59
|
+
lastDeployedAt: keyData?.lastUsedAt || content.updatedAt,
|
|
60
|
+
filesCount: content.files.length,
|
|
61
|
+
totalSize: content.files.reduce((sum, f)=>sum + f.content.length, 0),
|
|
62
|
+
files,
|
|
63
|
+
config: {
|
|
64
|
+
name: docsConfig.name,
|
|
65
|
+
favicon: docsConfig.favicon,
|
|
66
|
+
navigation: docsConfig.navigation ? 'configured' : 'default'
|
|
67
|
+
}
|
|
68
|
+
});
|
|
69
|
+
} catch (error) {
|
|
70
|
+
console.error('[Projects API] GET Error:', error);
|
|
71
|
+
return NextResponse.json({
|
|
72
|
+
error: 'Internal server error'
|
|
73
|
+
}, {
|
|
74
|
+
status: 500
|
|
75
|
+
});
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* DELETE /api/projects/[slug]
|
|
80
|
+
* Delete a project and all its content
|
|
81
|
+
*
|
|
82
|
+
* Headers:
|
|
83
|
+
* Authorization: Bearer <api_key>
|
|
84
|
+
*/ export async function DELETE(request, { params }) {
|
|
85
|
+
try {
|
|
86
|
+
const { slug } = await params;
|
|
87
|
+
// Validate API key
|
|
88
|
+
const authHeader = request.headers.get('Authorization');
|
|
89
|
+
const apiKey = authHeader?.replace('Bearer ', '');
|
|
90
|
+
if (!apiKey) {
|
|
91
|
+
return NextResponse.json({
|
|
92
|
+
error: 'API key required'
|
|
93
|
+
}, {
|
|
94
|
+
status: 401
|
|
95
|
+
});
|
|
96
|
+
}
|
|
97
|
+
// Validate the API key belongs to this project
|
|
98
|
+
const validatedSlug = await validateApiKey(apiKey);
|
|
99
|
+
if (validatedSlug !== slug) {
|
|
100
|
+
return NextResponse.json({
|
|
101
|
+
error: 'API key does not match this project'
|
|
102
|
+
}, {
|
|
103
|
+
status: 403
|
|
104
|
+
});
|
|
105
|
+
}
|
|
106
|
+
// Check project exists
|
|
107
|
+
const exists = await projectExists(slug);
|
|
108
|
+
if (!exists) {
|
|
109
|
+
return NextResponse.json({
|
|
110
|
+
error: 'Project not found'
|
|
111
|
+
}, {
|
|
112
|
+
status: 404
|
|
113
|
+
});
|
|
114
|
+
}
|
|
115
|
+
// Delete the project
|
|
116
|
+
await deleteProject(slug);
|
|
117
|
+
return NextResponse.json({
|
|
118
|
+
success: true,
|
|
119
|
+
message: `Project ${slug} has been deleted`
|
|
120
|
+
});
|
|
121
|
+
} catch (error) {
|
|
122
|
+
console.error('[Projects API] DELETE Error:', error);
|
|
123
|
+
return NextResponse.json({
|
|
124
|
+
error: 'Internal server error'
|
|
125
|
+
}, {
|
|
126
|
+
status: 500
|
|
127
|
+
});
|
|
128
|
+
}
|
|
129
|
+
}
|
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
import { NextResponse } from 'next/server';
|
|
2
|
+
import { validateApiKey, getProjectContent, getProjectApiKey } from '@/lib/storage/blob';
|
|
3
|
+
/**
|
|
4
|
+
* GET /api/projects/[slug]/stats
|
|
5
|
+
* Get project statistics
|
|
6
|
+
*
|
|
7
|
+
* Headers:
|
|
8
|
+
* Authorization: Bearer <api_key>
|
|
9
|
+
*/ export async function GET(request, { params }) {
|
|
10
|
+
try {
|
|
11
|
+
const { slug } = await params;
|
|
12
|
+
// Validate API key
|
|
13
|
+
const authHeader = request.headers.get('Authorization');
|
|
14
|
+
const apiKey = authHeader?.replace('Bearer ', '');
|
|
15
|
+
if (!apiKey) {
|
|
16
|
+
return NextResponse.json({
|
|
17
|
+
error: 'API key required'
|
|
18
|
+
}, {
|
|
19
|
+
status: 401
|
|
20
|
+
});
|
|
21
|
+
}
|
|
22
|
+
// Validate the API key belongs to this project
|
|
23
|
+
const validatedSlug = await validateApiKey(apiKey);
|
|
24
|
+
if (validatedSlug !== slug) {
|
|
25
|
+
return NextResponse.json({
|
|
26
|
+
error: 'API key does not match this project'
|
|
27
|
+
}, {
|
|
28
|
+
status: 403
|
|
29
|
+
});
|
|
30
|
+
}
|
|
31
|
+
// Get project details
|
|
32
|
+
const content = await getProjectContent(slug);
|
|
33
|
+
const keyData = await getProjectApiKey(slug);
|
|
34
|
+
if (!content) {
|
|
35
|
+
return NextResponse.json({
|
|
36
|
+
error: 'Project not found'
|
|
37
|
+
}, {
|
|
38
|
+
status: 404
|
|
39
|
+
});
|
|
40
|
+
}
|
|
41
|
+
// Calculate stats
|
|
42
|
+
const totalSize = content.files.reduce((sum, f)=>sum + f.content.length, 0);
|
|
43
|
+
const mdxFiles = content.files.filter((f)=>f.path.endsWith('.mdx') || f.path.endsWith('.md'));
|
|
44
|
+
const jsonFiles = content.files.filter((f)=>f.path.endsWith('.json'));
|
|
45
|
+
const otherFiles = content.files.filter((f)=>!f.path.endsWith('.mdx') && !f.path.endsWith('.md') && !f.path.endsWith('.json'));
|
|
46
|
+
// Calculate days since creation
|
|
47
|
+
const createdDate = new Date(content.createdAt);
|
|
48
|
+
const now = new Date();
|
|
49
|
+
const daysSinceCreation = Math.floor((now.getTime() - createdDate.getTime()) / (1000 * 60 * 60 * 24));
|
|
50
|
+
return NextResponse.json({
|
|
51
|
+
slug,
|
|
52
|
+
stats: {
|
|
53
|
+
totalFiles: content.files.length,
|
|
54
|
+
mdxPages: mdxFiles.length,
|
|
55
|
+
configFiles: jsonFiles.length,
|
|
56
|
+
otherFiles: otherFiles.length,
|
|
57
|
+
totalSizeBytes: totalSize,
|
|
58
|
+
totalSizeKB: Math.round(totalSize / 1024 * 10) / 10
|
|
59
|
+
},
|
|
60
|
+
deployment: {
|
|
61
|
+
createdAt: content.createdAt,
|
|
62
|
+
updatedAt: content.updatedAt,
|
|
63
|
+
lastDeployedAt: keyData?.lastUsedAt || content.updatedAt,
|
|
64
|
+
daysSinceCreation
|
|
65
|
+
},
|
|
66
|
+
// Placeholder for future analytics
|
|
67
|
+
analytics: {
|
|
68
|
+
pageViews: 'coming soon',
|
|
69
|
+
uniqueVisitors: 'coming soon'
|
|
70
|
+
}
|
|
71
|
+
});
|
|
72
|
+
} catch (error) {
|
|
73
|
+
console.error('[Projects Stats API] Error:', error);
|
|
74
|
+
return NextResponse.json({
|
|
75
|
+
error: 'Internal server error'
|
|
76
|
+
}, {
|
|
77
|
+
status: 500
|
|
78
|
+
});
|
|
79
|
+
}
|
|
80
|
+
}
|
|
@@ -0,0 +1,176 @@
|
|
|
1
|
+
import { NextResponse } from 'next/server';
|
|
2
|
+
import { isSubdomainRegistered, registerSubdomain, generateApiKey, storeProjectApiKey } from '@/lib/storage/blob';
|
|
3
|
+
// Reserved/blacklisted subdomains (same as in subdomains/check)
|
|
4
|
+
const BLACKLISTED_SUBDOMAINS = new Set([
|
|
5
|
+
'www',
|
|
6
|
+
'api',
|
|
7
|
+
'app',
|
|
8
|
+
'admin',
|
|
9
|
+
'dashboard',
|
|
10
|
+
'console',
|
|
11
|
+
'panel',
|
|
12
|
+
'manage',
|
|
13
|
+
'login',
|
|
14
|
+
'signin',
|
|
15
|
+
'signup',
|
|
16
|
+
'register',
|
|
17
|
+
'auth',
|
|
18
|
+
'oauth',
|
|
19
|
+
'sso',
|
|
20
|
+
'devdoc',
|
|
21
|
+
'brainfish',
|
|
22
|
+
'docs',
|
|
23
|
+
'documentation',
|
|
24
|
+
'help',
|
|
25
|
+
'support',
|
|
26
|
+
'status',
|
|
27
|
+
'blog',
|
|
28
|
+
'news',
|
|
29
|
+
'mail',
|
|
30
|
+
'email',
|
|
31
|
+
'smtp',
|
|
32
|
+
'ftp',
|
|
33
|
+
'cdn',
|
|
34
|
+
'static',
|
|
35
|
+
'assets',
|
|
36
|
+
'images',
|
|
37
|
+
'files',
|
|
38
|
+
'media',
|
|
39
|
+
'download',
|
|
40
|
+
'downloads',
|
|
41
|
+
'test',
|
|
42
|
+
'testing',
|
|
43
|
+
'dev',
|
|
44
|
+
'development',
|
|
45
|
+
'staging',
|
|
46
|
+
'prod',
|
|
47
|
+
'production',
|
|
48
|
+
'demo',
|
|
49
|
+
'example',
|
|
50
|
+
'sandbox',
|
|
51
|
+
'preview',
|
|
52
|
+
'secure',
|
|
53
|
+
'ssl',
|
|
54
|
+
'security',
|
|
55
|
+
'abuse',
|
|
56
|
+
'spam',
|
|
57
|
+
'postmaster',
|
|
58
|
+
'hostmaster',
|
|
59
|
+
'webmaster',
|
|
60
|
+
'null',
|
|
61
|
+
'undefined',
|
|
62
|
+
'true',
|
|
63
|
+
'false',
|
|
64
|
+
'root',
|
|
65
|
+
'system',
|
|
66
|
+
'localhost'
|
|
67
|
+
]);
|
|
68
|
+
/**
|
|
69
|
+
* POST /api/projects/register
|
|
70
|
+
* Register a new project and generate an API key
|
|
71
|
+
*
|
|
72
|
+
* Body:
|
|
73
|
+
* name: string - Project display name
|
|
74
|
+
* slug: string - Project slug (URL-safe identifier)
|
|
75
|
+
* subdomain: string - Desired subdomain for <subdomain>.devdoc.sh
|
|
76
|
+
*
|
|
77
|
+
* Returns:
|
|
78
|
+
* success: boolean
|
|
79
|
+
* projectId: string
|
|
80
|
+
* slug: string
|
|
81
|
+
* subdomain: string
|
|
82
|
+
* apiKey: string
|
|
83
|
+
* url: string
|
|
84
|
+
*/ export async function POST(request) {
|
|
85
|
+
try {
|
|
86
|
+
const body = await request.json();
|
|
87
|
+
const { name, slug, subdomain } = body;
|
|
88
|
+
// Validate required fields
|
|
89
|
+
if (!name || !slug || !subdomain) {
|
|
90
|
+
return NextResponse.json({
|
|
91
|
+
error: 'name, slug, and subdomain are required'
|
|
92
|
+
}, {
|
|
93
|
+
status: 400
|
|
94
|
+
});
|
|
95
|
+
}
|
|
96
|
+
// Normalize subdomain
|
|
97
|
+
const normalizedSubdomain = subdomain.toLowerCase().trim();
|
|
98
|
+
// Validate subdomain format
|
|
99
|
+
if (normalizedSubdomain.length < 3) {
|
|
100
|
+
return NextResponse.json({
|
|
101
|
+
error: 'Subdomain must be at least 3 characters'
|
|
102
|
+
}, {
|
|
103
|
+
status: 400
|
|
104
|
+
});
|
|
105
|
+
}
|
|
106
|
+
if (normalizedSubdomain.length > 63) {
|
|
107
|
+
return NextResponse.json({
|
|
108
|
+
error: 'Subdomain must be 63 characters or less'
|
|
109
|
+
}, {
|
|
110
|
+
status: 400
|
|
111
|
+
});
|
|
112
|
+
}
|
|
113
|
+
if (!/^[a-z0-9]([a-z0-9-]*[a-z0-9])?$/.test(normalizedSubdomain)) {
|
|
114
|
+
return NextResponse.json({
|
|
115
|
+
error: 'Subdomain must start and end with alphanumeric characters'
|
|
116
|
+
}, {
|
|
117
|
+
status: 400
|
|
118
|
+
});
|
|
119
|
+
}
|
|
120
|
+
// Check blacklist
|
|
121
|
+
if (BLACKLISTED_SUBDOMAINS.has(normalizedSubdomain)) {
|
|
122
|
+
return NextResponse.json({
|
|
123
|
+
error: `"${normalizedSubdomain}" is a reserved subdomain`
|
|
124
|
+
}, {
|
|
125
|
+
status: 400
|
|
126
|
+
});
|
|
127
|
+
}
|
|
128
|
+
// Use subdomain as the project identifier (slug)
|
|
129
|
+
// This ensures subdomain uniqueness = project uniqueness
|
|
130
|
+
const projectSlug = normalizedSubdomain;
|
|
131
|
+
// Check if subdomain already registered (O(1) lookup from registry)
|
|
132
|
+
const exists = await isSubdomainRegistered(projectSlug);
|
|
133
|
+
if (exists) {
|
|
134
|
+
return NextResponse.json({
|
|
135
|
+
error: `Project with subdomain "${normalizedSubdomain}" already exists`
|
|
136
|
+
}, {
|
|
137
|
+
status: 409
|
|
138
|
+
});
|
|
139
|
+
}
|
|
140
|
+
// Generate API key and project ID
|
|
141
|
+
const apiKey = generateApiKey();
|
|
142
|
+
const projectId = `proj_${projectSlug}_${Math.random().toString(36).substring(2, 8)}`;
|
|
143
|
+
// Register in domain registry (O(1) lookup for future checks)
|
|
144
|
+
await registerSubdomain(projectSlug, projectId, name, apiKey);
|
|
145
|
+
// Also store API key in project folder (for backwards compatibility)
|
|
146
|
+
await storeProjectApiKey(projectSlug, apiKey);
|
|
147
|
+
const response = {
|
|
148
|
+
success: true,
|
|
149
|
+
projectId,
|
|
150
|
+
slug: projectSlug,
|
|
151
|
+
subdomain: normalizedSubdomain,
|
|
152
|
+
apiKey,
|
|
153
|
+
url: `https://${normalizedSubdomain}.devdoc.sh`
|
|
154
|
+
};
|
|
155
|
+
console.log(`[Projects API] Registered new project: ${projectSlug}`);
|
|
156
|
+
return NextResponse.json(response, {
|
|
157
|
+
status: 201
|
|
158
|
+
});
|
|
159
|
+
} catch (error) {
|
|
160
|
+
console.error('[Projects API] Register Error:', error);
|
|
161
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
162
|
+
// Provide clearer error messages for common issues
|
|
163
|
+
let userMessage = 'Failed to register project';
|
|
164
|
+
if (message.includes('blob already exists')) {
|
|
165
|
+
userMessage = 'Project storage conflict. Please try again or use a different subdomain.';
|
|
166
|
+
} else if (message.includes('network') || message.includes('fetch')) {
|
|
167
|
+
userMessage = 'Network error connecting to storage service';
|
|
168
|
+
}
|
|
169
|
+
return NextResponse.json({
|
|
170
|
+
error: userMessage,
|
|
171
|
+
details: message
|
|
172
|
+
}, {
|
|
173
|
+
status: 500
|
|
174
|
+
});
|
|
175
|
+
}
|
|
176
|
+
}
|
|
@@ -0,0 +1,139 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* API Proxy Route
|
|
3
|
+
*
|
|
4
|
+
* Forwards requests to external APIs to bypass CORS restrictions
|
|
5
|
+
* and ensure auth headers are sent properly.
|
|
6
|
+
*/ import { NextResponse } from 'next/server';
|
|
7
|
+
// Headers that should not be forwarded
|
|
8
|
+
const EXCLUDED_REQUEST_HEADERS = [
|
|
9
|
+
'host',
|
|
10
|
+
'connection',
|
|
11
|
+
'content-length',
|
|
12
|
+
'transfer-encoding',
|
|
13
|
+
'keep-alive',
|
|
14
|
+
'upgrade',
|
|
15
|
+
'proxy-connection',
|
|
16
|
+
'proxy-authorization'
|
|
17
|
+
];
|
|
18
|
+
const EXCLUDED_RESPONSE_HEADERS = [
|
|
19
|
+
'transfer-encoding',
|
|
20
|
+
'connection',
|
|
21
|
+
'keep-alive',
|
|
22
|
+
'content-encoding'
|
|
23
|
+
];
|
|
24
|
+
export async function POST(request) {
|
|
25
|
+
try {
|
|
26
|
+
const body = await request.json();
|
|
27
|
+
const { url, method, headers, requestBody } = body;
|
|
28
|
+
if (!url) {
|
|
29
|
+
return NextResponse.json({
|
|
30
|
+
error: 'URL is required'
|
|
31
|
+
}, {
|
|
32
|
+
status: 400
|
|
33
|
+
});
|
|
34
|
+
}
|
|
35
|
+
// Validate URL to prevent SSRF attacks
|
|
36
|
+
const parsedUrl = new URL(url);
|
|
37
|
+
const allowedProtocols = [
|
|
38
|
+
'http:',
|
|
39
|
+
'https:'
|
|
40
|
+
];
|
|
41
|
+
if (!allowedProtocols.includes(parsedUrl.protocol)) {
|
|
42
|
+
return NextResponse.json({
|
|
43
|
+
error: 'Invalid URL protocol'
|
|
44
|
+
}, {
|
|
45
|
+
status: 400
|
|
46
|
+
});
|
|
47
|
+
}
|
|
48
|
+
// Build headers for the proxied request
|
|
49
|
+
const proxyHeaders = {};
|
|
50
|
+
for (const [key, value] of Object.entries(headers || {})){
|
|
51
|
+
if (!EXCLUDED_REQUEST_HEADERS.includes(key.toLowerCase())) {
|
|
52
|
+
proxyHeaders[key] = value;
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
// Make the proxied request
|
|
56
|
+
const startTime = Date.now();
|
|
57
|
+
const response = await fetch(url, {
|
|
58
|
+
method: method || 'GET',
|
|
59
|
+
headers: proxyHeaders,
|
|
60
|
+
body: requestBody || undefined
|
|
61
|
+
});
|
|
62
|
+
const responseTime = Date.now() - startTime;
|
|
63
|
+
// Read response body
|
|
64
|
+
const contentType = response.headers.get('content-type') || '';
|
|
65
|
+
let responseBody = null;
|
|
66
|
+
if (contentType.includes('application/json') || contentType.includes('text/')) {
|
|
67
|
+
responseBody = await response.text();
|
|
68
|
+
} else {
|
|
69
|
+
// For binary content, convert to base64
|
|
70
|
+
const buffer = await response.arrayBuffer();
|
|
71
|
+
responseBody = Buffer.from(buffer).toString('base64');
|
|
72
|
+
}
|
|
73
|
+
// Build response headers
|
|
74
|
+
const responseHeaders = {};
|
|
75
|
+
response.headers.forEach((value, key)=>{
|
|
76
|
+
if (!EXCLUDED_RESPONSE_HEADERS.includes(key.toLowerCase())) {
|
|
77
|
+
responseHeaders[key] = value;
|
|
78
|
+
}
|
|
79
|
+
});
|
|
80
|
+
return NextResponse.json({
|
|
81
|
+
status: response.status,
|
|
82
|
+
statusText: response.statusText,
|
|
83
|
+
headers: responseHeaders,
|
|
84
|
+
body: responseBody,
|
|
85
|
+
responseTime,
|
|
86
|
+
size: responseBody ? new Blob([
|
|
87
|
+
responseBody
|
|
88
|
+
]).size : 0
|
|
89
|
+
});
|
|
90
|
+
} catch (error) {
|
|
91
|
+
return NextResponse.json({
|
|
92
|
+
status: 0,
|
|
93
|
+
statusText: 'Proxy Error',
|
|
94
|
+
headers: {},
|
|
95
|
+
body: null,
|
|
96
|
+
responseTime: 0,
|
|
97
|
+
size: 0,
|
|
98
|
+
error: error instanceof Error ? error.message : 'Proxy request failed'
|
|
99
|
+
});
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
// Also support GET for simple requests
|
|
103
|
+
export async function GET(request) {
|
|
104
|
+
const url = request.nextUrl.searchParams.get('url');
|
|
105
|
+
if (!url) {
|
|
106
|
+
return NextResponse.json({
|
|
107
|
+
error: 'URL query parameter is required'
|
|
108
|
+
}, {
|
|
109
|
+
status: 400
|
|
110
|
+
});
|
|
111
|
+
}
|
|
112
|
+
try {
|
|
113
|
+
const parsedUrl = new URL(url);
|
|
114
|
+
const allowedProtocols = [
|
|
115
|
+
'http:',
|
|
116
|
+
'https:'
|
|
117
|
+
];
|
|
118
|
+
if (!allowedProtocols.includes(parsedUrl.protocol)) {
|
|
119
|
+
return NextResponse.json({
|
|
120
|
+
error: 'Invalid URL protocol'
|
|
121
|
+
}, {
|
|
122
|
+
status: 400
|
|
123
|
+
});
|
|
124
|
+
}
|
|
125
|
+
const response = await fetch(url);
|
|
126
|
+
const body = await response.text();
|
|
127
|
+
return NextResponse.json({
|
|
128
|
+
status: response.status,
|
|
129
|
+
statusText: response.statusText,
|
|
130
|
+
body
|
|
131
|
+
});
|
|
132
|
+
} catch (error) {
|
|
133
|
+
return NextResponse.json({
|
|
134
|
+
error: error instanceof Error ? error.message : 'Request failed'
|
|
135
|
+
}, {
|
|
136
|
+
status: 500
|
|
137
|
+
});
|
|
138
|
+
}
|
|
139
|
+
}
|
|
@@ -0,0 +1,156 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Streaming Proxy Route
|
|
3
|
+
*
|
|
4
|
+
* Forwards requests and streams responses back for SSE endpoints.
|
|
5
|
+
* Auto-detects SSE based on response content-type.
|
|
6
|
+
*/ // Headers that should not be forwarded
|
|
7
|
+
const EXCLUDED_REQUEST_HEADERS = [
|
|
8
|
+
'host',
|
|
9
|
+
'connection',
|
|
10
|
+
'content-length',
|
|
11
|
+
'transfer-encoding',
|
|
12
|
+
'keep-alive',
|
|
13
|
+
'upgrade',
|
|
14
|
+
'proxy-connection',
|
|
15
|
+
'proxy-authorization'
|
|
16
|
+
];
|
|
17
|
+
export async function POST(request) {
|
|
18
|
+
const startTime = Date.now();
|
|
19
|
+
try {
|
|
20
|
+
const body = await request.json();
|
|
21
|
+
const { url, method, headers, requestBody } = body;
|
|
22
|
+
if (!url) {
|
|
23
|
+
return new Response(JSON.stringify({
|
|
24
|
+
error: 'URL is required'
|
|
25
|
+
}), {
|
|
26
|
+
status: 400,
|
|
27
|
+
headers: {
|
|
28
|
+
'Content-Type': 'application/json'
|
|
29
|
+
}
|
|
30
|
+
});
|
|
31
|
+
}
|
|
32
|
+
// Validate URL
|
|
33
|
+
const parsedUrl = new URL(url);
|
|
34
|
+
const allowedProtocols = [
|
|
35
|
+
'http:',
|
|
36
|
+
'https:'
|
|
37
|
+
];
|
|
38
|
+
if (!allowedProtocols.includes(parsedUrl.protocol)) {
|
|
39
|
+
return new Response(JSON.stringify({
|
|
40
|
+
error: 'Invalid URL protocol'
|
|
41
|
+
}), {
|
|
42
|
+
status: 400,
|
|
43
|
+
headers: {
|
|
44
|
+
'Content-Type': 'application/json'
|
|
45
|
+
}
|
|
46
|
+
});
|
|
47
|
+
}
|
|
48
|
+
// Build headers for the proxied request
|
|
49
|
+
const proxyHeaders = {};
|
|
50
|
+
for (const [key, value] of Object.entries(headers || {})){
|
|
51
|
+
if (!EXCLUDED_REQUEST_HEADERS.includes(key.toLowerCase())) {
|
|
52
|
+
proxyHeaders[key] = value;
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
// Make the proxied request
|
|
56
|
+
const response = await fetch(url, {
|
|
57
|
+
method: method || 'GET',
|
|
58
|
+
headers: proxyHeaders,
|
|
59
|
+
body: requestBody || undefined
|
|
60
|
+
});
|
|
61
|
+
const contentType = response.headers.get('content-type') || '';
|
|
62
|
+
// Check if this is a streaming response (SSE or similar)
|
|
63
|
+
const isStreaming = contentType.includes('text/event-stream') || contentType.includes('application/x-ndjson') || contentType.includes('application/stream+json');
|
|
64
|
+
if (isStreaming && response.body) {
|
|
65
|
+
// For streaming responses, wrap in our SSE format
|
|
66
|
+
const encoder = new TextEncoder();
|
|
67
|
+
// Send metadata first, then stream chunks
|
|
68
|
+
const stream = new ReadableStream({
|
|
69
|
+
async start (controller) {
|
|
70
|
+
// Send metadata
|
|
71
|
+
const metadata = {
|
|
72
|
+
type: 'metadata',
|
|
73
|
+
status: response.status,
|
|
74
|
+
statusText: response.statusText,
|
|
75
|
+
headers: Object.fromEntries(response.headers.entries()),
|
|
76
|
+
contentType
|
|
77
|
+
};
|
|
78
|
+
controller.enqueue(encoder.encode(`data: ${JSON.stringify(metadata)}\n\n`));
|
|
79
|
+
const reader = response.body.getReader();
|
|
80
|
+
const decoder = new TextDecoder();
|
|
81
|
+
let totalSize = 0;
|
|
82
|
+
try {
|
|
83
|
+
while(true){
|
|
84
|
+
const { done, value } = await reader.read();
|
|
85
|
+
if (done) {
|
|
86
|
+
controller.enqueue(encoder.encode(`data: ${JSON.stringify({
|
|
87
|
+
type: 'done',
|
|
88
|
+
totalSize
|
|
89
|
+
})}\n\n`));
|
|
90
|
+
controller.close();
|
|
91
|
+
break;
|
|
92
|
+
}
|
|
93
|
+
const chunk = decoder.decode(value, {
|
|
94
|
+
stream: true
|
|
95
|
+
});
|
|
96
|
+
totalSize += value.length;
|
|
97
|
+
// Send chunk
|
|
98
|
+
const chunkEvent = {
|
|
99
|
+
type: 'chunk',
|
|
100
|
+
data: chunk,
|
|
101
|
+
size: value.length
|
|
102
|
+
};
|
|
103
|
+
controller.enqueue(encoder.encode(`data: ${JSON.stringify(chunkEvent)}\n\n`));
|
|
104
|
+
}
|
|
105
|
+
} catch (error) {
|
|
106
|
+
const errorEvent = {
|
|
107
|
+
type: 'error',
|
|
108
|
+
error: error instanceof Error ? error.message : 'Stream error'
|
|
109
|
+
};
|
|
110
|
+
controller.enqueue(encoder.encode(`data: ${JSON.stringify(errorEvent)}\n\n`));
|
|
111
|
+
controller.close();
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
});
|
|
115
|
+
return new Response(stream, {
|
|
116
|
+
headers: {
|
|
117
|
+
'Content-Type': 'text/event-stream',
|
|
118
|
+
'Cache-Control': 'no-cache',
|
|
119
|
+
'Connection': 'keep-alive'
|
|
120
|
+
}
|
|
121
|
+
});
|
|
122
|
+
}
|
|
123
|
+
// For non-streaming responses, return as JSON
|
|
124
|
+
const responseBody = await response.text();
|
|
125
|
+
const responseHeaders = {};
|
|
126
|
+
response.headers.forEach((value, key)=>{
|
|
127
|
+
responseHeaders[key] = value;
|
|
128
|
+
});
|
|
129
|
+
return new Response(JSON.stringify({
|
|
130
|
+
status: response.status,
|
|
131
|
+
statusText: response.statusText,
|
|
132
|
+
headers: responseHeaders,
|
|
133
|
+
body: responseBody,
|
|
134
|
+
size: new Blob([
|
|
135
|
+
responseBody
|
|
136
|
+
]).size,
|
|
137
|
+
responseTime: Date.now() - startTime,
|
|
138
|
+
isStreaming: false
|
|
139
|
+
}), {
|
|
140
|
+
headers: {
|
|
141
|
+
'Content-Type': 'application/json'
|
|
142
|
+
}
|
|
143
|
+
});
|
|
144
|
+
} catch (error) {
|
|
145
|
+
return new Response(JSON.stringify({
|
|
146
|
+
status: 0,
|
|
147
|
+
statusText: 'Proxy Error',
|
|
148
|
+
error: error instanceof Error ? error.message : 'Proxy request failed'
|
|
149
|
+
}), {
|
|
150
|
+
status: 500,
|
|
151
|
+
headers: {
|
|
152
|
+
'Content-Type': 'application/json'
|
|
153
|
+
}
|
|
154
|
+
});
|
|
155
|
+
}
|
|
156
|
+
}
|