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