@brainfish-ai/devdoc 0.1.21

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (268) hide show
  1. package/LICENSE +33 -0
  2. package/README.md +415 -0
  3. package/bin/devdoc.js +13 -0
  4. package/dist/cli/commands/build.d.ts +5 -0
  5. package/dist/cli/commands/build.js +87 -0
  6. package/dist/cli/commands/check.d.ts +1 -0
  7. package/dist/cli/commands/check.js +143 -0
  8. package/dist/cli/commands/create.d.ts +24 -0
  9. package/dist/cli/commands/create.js +387 -0
  10. package/dist/cli/commands/deploy.d.ts +9 -0
  11. package/dist/cli/commands/deploy.js +433 -0
  12. package/dist/cli/commands/dev.d.ts +6 -0
  13. package/dist/cli/commands/dev.js +139 -0
  14. package/dist/cli/commands/init.d.ts +11 -0
  15. package/dist/cli/commands/init.js +238 -0
  16. package/dist/cli/commands/keys.d.ts +12 -0
  17. package/dist/cli/commands/keys.js +165 -0
  18. package/dist/cli/commands/start.d.ts +5 -0
  19. package/dist/cli/commands/start.js +56 -0
  20. package/dist/cli/commands/upload.d.ts +13 -0
  21. package/dist/cli/commands/upload.js +238 -0
  22. package/dist/cli/commands/whoami.d.ts +8 -0
  23. package/dist/cli/commands/whoami.js +91 -0
  24. package/dist/cli/index.d.ts +1 -0
  25. package/dist/cli/index.js +106 -0
  26. package/dist/config/index.d.ts +80 -0
  27. package/dist/config/index.js +133 -0
  28. package/dist/constants.d.ts +9 -0
  29. package/dist/constants.js +13 -0
  30. package/dist/index.d.ts +7 -0
  31. package/dist/index.js +12 -0
  32. package/dist/utils/logger.d.ts +16 -0
  33. package/dist/utils/logger.js +61 -0
  34. package/dist/utils/paths.d.ts +16 -0
  35. package/dist/utils/paths.js +50 -0
  36. package/package.json +51 -0
  37. package/renderer/app/api/assets/[...path]/route.ts +123 -0
  38. package/renderer/app/api/assets/route.ts +124 -0
  39. package/renderer/app/api/assets/upload/route.ts +177 -0
  40. package/renderer/app/api/auth-schemes/route.ts +77 -0
  41. package/renderer/app/api/chat/route.ts +858 -0
  42. package/renderer/app/api/codegen/route.ts +72 -0
  43. package/renderer/app/api/collections/route.ts +1016 -0
  44. package/renderer/app/api/debug/route.ts +53 -0
  45. package/renderer/app/api/deploy/route.ts +234 -0
  46. package/renderer/app/api/device/route.ts +42 -0
  47. package/renderer/app/api/docs/route.ts +187 -0
  48. package/renderer/app/api/keys/regenerate/route.ts +80 -0
  49. package/renderer/app/api/openapi-spec/route.ts +151 -0
  50. package/renderer/app/api/projects/[slug]/route.ts +153 -0
  51. package/renderer/app/api/projects/[slug]/stats/route.ts +96 -0
  52. package/renderer/app/api/projects/register/route.ts +152 -0
  53. package/renderer/app/api/proxy/route.ts +149 -0
  54. package/renderer/app/api/proxy-stream/route.ts +168 -0
  55. package/renderer/app/api/redirects/route.ts +47 -0
  56. package/renderer/app/api/schema/route.ts +65 -0
  57. package/renderer/app/api/subdomains/check/route.ts +172 -0
  58. package/renderer/app/api/suggestions/route.ts +144 -0
  59. package/renderer/app/favicon.ico +0 -0
  60. package/renderer/app/globals.css +1103 -0
  61. package/renderer/app/layout.tsx +47 -0
  62. package/renderer/app/llms-full.txt/route.ts +346 -0
  63. package/renderer/app/llms.txt/route.ts +279 -0
  64. package/renderer/app/page.tsx +14 -0
  65. package/renderer/app/robots.txt/route.ts +84 -0
  66. package/renderer/app/sitemap.xml/route.ts +199 -0
  67. package/renderer/components/docs/index.ts +12 -0
  68. package/renderer/components/docs/mdx/accordion.tsx +169 -0
  69. package/renderer/components/docs/mdx/badge.tsx +132 -0
  70. package/renderer/components/docs/mdx/callouts.tsx +154 -0
  71. package/renderer/components/docs/mdx/cards.tsx +213 -0
  72. package/renderer/components/docs/mdx/changelog.tsx +120 -0
  73. package/renderer/components/docs/mdx/code-block.tsx +186 -0
  74. package/renderer/components/docs/mdx/code-group.tsx +421 -0
  75. package/renderer/components/docs/mdx/file-embeds.tsx +105 -0
  76. package/renderer/components/docs/mdx/frame.tsx +112 -0
  77. package/renderer/components/docs/mdx/highlight.tsx +151 -0
  78. package/renderer/components/docs/mdx/iframe.tsx +134 -0
  79. package/renderer/components/docs/mdx/image.tsx +235 -0
  80. package/renderer/components/docs/mdx/index.ts +204 -0
  81. package/renderer/components/docs/mdx/mermaid.tsx +240 -0
  82. package/renderer/components/docs/mdx/param-field.tsx +200 -0
  83. package/renderer/components/docs/mdx/steps.tsx +113 -0
  84. package/renderer/components/docs/mdx/tabs.tsx +86 -0
  85. package/renderer/components/docs/mdx-renderer.tsx +100 -0
  86. package/renderer/components/docs/navigation/breadcrumbs.tsx +76 -0
  87. package/renderer/components/docs/navigation/index.ts +8 -0
  88. package/renderer/components/docs/navigation/page-nav.tsx +64 -0
  89. package/renderer/components/docs/navigation/sidebar.tsx +515 -0
  90. package/renderer/components/docs/navigation/toc.tsx +113 -0
  91. package/renderer/components/docs/notice.tsx +105 -0
  92. package/renderer/components/docs-header.tsx +274 -0
  93. package/renderer/components/docs-viewer/agent/agent-chat.tsx +2076 -0
  94. package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +90 -0
  95. package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +49 -0
  96. package/renderer/components/docs-viewer/agent/cards/index.tsx +50 -0
  97. package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +212 -0
  98. package/renderer/components/docs-viewer/agent/cards/types.ts +84 -0
  99. package/renderer/components/docs-viewer/agent/chat-message.tsx +17 -0
  100. package/renderer/components/docs-viewer/agent/index.tsx +6 -0
  101. package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +119 -0
  102. package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +46 -0
  103. package/renderer/components/docs-viewer/agent/messages/index.ts +17 -0
  104. package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +721 -0
  105. package/renderer/components/docs-viewer/agent/messages/types.ts +61 -0
  106. package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +24 -0
  107. package/renderer/components/docs-viewer/agent/messages/user-message.tsx +51 -0
  108. package/renderer/components/docs-viewer/code-editor/index.tsx +2 -0
  109. package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +1283 -0
  110. package/renderer/components/docs-viewer/content/changelog-page.tsx +331 -0
  111. package/renderer/components/docs-viewer/content/doc-page.tsx +285 -0
  112. package/renderer/components/docs-viewer/content/documentation-viewer.tsx +17 -0
  113. package/renderer/components/docs-viewer/content/index.tsx +29 -0
  114. package/renderer/components/docs-viewer/content/introduction.tsx +21 -0
  115. package/renderer/components/docs-viewer/content/request-details.tsx +330 -0
  116. package/renderer/components/docs-viewer/content/sections/auth.tsx +69 -0
  117. package/renderer/components/docs-viewer/content/sections/body.tsx +66 -0
  118. package/renderer/components/docs-viewer/content/sections/headers.tsx +43 -0
  119. package/renderer/components/docs-viewer/content/sections/overview.tsx +40 -0
  120. package/renderer/components/docs-viewer/content/sections/parameters.tsx +43 -0
  121. package/renderer/components/docs-viewer/content/sections/responses.tsx +87 -0
  122. package/renderer/components/docs-viewer/global-auth-modal.tsx +352 -0
  123. package/renderer/components/docs-viewer/index.tsx +1466 -0
  124. package/renderer/components/docs-viewer/playground/auth-editor.tsx +280 -0
  125. package/renderer/components/docs-viewer/playground/body-editor.tsx +221 -0
  126. package/renderer/components/docs-viewer/playground/code-editor.tsx +224 -0
  127. package/renderer/components/docs-viewer/playground/code-snippet.tsx +387 -0
  128. package/renderer/components/docs-viewer/playground/graphql-playground.tsx +745 -0
  129. package/renderer/components/docs-viewer/playground/index.tsx +671 -0
  130. package/renderer/components/docs-viewer/playground/key-value-editor.tsx +261 -0
  131. package/renderer/components/docs-viewer/playground/method-selector.tsx +60 -0
  132. package/renderer/components/docs-viewer/playground/request-builder.tsx +179 -0
  133. package/renderer/components/docs-viewer/playground/request-tabs.tsx +237 -0
  134. package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +21 -0
  135. package/renderer/components/docs-viewer/playground/response-cards/index.tsx +93 -0
  136. package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +16 -0
  137. package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +23 -0
  138. package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +268 -0
  139. package/renderer/components/docs-viewer/playground/response-cards/types.ts +82 -0
  140. package/renderer/components/docs-viewer/playground/response-viewer.tsx +43 -0
  141. package/renderer/components/docs-viewer/search/index.ts +2 -0
  142. package/renderer/components/docs-viewer/search/search-dialog.tsx +331 -0
  143. package/renderer/components/docs-viewer/search/use-search.ts +117 -0
  144. package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +431 -0
  145. package/renderer/components/docs-viewer/shared/method-badge.tsx +41 -0
  146. package/renderer/components/docs-viewer/shared/schema-viewer.tsx +349 -0
  147. package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +239 -0
  148. package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +316 -0
  149. package/renderer/components/docs-viewer/sidebar/index.tsx +343 -0
  150. package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +202 -0
  151. package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +118 -0
  152. package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +226 -0
  153. package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +52 -0
  154. package/renderer/components/theme-provider.tsx +11 -0
  155. package/renderer/components/theme-toggle.tsx +76 -0
  156. package/renderer/components/ui/badge.tsx +46 -0
  157. package/renderer/components/ui/button.tsx +59 -0
  158. package/renderer/components/ui/dialog.tsx +118 -0
  159. package/renderer/components/ui/dropdown-menu.tsx +257 -0
  160. package/renderer/components/ui/input.tsx +21 -0
  161. package/renderer/components/ui/label.tsx +24 -0
  162. package/renderer/components/ui/navigation-menu.tsx +168 -0
  163. package/renderer/components/ui/select.tsx +190 -0
  164. package/renderer/components/ui/spinner.tsx +114 -0
  165. package/renderer/components/ui/tabs.tsx +66 -0
  166. package/renderer/components/ui/tooltip.tsx +61 -0
  167. package/renderer/hooks/use-code-copy.ts +88 -0
  168. package/renderer/hooks/use-openapi-title.ts +44 -0
  169. package/renderer/lib/api-docs/agent/index.ts +6 -0
  170. package/renderer/lib/api-docs/agent/indexer.ts +323 -0
  171. package/renderer/lib/api-docs/agent/spec-summary.ts +335 -0
  172. package/renderer/lib/api-docs/agent/types.ts +116 -0
  173. package/renderer/lib/api-docs/auth/auth-context.tsx +225 -0
  174. package/renderer/lib/api-docs/auth/auth-storage.ts +87 -0
  175. package/renderer/lib/api-docs/auth/crypto.ts +89 -0
  176. package/renderer/lib/api-docs/auth/index.ts +4 -0
  177. package/renderer/lib/api-docs/code-editor/db.ts +164 -0
  178. package/renderer/lib/api-docs/code-editor/hooks.ts +266 -0
  179. package/renderer/lib/api-docs/code-editor/index.ts +6 -0
  180. package/renderer/lib/api-docs/code-editor/mode-context.tsx +207 -0
  181. package/renderer/lib/api-docs/code-editor/types.ts +105 -0
  182. package/renderer/lib/api-docs/codegen/definitions.ts +297 -0
  183. package/renderer/lib/api-docs/codegen/har.ts +251 -0
  184. package/renderer/lib/api-docs/codegen/index.ts +159 -0
  185. package/renderer/lib/api-docs/factories.ts +151 -0
  186. package/renderer/lib/api-docs/index.ts +17 -0
  187. package/renderer/lib/api-docs/mobile-context.tsx +112 -0
  188. package/renderer/lib/api-docs/navigation-context.tsx +88 -0
  189. package/renderer/lib/api-docs/parsers/graphql/README.md +129 -0
  190. package/renderer/lib/api-docs/parsers/graphql/index.ts +91 -0
  191. package/renderer/lib/api-docs/parsers/graphql/parser.ts +491 -0
  192. package/renderer/lib/api-docs/parsers/graphql/transformer.ts +246 -0
  193. package/renderer/lib/api-docs/parsers/graphql/types.ts +283 -0
  194. package/renderer/lib/api-docs/parsers/openapi/README.md +32 -0
  195. package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +60 -0
  196. package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +574 -0
  197. package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +403 -0
  198. package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +232 -0
  199. package/renderer/lib/api-docs/parsers/openapi/index.ts +171 -0
  200. package/renderer/lib/api-docs/parsers/openapi/transformer.ts +277 -0
  201. package/renderer/lib/api-docs/parsers/openapi/validator.ts +31 -0
  202. package/renderer/lib/api-docs/playground/context.tsx +107 -0
  203. package/renderer/lib/api-docs/playground/navigation-context.tsx +124 -0
  204. package/renderer/lib/api-docs/playground/request-builder.ts +223 -0
  205. package/renderer/lib/api-docs/playground/request-runner.ts +282 -0
  206. package/renderer/lib/api-docs/playground/types.ts +35 -0
  207. package/renderer/lib/api-docs/types.ts +269 -0
  208. package/renderer/lib/api-docs/utils.ts +311 -0
  209. package/renderer/lib/cache.ts +193 -0
  210. package/renderer/lib/docs/config/index.ts +29 -0
  211. package/renderer/lib/docs/config/loader.ts +142 -0
  212. package/renderer/lib/docs/config/schema.ts +298 -0
  213. package/renderer/lib/docs/index.ts +12 -0
  214. package/renderer/lib/docs/mdx/compiler.ts +176 -0
  215. package/renderer/lib/docs/mdx/frontmatter.ts +80 -0
  216. package/renderer/lib/docs/mdx/index.ts +26 -0
  217. package/renderer/lib/docs/navigation/generator.ts +348 -0
  218. package/renderer/lib/docs/navigation/index.ts +12 -0
  219. package/renderer/lib/docs/navigation/types.ts +123 -0
  220. package/renderer/lib/docs-navigation-context.tsx +80 -0
  221. package/renderer/lib/multi-tenant/context.ts +105 -0
  222. package/renderer/lib/storage/blob.ts +845 -0
  223. package/renderer/lib/utils.ts +6 -0
  224. package/renderer/next.config.ts +76 -0
  225. package/renderer/package.json +66 -0
  226. package/renderer/postcss.config.mjs +5 -0
  227. package/renderer/public/assets/images/screenshot.png +0 -0
  228. package/renderer/public/assets/logo/dark.svg +9 -0
  229. package/renderer/public/assets/logo/light.svg +9 -0
  230. package/renderer/public/assets/logo.svg +9 -0
  231. package/renderer/public/file.svg +1 -0
  232. package/renderer/public/globe.svg +1 -0
  233. package/renderer/public/icon.png +0 -0
  234. package/renderer/public/logo.svg +9 -0
  235. package/renderer/public/window.svg +1 -0
  236. package/renderer/tsconfig.json +28 -0
  237. package/templates/basic/README.md +139 -0
  238. package/templates/basic/assets/favicon.svg +4 -0
  239. package/templates/basic/assets/logo.svg +9 -0
  240. package/templates/basic/docs.json +47 -0
  241. package/templates/basic/guides/configuration.mdx +149 -0
  242. package/templates/basic/guides/overview.mdx +96 -0
  243. package/templates/basic/index.mdx +39 -0
  244. package/templates/basic/package.json +14 -0
  245. package/templates/basic/quickstart.mdx +92 -0
  246. package/templates/basic/vercel.json +6 -0
  247. package/templates/graphql/README.md +139 -0
  248. package/templates/graphql/api-reference/schema.graphql +305 -0
  249. package/templates/graphql/assets/favicon.svg +4 -0
  250. package/templates/graphql/assets/logo.svg +9 -0
  251. package/templates/graphql/docs.json +54 -0
  252. package/templates/graphql/guides/configuration.mdx +149 -0
  253. package/templates/graphql/guides/overview.mdx +96 -0
  254. package/templates/graphql/index.mdx +39 -0
  255. package/templates/graphql/package.json +14 -0
  256. package/templates/graphql/quickstart.mdx +92 -0
  257. package/templates/graphql/vercel.json +6 -0
  258. package/templates/openapi/README.md +139 -0
  259. package/templates/openapi/api-reference/openapi.json +419 -0
  260. package/templates/openapi/assets/favicon.svg +4 -0
  261. package/templates/openapi/assets/logo.svg +9 -0
  262. package/templates/openapi/docs.json +61 -0
  263. package/templates/openapi/guides/configuration.mdx +149 -0
  264. package/templates/openapi/guides/overview.mdx +96 -0
  265. package/templates/openapi/index.mdx +39 -0
  266. package/templates/openapi/package.json +14 -0
  267. package/templates/openapi/quickstart.mdx +92 -0
  268. package/templates/openapi/vercel.json +6 -0
