@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,403 @@
1
+ /**
2
+ * OpenAPI Body Extractors
3
+ *
4
+ * Ported from Hoppscotch's openapi/index.ts
5
+ * Handles request body parsing for v2 and v3
6
+ */
7
+
8
+ /* eslint-disable @typescript-eslint/no-explicit-any */
9
+
10
+ import type {
11
+ OpenAPI,
12
+ OpenAPIV2,
13
+ OpenAPIV3,
14
+ OpenAPIV3_1 as OpenAPIV31,
15
+ } from 'openapi-types'
16
+ import type {
17
+ BrainfishRESTReqBody,
18
+ FormDataKeyValue,
19
+ } from '../../../types'
20
+ import { knownContentTypes, isOpenAPIV3Operation, type OpenAPIOperationType } from './index'
21
+
22
+ /**
23
+ * Resolve a $ref reference in the schema
24
+ */
25
+ function resolveRef(doc: OpenAPI.Document, ref: string): OpenAPIV3.SchemaObject | null {
26
+ if (!ref.startsWith('#/')) return null
27
+
28
+ try {
29
+ const path = ref.slice(2).split('/')
30
+ let current: any = doc
31
+
32
+ for (const segment of path) {
33
+ if (current && typeof current === 'object' && segment in current) {
34
+ current = current[segment]
35
+ } else {
36
+ return null
37
+ }
38
+ }
39
+
40
+ return current as OpenAPIV3.SchemaObject
41
+ } catch {
42
+ return null
43
+ }
44
+ }
45
+
46
+ /**
47
+ * Generate a sample value based on schema type
48
+ */
49
+ function generateSampleValue(
50
+ schema: OpenAPIV3.SchemaObject | OpenAPIV31.SchemaObject | OpenAPIV3.ReferenceObject,
51
+ doc: OpenAPI.Document,
52
+ required: boolean = false
53
+ ): any {
54
+ try {
55
+ // Handle $ref
56
+ if ('$ref' in schema) {
57
+ const resolved = resolveRef(doc, schema.$ref)
58
+ if (resolved) {
59
+ return generateSampleValue(resolved, doc, required)
60
+ }
61
+ return null
62
+ }
63
+
64
+ const schemaObj = schema as OpenAPIV3.SchemaObject
65
+
66
+ // Use example if provided
67
+ if (schemaObj.example !== undefined) {
68
+ return schemaObj.example
69
+ }
70
+
71
+ // Use default if provided
72
+ if (schemaObj.default !== undefined) {
73
+ return schemaObj.default
74
+ }
75
+
76
+ // Use enum first value if provided
77
+ if (schemaObj.enum && schemaObj.enum.length > 0) {
78
+ return schemaObj.enum[0]
79
+ }
80
+
81
+ // Generate based on type
82
+ switch (schemaObj.type) {
83
+ case 'string':
84
+ if (schemaObj.format === 'date') return '2024-01-01'
85
+ if (schemaObj.format === 'date-time') return '2024-01-01T00:00:00Z'
86
+ if (schemaObj.format === 'email') return 'user@example.com'
87
+ if (schemaObj.format === 'uri' || schemaObj.format === 'url') return 'https://example.com'
88
+ if (schemaObj.format === 'uuid') return '00000000-0000-0000-0000-000000000000'
89
+ return required ? 'string' : ''
90
+
91
+ case 'number':
92
+ case 'integer':
93
+ if (schemaObj.minimum !== undefined) return schemaObj.minimum
94
+ return 0
95
+
96
+ case 'boolean':
97
+ return false
98
+
99
+ case 'array':
100
+ if (schemaObj.items) {
101
+ return [generateSampleValue(schemaObj.items, doc, true)]
102
+ }
103
+ return []
104
+
105
+ case 'object':
106
+ return generateObjectFromSchema(schemaObj, doc)
107
+
108
+ default:
109
+ // If no type but has properties, treat as object
110
+ if (schemaObj.properties) {
111
+ return generateObjectFromSchema(schemaObj, doc)
112
+ }
113
+ return null
114
+ }
115
+ } catch {
116
+ return null
117
+ }
118
+ }
119
+
120
+ /**
121
+ * Generate a sample object from an OpenAPI schema
122
+ * Required fields are listed first
123
+ */
124
+ function generateObjectFromSchema(
125
+ schema: OpenAPIV3.SchemaObject | OpenAPIV31.SchemaObject,
126
+ doc: OpenAPI.Document
127
+ ): Record<string, any> {
128
+ try {
129
+ const result: Record<string, any> = {}
130
+ const requiredFields = new Set(schema.required || [])
131
+ const properties = schema.properties || {}
132
+
133
+ // Sort properties to put required fields first
134
+ const sortedEntries = Object.entries(properties).sort(([keyA], [keyB]) => {
135
+ const aRequired = requiredFields.has(keyA)
136
+ const bRequired = requiredFields.has(keyB)
137
+ if (aRequired && !bRequired) return -1
138
+ if (!aRequired && bRequired) return 1
139
+ return 0
140
+ })
141
+
142
+ for (const [key, propSchema] of sortedEntries) {
143
+ const isRequired = requiredFields.has(key)
144
+ const value = generateSampleValue(propSchema as OpenAPIV3.SchemaObject, doc, isRequired)
145
+ if (value !== null) {
146
+ result[key] = value
147
+ }
148
+ }
149
+
150
+ return result
151
+ } catch {
152
+ return {}
153
+ }
154
+ }
155
+
156
+ /**
157
+ * Generate example from OpenAPI v3 media object
158
+ */
159
+ function generateExampleFromMedia(
160
+ media: OpenAPIV3.MediaTypeObject | OpenAPIV31.MediaTypeObject,
161
+ doc: OpenAPI.Document
162
+ ): any {
163
+ try {
164
+ // First check for explicit examples
165
+ if (media.example !== undefined) {
166
+ return media.example
167
+ }
168
+
169
+ if (media.examples && Object.keys(media.examples).length > 0) {
170
+ const firstExampleKey = Object.keys(media.examples)[0]
171
+ const firstExample = media.examples[firstExampleKey]
172
+ if ('value' in firstExample) {
173
+ return (firstExample as OpenAPIV3.ExampleObject).value
174
+ }
175
+ }
176
+
177
+ // Generate from schema
178
+ const schema = media.schema as OpenAPIV3.SchemaObject | OpenAPIV31.SchemaObject | OpenAPIV3.ReferenceObject | undefined
179
+ if (!schema) return null
180
+
181
+ // Handle $ref at schema level
182
+ if ('$ref' in schema) {
183
+ const resolved = resolveRef(doc, schema.$ref)
184
+ if (resolved) {
185
+ if (resolved.type === 'object' || resolved.properties) {
186
+ return generateObjectFromSchema(resolved, doc)
187
+ }
188
+ return generateSampleValue(resolved, doc, true)
189
+ }
190
+ return null
191
+ }
192
+
193
+ if (schema.type === 'object' || schema.properties) {
194
+ return generateObjectFromSchema(schema, doc)
195
+ }
196
+
197
+ return generateSampleValue(schema, doc, true)
198
+ } catch {
199
+ return null
200
+ }
201
+ }
202
+
203
+ // Legacy function for v2 compatibility
204
+ function generateRequestBodyExampleFromOpenAPIV2Body(
205
+ op: OpenAPIV2.OperationObject,
206
+ doc: OpenAPI.Document
207
+ ): string | null {
208
+ try {
209
+ const bodyParam = (op.parameters ?? []).find(
210
+ (param) => (param as OpenAPIV2.Parameter).in === 'body'
211
+ ) as OpenAPIV2.InBodyParameterObject | undefined
212
+
213
+ if (!bodyParam?.schema) return null
214
+
215
+ const schema = bodyParam.schema as OpenAPIV3.SchemaObject
216
+ if (schema.type === 'object' || schema.properties) {
217
+ const example = generateObjectFromSchema(schema, doc)
218
+ return JSON.stringify(example, null, 2)
219
+ }
220
+
221
+ return null
222
+ } catch {
223
+ return null
224
+ }
225
+ }
226
+
227
+ function generateV3Example(media: OpenAPIV3.MediaTypeObject, doc: OpenAPI.Document): any {
228
+ return generateExampleFromMedia(media, doc)
229
+ }
230
+
231
+ function generateV31Example(media: OpenAPIV31.MediaTypeObject, doc: OpenAPI.Document): any {
232
+ return generateExampleFromMedia(media, doc)
233
+ }
234
+
235
+ /**
236
+ * Parses OpenAPI v2 request body
237
+ */
238
+ export function parseOpenAPIV2Body(
239
+ op: OpenAPIV2.OperationObject,
240
+ doc?: OpenAPI.Document
241
+ ): BrainfishRESTReqBody {
242
+ const obj = (op.consumes ?? [])[0] as string | undefined
243
+
244
+ // Not a content-type we support
245
+ if (!obj || !(obj in knownContentTypes)) {
246
+ return { contentType: null, body: null }
247
+ }
248
+
249
+ // For form data types, extract form fields
250
+ if (
251
+ obj === 'multipart/form-data' ||
252
+ obj === 'application/x-www-form-urlencoded'
253
+ ) {
254
+ const formDataValues = ((op.parameters ?? []) as OpenAPIV2.Parameter[])
255
+ .filter((param) => param.in === 'formData')
256
+ .map(
257
+ (param) =>
258
+ <FormDataKeyValue>{
259
+ key: param.name,
260
+ isFile: param.type === 'file',
261
+ value: '',
262
+ active: true,
263
+ }
264
+ )
265
+
266
+ return obj === 'application/x-www-form-urlencoded'
267
+ ? {
268
+ contentType: obj,
269
+ body: formDataValues.map(({ key }) => `${key}: `).join('\n'),
270
+ }
271
+ : { contentType: obj, body: formDataValues }
272
+ }
273
+
274
+ // For other content types (JSON, XML, etc.)
275
+ const bodyParam = (op.parameters ?? []).find(
276
+ (param) => (param as OpenAPIV2.Parameter).in === 'body'
277
+ ) as OpenAPIV2.InBodyParameterObject | undefined
278
+
279
+ if (bodyParam && doc) {
280
+ const result = generateRequestBodyExampleFromOpenAPIV2Body(op, doc)
281
+ if (result) {
282
+ return {
283
+ contentType: obj as any,
284
+ body: result,
285
+ }
286
+ }
287
+ }
288
+
289
+ // Fallback to empty body for textual content types
290
+ return { contentType: obj as any, body: '' }
291
+ }
292
+
293
+ /**
294
+ * Parses OpenAPI v3 form data body
295
+ */
296
+ export function parseOpenAPIV3BodyFormData(
297
+ contentType: 'multipart/form-data' | 'application/x-www-form-urlencoded',
298
+ mediaObj: OpenAPIV3.MediaTypeObject | OpenAPIV31.MediaTypeObject
299
+ ): BrainfishRESTReqBody {
300
+ const schema = mediaObj.schema as
301
+ | OpenAPIV3.SchemaObject
302
+ | OpenAPIV31.SchemaObject
303
+ | undefined
304
+
305
+ if (!schema || schema.type !== 'object') {
306
+ return contentType === 'application/x-www-form-urlencoded'
307
+ ? { contentType, body: '' }
308
+ : { contentType, body: [] }
309
+ }
310
+
311
+ const keys = Object.keys(schema.properties ?? {})
312
+
313
+ if (contentType === 'application/x-www-form-urlencoded') {
314
+ return {
315
+ contentType,
316
+ body: keys.map((key) => `${key}: `).join('\n'),
317
+ }
318
+ }
319
+
320
+ return {
321
+ contentType,
322
+ body: keys.map(
323
+ (key) => <FormDataKeyValue>{ key, value: '', isFile: false, active: true }
324
+ ),
325
+ }
326
+ }
327
+
328
+ /**
329
+ * Parses OpenAPI v3 request body
330
+ */
331
+ export function parseOpenAPIV3Body(
332
+ doc: OpenAPI.Document,
333
+ op: OpenAPIV3.OperationObject | OpenAPIV31.OperationObject
334
+ ): BrainfishRESTReqBody {
335
+ const objs = Object.entries(
336
+ (
337
+ op.requestBody as
338
+ | OpenAPIV3.RequestBodyObject
339
+ | OpenAPIV31.RequestBodyObject
340
+ | undefined
341
+ )?.content ?? {}
342
+ )
343
+
344
+ if (objs.length === 0) return { contentType: null, body: null }
345
+
346
+ // We only take the first definition
347
+ const [contentType, media]: [
348
+ string,
349
+ OpenAPIV3.MediaTypeObject | OpenAPIV31.MediaTypeObject,
350
+ ] = objs[0]
351
+
352
+ if (!(contentType in knownContentTypes)) {
353
+ return { contentType: null, body: null }
354
+ }
355
+
356
+ // Handle form data types
357
+ if (
358
+ contentType === 'multipart/form-data' ||
359
+ contentType === 'application/x-www-form-urlencoded'
360
+ ) {
361
+ return parseOpenAPIV3BodyFormData(contentType, media)
362
+ }
363
+
364
+ // For other content types (JSON, XML, etc.), try to generate sample from schema
365
+ try {
366
+ const docAny = doc as any
367
+ const isV31 = docAny.openapi && docAny.openapi.startsWith('3.1')
368
+
369
+ let sampleBody: any
370
+ if (isV31) {
371
+ sampleBody = generateV31Example(media as any, doc)
372
+ } else {
373
+ sampleBody = generateV3Example(media as any, doc)
374
+ }
375
+
376
+ if (sampleBody !== null && sampleBody !== undefined) {
377
+ return {
378
+ contentType: contentType as any,
379
+ body:
380
+ typeof sampleBody === 'string'
381
+ ? sampleBody
382
+ : JSON.stringify(sampleBody, null, 2),
383
+ }
384
+ }
385
+ } catch {
386
+ // Fall through to empty body
387
+ }
388
+
389
+ // Fallback to empty body for textual content types
390
+ return { contentType: contentType as any, body: '' }
391
+ }
392
+
393
+ /**
394
+ * Parses OpenAPI request body (v2 or v3)
395
+ */
396
+ export function parseOpenAPIBody(
397
+ doc: OpenAPI.Document,
398
+ op: OpenAPIOperationType
399
+ ): BrainfishRESTReqBody {
400
+ return isOpenAPIV3Operation(doc, op)
401
+ ? parseOpenAPIV3Body(doc, op)
402
+ : parseOpenAPIV2Body(op as OpenAPIV2.OperationObject, doc)
403
+ }
@@ -0,0 +1,232 @@
1
+ /**
2
+ * OpenAPI Extractors
3
+ *
4
+ * Ported from Hoppscotch's openapi/index.ts
5
+ * Converts fp-ts functional programming to native TypeScript
6
+ */
7
+
8
+ /* eslint-disable @typescript-eslint/no-unused-vars */
9
+
10
+ import type {
11
+ OpenAPI,
12
+ OpenAPIV2,
13
+ OpenAPIV3,
14
+ OpenAPIV3_1 as OpenAPIV31,
15
+ } from 'openapi-types'
16
+ import type {
17
+ BrainfishRESTParam,
18
+ BrainfishRESTHeader,
19
+ BrainfishRESTRequestVariable,
20
+ BrainfishRESTRequestResponses,
21
+ BrainfishRESTResponseOriginalRequest,
22
+ } from '../../../types'
23
+ import { isNumeric, getStatusCodeReasonPhrase } from '../../../utils'
24
+
25
+ // Type definitions
26
+ export type OpenAPIPathInfoType =
27
+ | OpenAPIV2.PathItemObject<Record<string, unknown>>
28
+ | OpenAPIV3.PathItemObject<Record<string, unknown>>
29
+ | OpenAPIV31.PathItemObject<Record<string, unknown>>
30
+
31
+ export type OpenAPIParamsType =
32
+ | OpenAPIV2.ParameterObject
33
+ | OpenAPIV3.ParameterObject
34
+ | OpenAPIV31.ParameterObject
35
+
36
+ export type OpenAPIOperationType =
37
+ | OpenAPIV2.OperationObject
38
+ | OpenAPIV3.OperationObject
39
+ | OpenAPIV31.OperationObject
40
+
41
+ // Known content types (from Hoppscotch)
42
+ const knownContentTypes: Record<string, boolean> = {
43
+ 'application/json': true,
44
+ 'application/xml': true,
45
+ 'text/plain': true,
46
+ 'text/html': true,
47
+ 'application/x-www-form-urlencoded': true,
48
+ 'multipart/form-data': true,
49
+ 'application/octet-stream': true,
50
+ }
51
+
52
+ /**
53
+ * Replaces OpenAPI path templating ({id}) with Brainfish templating (<<id>>)
54
+ */
55
+ export function replaceOpenApiPathTemplating(path: string): string {
56
+ return path.replace(/{/g, '<<').replace(/}/g, '>>')
57
+ }
58
+
59
+ /**
60
+ * Parses OpenAPI query parameters
61
+ */
62
+ export function parseOpenAPIParams(
63
+ params: OpenAPIParamsType[]
64
+ ): BrainfishRESTParam[] {
65
+ return params
66
+ .filter((param) => param.in === 'query')
67
+ .map((param) => ({
68
+ key: param.name,
69
+ value: '', // TODO: Can we parse default values?
70
+ active: true,
71
+ description: param.description ?? '',
72
+ }))
73
+ }
74
+
75
+ /**
76
+ * Parses OpenAPI path variables
77
+ */
78
+ export function parseOpenAPIVariables(
79
+ variables: OpenAPIParamsType[]
80
+ ): BrainfishRESTRequestVariable[] {
81
+ return variables
82
+ .filter((param) => param.in === 'path')
83
+ .map((param) => ({
84
+ key: param.name,
85
+ value: '', // TODO: Can we parse default values?
86
+ active: true,
87
+ }))
88
+ }
89
+
90
+ /**
91
+ * Parses OpenAPI header parameters
92
+ */
93
+ export function parseOpenAPIHeaders(
94
+ params: OpenAPIParamsType[]
95
+ ): BrainfishRESTHeader[] {
96
+ return params
97
+ .filter((param) => param.in === 'header')
98
+ .map((header) => ({
99
+ key: header.name,
100
+ value: '', // TODO: Can we parse default values?
101
+ active: true,
102
+ description: header.description ?? '',
103
+ }))
104
+ }
105
+
106
+ /**
107
+ * Parses OpenAPI v3 responses
108
+ */
109
+ export function parseOpenAPIV3Responses(
110
+ op: OpenAPIV3.OperationObject | OpenAPIV31.OperationObject,
111
+ originalRequest: BrainfishRESTResponseOriginalRequest
112
+ ): BrainfishRESTRequestResponses {
113
+ const responses = op.responses
114
+ if (!responses) return {}
115
+
116
+ const res: BrainfishRESTRequestResponses = {}
117
+
118
+ for (const [key, value] of Object.entries(responses)) {
119
+ const response = value as
120
+ | OpenAPIV3.ResponseObject
121
+ | OpenAPIV31.ResponseObject
122
+
123
+ const contentType = Object.keys(response.content ?? {})[0]
124
+ const body = response.content?.[contentType]
125
+
126
+ const name = response.description ?? key
127
+ const code = isNumeric(key) ? Number(key) : 200
128
+ const status = getStatusCodeReasonPhrase(code)
129
+
130
+ const headers: BrainfishRESTHeader[] = [
131
+ {
132
+ key: 'content-type',
133
+ value: contentType ?? 'application/json',
134
+ description: '',
135
+ active: true,
136
+ },
137
+ ]
138
+
139
+ let stringifiedBody = ''
140
+
141
+ try {
142
+ stringifiedBody = JSON.stringify(body ?? '')
143
+ } catch (e) {
144
+ // Ignore circular reference errors
145
+ }
146
+
147
+ res[name] = {
148
+ name,
149
+ status,
150
+ code,
151
+ headers,
152
+ body: stringifiedBody,
153
+ originalRequest,
154
+ }
155
+ }
156
+
157
+ return res
158
+ }
159
+
160
+ /**
161
+ * Parses OpenAPI v2 responses
162
+ */
163
+ export function parseOpenAPIV2Responses(
164
+ op: OpenAPIV2.OperationObject,
165
+ originalRequest: BrainfishRESTResponseOriginalRequest
166
+ ): BrainfishRESTRequestResponses {
167
+ const responses = op.responses
168
+ if (!responses) return {}
169
+
170
+ const res: BrainfishRESTRequestResponses = {}
171
+
172
+ for (const [key, value] of Object.entries(responses)) {
173
+ const response = value as OpenAPIV2.ResponseObject
174
+
175
+ const contentType = Object.keys(response.examples ?? {})[0]
176
+ const body = response.examples?.[contentType]
177
+
178
+ const name = response.description ?? key
179
+ const code = isNumeric(Number(key)) ? Number(key) : 200
180
+ const status = getStatusCodeReasonPhrase(code)
181
+
182
+ const headers: BrainfishRESTHeader[] = [
183
+ {
184
+ key: 'content-type',
185
+ value: contentType ?? 'application/json',
186
+ description: '',
187
+ active: true,
188
+ },
189
+ ]
190
+
191
+ res[name] = {
192
+ name,
193
+ status,
194
+ code,
195
+ headers,
196
+ body: body ?? '',
197
+ originalRequest,
198
+ }
199
+ }
200
+
201
+ return res
202
+ }
203
+
204
+ /**
205
+ * Parses OpenAPI responses (v2 or v3)
206
+ */
207
+ export function parseOpenAPIResponses(
208
+ doc: OpenAPI.Document,
209
+ op: OpenAPIOperationType,
210
+ originalRequest: BrainfishRESTResponseOriginalRequest
211
+ ): BrainfishRESTRequestResponses {
212
+ return isOpenAPIV3Operation(doc, op)
213
+ ? parseOpenAPIV3Responses(op, originalRequest)
214
+ : parseOpenAPIV2Responses(op as OpenAPIV2.OperationObject, originalRequest)
215
+ }
216
+
217
+ /**
218
+ * Checks if an operation is OpenAPI v3
219
+ */
220
+ export function isOpenAPIV3Operation(
221
+ doc: OpenAPI.Document,
222
+ op: OpenAPIOperationType
223
+ ): op is OpenAPIV3.OperationObject | OpenAPIV31.OperationObject {
224
+ return (
225
+ 'openapi' in doc &&
226
+ typeof doc.openapi === 'string' &&
227
+ doc.openapi.startsWith('3.')
228
+ )
229
+ }
230
+
231
+ // Export knownContentTypes for use in body parsers
232
+ export { knownContentTypes }