@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,574 @@
1
+ /**
2
+ * OpenAPI Auth Extractors
3
+ *
4
+ * Ported from Hoppscotch's openapi/index.ts
5
+ * Handles authentication scheme 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 { BrainfishRESTAuth } from '../../../types'
17
+ import { isOpenAPIV3Operation } from './index'
18
+
19
+ /**
20
+ * Resolves OpenAPI v3 security scheme object to Brainfish auth
21
+ */
22
+ function resolveOpenAPIV3SecurityObj(
23
+ scheme: OpenAPIV3.SecuritySchemeObject | OpenAPIV31.SecuritySchemeObject,
24
+ _schemeData: string[] // Used for OAuth to pass params
25
+ ): BrainfishRESTAuth {
26
+ if (scheme.type === 'http') {
27
+ if (scheme.scheme === 'basic') {
28
+ return { authType: 'basic', authActive: true, username: '', password: '' }
29
+ } else if (scheme.scheme === 'bearer') {
30
+ return { authType: 'bearer', authActive: true, token: '' }
31
+ }
32
+ return { authType: 'none', authActive: true }
33
+ } else if (scheme.type === 'apiKey') {
34
+ if (scheme.in === 'header') {
35
+ return {
36
+ authType: 'api-key',
37
+ authActive: true,
38
+ addTo: 'HEADERS',
39
+ key: scheme.name,
40
+ value: '',
41
+ }
42
+ } else if (scheme.in === 'query') {
43
+ return {
44
+ authType: 'api-key',
45
+ authActive: true,
46
+ addTo: 'QUERY_PARAMS',
47
+ key: scheme.name,
48
+ value: '',
49
+ }
50
+ }
51
+ } else if (scheme.type === 'oauth2') {
52
+ // Select flow on first come basis: authorizationCode > implicit > password > clientCredentials
53
+ if (scheme.flows?.authorizationCode) {
54
+ return {
55
+ authType: 'oauth-2',
56
+ authActive: true,
57
+ grantTypeInfo: {
58
+ grantType: 'AUTHORIZATION_CODE',
59
+ authEndpoint: scheme.flows.authorizationCode.authorizationUrl ?? '',
60
+ clientID: '',
61
+ scopes: _schemeData.join(' '),
62
+ token: '',
63
+ isPKCE: false,
64
+ tokenEndpoint: scheme.flows.authorizationCode.tokenUrl ?? '',
65
+ clientSecret: '',
66
+ authRequestParams: [],
67
+ refreshRequestParams: [],
68
+ tokenRequestParams: [],
69
+ },
70
+ addTo: 'HEADERS',
71
+ }
72
+ } else if (scheme.flows?.implicit) {
73
+ return {
74
+ authType: 'oauth-2',
75
+ authActive: true,
76
+ grantTypeInfo: {
77
+ grantType: 'IMPLICIT',
78
+ authEndpoint: scheme.flows.implicit.authorizationUrl ?? '',
79
+ clientID: '',
80
+ token: '',
81
+ scopes: _schemeData.join(' '),
82
+ authRequestParams: [],
83
+ refreshRequestParams: [],
84
+ },
85
+ addTo: 'HEADERS',
86
+ }
87
+ } else if (scheme.flows?.password) {
88
+ return {
89
+ authType: 'oauth-2',
90
+ authActive: true,
91
+ grantTypeInfo: {
92
+ grantType: 'PASSWORD',
93
+ clientID: '',
94
+ authEndpoint: scheme.flows.password.tokenUrl ?? '',
95
+ clientSecret: '',
96
+ password: '',
97
+ username: '',
98
+ token: '',
99
+ scopes: _schemeData.join(' '),
100
+ refreshRequestParams: [],
101
+ tokenRequestParams: [],
102
+ },
103
+ addTo: 'HEADERS',
104
+ }
105
+ } else if (scheme.flows?.clientCredentials) {
106
+ return {
107
+ authType: 'oauth-2',
108
+ authActive: true,
109
+ grantTypeInfo: {
110
+ grantType: 'CLIENT_CREDENTIALS',
111
+ authEndpoint: scheme.flows.clientCredentials.tokenUrl ?? '',
112
+ clientID: '',
113
+ clientSecret: '',
114
+ scopes: _schemeData.join(' '),
115
+ token: '',
116
+ clientAuthentication: 'IN_BODY',
117
+ refreshRequestParams: [],
118
+ tokenRequestParams: [],
119
+ },
120
+ addTo: 'HEADERS',
121
+ }
122
+ }
123
+ // Default OAuth2
124
+ return {
125
+ authType: 'oauth-2',
126
+ authActive: true,
127
+ grantTypeInfo: {
128
+ grantType: 'AUTHORIZATION_CODE',
129
+ authEndpoint: '',
130
+ clientID: '',
131
+ scopes: _schemeData.join(' '),
132
+ token: '',
133
+ isPKCE: false,
134
+ tokenEndpoint: '',
135
+ clientSecret: '',
136
+ authRequestParams: [],
137
+ refreshRequestParams: [],
138
+ tokenRequestParams: [],
139
+ },
140
+ addTo: 'HEADERS',
141
+ }
142
+ } else if (scheme.type === 'openIdConnect') {
143
+ return {
144
+ authType: 'oauth-2',
145
+ authActive: true,
146
+ grantTypeInfo: {
147
+ grantType: 'AUTHORIZATION_CODE',
148
+ authEndpoint: '',
149
+ clientID: '',
150
+ scopes: _schemeData.join(' '),
151
+ token: '',
152
+ isPKCE: false,
153
+ tokenEndpoint: '',
154
+ clientSecret: '',
155
+ authRequestParams: [],
156
+ refreshRequestParams: [],
157
+ tokenRequestParams: [],
158
+ },
159
+ addTo: 'HEADERS',
160
+ }
161
+ }
162
+
163
+ return { authType: 'none', authActive: true }
164
+ }
165
+
166
+ /**
167
+ * Resolves OpenAPI v3 security scheme
168
+ */
169
+ function resolveOpenAPIV3SecurityScheme(
170
+ doc: OpenAPIV3.Document | OpenAPIV31.Document,
171
+ schemeName: string,
172
+ schemeData: string[]
173
+ ): BrainfishRESTAuth {
174
+ const scheme = doc.components?.securitySchemes?.[schemeName] as
175
+ | OpenAPIV3.SecuritySchemeObject
176
+ | undefined
177
+
178
+ if (!scheme) return { authType: 'none', authActive: true }
179
+ return resolveOpenAPIV3SecurityObj(scheme, schemeData)
180
+ }
181
+
182
+ /**
183
+ * Resolves OpenAPI v3 security requirements
184
+ */
185
+ function resolveOpenAPIV3Security(
186
+ doc: OpenAPIV3.Document | OpenAPIV31.Document,
187
+ security:
188
+ | OpenAPIV3.SecurityRequirementObject[]
189
+ | OpenAPIV31.SecurityRequirementObject[]
190
+ ): BrainfishRESTAuth {
191
+ // Only consider the first security requirement
192
+ const sec = security[0] as OpenAPIV3.SecurityRequirementObject | undefined
193
+
194
+ if (!sec) return { authType: 'none', authActive: true }
195
+
196
+ // Only consider the first security condition within the first condition
197
+ const [schemeName, schemeData] = (Object.entries(sec)[0] ?? [
198
+ undefined,
199
+ undefined,
200
+ ]) as [string | undefined, string[] | undefined]
201
+
202
+ if (!schemeName || !schemeData) return { authType: 'none', authActive: true }
203
+
204
+ return resolveOpenAPIV3SecurityScheme(doc, schemeName, schemeData)
205
+ }
206
+
207
+ /**
208
+ * Parses OpenAPI v3 authentication
209
+ */
210
+ export function parseOpenAPIV3Auth(
211
+ doc: OpenAPIV3.Document | OpenAPIV31.Document,
212
+ op: OpenAPIV3.OperationObject | OpenAPIV31.OperationObject
213
+ ): BrainfishRESTAuth {
214
+ const rootAuth = doc.security
215
+ ? resolveOpenAPIV3Security(doc, doc.security)
216
+ : undefined
217
+ const opAuth = op.security
218
+ ? resolveOpenAPIV3Security(doc, op.security)
219
+ : undefined
220
+
221
+ return opAuth ?? rootAuth ?? { authType: 'none', authActive: true }
222
+ }
223
+
224
+ /**
225
+ * Resolves OpenAPI v2 security scheme
226
+ */
227
+ function resolveOpenAPIV2SecurityScheme(
228
+ scheme: OpenAPIV2.SecuritySchemeObject,
229
+ _schemeData: string[]
230
+ ): BrainfishRESTAuth {
231
+ if (scheme.type === 'basic') {
232
+ return { authType: 'basic', authActive: true, username: '', password: '' }
233
+ } else if (scheme.type === 'apiKey') {
234
+ // V2 only supports in: header and in: query
235
+ return {
236
+ authType: 'api-key',
237
+ addTo: scheme.in === 'header' ? 'HEADERS' : 'QUERY_PARAMS',
238
+ authActive: true,
239
+ key: scheme.name,
240
+ value: '',
241
+ }
242
+ } else if (scheme.type === 'oauth2') {
243
+ // Select flow: accessCode > implicit > password > application
244
+ if (scheme.flow === 'accessCode') {
245
+ return {
246
+ authType: 'oauth-2',
247
+ authActive: true,
248
+ grantTypeInfo: {
249
+ authEndpoint: scheme.authorizationUrl ?? '',
250
+ clientID: '',
251
+ clientSecret: '',
252
+ grantType: 'AUTHORIZATION_CODE',
253
+ scopes: _schemeData.join(' '),
254
+ token: '',
255
+ isPKCE: false,
256
+ tokenEndpoint: scheme.tokenUrl ?? '',
257
+ authRequestParams: [],
258
+ refreshRequestParams: [],
259
+ tokenRequestParams: [],
260
+ },
261
+ addTo: 'HEADERS',
262
+ }
263
+ } else if (scheme.flow === 'implicit') {
264
+ return {
265
+ authType: 'oauth-2',
266
+ authActive: true,
267
+ grantTypeInfo: {
268
+ authEndpoint: scheme.authorizationUrl ?? '',
269
+ clientID: '',
270
+ grantType: 'IMPLICIT',
271
+ scopes: _schemeData.join(' '),
272
+ token: '',
273
+ authRequestParams: [],
274
+ refreshRequestParams: [],
275
+ },
276
+ addTo: 'HEADERS',
277
+ }
278
+ } else if (scheme.flow === 'application') {
279
+ return {
280
+ authType: 'oauth-2',
281
+ authActive: true,
282
+ grantTypeInfo: {
283
+ authEndpoint: scheme.tokenUrl ?? '',
284
+ clientID: '',
285
+ clientSecret: '',
286
+ grantType: 'CLIENT_CREDENTIALS',
287
+ scopes: _schemeData.join(' '),
288
+ token: '',
289
+ clientAuthentication: 'IN_BODY',
290
+ refreshRequestParams: [],
291
+ tokenRequestParams: [],
292
+ },
293
+ addTo: 'HEADERS',
294
+ }
295
+ } else if (scheme.flow === 'password') {
296
+ return {
297
+ authType: 'oauth-2',
298
+ authActive: true,
299
+ grantTypeInfo: {
300
+ grantType: 'PASSWORD',
301
+ authEndpoint: scheme.tokenUrl ?? '',
302
+ clientID: '',
303
+ clientSecret: '',
304
+ password: '',
305
+ scopes: _schemeData.join(' '),
306
+ token: '',
307
+ username: '',
308
+ refreshRequestParams: [],
309
+ tokenRequestParams: [],
310
+ },
311
+ addTo: 'HEADERS',
312
+ }
313
+ }
314
+ // Default OAuth2
315
+ return {
316
+ authType: 'oauth-2',
317
+ authActive: true,
318
+ grantTypeInfo: {
319
+ authEndpoint: '',
320
+ clientID: '',
321
+ clientSecret: '',
322
+ grantType: 'AUTHORIZATION_CODE',
323
+ scopes: _schemeData.join(' '),
324
+ token: '',
325
+ isPKCE: false,
326
+ tokenEndpoint: '',
327
+ authRequestParams: [],
328
+ refreshRequestParams: [],
329
+ tokenRequestParams: [],
330
+ },
331
+ addTo: 'HEADERS',
332
+ }
333
+ }
334
+
335
+ return { authType: 'none', authActive: true }
336
+ }
337
+
338
+ /**
339
+ * Resolves OpenAPI v2 security definition
340
+ */
341
+ function resolveOpenAPIV2SecurityDef(
342
+ doc: OpenAPIV2.Document,
343
+ schemeName: string,
344
+ schemeData: string[]
345
+ ): BrainfishRESTAuth {
346
+ const scheme = Object.entries(doc.securityDefinitions ?? {}).find(
347
+ ([name]) => schemeName === name
348
+ )
349
+
350
+ if (!scheme) return { authType: 'none', authActive: true }
351
+
352
+ const schemeObj = scheme[1]
353
+ return resolveOpenAPIV2SecurityScheme(schemeObj, schemeData)
354
+ }
355
+
356
+ /**
357
+ * Resolves OpenAPI v2 security requirements
358
+ */
359
+ function resolveOpenAPIV2Security(
360
+ doc: OpenAPIV2.Document,
361
+ security: OpenAPIV2.SecurityRequirementObject[]
362
+ ): BrainfishRESTAuth {
363
+ // Only consider the first security requirement
364
+ const sec = security[0] as OpenAPIV2.SecurityRequirementObject | undefined
365
+
366
+ if (!sec) return { authType: 'none', authActive: true }
367
+
368
+ // Only consider the first security condition within the first condition
369
+ const [schemeName, schemeData] = (Object.entries(sec)[0] ?? [
370
+ undefined,
371
+ undefined,
372
+ ]) as [string | undefined, string[] | undefined]
373
+
374
+ if (!schemeName || !schemeData) return { authType: 'none', authActive: true }
375
+
376
+ return resolveOpenAPIV2SecurityDef(doc, schemeName, schemeData)
377
+ }
378
+
379
+ /**
380
+ * Parses OpenAPI v2 authentication
381
+ */
382
+ export function parseOpenAPIV2Auth(
383
+ doc: OpenAPIV2.Document,
384
+ op: OpenAPIV2.OperationObject
385
+ ): BrainfishRESTAuth {
386
+ const rootAuth = doc.security
387
+ ? resolveOpenAPIV2Security(doc, doc.security)
388
+ : undefined
389
+ const opAuth = op.security
390
+ ? resolveOpenAPIV2Security(doc, op.security)
391
+ : undefined
392
+
393
+ return opAuth ?? rootAuth ?? { authType: 'none', authActive: true }
394
+ }
395
+
396
+ /**
397
+ * Parses OpenAPI authentication (v2 or v3)
398
+ */
399
+ export function parseOpenAPIAuth(
400
+ doc: OpenAPI.Document,
401
+ op: any // OpenAPIOperationType
402
+ ): BrainfishRESTAuth {
403
+ return isOpenAPIV3Operation(doc, op)
404
+ ? parseOpenAPIV3Auth(
405
+ doc as OpenAPIV3.Document | OpenAPIV31.Document,
406
+ op
407
+ )
408
+ : parseOpenAPIV2Auth(doc as OpenAPIV2.Document, op)
409
+ }
410
+
411
+ /**
412
+ * Extract all security schemes from an OpenAPI document
413
+ * Returns them in a format suitable for the auth modal
414
+ */
415
+ export interface SecuritySchemeInfo {
416
+ name: string
417
+ type: 'basic' | 'bearer' | 'api-key' | 'oauth2' | 'none'
418
+ auth: BrainfishRESTAuth
419
+ description?: string
420
+ }
421
+
422
+ export function extractSecuritySchemes(doc: OpenAPI.Document): SecuritySchemeInfo[] {
423
+ const schemes: SecuritySchemeInfo[] = []
424
+
425
+ // Check if it's v3
426
+ const v3Doc = doc as OpenAPIV3.Document | OpenAPIV31.Document
427
+ if (v3Doc.components?.securitySchemes) {
428
+ for (const [name, schemeRef] of Object.entries(v3Doc.components.securitySchemes)) {
429
+ // Skip references for now
430
+ if ('$ref' in schemeRef) continue
431
+
432
+ const scheme = schemeRef as OpenAPIV3.SecuritySchemeObject
433
+ const auth = resolveOpenAPIV3SecurityObj(scheme, [])
434
+
435
+ let type: SecuritySchemeInfo['type'] = 'none'
436
+ let description = scheme.description
437
+
438
+ if (scheme.type === 'http') {
439
+ if (scheme.scheme === 'basic') type = 'basic'
440
+ else if (scheme.scheme === 'bearer') type = 'bearer'
441
+ } else if (scheme.type === 'apiKey') {
442
+ type = 'api-key'
443
+ description = description || `API Key in ${scheme.in}: ${scheme.name}`
444
+ } else if (scheme.type === 'oauth2') {
445
+ type = 'oauth2'
446
+ }
447
+
448
+ if (type !== 'none') {
449
+ schemes.push({ name, type, auth, description })
450
+ }
451
+ }
452
+ }
453
+
454
+ // Check if it's v2
455
+ const v2Doc = doc as OpenAPIV2.Document
456
+ if (v2Doc.securityDefinitions) {
457
+ for (const [name, scheme] of Object.entries(v2Doc.securityDefinitions)) {
458
+ const auth = resolveOpenAPIV2SecurityScheme(scheme, [])
459
+
460
+ let type: SecuritySchemeInfo['type'] = 'none'
461
+ let description = scheme.description
462
+
463
+ if (scheme.type === 'basic') {
464
+ type = 'basic'
465
+ } else if (scheme.type === 'apiKey') {
466
+ type = 'api-key'
467
+ description = description || `API Key in ${scheme.in}: ${scheme.name}`
468
+ } else if (scheme.type === 'oauth2') {
469
+ type = 'oauth2'
470
+ }
471
+
472
+ if (type !== 'none') {
473
+ schemes.push({ name, type, auth, description })
474
+ }
475
+ }
476
+ }
477
+
478
+ return schemes
479
+ }
480
+
481
+ /**
482
+ * Get the default/recommended auth from the spec
483
+ * Based on the global security requirement
484
+ */
485
+ export function getDefaultAuth(doc: OpenAPI.Document): BrainfishRESTAuth | null {
486
+ // Check v3 global security
487
+ const v3Doc = doc as OpenAPIV3.Document | OpenAPIV31.Document
488
+ if (v3Doc.security && v3Doc.security.length > 0) {
489
+ return resolveOpenAPIV3Security(v3Doc, v3Doc.security)
490
+ }
491
+
492
+ // Check v2 global security
493
+ const v2Doc = doc as OpenAPIV2.Document
494
+ if (v2Doc.security && v2Doc.security.length > 0) {
495
+ return resolveOpenAPIV2Security(v2Doc, v2Doc.security)
496
+ }
497
+
498
+ // If no global security, return the first scheme found
499
+ const schemes = extractSecuritySchemes(doc)
500
+ if (schemes.length > 0) {
501
+ return schemes[0].auth
502
+ }
503
+
504
+ return null
505
+ }
506
+
507
+ /**
508
+ * Header info extracted from security schemes
509
+ */
510
+ export interface SecurityHeader {
511
+ key: string
512
+ value: string
513
+ description: string
514
+ active: boolean
515
+ required: boolean
516
+ }
517
+
518
+ /**
519
+ * Extract additional security headers required by an endpoint
520
+ * These are API key headers that are NOT the primary auth (handled by global auth)
521
+ *
522
+ * For example, if an endpoint requires both `access-token` and `agent-key`,
523
+ * the primary auth handles `access-token`, and this returns `agent-key` as additional header
524
+ */
525
+ export function extractSecurityHeaders(
526
+ doc: OpenAPI.Document,
527
+ op: any, // OpenAPIOperationType
528
+ primaryAuthKey?: string // The key handled by global auth (e.g., 'access-token')
529
+ ): SecurityHeader[] {
530
+ const headers: SecurityHeader[] = []
531
+
532
+ try {
533
+ const v3Doc = doc as OpenAPIV3.Document | OpenAPIV31.Document
534
+ const v3Op = op as OpenAPIV3.OperationObject | OpenAPIV31.OperationObject
535
+
536
+ // Get security requirements (operation-level or global)
537
+ const security = v3Op.security ?? v3Doc.security ?? []
538
+
539
+ if (security.length === 0) return []
540
+
541
+ // Process the first security requirement (they are OR'd, we use first)
542
+ const secReq = security[0]
543
+ if (!secReq) return []
544
+
545
+ // Get all required scheme names
546
+ const schemeNames = Object.keys(secReq)
547
+
548
+ for (const schemeName of schemeNames) {
549
+ const scheme = v3Doc.components?.securitySchemes?.[schemeName] as
550
+ | OpenAPIV3.SecuritySchemeObject
551
+ | undefined
552
+
553
+ if (!scheme) continue
554
+
555
+ // Only handle API keys in headers
556
+ if (scheme.type === 'apiKey' && scheme.in === 'header') {
557
+ // Skip if this is the primary auth key
558
+ if (primaryAuthKey && scheme.name === primaryAuthKey) continue
559
+
560
+ headers.push({
561
+ key: scheme.name,
562
+ value: '',
563
+ description: scheme.description || `Required: ${schemeName}`,
564
+ active: true,
565
+ required: true,
566
+ })
567
+ }
568
+ }
569
+ } catch {
570
+ // Return empty on error
571
+ }
572
+
573
+ return headers
574
+ }