@@ -0,0 +1,151 @@
1
+ import { NextResponse } from 'next/server'
2
+ import { CacheUtils } from '@/lib/cache'
3
+
4
+ // Configuration - you can set these as environment variables
5
+ const BRAINFISH_API_BASE_URL = process.env.BRAINFISH_API_BASE_URL || 'https://api.brainfish.ai/api'
6
+ const BRAINFISH_CATALOG_ID = process.env.BRAINFISH_CATALOG_ID || 'your_catalog_id_here'
7
+ const BRAINFISH_JWT_TOKEN = process.env.BRAINFISH_JWT_TOKEN || 'your_jwt_token_here'
8
+
9
+ // Cache configuration
10
+ const CACHE_KEY = `openapi-spec-${BRAINFISH_CATALOG_ID}`
11
+ const CACHE_TTL = 60 * 5 // 5 minutes
12
+ const FALLBACK_CACHE_KEY = `${CACHE_KEY}-fallback`
13
+
14
+ interface OpenApiSpec {
15
+ openapi: string;
16
+ info: {
17
+ title: string;
18
+ version: string;
19
+ description: string;
20
+ };
21
+ paths: Record<string, unknown>;
22
+ servers: Array<{
23
+ url: string;
24
+ description: string;
25
+ }>;
26
+ }
27
+
28
+ async function fetchOpenApiSpec(): Promise<OpenApiSpec> {
29
+ // First, try to get from cache
30
+ const cachedSpec = await CacheUtils.get<OpenApiSpec>(CACHE_KEY)
31
+ if (cachedSpec) {
32
+ return cachedSpec
33
+ }
34
+
35
+ // If not in cache, try to fetch from API
36
+ try {
37
+ // IMPORTANT: Use dots (.) not slashes (/) - e.g., catalogs.openapi-spec NOT catalogs/openapi-spec
38
+ const url = `${BRAINFISH_API_BASE_URL}/catalogs.openapi-spec`
39
+
40
+ console.log('[OpenAPI] Fetching from:', url)
41
+ console.log('[OpenAPI] Request body:', { catalogId: BRAINFISH_CATALOG_ID })
42
+
43
+ const response = await fetch(url, {
44
+ method: 'POST',
45
+ headers: {
46
+ 'Authorization': `Bearer ${BRAINFISH_JWT_TOKEN}`,
47
+ 'Accept': 'application/json',
48
+ 'Content-Type': 'application/json',
49
+ 'User-Agent': 'Brainfish-API-Docs/1.0',
50
+ },
51
+ body: JSON.stringify({
52
+ catalogId: BRAINFISH_CATALOG_ID
53
+ }),
54
+ // Ensure fetch doesn't cache
55
+ cache: 'no-store',
56
+ })
57
+
58
+ console.log('[OpenAPI] Response status:', response.status, response.statusText)
59
+ console.log('[OpenAPI] Response headers:', Object.fromEntries(response.headers.entries()))
60
+
61
+ if (!response.ok) {
62
+ // Try to get error details from response body
63
+ let errorBody = ''
64
+ try {
65
+ errorBody = await response.text()
66
+ console.log('[OpenAPI] Error response body:', errorBody)
67
+ } catch {
68
+ console.log('[OpenAPI] Could not read error response body')
69
+ }
70
+
71
+ throw new Error(`Failed to fetch OpenAPI spec: ${response.status} ${response.statusText}${errorBody ? ` - ${errorBody}` : ''}`)
72
+ }
73
+
74
+ const spec = await response.json() as OpenApiSpec
75
+
76
+ // Cache the successful response
77
+ await CacheUtils.set(CACHE_KEY, spec, CACHE_TTL)
78
+ // Also store as fallback (with longer TTL)
79
+ await CacheUtils.set(FALLBACK_CACHE_KEY, spec, 60 * 60 * 24 * 365)
80
+ return spec
81
+ } catch (error) {
82
+ const errorMessage = error instanceof Error ? error.message : 'Unknown error'
83
+ console.error('[OpenAPI] Error fetching spec:', errorMessage)
84
+
85
+ // Try to get fallback from cache
86
+ const fallbackSpec = await CacheUtils.get<OpenApiSpec>(FALLBACK_CACHE_KEY)
87
+ if (fallbackSpec) {
88
+ console.warn('[OpenAPI] Using fallback cache due to API error')
89
+ return fallbackSpec
90
+ }
91
+
92
+ // If no fallback available, return basic spec
93
+ console.warn('[OpenAPI] No fallback available, using basic spec')
94
+ return {
95
+ openapi: '3.0.0',
96
+ info: {
97
+ title: 'Brainfish API',
98
+ version: '1.0.0',
99
+ description: 'API temporarily unavailable. Please try again later.',
100
+ },
101
+ paths: {},
102
+ servers: [
103
+ {
104
+ url: BRAINFISH_API_BASE_URL.replace('/api', ''),
105
+ description: 'Brainfish API'
106
+ }
107
+ ]
108
+ }
109
+ }
110
+ }
111
+
112
+ export async function GET() {
113
+ try {
114
+ const spec = await fetchOpenApiSpec()
115
+
116
+ return NextResponse.json(spec, {
117
+ headers: {
118
+ 'Content-Type': 'application/json',
119
+ 'Cache-Control': 'no-cache, no-store, must-revalidate', // Disable browser caching
120
+ },
121
+ })
122
+ } catch (error) {
123
+ console.error('[OpenAPI] Unexpected error in endpoint:', error)
124
+
125
+ // As a last resort, try the fallback cache
126
+ const fallbackSpec = await CacheUtils.get<OpenApiSpec>(FALLBACK_CACHE_KEY)
127
+ if (fallbackSpec) {
128
+ console.warn('[OpenAPI] Using fallback cache in error handler')
129
+ return NextResponse.json(fallbackSpec, {
130
+ headers: {
131
+ 'Content-Type': 'application/json',
132
+ 'Cache-Control': 'no-cache, no-store, must-revalidate',
133
+ },
134
+ })
135
+ }
136
+
137
+ return NextResponse.json(
138
+ {
139
+ error: 'OpenAPI specification temporarily unavailable',
140
+ openapi: '3.0.0',
141
+ info: {
142
+ title: 'Brainfish API',
143
+ version: '1.0.0',
144
+ description: 'Service temporarily unavailable. Please try again later.',
145
+ },
146
+ paths: {},
147
+ },
148
+ { status: 503 } // Service Unavailable
149
+ )
150
+ }
151
+ }
@@ -0,0 +1,153 @@
1
+ import { NextRequest, NextResponse } from 'next/server'
2
+ import {
3
+ validateApiKey,
4
+ getProjectContent,
5
+ getProjectApiKey,
6
+ deleteProject,
7
+ projectExists
8
+ } from '@/lib/storage/blob'
9
+
10
+ interface RouteParams {
11
+ params: Promise<{ slug: string }>
12
+ }
13
+
14
+ /**
15
+ * GET /api/projects/[slug]
16
+ * Get detailed project information
17
+ *
18
+ * Headers:
19
+ * Authorization: Bearer <api_key>
20
+ */
21
+ export async function GET(request: NextRequest, { params }: RouteParams) {
22
+ try {
23
+ const { slug } = await params
24
+
25
+ // Validate API key
26
+ const authHeader = request.headers.get('Authorization')
27
+ const apiKey = authHeader?.replace('Bearer ', '')
28
+
29
+ if (!apiKey) {
30
+ return NextResponse.json(
31
+ { error: 'API key required' },
32
+ { status: 401 }
33
+ )
34
+ }
35
+
36
+ // Validate the API key belongs to this project
37
+ const validatedSlug = await validateApiKey(apiKey)
38
+
39
+ if (validatedSlug !== slug) {
40
+ return NextResponse.json(
41
+ { error: 'API key does not match this project' },
42
+ { status: 403 }
43
+ )
44
+ }
45
+
46
+ // Get project details
47
+ const content = await getProjectContent(slug)
48
+ const keyData = await getProjectApiKey(slug)
49
+
50
+ if (!content) {
51
+ return NextResponse.json(
52
+ { error: 'Project not found' },
53
+ { status: 404 }
54
+ )
55
+ }
56
+
57
+ // Parse docs.json
58
+ let docsConfig: Record<string, unknown> = {}
59
+ try {
60
+ docsConfig = JSON.parse(content.docsJson)
61
+ } catch {
62
+ // Ignore parse errors
63
+ }
64
+
65
+ // Build file list (without content for security/size)
66
+ const files = content.files.map(f => ({
67
+ path: f.path,
68
+ size: f.content.length,
69
+ }))
70
+
71
+ return NextResponse.json({
72
+ slug: content.slug,
73
+ name: content.name,
74
+ url: `https://${slug}.devdoc.sh`,
75
+ createdAt: content.createdAt,
76
+ updatedAt: content.updatedAt,
77
+ lastDeployedAt: keyData?.lastUsedAt || content.updatedAt,
78
+ filesCount: content.files.length,
79
+ totalSize: content.files.reduce((sum, f) => sum + f.content.length, 0),
80
+ files,
81
+ config: {
82
+ name: docsConfig.name,
83
+ favicon: docsConfig.favicon,
84
+ navigation: docsConfig.navigation ? 'configured' : 'default',
85
+ },
86
+ })
87
+
88
+ } catch (error) {
89
+ console.error('[Projects API] GET Error:', error)
90
+ return NextResponse.json(
91
+ { error: 'Internal server error' },
92
+ { status: 500 }
93
+ )
94
+ }
95
+ }
96
+
97
+ /**
98
+ * DELETE /api/projects/[slug]
99
+ * Delete a project and all its content
100
+ *
101
+ * Headers:
102
+ * Authorization: Bearer <api_key>
103
+ */
104
+ export async function DELETE(request: NextRequest, { params }: RouteParams) {
105
+ try {
106
+ const { slug } = await params
107
+
108
+ // Validate API key
109
+ const authHeader = request.headers.get('Authorization')
110
+ const apiKey = authHeader?.replace('Bearer ', '')
111
+
112
+ if (!apiKey) {
113
+ return NextResponse.json(
114
+ { error: 'API key required' },
115
+ { status: 401 }
116
+ )
117
+ }
118
+
119
+ // Validate the API key belongs to this project
120
+ const validatedSlug = await validateApiKey(apiKey)
121
+
122
+ if (validatedSlug !== slug) {
123
+ return NextResponse.json(
124
+ { error: 'API key does not match this project' },
125
+ { status: 403 }
126
+ )
127
+ }
128
+
129
+ // Check project exists
130
+ const exists = await projectExists(slug)
131
+ if (!exists) {
132
+ return NextResponse.json(
133
+ { error: 'Project not found' },
134
+ { status: 404 }
135
+ )
136
+ }
137
+
138
+ // Delete the project
139
+ await deleteProject(slug)
140
+
141
+ return NextResponse.json({
142
+ success: true,
143
+ message: `Project ${slug} has been deleted`,
144
+ })
145
+
146
+ } catch (error) {
147
+ console.error('[Projects API] DELETE Error:', error)
148
+ return NextResponse.json(
149
+ { error: 'Internal server error' },
150
+ { status: 500 }
151
+ )
152
+ }
153
+ }
@@ -0,0 +1,96 @@
1
+ import { NextRequest, NextResponse } from 'next/server'
2
+ import {
3
+ validateApiKey,
4
+ getProjectContent,
5
+ getProjectApiKey,
6
+ } from '@/lib/storage/blob'
7
+
8
+ interface RouteParams {
9
+ params: Promise<{ slug: string }>
10
+ }
11
+
12
+ /**
13
+ * GET /api/projects/[slug]/stats
14
+ * Get project statistics
15
+ *
16
+ * Headers:
17
+ * Authorization: Bearer <api_key>
18
+ */
19
+ export async function GET(request: NextRequest, { params }: RouteParams) {
20
+ try {
21
+ const { slug } = await params
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
+ // Validate the API key belongs to this project
35
+ const validatedSlug = await validateApiKey(apiKey)
36
+
37
+ if (validatedSlug !== slug) {
38
+ return NextResponse.json(
39
+ { error: 'API key does not match this project' },
40
+ { status: 403 }
41
+ )
42
+ }
43
+
44
+ // Get project details
45
+ const content = await getProjectContent(slug)
46
+ const keyData = await getProjectApiKey(slug)
47
+
48
+ if (!content) {
49
+ return NextResponse.json(
50
+ { error: 'Project not found' },
51
+ { status: 404 }
52
+ )
53
+ }
54
+
55
+ // Calculate stats
56
+ const totalSize = content.files.reduce((sum, f) => sum + f.content.length, 0)
57
+ const mdxFiles = content.files.filter(f => f.path.endsWith('.mdx') || f.path.endsWith('.md'))
58
+ const jsonFiles = content.files.filter(f => f.path.endsWith('.json'))
59
+ const otherFiles = content.files.filter(f => !f.path.endsWith('.mdx') && !f.path.endsWith('.md') && !f.path.endsWith('.json'))
60
+
61
+ // Calculate days since creation
62
+ const createdDate = new Date(content.createdAt)
63
+ const now = new Date()
64
+ const daysSinceCreation = Math.floor((now.getTime() - createdDate.getTime()) / (1000 * 60 * 60 * 24))
65
+
66
+ return NextResponse.json({
67
+ slug,
68
+ stats: {
69
+ totalFiles: content.files.length,
70
+ mdxPages: mdxFiles.length,
71
+ configFiles: jsonFiles.length,
72
+ otherFiles: otherFiles.length,
73
+ totalSizeBytes: totalSize,
74
+ totalSizeKB: Math.round(totalSize / 1024 * 10) / 10,
75
+ },
76
+ deployment: {
77
+ createdAt: content.createdAt,
78
+ updatedAt: content.updatedAt,
79
+ lastDeployedAt: keyData?.lastUsedAt || content.updatedAt,
80
+ daysSinceCreation,
81
+ },
82
+ // Placeholder for future analytics
83
+ analytics: {
84
+ pageViews: 'coming soon',
85
+ uniqueVisitors: 'coming soon',
86
+ },
87
+ })
88
+
89
+ } catch (error) {
90
+ console.error('[Projects Stats API] Error:', error)
91
+ return NextResponse.json(
92
+ { error: 'Internal server error' },
93
+ { status: 500 }
94
+ )
95
+ }
96
+ }
@@ -0,0 +1,152 @@
1
+ import { NextRequest, NextResponse } from 'next/server'
2
+ import {
3
+ isSubdomainRegistered,
4
+ registerSubdomain,
5
+ generateApiKey,
6
+ storeProjectApiKey,
7
+ } from '@/lib/storage/blob'
8
+
9
+ // Reserved/blacklisted subdomains (same as in subdomains/check)
10
+ const BLACKLISTED_SUBDOMAINS = new Set([
11
+ 'www', 'api', 'app', 'admin', 'dashboard', 'console', 'panel', 'manage',
12
+ 'login', 'signin', 'signup', 'register', 'auth', 'oauth', 'sso',
13
+ 'devdoc', 'brainfish', 'docs', 'documentation', 'help', 'support', 'status', 'blog', 'news',
14
+ 'mail', 'email', 'smtp', 'ftp', 'cdn', 'static', 'assets', 'images', 'files', 'media', 'download', 'downloads',
15
+ 'test', 'testing', 'dev', 'development', 'staging', 'prod', 'production', 'demo', 'example', 'sandbox', 'preview',
16
+ 'secure', 'ssl', 'security', 'abuse', 'spam', 'postmaster', 'hostmaster', 'webmaster',
17
+ 'null', 'undefined', 'true', 'false', 'root', 'system', 'localhost',
18
+ ])
19
+
20
+ interface RegisterRequest {
21
+ name: string
22
+ slug: string
23
+ subdomain: string
24
+ }
25
+
26
+ interface RegisterResponse {
27
+ success: boolean
28
+ projectId: string
29
+ slug: string
30
+ subdomain: string
31
+ apiKey: string
32
+ url: string
33
+ }
34
+
35
+ /**
36
+ * POST /api/projects/register
37
+ * Register a new project and generate an API key
38
+ *
39
+ * Body:
40
+ * name: string - Project display name
41
+ * slug: string - Project slug (URL-safe identifier)
42
+ * subdomain: string - Desired subdomain for <subdomain>.devdoc.sh
43
+ *
44
+ * Returns:
45
+ * success: boolean
46
+ * projectId: string
47
+ * slug: string
48
+ * subdomain: string
49
+ * apiKey: string
50
+ * url: string
51
+ */
52
+ export async function POST(request: NextRequest) {
53
+ try {
54
+ const body = await request.json() as RegisterRequest
55
+ const { name, slug, subdomain } = body
56
+
57
+ // Validate required fields
58
+ if (!name || !slug || !subdomain) {
59
+ return NextResponse.json(
60
+ { error: 'name, slug, and subdomain are required' },
61
+ { status: 400 }
62
+ )
63
+ }
64
+
65
+ // Normalize subdomain
66
+ const normalizedSubdomain = subdomain.toLowerCase().trim()
67
+
68
+ // Validate subdomain format
69
+ if (normalizedSubdomain.length < 3) {
70
+ return NextResponse.json(
71
+ { error: 'Subdomain must be at least 3 characters' },
72
+ { status: 400 }
73
+ )
74
+ }
75
+
76
+ if (normalizedSubdomain.length > 63) {
77
+ return NextResponse.json(
78
+ { error: 'Subdomain must be 63 characters or less' },
79
+ { status: 400 }
80
+ )
81
+ }
82
+
83
+ if (!/^[a-z0-9]([a-z0-9-]*[a-z0-9])?$/.test(normalizedSubdomain)) {
84
+ return NextResponse.json(
85
+ { error: 'Subdomain must start and end with alphanumeric characters' },
86
+ { status: 400 }
87
+ )
88
+ }
89
+
90
+ // Check blacklist
91
+ if (BLACKLISTED_SUBDOMAINS.has(normalizedSubdomain)) {
92
+ return NextResponse.json(
93
+ { error: `"${normalizedSubdomain}" is a reserved subdomain` },
94
+ { status: 400 }
95
+ )
96
+ }
97
+
98
+ // Use subdomain as the project identifier (slug)
99
+ // This ensures subdomain uniqueness = project uniqueness
100
+ const projectSlug = normalizedSubdomain
101
+
102
+ // Check if subdomain already registered (O(1) lookup from registry)
103
+ const exists = await isSubdomainRegistered(projectSlug)
104
+
105
+ if (exists) {
106
+ return NextResponse.json(
107
+ { error: `Project with subdomain "${normalizedSubdomain}" already exists` },
108
+ { status: 409 }
109
+ )
110
+ }
111
+
112
+ // Generate API key and project ID
113
+ const apiKey = generateApiKey()
114
+ const projectId = `proj_${projectSlug}_${Math.random().toString(36).substring(2, 8)}`
115
+
116
+ // Register in domain registry (O(1) lookup for future checks)
117
+ await registerSubdomain(projectSlug, projectId, name, apiKey)
118
+
119
+ // Also store API key in project folder (for backwards compatibility)
120
+ await storeProjectApiKey(projectSlug, apiKey)
121
+
122
+ const response: RegisterResponse = {
123
+ success: true,
124
+ projectId,
125
+ slug: projectSlug,
126
+ subdomain: normalizedSubdomain,
127
+ apiKey,
128
+ url: `https://${normalizedSubdomain}.devdoc.sh`,
129
+ }
130
+
131
+ console.log(`[Projects API] Registered new project: ${projectSlug}`)
132
+
133
+ return NextResponse.json(response, { status: 201 })
134
+
135
+ } catch (error) {
136
+ console.error('[Projects API] Register Error:', error)
137
+ const message = error instanceof Error ? error.message : String(error)
138
+
139
+ // Provide clearer error messages for common issues
140
+ let userMessage = 'Failed to register project'
141
+ if (message.includes('blob already exists')) {
142
+ userMessage = 'Project storage conflict. Please try again or use a different subdomain.'
143
+ } else if (message.includes('network') || message.includes('fetch')) {
144
+ userMessage = 'Network error connecting to storage service'
145
+ }
146
+
147
+ return NextResponse.json(
148
+ { error: userMessage, details: message },
149
+ { status: 500 }
150
+ )
151
+ }
152
+ }
@@ -0,0 +1,149 @@
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
+ */
7
+
8
+ import { NextRequest, NextResponse } from 'next/server'
9
+
10
+ // Headers that should not be forwarded
11
+ const EXCLUDED_REQUEST_HEADERS = [
12
+ 'host',
13
+ 'connection',
14
+ 'content-length',
15
+ 'transfer-encoding',
16
+ 'keep-alive',
17
+ 'upgrade',
18
+ 'proxy-connection',
19
+ 'proxy-authorization',
20
+ ]
21
+
22
+ const EXCLUDED_RESPONSE_HEADERS = [
23
+ 'transfer-encoding',
24
+ 'connection',
25
+ 'keep-alive',
26
+ 'content-encoding',
27
+ ]
28
+
29
+ export async function POST(request: NextRequest) {
30
+ try {
31
+ const body = await request.json()
32
+
33
+ const { url, method, headers, requestBody } = body as {
34
+ url: string
35
+ method: string
36
+ headers: Record<string, string>
37
+ requestBody?: string | null
38
+ }
39
+
40
+ if (!url) {
41
+ return NextResponse.json(
42
+ { error: 'URL is required' },
43
+ { status: 400 }
44
+ )
45
+ }
46
+
47
+ // Validate URL to prevent SSRF attacks
48
+ const parsedUrl = new URL(url)
49
+ const allowedProtocols = ['http:', 'https:']
50
+ if (!allowedProtocols.includes(parsedUrl.protocol)) {
51
+ return NextResponse.json(
52
+ { error: 'Invalid URL protocol' },
53
+ { status: 400 }
54
+ )
55
+ }
56
+
57
+ // Build headers for the proxied request
58
+ const proxyHeaders: Record<string, string> = {}
59
+ for (const [key, value] of Object.entries(headers || {})) {
60
+ if (!EXCLUDED_REQUEST_HEADERS.includes(key.toLowerCase())) {
61
+ proxyHeaders[key] = value
62
+ }
63
+ }
64
+
65
+ // Make the proxied request
66
+ const startTime = Date.now()
67
+ const response = await fetch(url, {
68
+ method: method || 'GET',
69
+ headers: proxyHeaders,
70
+ body: requestBody || undefined,
71
+ })
72
+ const responseTime = Date.now() - startTime
73
+
74
+ // Read response body
75
+ const contentType = response.headers.get('content-type') || ''
76
+ let responseBody: string | null = null
77
+
78
+ if (contentType.includes('application/json') || contentType.includes('text/')) {
79
+ responseBody = await response.text()
80
+ } else {
81
+ // For binary content, convert to base64
82
+ const buffer = await response.arrayBuffer()
83
+ responseBody = Buffer.from(buffer).toString('base64')
84
+ }
85
+
86
+ // Build response headers
87
+ const responseHeaders: Record<string, string> = {}
88
+ response.headers.forEach((value, key) => {
89
+ if (!EXCLUDED_RESPONSE_HEADERS.includes(key.toLowerCase())) {
90
+ responseHeaders[key] = value
91
+ }
92
+ })
93
+
94
+ return NextResponse.json({
95
+ status: response.status,
96
+ statusText: response.statusText,
97
+ headers: responseHeaders,
98
+ body: responseBody,
99
+ responseTime,
100
+ size: responseBody ? new Blob([responseBody]).size : 0,
101
+ })
102
+ } catch (error) {
103
+ return NextResponse.json({
104
+ status: 0,
105
+ statusText: 'Proxy Error',
106
+ headers: {},
107
+ body: null,
108
+ responseTime: 0,
109
+ size: 0,
110
+ error: error instanceof Error ? error.message : 'Proxy request failed',
111
+ })
112
+ }
113
+ }
114
+
115
+ // Also support GET for simple requests
116
+ export async function GET(request: NextRequest) {
117
+ const url = request.nextUrl.searchParams.get('url')
118
+
119
+ if (!url) {
120
+ return NextResponse.json(
121
+ { error: 'URL query parameter is required' },
122
+ { status: 400 }
123
+ )
124
+ }
125
+
126
+ try {
127
+ const parsedUrl = new URL(url)
128
+ const allowedProtocols = ['http:', 'https:']
129
+ if (!allowedProtocols.includes(parsedUrl.protocol)) {
130
+ return NextResponse.json(
131
+ { error: 'Invalid URL protocol' },
132
+ { status: 400 }
133
+ )
134
+ }
135
+
136
+ const response = await fetch(url)
137
+ const body = await response.text()
138
+
139
+ return NextResponse.json({
140
+ status: response.status,
141
+ statusText: response.statusText,
142
+ body,
143
+ })
144
+ } catch (error) {
145
+ return NextResponse.json({
146
+ error: error instanceof Error ? error.message : 'Request failed',
147
+ }, { status: 500 })
148
+ }
149
+ }