@brainfish-ai/devdoc 0.1.41 → 0.1.43
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/ai-agents/.claude/skills/bootstrap-docs/SKILL.md +710 -79
- package/ai-agents/.claude/skills/check-docs/SKILL.md +83 -8
- package/ai-agents/.claude/skills/create-doc/SKILL.md +267 -55
- package/ai-agents/.claude/skills/update-doc/SKILL.md +162 -63
- package/ai-agents/.cursor/rules/devdoc-bootstrap.mdc +145 -15
- package/ai-agents/.cursor/rules/devdoc-create.mdc +108 -57
- package/ai-agents/.cursor/rules/devdoc-update.mdc +93 -70
- package/ai-agents/.cursor/rules/devdoc.mdc +21 -0
- package/ai-agents/schemas/docs.schema.json +332 -0
- package/ai-agents/schemas/theme.schema.json +243 -0
- package/dist/cli/commands/create.js +4 -9
- package/dist/cli/commands/deploy.js +50 -25
- package/dist/cli/commands/dev.js +19 -10
- package/package.json +3 -2
- package/renderer/app/api/assets/[...path]/route.js +108 -0
- package/renderer/app/api/assets/route.js +114 -0
- package/renderer/app/api/assets/upload/route.js +163 -0
- package/renderer/app/api/auth-schemes/route.js +58 -0
- package/renderer/app/api/chat/route.js +759 -0
- package/renderer/app/api/codegen/route.js +52 -0
- package/renderer/app/api/collections/route.js +675 -0
- package/renderer/app/api/debug/route.js +47 -0
- package/renderer/app/api/deploy/route.js +199 -0
- package/renderer/app/api/device/route.js +36 -0
- package/renderer/app/api/docs/route.js +205 -0
- package/renderer/app/api/domains/add/route.js +121 -0
- package/renderer/app/api/domains/lookup/route.js +43 -0
- package/renderer/app/api/domains/remove/route.js +89 -0
- package/renderer/app/api/domains/status/route.js +140 -0
- package/renderer/app/api/domains/verify/route.js +168 -0
- package/renderer/app/api/keys/regenerate/route.js +71 -0
- package/renderer/app/api/local-assets/[...path]/route.js +108 -0
- package/renderer/app/api/openapi-spec/route.js +73 -0
- package/renderer/app/api/projects/[slug]/route.js +129 -0
- package/renderer/app/api/projects/[slug]/stats/route.js +80 -0
- package/renderer/app/api/projects/register/route.js +176 -0
- package/renderer/app/api/proxy/route.js +139 -0
- package/renderer/app/api/proxy-stream/route.js +156 -0
- package/renderer/app/api/redirects/route.js +35 -0
- package/renderer/app/api/schema/route.js +85 -0
- package/renderer/app/api/subdomains/check/route.js +158 -0
- package/renderer/app/api/suggestions/route.js +175 -0
- package/renderer/app/layout.js +47 -0
- package/renderer/app/llms-full.txt/route.js +257 -0
- package/renderer/app/llms.txt/route.js +219 -0
- package/renderer/app/page.js +12 -0
- package/renderer/app/robots.txt/route.js +66 -0
- package/renderer/app/sitemap.xml/route.js +145 -0
- package/renderer/components/docs/index.js +8 -0
- package/renderer/components/docs/mdx/accordion.js +113 -0
- package/renderer/components/docs/mdx/badge.js +72 -0
- package/renderer/components/docs/mdx/callouts.js +137 -0
- package/renderer/components/docs/mdx/cards.js +175 -0
- package/renderer/components/docs/mdx/changelog.js +100 -0
- package/renderer/components/docs/mdx/code-block.js +147 -0
- package/renderer/components/docs/mdx/code-group.js +287 -0
- package/renderer/components/docs/mdx/file-embeds.js +82 -0
- package/renderer/components/docs/mdx/frame.js +59 -0
- package/renderer/components/docs/mdx/highlight.js +90 -0
- package/renderer/components/docs/mdx/iframe.js +69 -0
- package/renderer/components/docs/mdx/image.js +135 -0
- package/renderer/components/docs/mdx/index.js +134 -0
- package/renderer/components/docs/mdx/landing.js +315 -0
- package/renderer/components/docs/mdx/mermaid.js +212 -0
- package/renderer/components/docs/mdx/param-field.js +112 -0
- package/renderer/components/docs/mdx/steps.js +74 -0
- package/renderer/components/docs/mdx/tabs.js +50 -0
- package/renderer/components/docs/mdx-renderer.js +77 -0
- package/renderer/components/docs/navigation/breadcrumbs.js +64 -0
- package/renderer/components/docs/navigation/index.js +6 -0
- package/renderer/components/docs/navigation/page-nav.js +57 -0
- package/renderer/components/docs/navigation/sidebar.js +375 -0
- package/renderer/components/docs/navigation/toc.js +89 -0
- package/renderer/components/docs/notice.js +77 -0
- package/renderer/components/docs-header.js +202 -0
- package/renderer/components/docs-viewer/agent/agent-chat.js +1930 -0
- package/renderer/components/docs-viewer/agent/cards/debug-context-card.js +107 -0
- package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.js +57 -0
- package/renderer/components/docs-viewer/agent/cards/index.js +45 -0
- package/renderer/components/docs-viewer/agent/cards/response-options-card.js +154 -0
- package/renderer/components/docs-viewer/agent/cards/types.js +22 -0
- package/renderer/components/docs-viewer/agent/chat-message.js +2 -0
- package/renderer/components/docs-viewer/agent/index.js +4 -0
- package/renderer/components/docs-viewer/agent/messages/assistant-message.js +108 -0
- package/renderer/components/docs-viewer/agent/messages/chat-message.js +34 -0
- package/renderer/components/docs-viewer/agent/messages/index.js +6 -0
- package/renderer/components/docs-viewer/agent/messages/tool-call-display.js +1065 -0
- package/renderer/components/docs-viewer/agent/messages/types.js +2 -0
- package/renderer/components/docs-viewer/agent/messages/typing-indicator.js +26 -0
- package/renderer/components/docs-viewer/agent/messages/user-message.js +37 -0
- package/renderer/components/docs-viewer/code-editor/{index.tsx → index.js} +1 -1
- package/renderer/components/docs-viewer/code-editor/notes-mode.js +1338 -0
- package/renderer/components/docs-viewer/content/changelog-page.js +297 -0
- package/renderer/components/docs-viewer/content/doc-page.js +264 -0
- package/renderer/components/docs-viewer/content/documentation-viewer.js +14 -0
- package/renderer/components/docs-viewer/content/index.js +29 -0
- package/renderer/components/docs-viewer/content/not-found-page.js +300 -0
- package/renderer/components/docs-viewer/content/request-details.js +528 -0
- package/renderer/components/docs-viewer/content/sections/auth.js +108 -0
- package/renderer/components/docs-viewer/content/sections/body.js +80 -0
- package/renderer/components/docs-viewer/content/sections/headers.js +64 -0
- package/renderer/components/docs-viewer/content/sections/overview.js +56 -0
- package/renderer/components/docs-viewer/content/sections/parameters.js +64 -0
- package/renderer/components/docs-viewer/content/sections/responses.js +91 -0
- package/renderer/components/docs-viewer/global-auth-modal.js +427 -0
- package/renderer/components/docs-viewer/index.js +1552 -0
- package/renderer/components/docs-viewer/playground/auth-editor.js +418 -0
- package/renderer/components/docs-viewer/playground/body-editor.js +240 -0
- package/renderer/components/docs-viewer/playground/code-editor.js +135 -0
- package/renderer/components/docs-viewer/playground/code-snippet.js +393 -0
- package/renderer/components/docs-viewer/playground/graphql-playground.js +734 -0
- package/renderer/components/docs-viewer/playground/index.js +682 -0
- package/renderer/components/docs-viewer/playground/key-value-editor.js +317 -0
- package/renderer/components/docs-viewer/playground/method-selector.js +65 -0
- package/renderer/components/docs-viewer/playground/request-builder.js +181 -0
- package/renderer/components/docs-viewer/playground/request-tabs.js +240 -0
- package/renderer/components/docs-viewer/playground/response-cards/idle-card.js +42 -0
- package/renderer/components/docs-viewer/playground/response-cards/index.js +72 -0
- package/renderer/components/docs-viewer/playground/response-cards/loading-card.js +24 -0
- package/renderer/components/docs-viewer/playground/response-cards/network-error-card.js +28 -0
- package/renderer/components/docs-viewer/playground/response-cards/response-body-card.js +308 -0
- package/renderer/components/docs-viewer/playground/response-cards/types.js +9 -0
- package/renderer/components/docs-viewer/playground/response-viewer.js +18 -0
- package/renderer/components/docs-viewer/search/index.js +2 -0
- package/renderer/components/docs-viewer/search/search-dialog.js +367 -0
- package/renderer/components/docs-viewer/search/use-search.js +89 -0
- package/renderer/components/docs-viewer/shared/markdown-renderer.js +423 -0
- package/renderer/components/docs-viewer/shared/method-badge.js +23 -0
- package/renderer/components/docs-viewer/shared/schema-viewer.js +321 -0
- package/renderer/components/docs-viewer/sidebar/collection-tree.js +222 -0
- package/renderer/components/docs-viewer/sidebar/endpoint-options.js +512 -0
- package/renderer/components/docs-viewer/sidebar/index.js +196 -0
- package/renderer/components/docs-viewer/sidebar/right-sidebar.js +163 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-group.js +87 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-item.js +172 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-section.js +31 -0
- package/renderer/components/theme-provider.js +10 -0
- package/renderer/components/theme-toggle.js +106 -0
- package/renderer/components/ui/badge.js +29 -0
- package/renderer/components/ui/button.js +40 -0
- package/renderer/components/ui/dialog.js +50 -0
- package/renderer/components/ui/dropdown-menu.js +143 -0
- package/renderer/components/ui/input.js +12 -0
- package/renderer/components/ui/label.js +13 -0
- package/renderer/components/ui/navigation-menu.js +83 -0
- package/renderer/components/ui/select.js +116 -0
- package/renderer/components/ui/spinner.js +92 -0
- package/renderer/components/ui/tabs.js +34 -0
- package/renderer/components/ui/tooltip.js +43 -0
- package/renderer/hooks/use-code-copy.js +76 -0
- package/renderer/hooks/use-openapi-title.js +33 -0
- package/renderer/lib/api-docs/agent/index.js +4 -0
- package/renderer/lib/api-docs/agent/indexer.js +254 -0
- package/renderer/lib/api-docs/agent/spec-summary.js +227 -0
- package/renderer/lib/api-docs/agent/types.js +5 -0
- package/renderer/lib/api-docs/auth/auth-context.js +157 -0
- package/renderer/lib/api-docs/auth/auth-storage.js +66 -0
- package/renderer/lib/api-docs/auth/crypto.js +64 -0
- package/renderer/lib/api-docs/auth/index.js +3 -0
- package/renderer/lib/api-docs/code-editor/db.js +145 -0
- package/renderer/lib/api-docs/code-editor/hooks.js +254 -0
- package/renderer/lib/api-docs/code-editor/{index.ts → index.js} +3 -4
- package/renderer/lib/api-docs/code-editor/mode-context.js +154 -0
- package/renderer/lib/api-docs/code-editor/types.js +53 -0
- package/renderer/lib/api-docs/codegen/definitions.js +258 -0
- package/renderer/lib/api-docs/codegen/har.js +171 -0
- package/renderer/lib/api-docs/codegen/index.js +118 -0
- package/renderer/lib/api-docs/factories.js +136 -0
- package/renderer/lib/api-docs/{index.ts → index.js} +5 -10
- package/renderer/lib/api-docs/mobile-context.js +79 -0
- package/renderer/lib/api-docs/navigation-context.js +62 -0
- package/renderer/lib/api-docs/parsers/graphql/index.js +50 -0
- package/renderer/lib/api-docs/parsers/graphql/parser.js +350 -0
- package/renderer/lib/api-docs/parsers/graphql/transformer.js +215 -0
- package/renderer/lib/api-docs/parsers/graphql/types.js +46 -0
- package/renderer/lib/api-docs/parsers/openapi/dereferencer.js +43 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/auth.js +486 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/body.js +295 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/index.js +132 -0
- package/renderer/lib/api-docs/parsers/openapi/index.js +127 -0
- package/renderer/lib/api-docs/parsers/openapi/transformer.js +192 -0
- package/renderer/lib/api-docs/parsers/openapi/validator.js +24 -0
- package/renderer/lib/api-docs/playground/context.js +65 -0
- package/renderer/lib/api-docs/playground/navigation-context.js +74 -0
- package/renderer/lib/api-docs/playground/request-builder.js +163 -0
- package/renderer/lib/api-docs/playground/request-runner.js +224 -0
- package/renderer/lib/api-docs/playground/types.js +5 -0
- package/renderer/lib/api-docs/types.js +23 -0
- package/renderer/lib/api-docs/utils.js +212 -0
- package/renderer/lib/cache.js +157 -0
- package/renderer/lib/docs/config/domain-schema.js +161 -0
- package/renderer/lib/docs/config/index.js +5 -0
- package/renderer/lib/docs/config/loader.js +113 -0
- package/renderer/lib/docs/config/schema.js +269 -0
- package/renderer/lib/docs/index.js +8 -0
- package/renderer/lib/docs/mdx/compiler.js +128 -0
- package/renderer/lib/docs/mdx/frontmatter.js +73 -0
- package/renderer/lib/docs/mdx/index.js +8 -0
- package/renderer/lib/docs/navigation/generator.js +269 -0
- package/renderer/lib/docs/navigation/index.js +4 -0
- package/renderer/lib/docs/navigation/types.js +9 -0
- package/renderer/lib/docs-navigation-context.js +40 -0
- package/renderer/lib/multi-tenant/context.js +80 -0
- package/renderer/lib/storage/blob.js +767 -0
- package/renderer/lib/utils/icons.js +30 -0
- package/renderer/lib/utils.js +5 -0
- package/renderer/next.config.js +62 -0
- package/renderer/tsconfig.json +23 -5
- package/renderer/app/api/assets/[...path]/route.ts +0 -123
- package/renderer/app/api/assets/route.ts +0 -124
- package/renderer/app/api/assets/upload/route.ts +0 -177
- package/renderer/app/api/auth-schemes/route.ts +0 -77
- package/renderer/app/api/chat/route.ts +0 -858
- package/renderer/app/api/codegen/route.ts +0 -72
- package/renderer/app/api/collections/route.ts +0 -1002
- package/renderer/app/api/debug/route.ts +0 -53
- package/renderer/app/api/deploy/route.ts +0 -234
- package/renderer/app/api/device/route.ts +0 -42
- package/renderer/app/api/docs/route.ts +0 -201
- package/renderer/app/api/domains/add/route.ts +0 -132
- package/renderer/app/api/domains/lookup/route.ts +0 -43
- package/renderer/app/api/domains/remove/route.ts +0 -100
- package/renderer/app/api/domains/status/route.ts +0 -158
- package/renderer/app/api/domains/verify/route.ts +0 -181
- package/renderer/app/api/keys/regenerate/route.ts +0 -80
- package/renderer/app/api/local-assets/[...path]/route.ts +0 -122
- package/renderer/app/api/openapi-spec/route.ts +0 -151
- package/renderer/app/api/projects/[slug]/route.ts +0 -153
- package/renderer/app/api/projects/[slug]/stats/route.ts +0 -96
- package/renderer/app/api/projects/register/route.ts +0 -152
- package/renderer/app/api/proxy/route.ts +0 -149
- package/renderer/app/api/proxy-stream/route.ts +0 -168
- package/renderer/app/api/redirects/route.ts +0 -47
- package/renderer/app/api/schema/route.ts +0 -73
- package/renderer/app/api/subdomains/check/route.ts +0 -172
- package/renderer/app/api/suggestions/route.ts +0 -144
- package/renderer/app/layout.tsx +0 -54
- package/renderer/app/llms-full.txt/route.ts +0 -346
- package/renderer/app/llms.txt/route.ts +0 -279
- package/renderer/app/page.tsx +0 -14
- package/renderer/app/robots.txt/route.ts +0 -84
- package/renderer/app/sitemap.xml/route.ts +0 -199
- package/renderer/components/docs/index.ts +0 -12
- package/renderer/components/docs/mdx/accordion.tsx +0 -169
- package/renderer/components/docs/mdx/badge.tsx +0 -132
- package/renderer/components/docs/mdx/callouts.tsx +0 -154
- package/renderer/components/docs/mdx/cards.tsx +0 -241
- package/renderer/components/docs/mdx/changelog.tsx +0 -120
- package/renderer/components/docs/mdx/code-block.tsx +0 -186
- package/renderer/components/docs/mdx/code-group.tsx +0 -421
- package/renderer/components/docs/mdx/file-embeds.tsx +0 -105
- package/renderer/components/docs/mdx/frame.tsx +0 -112
- package/renderer/components/docs/mdx/highlight.tsx +0 -151
- package/renderer/components/docs/mdx/iframe.tsx +0 -134
- package/renderer/components/docs/mdx/image.tsx +0 -235
- package/renderer/components/docs/mdx/index.ts +0 -237
- package/renderer/components/docs/mdx/landing.tsx +0 -684
- package/renderer/components/docs/mdx/mermaid.tsx +0 -240
- package/renderer/components/docs/mdx/param-field.tsx +0 -200
- package/renderer/components/docs/mdx/steps.tsx +0 -113
- package/renderer/components/docs/mdx/tabs.tsx +0 -86
- package/renderer/components/docs/mdx-renderer.tsx +0 -100
- package/renderer/components/docs/navigation/breadcrumbs.tsx +0 -76
- package/renderer/components/docs/navigation/index.ts +0 -8
- package/renderer/components/docs/navigation/page-nav.tsx +0 -64
- package/renderer/components/docs/navigation/sidebar.tsx +0 -515
- package/renderer/components/docs/navigation/toc.tsx +0 -113
- package/renderer/components/docs/notice.tsx +0 -105
- package/renderer/components/docs-header.tsx +0 -278
- package/renderer/components/docs-viewer/agent/agent-chat.tsx +0 -2076
- package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +0 -90
- package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +0 -49
- package/renderer/components/docs-viewer/agent/cards/index.tsx +0 -50
- package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +0 -212
- package/renderer/components/docs-viewer/agent/cards/types.ts +0 -84
- package/renderer/components/docs-viewer/agent/chat-message.tsx +0 -17
- package/renderer/components/docs-viewer/agent/index.tsx +0 -6
- package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +0 -119
- package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +0 -46
- package/renderer/components/docs-viewer/agent/messages/index.ts +0 -17
- package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +0 -721
- package/renderer/components/docs-viewer/agent/messages/types.ts +0 -61
- package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +0 -24
- package/renderer/components/docs-viewer/agent/messages/user-message.tsx +0 -51
- package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +0 -1283
- package/renderer/components/docs-viewer/content/changelog-page.tsx +0 -331
- package/renderer/components/docs-viewer/content/doc-page.tsx +0 -367
- package/renderer/components/docs-viewer/content/documentation-viewer.tsx +0 -17
- package/renderer/components/docs-viewer/content/index.tsx +0 -29
- package/renderer/components/docs-viewer/content/not-found-page.tsx +0 -330
- package/renderer/components/docs-viewer/content/request-details.tsx +0 -330
- package/renderer/components/docs-viewer/content/sections/auth.tsx +0 -69
- package/renderer/components/docs-viewer/content/sections/body.tsx +0 -66
- package/renderer/components/docs-viewer/content/sections/headers.tsx +0 -43
- package/renderer/components/docs-viewer/content/sections/overview.tsx +0 -40
- package/renderer/components/docs-viewer/content/sections/parameters.tsx +0 -43
- package/renderer/components/docs-viewer/content/sections/responses.tsx +0 -87
- package/renderer/components/docs-viewer/global-auth-modal.tsx +0 -352
- package/renderer/components/docs-viewer/index.tsx +0 -1662
- package/renderer/components/docs-viewer/playground/auth-editor.tsx +0 -280
- package/renderer/components/docs-viewer/playground/body-editor.tsx +0 -221
- package/renderer/components/docs-viewer/playground/code-editor.tsx +0 -224
- package/renderer/components/docs-viewer/playground/code-snippet.tsx +0 -387
- package/renderer/components/docs-viewer/playground/graphql-playground.tsx +0 -745
- package/renderer/components/docs-viewer/playground/index.tsx +0 -671
- package/renderer/components/docs-viewer/playground/key-value-editor.tsx +0 -261
- package/renderer/components/docs-viewer/playground/method-selector.tsx +0 -60
- package/renderer/components/docs-viewer/playground/request-builder.tsx +0 -179
- package/renderer/components/docs-viewer/playground/request-tabs.tsx +0 -237
- package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +0 -21
- package/renderer/components/docs-viewer/playground/response-cards/index.tsx +0 -93
- package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +0 -16
- package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +0 -23
- package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +0 -268
- package/renderer/components/docs-viewer/playground/response-cards/types.ts +0 -82
- package/renderer/components/docs-viewer/playground/response-viewer.tsx +0 -43
- package/renderer/components/docs-viewer/search/index.ts +0 -2
- package/renderer/components/docs-viewer/search/search-dialog.tsx +0 -331
- package/renderer/components/docs-viewer/search/use-search.ts +0 -117
- package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +0 -431
- package/renderer/components/docs-viewer/shared/method-badge.tsx +0 -41
- package/renderer/components/docs-viewer/shared/schema-viewer.tsx +0 -349
- package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +0 -259
- package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +0 -316
- package/renderer/components/docs-viewer/sidebar/index.tsx +0 -282
- package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +0 -202
- package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +0 -118
- package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +0 -212
- package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +0 -38
- package/renderer/components/theme-provider.tsx +0 -11
- package/renderer/components/theme-toggle.tsx +0 -76
- package/renderer/components/ui/badge.tsx +0 -46
- package/renderer/components/ui/button.tsx +0 -59
- package/renderer/components/ui/dialog.tsx +0 -118
- package/renderer/components/ui/dropdown-menu.tsx +0 -257
- package/renderer/components/ui/input.tsx +0 -21
- package/renderer/components/ui/label.tsx +0 -24
- package/renderer/components/ui/navigation-menu.tsx +0 -168
- package/renderer/components/ui/select.tsx +0 -190
- package/renderer/components/ui/spinner.tsx +0 -114
- package/renderer/components/ui/tabs.tsx +0 -66
- package/renderer/components/ui/tooltip.tsx +0 -61
- package/renderer/hooks/use-code-copy.ts +0 -88
- package/renderer/hooks/use-openapi-title.ts +0 -44
- package/renderer/lib/api-docs/agent/index.ts +0 -6
- package/renderer/lib/api-docs/agent/indexer.ts +0 -323
- package/renderer/lib/api-docs/agent/spec-summary.ts +0 -335
- package/renderer/lib/api-docs/agent/types.ts +0 -116
- package/renderer/lib/api-docs/auth/auth-context.tsx +0 -225
- package/renderer/lib/api-docs/auth/auth-storage.ts +0 -87
- package/renderer/lib/api-docs/auth/crypto.ts +0 -89
- package/renderer/lib/api-docs/auth/index.ts +0 -4
- package/renderer/lib/api-docs/code-editor/db.ts +0 -164
- package/renderer/lib/api-docs/code-editor/hooks.ts +0 -266
- package/renderer/lib/api-docs/code-editor/mode-context.tsx +0 -207
- package/renderer/lib/api-docs/code-editor/types.ts +0 -105
- package/renderer/lib/api-docs/codegen/definitions.ts +0 -297
- package/renderer/lib/api-docs/codegen/har.ts +0 -251
- package/renderer/lib/api-docs/codegen/index.ts +0 -159
- package/renderer/lib/api-docs/factories.ts +0 -170
- package/renderer/lib/api-docs/mobile-context.tsx +0 -112
- package/renderer/lib/api-docs/navigation-context.tsx +0 -88
- package/renderer/lib/api-docs/parsers/graphql/README.md +0 -129
- package/renderer/lib/api-docs/parsers/graphql/index.ts +0 -91
- package/renderer/lib/api-docs/parsers/graphql/parser.ts +0 -491
- package/renderer/lib/api-docs/parsers/graphql/transformer.ts +0 -246
- package/renderer/lib/api-docs/parsers/graphql/types.ts +0 -283
- package/renderer/lib/api-docs/parsers/openapi/README.md +0 -32
- package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +0 -60
- package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +0 -574
- package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +0 -403
- package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +0 -232
- package/renderer/lib/api-docs/parsers/openapi/index.ts +0 -171
- package/renderer/lib/api-docs/parsers/openapi/transformer.ts +0 -278
- package/renderer/lib/api-docs/parsers/openapi/validator.ts +0 -31
- package/renderer/lib/api-docs/playground/context.tsx +0 -107
- package/renderer/lib/api-docs/playground/navigation-context.tsx +0 -124
- package/renderer/lib/api-docs/playground/request-builder.ts +0 -223
- package/renderer/lib/api-docs/playground/request-runner.ts +0 -282
- package/renderer/lib/api-docs/playground/types.ts +0 -35
- package/renderer/lib/api-docs/types.ts +0 -269
- package/renderer/lib/api-docs/utils.ts +0 -311
- package/renderer/lib/cache.ts +0 -193
- package/renderer/lib/docs/config/domain-schema.ts +0 -260
- package/renderer/lib/docs/config/index.ts +0 -43
- package/renderer/lib/docs/config/loader.ts +0 -142
- package/renderer/lib/docs/config/schema.ts +0 -308
- package/renderer/lib/docs/index.ts +0 -12
- package/renderer/lib/docs/mdx/compiler.ts +0 -176
- package/renderer/lib/docs/mdx/frontmatter.ts +0 -91
- package/renderer/lib/docs/mdx/index.ts +0 -26
- package/renderer/lib/docs/navigation/generator.ts +0 -348
- package/renderer/lib/docs/navigation/index.ts +0 -12
- package/renderer/lib/docs/navigation/types.ts +0 -123
- package/renderer/lib/docs-navigation-context.tsx +0 -80
- package/renderer/lib/multi-tenant/context.ts +0 -105
- package/renderer/lib/storage/blob.ts +0 -1083
- package/renderer/lib/utils/icons.ts +0 -48
- package/renderer/lib/utils.ts +0 -6
- package/renderer/next.config.ts +0 -76
|
@@ -0,0 +1,295 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* OpenAPI Body Extractors
|
|
3
|
+
*
|
|
4
|
+
* Ported from Hoppscotch's openapi/index.ts
|
|
5
|
+
* Handles request body parsing for v2 and v3
|
|
6
|
+
*/ /* eslint-disable @typescript-eslint/no-explicit-any */ import { knownContentTypes, isOpenAPIV3Operation } from './index';
|
|
7
|
+
/**
|
|
8
|
+
* Resolve a $ref reference in the schema
|
|
9
|
+
*/ function resolveRef(doc, ref) {
|
|
10
|
+
if (!ref.startsWith('#/')) return null;
|
|
11
|
+
try {
|
|
12
|
+
const path = ref.slice(2).split('/');
|
|
13
|
+
let current = doc;
|
|
14
|
+
for (const segment of path){
|
|
15
|
+
if (current && typeof current === 'object' && segment in current) {
|
|
16
|
+
current = current[segment];
|
|
17
|
+
} else {
|
|
18
|
+
return null;
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
return current;
|
|
22
|
+
} catch {
|
|
23
|
+
return null;
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* Generate a sample value based on schema type
|
|
28
|
+
*/ function generateSampleValue(schema, doc, required = false) {
|
|
29
|
+
try {
|
|
30
|
+
// Handle $ref
|
|
31
|
+
if ('$ref' in schema) {
|
|
32
|
+
const resolved = resolveRef(doc, schema.$ref);
|
|
33
|
+
if (resolved) {
|
|
34
|
+
return generateSampleValue(resolved, doc, required);
|
|
35
|
+
}
|
|
36
|
+
return null;
|
|
37
|
+
}
|
|
38
|
+
const schemaObj = schema;
|
|
39
|
+
// Use example if provided
|
|
40
|
+
if (schemaObj.example !== undefined) {
|
|
41
|
+
return schemaObj.example;
|
|
42
|
+
}
|
|
43
|
+
// Use default if provided
|
|
44
|
+
if (schemaObj.default !== undefined) {
|
|
45
|
+
return schemaObj.default;
|
|
46
|
+
}
|
|
47
|
+
// Use enum first value if provided
|
|
48
|
+
if (schemaObj.enum && schemaObj.enum.length > 0) {
|
|
49
|
+
return schemaObj.enum[0];
|
|
50
|
+
}
|
|
51
|
+
// Generate based on type
|
|
52
|
+
switch(schemaObj.type){
|
|
53
|
+
case 'string':
|
|
54
|
+
if (schemaObj.format === 'date') return '2024-01-01';
|
|
55
|
+
if (schemaObj.format === 'date-time') return '2024-01-01T00:00:00Z';
|
|
56
|
+
if (schemaObj.format === 'email') return 'user@example.com';
|
|
57
|
+
if (schemaObj.format === 'uri' || schemaObj.format === 'url') return 'https://example.com';
|
|
58
|
+
if (schemaObj.format === 'uuid') return '00000000-0000-0000-0000-000000000000';
|
|
59
|
+
return required ? 'string' : '';
|
|
60
|
+
case 'number':
|
|
61
|
+
case 'integer':
|
|
62
|
+
if (schemaObj.minimum !== undefined) return schemaObj.minimum;
|
|
63
|
+
return 0;
|
|
64
|
+
case 'boolean':
|
|
65
|
+
return false;
|
|
66
|
+
case 'array':
|
|
67
|
+
if (schemaObj.items) {
|
|
68
|
+
return [
|
|
69
|
+
generateSampleValue(schemaObj.items, doc, true)
|
|
70
|
+
];
|
|
71
|
+
}
|
|
72
|
+
return [];
|
|
73
|
+
case 'object':
|
|
74
|
+
return generateObjectFromSchema(schemaObj, doc);
|
|
75
|
+
default:
|
|
76
|
+
// If no type but has properties, treat as object
|
|
77
|
+
if (schemaObj.properties) {
|
|
78
|
+
return generateObjectFromSchema(schemaObj, doc);
|
|
79
|
+
}
|
|
80
|
+
return null;
|
|
81
|
+
}
|
|
82
|
+
} catch {
|
|
83
|
+
return null;
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
/**
|
|
87
|
+
* Generate a sample object from an OpenAPI schema
|
|
88
|
+
* Required fields are listed first
|
|
89
|
+
*/ function generateObjectFromSchema(schema, doc) {
|
|
90
|
+
try {
|
|
91
|
+
const result = {};
|
|
92
|
+
const requiredFields = new Set(schema.required || []);
|
|
93
|
+
const properties = schema.properties || {};
|
|
94
|
+
// Sort properties to put required fields first
|
|
95
|
+
const sortedEntries = Object.entries(properties).sort(([keyA], [keyB])=>{
|
|
96
|
+
const aRequired = requiredFields.has(keyA);
|
|
97
|
+
const bRequired = requiredFields.has(keyB);
|
|
98
|
+
if (aRequired && !bRequired) return -1;
|
|
99
|
+
if (!aRequired && bRequired) return 1;
|
|
100
|
+
return 0;
|
|
101
|
+
});
|
|
102
|
+
for (const [key, propSchema] of sortedEntries){
|
|
103
|
+
const isRequired = requiredFields.has(key);
|
|
104
|
+
const value = generateSampleValue(propSchema, doc, isRequired);
|
|
105
|
+
if (value !== null) {
|
|
106
|
+
result[key] = value;
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
return result;
|
|
110
|
+
} catch {
|
|
111
|
+
return {};
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
/**
|
|
115
|
+
* Generate example from OpenAPI v3 media object
|
|
116
|
+
*/ function generateExampleFromMedia(media, doc) {
|
|
117
|
+
try {
|
|
118
|
+
// First check for explicit examples
|
|
119
|
+
if (media.example !== undefined) {
|
|
120
|
+
return media.example;
|
|
121
|
+
}
|
|
122
|
+
if (media.examples && Object.keys(media.examples).length > 0) {
|
|
123
|
+
const firstExampleKey = Object.keys(media.examples)[0];
|
|
124
|
+
const firstExample = media.examples[firstExampleKey];
|
|
125
|
+
if ('value' in firstExample) {
|
|
126
|
+
return firstExample.value;
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
// Generate from schema
|
|
130
|
+
const schema = media.schema;
|
|
131
|
+
if (!schema) return null;
|
|
132
|
+
// Handle $ref at schema level
|
|
133
|
+
if ('$ref' in schema) {
|
|
134
|
+
const resolved = resolveRef(doc, schema.$ref);
|
|
135
|
+
if (resolved) {
|
|
136
|
+
if (resolved.type === 'object' || resolved.properties) {
|
|
137
|
+
return generateObjectFromSchema(resolved, doc);
|
|
138
|
+
}
|
|
139
|
+
return generateSampleValue(resolved, doc, true);
|
|
140
|
+
}
|
|
141
|
+
return null;
|
|
142
|
+
}
|
|
143
|
+
if (schema.type === 'object' || schema.properties) {
|
|
144
|
+
return generateObjectFromSchema(schema, doc);
|
|
145
|
+
}
|
|
146
|
+
return generateSampleValue(schema, doc, true);
|
|
147
|
+
} catch {
|
|
148
|
+
return null;
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
// Legacy function for v2 compatibility
|
|
152
|
+
function generateRequestBodyExampleFromOpenAPIV2Body(op, doc) {
|
|
153
|
+
try {
|
|
154
|
+
const bodyParam = (op.parameters ?? []).find((param)=>param.in === 'body');
|
|
155
|
+
if (!bodyParam?.schema) return null;
|
|
156
|
+
const schema = bodyParam.schema;
|
|
157
|
+
if (schema.type === 'object' || schema.properties) {
|
|
158
|
+
const example = generateObjectFromSchema(schema, doc);
|
|
159
|
+
return JSON.stringify(example, null, 2);
|
|
160
|
+
}
|
|
161
|
+
return null;
|
|
162
|
+
} catch {
|
|
163
|
+
return null;
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
function generateV3Example(media, doc) {
|
|
167
|
+
return generateExampleFromMedia(media, doc);
|
|
168
|
+
}
|
|
169
|
+
function generateV31Example(media, doc) {
|
|
170
|
+
return generateExampleFromMedia(media, doc);
|
|
171
|
+
}
|
|
172
|
+
/**
|
|
173
|
+
* Parses OpenAPI v2 request body
|
|
174
|
+
*/ export function parseOpenAPIV2Body(op, doc) {
|
|
175
|
+
const obj = (op.consumes ?? [])[0];
|
|
176
|
+
// Not a content-type we support
|
|
177
|
+
if (!obj || !(obj in knownContentTypes)) {
|
|
178
|
+
return {
|
|
179
|
+
contentType: null,
|
|
180
|
+
body: null
|
|
181
|
+
};
|
|
182
|
+
}
|
|
183
|
+
// For form data types, extract form fields
|
|
184
|
+
if (obj === 'multipart/form-data' || obj === 'application/x-www-form-urlencoded') {
|
|
185
|
+
const formDataValues = (op.parameters ?? []).filter((param)=>param.in === 'formData').map((param)=>({
|
|
186
|
+
key: param.name,
|
|
187
|
+
isFile: param.type === 'file',
|
|
188
|
+
value: '',
|
|
189
|
+
active: true
|
|
190
|
+
}));
|
|
191
|
+
return obj === 'application/x-www-form-urlencoded' ? {
|
|
192
|
+
contentType: obj,
|
|
193
|
+
body: formDataValues.map(({ key })=>`${key}: `).join('\n')
|
|
194
|
+
} : {
|
|
195
|
+
contentType: obj,
|
|
196
|
+
body: formDataValues
|
|
197
|
+
};
|
|
198
|
+
}
|
|
199
|
+
// For other content types (JSON, XML, etc.)
|
|
200
|
+
const bodyParam = (op.parameters ?? []).find((param)=>param.in === 'body');
|
|
201
|
+
if (bodyParam && doc) {
|
|
202
|
+
const result = generateRequestBodyExampleFromOpenAPIV2Body(op, doc);
|
|
203
|
+
if (result) {
|
|
204
|
+
return {
|
|
205
|
+
contentType: obj,
|
|
206
|
+
body: result
|
|
207
|
+
};
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
// Fallback to empty body for textual content types
|
|
211
|
+
return {
|
|
212
|
+
contentType: obj,
|
|
213
|
+
body: ''
|
|
214
|
+
};
|
|
215
|
+
}
|
|
216
|
+
/**
|
|
217
|
+
* Parses OpenAPI v3 form data body
|
|
218
|
+
*/ export function parseOpenAPIV3BodyFormData(contentType, mediaObj) {
|
|
219
|
+
const schema = mediaObj.schema;
|
|
220
|
+
if (!schema || schema.type !== 'object') {
|
|
221
|
+
return contentType === 'application/x-www-form-urlencoded' ? {
|
|
222
|
+
contentType,
|
|
223
|
+
body: ''
|
|
224
|
+
} : {
|
|
225
|
+
contentType,
|
|
226
|
+
body: []
|
|
227
|
+
};
|
|
228
|
+
}
|
|
229
|
+
const keys = Object.keys(schema.properties ?? {});
|
|
230
|
+
if (contentType === 'application/x-www-form-urlencoded') {
|
|
231
|
+
return {
|
|
232
|
+
contentType,
|
|
233
|
+
body: keys.map((key)=>`${key}: `).join('\n')
|
|
234
|
+
};
|
|
235
|
+
}
|
|
236
|
+
return {
|
|
237
|
+
contentType,
|
|
238
|
+
body: keys.map((key)=>({
|
|
239
|
+
key,
|
|
240
|
+
value: '',
|
|
241
|
+
isFile: false,
|
|
242
|
+
active: true
|
|
243
|
+
}))
|
|
244
|
+
};
|
|
245
|
+
}
|
|
246
|
+
/**
|
|
247
|
+
* Parses OpenAPI v3 request body
|
|
248
|
+
*/ export function parseOpenAPIV3Body(doc, op) {
|
|
249
|
+
const objs = Object.entries(op.requestBody?.content ?? {});
|
|
250
|
+
if (objs.length === 0) return {
|
|
251
|
+
contentType: null,
|
|
252
|
+
body: null
|
|
253
|
+
};
|
|
254
|
+
// We only take the first definition
|
|
255
|
+
const [contentType, media] = objs[0];
|
|
256
|
+
if (!(contentType in knownContentTypes)) {
|
|
257
|
+
return {
|
|
258
|
+
contentType: null,
|
|
259
|
+
body: null
|
|
260
|
+
};
|
|
261
|
+
}
|
|
262
|
+
// Handle form data types
|
|
263
|
+
if (contentType === 'multipart/form-data' || contentType === 'application/x-www-form-urlencoded') {
|
|
264
|
+
return parseOpenAPIV3BodyFormData(contentType, media);
|
|
265
|
+
}
|
|
266
|
+
// For other content types (JSON, XML, etc.), try to generate sample from schema
|
|
267
|
+
try {
|
|
268
|
+
const docAny = doc;
|
|
269
|
+
const isV31 = docAny.openapi && docAny.openapi.startsWith('3.1');
|
|
270
|
+
let sampleBody;
|
|
271
|
+
if (isV31) {
|
|
272
|
+
sampleBody = generateV31Example(media, doc);
|
|
273
|
+
} else {
|
|
274
|
+
sampleBody = generateV3Example(media, doc);
|
|
275
|
+
}
|
|
276
|
+
if (sampleBody !== null && sampleBody !== undefined) {
|
|
277
|
+
return {
|
|
278
|
+
contentType: contentType,
|
|
279
|
+
body: typeof sampleBody === 'string' ? sampleBody : JSON.stringify(sampleBody, null, 2)
|
|
280
|
+
};
|
|
281
|
+
}
|
|
282
|
+
} catch {
|
|
283
|
+
// Fall through to empty body
|
|
284
|
+
}
|
|
285
|
+
// Fallback to empty body for textual content types
|
|
286
|
+
return {
|
|
287
|
+
contentType: contentType,
|
|
288
|
+
body: ''
|
|
289
|
+
};
|
|
290
|
+
}
|
|
291
|
+
/**
|
|
292
|
+
* Parses OpenAPI request body (v2 or v3)
|
|
293
|
+
*/ export function parseOpenAPIBody(doc, op) {
|
|
294
|
+
return isOpenAPIV3Operation(doc, op) ? parseOpenAPIV3Body(doc, op) : parseOpenAPIV2Body(op, doc);
|
|
295
|
+
}
|
|
@@ -0,0 +1,132 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* OpenAPI Extractors
|
|
3
|
+
*
|
|
4
|
+
* Ported from Hoppscotch's openapi/index.ts
|
|
5
|
+
* Converts fp-ts functional programming to native TypeScript
|
|
6
|
+
*/ /* eslint-disable @typescript-eslint/no-unused-vars */ import { isNumeric, getStatusCodeReasonPhrase } from '../../../utils';
|
|
7
|
+
// Known content types (from Hoppscotch)
|
|
8
|
+
const knownContentTypes = {
|
|
9
|
+
'application/json': true,
|
|
10
|
+
'application/xml': true,
|
|
11
|
+
'text/plain': true,
|
|
12
|
+
'text/html': true,
|
|
13
|
+
'application/x-www-form-urlencoded': true,
|
|
14
|
+
'multipart/form-data': true,
|
|
15
|
+
'application/octet-stream': true
|
|
16
|
+
};
|
|
17
|
+
/**
|
|
18
|
+
* Replaces OpenAPI path templating ({id}) with Brainfish templating (<<id>>)
|
|
19
|
+
*/ export function replaceOpenApiPathTemplating(path) {
|
|
20
|
+
return path.replace(/{/g, '<<').replace(/}/g, '>>');
|
|
21
|
+
}
|
|
22
|
+
/**
|
|
23
|
+
* Parses OpenAPI query parameters
|
|
24
|
+
*/ export function parseOpenAPIParams(params) {
|
|
25
|
+
return params.filter((param)=>param.in === 'query').map((param)=>({
|
|
26
|
+
key: param.name,
|
|
27
|
+
value: '',
|
|
28
|
+
active: true,
|
|
29
|
+
description: param.description ?? ''
|
|
30
|
+
}));
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Parses OpenAPI path variables
|
|
34
|
+
*/ export function parseOpenAPIVariables(variables) {
|
|
35
|
+
return variables.filter((param)=>param.in === 'path').map((param)=>({
|
|
36
|
+
key: param.name,
|
|
37
|
+
value: '',
|
|
38
|
+
active: true
|
|
39
|
+
}));
|
|
40
|
+
}
|
|
41
|
+
/**
|
|
42
|
+
* Parses OpenAPI header parameters
|
|
43
|
+
*/ export function parseOpenAPIHeaders(params) {
|
|
44
|
+
return params.filter((param)=>param.in === 'header').map((header)=>({
|
|
45
|
+
key: header.name,
|
|
46
|
+
value: '',
|
|
47
|
+
active: true,
|
|
48
|
+
description: header.description ?? ''
|
|
49
|
+
}));
|
|
50
|
+
}
|
|
51
|
+
/**
|
|
52
|
+
* Parses OpenAPI v3 responses
|
|
53
|
+
*/ export function parseOpenAPIV3Responses(op, originalRequest) {
|
|
54
|
+
const responses = op.responses;
|
|
55
|
+
if (!responses) return {};
|
|
56
|
+
const res = {};
|
|
57
|
+
for (const [key, value] of Object.entries(responses)){
|
|
58
|
+
const response = value;
|
|
59
|
+
const contentType = Object.keys(response.content ?? {})[0];
|
|
60
|
+
const body = response.content?.[contentType];
|
|
61
|
+
const name = response.description ?? key;
|
|
62
|
+
const code = isNumeric(key) ? Number(key) : 200;
|
|
63
|
+
const status = getStatusCodeReasonPhrase(code);
|
|
64
|
+
const headers = [
|
|
65
|
+
{
|
|
66
|
+
key: 'content-type',
|
|
67
|
+
value: contentType ?? 'application/json',
|
|
68
|
+
description: '',
|
|
69
|
+
active: true
|
|
70
|
+
}
|
|
71
|
+
];
|
|
72
|
+
let stringifiedBody = '';
|
|
73
|
+
try {
|
|
74
|
+
stringifiedBody = JSON.stringify(body ?? '');
|
|
75
|
+
} catch (e) {
|
|
76
|
+
// Ignore circular reference errors
|
|
77
|
+
}
|
|
78
|
+
res[name] = {
|
|
79
|
+
name,
|
|
80
|
+
status,
|
|
81
|
+
code,
|
|
82
|
+
headers,
|
|
83
|
+
body: stringifiedBody,
|
|
84
|
+
originalRequest
|
|
85
|
+
};
|
|
86
|
+
}
|
|
87
|
+
return res;
|
|
88
|
+
}
|
|
89
|
+
/**
|
|
90
|
+
* Parses OpenAPI v2 responses
|
|
91
|
+
*/ export function parseOpenAPIV2Responses(op, originalRequest) {
|
|
92
|
+
const responses = op.responses;
|
|
93
|
+
if (!responses) return {};
|
|
94
|
+
const res = {};
|
|
95
|
+
for (const [key, value] of Object.entries(responses)){
|
|
96
|
+
const response = value;
|
|
97
|
+
const contentType = Object.keys(response.examples ?? {})[0];
|
|
98
|
+
const body = response.examples?.[contentType];
|
|
99
|
+
const name = response.description ?? key;
|
|
100
|
+
const code = isNumeric(Number(key)) ? Number(key) : 200;
|
|
101
|
+
const status = getStatusCodeReasonPhrase(code);
|
|
102
|
+
const headers = [
|
|
103
|
+
{
|
|
104
|
+
key: 'content-type',
|
|
105
|
+
value: contentType ?? 'application/json',
|
|
106
|
+
description: '',
|
|
107
|
+
active: true
|
|
108
|
+
}
|
|
109
|
+
];
|
|
110
|
+
res[name] = {
|
|
111
|
+
name,
|
|
112
|
+
status,
|
|
113
|
+
code,
|
|
114
|
+
headers,
|
|
115
|
+
body: body ?? '',
|
|
116
|
+
originalRequest
|
|
117
|
+
};
|
|
118
|
+
}
|
|
119
|
+
return res;
|
|
120
|
+
}
|
|
121
|
+
/**
|
|
122
|
+
* Parses OpenAPI responses (v2 or v3)
|
|
123
|
+
*/ export function parseOpenAPIResponses(doc, op, originalRequest) {
|
|
124
|
+
return isOpenAPIV3Operation(doc, op) ? parseOpenAPIV3Responses(op, originalRequest) : parseOpenAPIV2Responses(op, originalRequest);
|
|
125
|
+
}
|
|
126
|
+
/**
|
|
127
|
+
* Checks if an operation is OpenAPI v3
|
|
128
|
+
*/ export function isOpenAPIV3Operation(doc, op) {
|
|
129
|
+
return 'openapi' in doc && typeof doc.openapi === 'string' && doc.openapi.startsWith('3.');
|
|
130
|
+
}
|
|
131
|
+
// Export knownContentTypes for use in body parsers
|
|
132
|
+
export { knownContentTypes };
|
|
@@ -0,0 +1,127 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* OpenAPI Importer
|
|
3
|
+
*
|
|
4
|
+
* Main entry point for importing OpenAPI specifications
|
|
5
|
+
* Ported from Hoppscotch's openapi/index.ts
|
|
6
|
+
*/ import yaml from 'js-yaml';
|
|
7
|
+
import { objectHasProperty } from '../../utils';
|
|
8
|
+
import { validateOpenAPISpec } from './validator';
|
|
9
|
+
import { dereferenceOpenAPISpec, hasUnresolvedRefs } from './dereferencer';
|
|
10
|
+
import { convertOpenAPIToCollection } from './transformer';
|
|
11
|
+
export const OPENAPI_DEREF_ERROR = 'openapi/deref_error';
|
|
12
|
+
export const IMPORTER_INVALID_FILE_FORMAT = 'importer/invalid_file_format';
|
|
13
|
+
/**
|
|
14
|
+
* Safely parses JSON string
|
|
15
|
+
*/ function safeParseJSON(str) {
|
|
16
|
+
try {
|
|
17
|
+
return JSON.parse(str);
|
|
18
|
+
} catch {
|
|
19
|
+
return null;
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
/**
|
|
23
|
+
* Safely parses YAML string
|
|
24
|
+
*/ function safeParseYAML(str) {
|
|
25
|
+
try {
|
|
26
|
+
return yaml.load(str);
|
|
27
|
+
} catch {
|
|
28
|
+
return null;
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
/**
|
|
32
|
+
* Parses OpenAPI document content (JSON or YAML)
|
|
33
|
+
*/ function parseOpenAPIDocContent(str) {
|
|
34
|
+
const jsonResult = safeParseJSON(str);
|
|
35
|
+
if (jsonResult !== null) {
|
|
36
|
+
return jsonResult;
|
|
37
|
+
}
|
|
38
|
+
return safeParseYAML(str);
|
|
39
|
+
}
|
|
40
|
+
/**
|
|
41
|
+
* Checks if document is OpenAPI v2
|
|
42
|
+
*/ function isOpenAPIV2Document(doc) {
|
|
43
|
+
return objectHasProperty(doc, 'swagger') && typeof doc.swagger === 'string' && doc.swagger === '2.0';
|
|
44
|
+
}
|
|
45
|
+
/**
|
|
46
|
+
* Checks if document is OpenAPI v3
|
|
47
|
+
*/ function isOpenAPIV3Document(doc) {
|
|
48
|
+
return objectHasProperty(doc, 'openapi') && typeof doc.openapi === 'string' && doc.openapi.startsWith('3.');
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* Imports OpenAPI specification from file contents or document
|
|
52
|
+
*
|
|
53
|
+
* @param input - OpenAPI document object or array of file contents (strings)
|
|
54
|
+
* @returns Promise resolving to BrainfishCollection array
|
|
55
|
+
*/ export async function importOpenAPISpec(input) {
|
|
56
|
+
let docArr;
|
|
57
|
+
// Handle different input types
|
|
58
|
+
if (typeof input === 'string') {
|
|
59
|
+
// Single file content string
|
|
60
|
+
const parsed = parseOpenAPIDocContent(input);
|
|
61
|
+
if (!parsed) {
|
|
62
|
+
throw new Error(IMPORTER_INVALID_FILE_FORMAT);
|
|
63
|
+
}
|
|
64
|
+
docArr = [
|
|
65
|
+
parsed
|
|
66
|
+
];
|
|
67
|
+
} else if (Array.isArray(input)) {
|
|
68
|
+
// Array of file content strings
|
|
69
|
+
docArr = input.map((str)=>parseOpenAPIDocContent(str)).filter((doc)=>doc !== null);
|
|
70
|
+
if (docArr.length === 0) {
|
|
71
|
+
throw new Error(IMPORTER_INVALID_FILE_FORMAT);
|
|
72
|
+
}
|
|
73
|
+
} else {
|
|
74
|
+
// Already a parsed OpenAPI document
|
|
75
|
+
docArr = [
|
|
76
|
+
input
|
|
77
|
+
];
|
|
78
|
+
}
|
|
79
|
+
// Validate documents
|
|
80
|
+
const validatedDocs = [];
|
|
81
|
+
for (const docObj of docArr){
|
|
82
|
+
try {
|
|
83
|
+
// More lenient check - if it has paths, we'll try to import it
|
|
84
|
+
const isValidOpenAPISpec = objectHasProperty(docObj, 'paths') && (isOpenAPIV2Document(docObj) || isOpenAPIV3Document(docObj) || objectHasProperty(docObj, 'info'));
|
|
85
|
+
if (!isValidOpenAPISpec) {
|
|
86
|
+
throw new Error('INVALID_OPENAPI_SPEC');
|
|
87
|
+
}
|
|
88
|
+
try {
|
|
89
|
+
const validatedDoc = await validateOpenAPISpec(docObj);
|
|
90
|
+
validatedDocs.push(validatedDoc);
|
|
91
|
+
} catch (validationError) {
|
|
92
|
+
// If validation fails but it has basic OpenAPI structure, add it anyway
|
|
93
|
+
if (objectHasProperty(docObj, 'paths')) {
|
|
94
|
+
validatedDocs.push(docObj);
|
|
95
|
+
} else {
|
|
96
|
+
throw validationError;
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
} catch (err) {
|
|
100
|
+
if (err instanceof Error && err.message === 'INVALID_OPENAPI_SPEC') {
|
|
101
|
+
throw new Error('INVALID_OPENAPI_SPEC');
|
|
102
|
+
}
|
|
103
|
+
// Continue with other documents
|
|
104
|
+
console.warn('Failed to validate document:', err);
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
if (validatedDocs.length === 0) {
|
|
108
|
+
throw new Error(IMPORTER_INVALID_FILE_FORMAT);
|
|
109
|
+
}
|
|
110
|
+
// Dereference documents
|
|
111
|
+
const dereferencedDocs = [];
|
|
112
|
+
for (const docObj of validatedDocs){
|
|
113
|
+
try {
|
|
114
|
+
const dereferencedDoc = await dereferenceOpenAPISpec(docObj);
|
|
115
|
+
dereferencedDocs.push(dereferencedDoc);
|
|
116
|
+
} catch {
|
|
117
|
+
// Check if the document has unresolved references
|
|
118
|
+
if (hasUnresolvedRefs(docObj)) {
|
|
119
|
+
console.warn('Document contains unresolved references which may affect import quality');
|
|
120
|
+
}
|
|
121
|
+
// If dereferencing fails, use the original document
|
|
122
|
+
dereferencedDocs.push(docObj);
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
// Convert to BrainfishCollections
|
|
126
|
+
return convertOpenAPIToCollection(dereferencedDocs);
|
|
127
|
+
}
|