@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.
Files changed (400) hide show
  1. package/ai-agents/.claude/skills/bootstrap-docs/SKILL.md +710 -79
  2. package/ai-agents/.claude/skills/check-docs/SKILL.md +83 -8
  3. package/ai-agents/.claude/skills/create-doc/SKILL.md +267 -55
  4. package/ai-agents/.claude/skills/update-doc/SKILL.md +162 -63
  5. package/ai-agents/.cursor/rules/devdoc-bootstrap.mdc +145 -15
  6. package/ai-agents/.cursor/rules/devdoc-create.mdc +108 -57
  7. package/ai-agents/.cursor/rules/devdoc-update.mdc +93 -70
  8. package/ai-agents/.cursor/rules/devdoc.mdc +21 -0
  9. package/ai-agents/schemas/docs.schema.json +332 -0
  10. package/ai-agents/schemas/theme.schema.json +243 -0
  11. package/dist/cli/commands/create.js +4 -9
  12. package/dist/cli/commands/deploy.js +50 -25
  13. package/dist/cli/commands/dev.js +19 -10
  14. package/package.json +3 -2
  15. package/renderer/app/api/assets/[...path]/route.js +108 -0
  16. package/renderer/app/api/assets/route.js +114 -0
  17. package/renderer/app/api/assets/upload/route.js +163 -0
  18. package/renderer/app/api/auth-schemes/route.js +58 -0
  19. package/renderer/app/api/chat/route.js +759 -0
  20. package/renderer/app/api/codegen/route.js +52 -0
  21. package/renderer/app/api/collections/route.js +675 -0
  22. package/renderer/app/api/debug/route.js +47 -0
  23. package/renderer/app/api/deploy/route.js +199 -0
  24. package/renderer/app/api/device/route.js +36 -0
  25. package/renderer/app/api/docs/route.js +205 -0
  26. package/renderer/app/api/domains/add/route.js +121 -0
  27. package/renderer/app/api/domains/lookup/route.js +43 -0
  28. package/renderer/app/api/domains/remove/route.js +89 -0
  29. package/renderer/app/api/domains/status/route.js +140 -0
  30. package/renderer/app/api/domains/verify/route.js +168 -0
  31. package/renderer/app/api/keys/regenerate/route.js +71 -0
  32. package/renderer/app/api/local-assets/[...path]/route.js +108 -0
  33. package/renderer/app/api/openapi-spec/route.js +73 -0
  34. package/renderer/app/api/projects/[slug]/route.js +129 -0
  35. package/renderer/app/api/projects/[slug]/stats/route.js +80 -0
  36. package/renderer/app/api/projects/register/route.js +176 -0
  37. package/renderer/app/api/proxy/route.js +139 -0
  38. package/renderer/app/api/proxy-stream/route.js +156 -0
  39. package/renderer/app/api/redirects/route.js +35 -0
  40. package/renderer/app/api/schema/route.js +85 -0
  41. package/renderer/app/api/subdomains/check/route.js +158 -0
  42. package/renderer/app/api/suggestions/route.js +175 -0
  43. package/renderer/app/layout.js +47 -0
  44. package/renderer/app/llms-full.txt/route.js +257 -0
  45. package/renderer/app/llms.txt/route.js +219 -0
  46. package/renderer/app/page.js +12 -0
  47. package/renderer/app/robots.txt/route.js +66 -0
  48. package/renderer/app/sitemap.xml/route.js +145 -0
  49. package/renderer/components/docs/index.js +8 -0
  50. package/renderer/components/docs/mdx/accordion.js +113 -0
  51. package/renderer/components/docs/mdx/badge.js +72 -0
  52. package/renderer/components/docs/mdx/callouts.js +137 -0
  53. package/renderer/components/docs/mdx/cards.js +175 -0
  54. package/renderer/components/docs/mdx/changelog.js +100 -0
  55. package/renderer/components/docs/mdx/code-block.js +147 -0
  56. package/renderer/components/docs/mdx/code-group.js +287 -0
  57. package/renderer/components/docs/mdx/file-embeds.js +82 -0
  58. package/renderer/components/docs/mdx/frame.js +59 -0
  59. package/renderer/components/docs/mdx/highlight.js +90 -0
  60. package/renderer/components/docs/mdx/iframe.js +69 -0
  61. package/renderer/components/docs/mdx/image.js +135 -0
  62. package/renderer/components/docs/mdx/index.js +134 -0
  63. package/renderer/components/docs/mdx/landing.js +315 -0
  64. package/renderer/components/docs/mdx/mermaid.js +212 -0
  65. package/renderer/components/docs/mdx/param-field.js +112 -0
  66. package/renderer/components/docs/mdx/steps.js +74 -0
  67. package/renderer/components/docs/mdx/tabs.js +50 -0
  68. package/renderer/components/docs/mdx-renderer.js +77 -0
  69. package/renderer/components/docs/navigation/breadcrumbs.js +64 -0
  70. package/renderer/components/docs/navigation/index.js +6 -0
  71. package/renderer/components/docs/navigation/page-nav.js +57 -0
  72. package/renderer/components/docs/navigation/sidebar.js +375 -0
  73. package/renderer/components/docs/navigation/toc.js +89 -0
  74. package/renderer/components/docs/notice.js +77 -0
  75. package/renderer/components/docs-header.js +202 -0
  76. package/renderer/components/docs-viewer/agent/agent-chat.js +1930 -0
  77. package/renderer/components/docs-viewer/agent/cards/debug-context-card.js +107 -0
  78. package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.js +57 -0
  79. package/renderer/components/docs-viewer/agent/cards/index.js +45 -0
  80. package/renderer/components/docs-viewer/agent/cards/response-options-card.js +154 -0
  81. package/renderer/components/docs-viewer/agent/cards/types.js +22 -0
  82. package/renderer/components/docs-viewer/agent/chat-message.js +2 -0
  83. package/renderer/components/docs-viewer/agent/index.js +4 -0
  84. package/renderer/components/docs-viewer/agent/messages/assistant-message.js +108 -0
  85. package/renderer/components/docs-viewer/agent/messages/chat-message.js +34 -0
  86. package/renderer/components/docs-viewer/agent/messages/index.js +6 -0
  87. package/renderer/components/docs-viewer/agent/messages/tool-call-display.js +1065 -0
  88. package/renderer/components/docs-viewer/agent/messages/types.js +2 -0
  89. package/renderer/components/docs-viewer/agent/messages/typing-indicator.js +26 -0
  90. package/renderer/components/docs-viewer/agent/messages/user-message.js +37 -0
  91. package/renderer/components/docs-viewer/code-editor/{index.tsx → index.js} +1 -1
  92. package/renderer/components/docs-viewer/code-editor/notes-mode.js +1338 -0
  93. package/renderer/components/docs-viewer/content/changelog-page.js +297 -0
  94. package/renderer/components/docs-viewer/content/doc-page.js +264 -0
  95. package/renderer/components/docs-viewer/content/documentation-viewer.js +14 -0
  96. package/renderer/components/docs-viewer/content/index.js +29 -0
  97. package/renderer/components/docs-viewer/content/not-found-page.js +300 -0
  98. package/renderer/components/docs-viewer/content/request-details.js +528 -0
  99. package/renderer/components/docs-viewer/content/sections/auth.js +108 -0
  100. package/renderer/components/docs-viewer/content/sections/body.js +80 -0
  101. package/renderer/components/docs-viewer/content/sections/headers.js +64 -0
  102. package/renderer/components/docs-viewer/content/sections/overview.js +56 -0
  103. package/renderer/components/docs-viewer/content/sections/parameters.js +64 -0
  104. package/renderer/components/docs-viewer/content/sections/responses.js +91 -0
  105. package/renderer/components/docs-viewer/global-auth-modal.js +427 -0
  106. package/renderer/components/docs-viewer/index.js +1552 -0
  107. package/renderer/components/docs-viewer/playground/auth-editor.js +418 -0
  108. package/renderer/components/docs-viewer/playground/body-editor.js +240 -0
  109. package/renderer/components/docs-viewer/playground/code-editor.js +135 -0
  110. package/renderer/components/docs-viewer/playground/code-snippet.js +393 -0
  111. package/renderer/components/docs-viewer/playground/graphql-playground.js +734 -0
  112. package/renderer/components/docs-viewer/playground/index.js +682 -0
  113. package/renderer/components/docs-viewer/playground/key-value-editor.js +317 -0
  114. package/renderer/components/docs-viewer/playground/method-selector.js +65 -0
  115. package/renderer/components/docs-viewer/playground/request-builder.js +181 -0
  116. package/renderer/components/docs-viewer/playground/request-tabs.js +240 -0
  117. package/renderer/components/docs-viewer/playground/response-cards/idle-card.js +42 -0
  118. package/renderer/components/docs-viewer/playground/response-cards/index.js +72 -0
  119. package/renderer/components/docs-viewer/playground/response-cards/loading-card.js +24 -0
  120. package/renderer/components/docs-viewer/playground/response-cards/network-error-card.js +28 -0
  121. package/renderer/components/docs-viewer/playground/response-cards/response-body-card.js +308 -0
  122. package/renderer/components/docs-viewer/playground/response-cards/types.js +9 -0
  123. package/renderer/components/docs-viewer/playground/response-viewer.js +18 -0
  124. package/renderer/components/docs-viewer/search/index.js +2 -0
  125. package/renderer/components/docs-viewer/search/search-dialog.js +367 -0
  126. package/renderer/components/docs-viewer/search/use-search.js +89 -0
  127. package/renderer/components/docs-viewer/shared/markdown-renderer.js +423 -0
  128. package/renderer/components/docs-viewer/shared/method-badge.js +23 -0
  129. package/renderer/components/docs-viewer/shared/schema-viewer.js +321 -0
  130. package/renderer/components/docs-viewer/sidebar/collection-tree.js +222 -0
  131. package/renderer/components/docs-viewer/sidebar/endpoint-options.js +512 -0
  132. package/renderer/components/docs-viewer/sidebar/index.js +196 -0
  133. package/renderer/components/docs-viewer/sidebar/right-sidebar.js +163 -0
  134. package/renderer/components/docs-viewer/sidebar/sidebar-group.js +87 -0
  135. package/renderer/components/docs-viewer/sidebar/sidebar-item.js +172 -0
  136. package/renderer/components/docs-viewer/sidebar/sidebar-section.js +31 -0
  137. package/renderer/components/theme-provider.js +10 -0
  138. package/renderer/components/theme-toggle.js +106 -0
  139. package/renderer/components/ui/badge.js +29 -0
  140. package/renderer/components/ui/button.js +40 -0
  141. package/renderer/components/ui/dialog.js +50 -0
  142. package/renderer/components/ui/dropdown-menu.js +143 -0
  143. package/renderer/components/ui/input.js +12 -0
  144. package/renderer/components/ui/label.js +13 -0
  145. package/renderer/components/ui/navigation-menu.js +83 -0
  146. package/renderer/components/ui/select.js +116 -0
  147. package/renderer/components/ui/spinner.js +92 -0
  148. package/renderer/components/ui/tabs.js +34 -0
  149. package/renderer/components/ui/tooltip.js +43 -0
  150. package/renderer/hooks/use-code-copy.js +76 -0
  151. package/renderer/hooks/use-openapi-title.js +33 -0
  152. package/renderer/lib/api-docs/agent/index.js +4 -0
  153. package/renderer/lib/api-docs/agent/indexer.js +254 -0
  154. package/renderer/lib/api-docs/agent/spec-summary.js +227 -0
  155. package/renderer/lib/api-docs/agent/types.js +5 -0
  156. package/renderer/lib/api-docs/auth/auth-context.js +157 -0
  157. package/renderer/lib/api-docs/auth/auth-storage.js +66 -0
  158. package/renderer/lib/api-docs/auth/crypto.js +64 -0
  159. package/renderer/lib/api-docs/auth/index.js +3 -0
  160. package/renderer/lib/api-docs/code-editor/db.js +145 -0
  161. package/renderer/lib/api-docs/code-editor/hooks.js +254 -0
  162. package/renderer/lib/api-docs/code-editor/{index.ts → index.js} +3 -4
  163. package/renderer/lib/api-docs/code-editor/mode-context.js +154 -0
  164. package/renderer/lib/api-docs/code-editor/types.js +53 -0
  165. package/renderer/lib/api-docs/codegen/definitions.js +258 -0
  166. package/renderer/lib/api-docs/codegen/har.js +171 -0
  167. package/renderer/lib/api-docs/codegen/index.js +118 -0
  168. package/renderer/lib/api-docs/factories.js +136 -0
  169. package/renderer/lib/api-docs/{index.ts → index.js} +5 -10
  170. package/renderer/lib/api-docs/mobile-context.js +79 -0
  171. package/renderer/lib/api-docs/navigation-context.js +62 -0
  172. package/renderer/lib/api-docs/parsers/graphql/index.js +50 -0
  173. package/renderer/lib/api-docs/parsers/graphql/parser.js +350 -0
  174. package/renderer/lib/api-docs/parsers/graphql/transformer.js +215 -0
  175. package/renderer/lib/api-docs/parsers/graphql/types.js +46 -0
  176. package/renderer/lib/api-docs/parsers/openapi/dereferencer.js +43 -0
  177. package/renderer/lib/api-docs/parsers/openapi/extractors/auth.js +486 -0
  178. package/renderer/lib/api-docs/parsers/openapi/extractors/body.js +295 -0
  179. package/renderer/lib/api-docs/parsers/openapi/extractors/index.js +132 -0
  180. package/renderer/lib/api-docs/parsers/openapi/index.js +127 -0
  181. package/renderer/lib/api-docs/parsers/openapi/transformer.js +192 -0
  182. package/renderer/lib/api-docs/parsers/openapi/validator.js +24 -0
  183. package/renderer/lib/api-docs/playground/context.js +65 -0
  184. package/renderer/lib/api-docs/playground/navigation-context.js +74 -0
  185. package/renderer/lib/api-docs/playground/request-builder.js +163 -0
  186. package/renderer/lib/api-docs/playground/request-runner.js +224 -0
  187. package/renderer/lib/api-docs/playground/types.js +5 -0
  188. package/renderer/lib/api-docs/types.js +23 -0
  189. package/renderer/lib/api-docs/utils.js +212 -0
  190. package/renderer/lib/cache.js +157 -0
  191. package/renderer/lib/docs/config/domain-schema.js +161 -0
  192. package/renderer/lib/docs/config/index.js +5 -0
  193. package/renderer/lib/docs/config/loader.js +113 -0
  194. package/renderer/lib/docs/config/schema.js +269 -0
  195. package/renderer/lib/docs/index.js +8 -0
  196. package/renderer/lib/docs/mdx/compiler.js +128 -0
  197. package/renderer/lib/docs/mdx/frontmatter.js +73 -0
  198. package/renderer/lib/docs/mdx/index.js +8 -0
  199. package/renderer/lib/docs/navigation/generator.js +269 -0
  200. package/renderer/lib/docs/navigation/index.js +4 -0
  201. package/renderer/lib/docs/navigation/types.js +9 -0
  202. package/renderer/lib/docs-navigation-context.js +40 -0
  203. package/renderer/lib/multi-tenant/context.js +80 -0
  204. package/renderer/lib/storage/blob.js +767 -0
  205. package/renderer/lib/utils/icons.js +30 -0
  206. package/renderer/lib/utils.js +5 -0
  207. package/renderer/next.config.js +62 -0
  208. package/renderer/tsconfig.json +23 -5
  209. package/renderer/app/api/assets/[...path]/route.ts +0 -123
  210. package/renderer/app/api/assets/route.ts +0 -124
  211. package/renderer/app/api/assets/upload/route.ts +0 -177
  212. package/renderer/app/api/auth-schemes/route.ts +0 -77
  213. package/renderer/app/api/chat/route.ts +0 -858
  214. package/renderer/app/api/codegen/route.ts +0 -72
  215. package/renderer/app/api/collections/route.ts +0 -1002
  216. package/renderer/app/api/debug/route.ts +0 -53
  217. package/renderer/app/api/deploy/route.ts +0 -234
  218. package/renderer/app/api/device/route.ts +0 -42
  219. package/renderer/app/api/docs/route.ts +0 -201
  220. package/renderer/app/api/domains/add/route.ts +0 -132
  221. package/renderer/app/api/domains/lookup/route.ts +0 -43
  222. package/renderer/app/api/domains/remove/route.ts +0 -100
  223. package/renderer/app/api/domains/status/route.ts +0 -158
  224. package/renderer/app/api/domains/verify/route.ts +0 -181
  225. package/renderer/app/api/keys/regenerate/route.ts +0 -80
  226. package/renderer/app/api/local-assets/[...path]/route.ts +0 -122
  227. package/renderer/app/api/openapi-spec/route.ts +0 -151
  228. package/renderer/app/api/projects/[slug]/route.ts +0 -153
  229. package/renderer/app/api/projects/[slug]/stats/route.ts +0 -96
  230. package/renderer/app/api/projects/register/route.ts +0 -152
  231. package/renderer/app/api/proxy/route.ts +0 -149
  232. package/renderer/app/api/proxy-stream/route.ts +0 -168
  233. package/renderer/app/api/redirects/route.ts +0 -47
  234. package/renderer/app/api/schema/route.ts +0 -73
  235. package/renderer/app/api/subdomains/check/route.ts +0 -172
  236. package/renderer/app/api/suggestions/route.ts +0 -144
  237. package/renderer/app/layout.tsx +0 -54
  238. package/renderer/app/llms-full.txt/route.ts +0 -346
  239. package/renderer/app/llms.txt/route.ts +0 -279
  240. package/renderer/app/page.tsx +0 -14
  241. package/renderer/app/robots.txt/route.ts +0 -84
  242. package/renderer/app/sitemap.xml/route.ts +0 -199
  243. package/renderer/components/docs/index.ts +0 -12
  244. package/renderer/components/docs/mdx/accordion.tsx +0 -169
  245. package/renderer/components/docs/mdx/badge.tsx +0 -132
  246. package/renderer/components/docs/mdx/callouts.tsx +0 -154
  247. package/renderer/components/docs/mdx/cards.tsx +0 -241
  248. package/renderer/components/docs/mdx/changelog.tsx +0 -120
  249. package/renderer/components/docs/mdx/code-block.tsx +0 -186
  250. package/renderer/components/docs/mdx/code-group.tsx +0 -421
  251. package/renderer/components/docs/mdx/file-embeds.tsx +0 -105
  252. package/renderer/components/docs/mdx/frame.tsx +0 -112
  253. package/renderer/components/docs/mdx/highlight.tsx +0 -151
  254. package/renderer/components/docs/mdx/iframe.tsx +0 -134
  255. package/renderer/components/docs/mdx/image.tsx +0 -235
  256. package/renderer/components/docs/mdx/index.ts +0 -237
  257. package/renderer/components/docs/mdx/landing.tsx +0 -684
  258. package/renderer/components/docs/mdx/mermaid.tsx +0 -240
  259. package/renderer/components/docs/mdx/param-field.tsx +0 -200
  260. package/renderer/components/docs/mdx/steps.tsx +0 -113
  261. package/renderer/components/docs/mdx/tabs.tsx +0 -86
  262. package/renderer/components/docs/mdx-renderer.tsx +0 -100
  263. package/renderer/components/docs/navigation/breadcrumbs.tsx +0 -76
  264. package/renderer/components/docs/navigation/index.ts +0 -8
  265. package/renderer/components/docs/navigation/page-nav.tsx +0 -64
  266. package/renderer/components/docs/navigation/sidebar.tsx +0 -515
  267. package/renderer/components/docs/navigation/toc.tsx +0 -113
  268. package/renderer/components/docs/notice.tsx +0 -105
  269. package/renderer/components/docs-header.tsx +0 -278
  270. package/renderer/components/docs-viewer/agent/agent-chat.tsx +0 -2076
  271. package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +0 -90
  272. package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +0 -49
  273. package/renderer/components/docs-viewer/agent/cards/index.tsx +0 -50
  274. package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +0 -212
  275. package/renderer/components/docs-viewer/agent/cards/types.ts +0 -84
  276. package/renderer/components/docs-viewer/agent/chat-message.tsx +0 -17
  277. package/renderer/components/docs-viewer/agent/index.tsx +0 -6
  278. package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +0 -119
  279. package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +0 -46
  280. package/renderer/components/docs-viewer/agent/messages/index.ts +0 -17
  281. package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +0 -721
  282. package/renderer/components/docs-viewer/agent/messages/types.ts +0 -61
  283. package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +0 -24
  284. package/renderer/components/docs-viewer/agent/messages/user-message.tsx +0 -51
  285. package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +0 -1283
  286. package/renderer/components/docs-viewer/content/changelog-page.tsx +0 -331
  287. package/renderer/components/docs-viewer/content/doc-page.tsx +0 -367
  288. package/renderer/components/docs-viewer/content/documentation-viewer.tsx +0 -17
  289. package/renderer/components/docs-viewer/content/index.tsx +0 -29
  290. package/renderer/components/docs-viewer/content/not-found-page.tsx +0 -330
  291. package/renderer/components/docs-viewer/content/request-details.tsx +0 -330
  292. package/renderer/components/docs-viewer/content/sections/auth.tsx +0 -69
  293. package/renderer/components/docs-viewer/content/sections/body.tsx +0 -66
  294. package/renderer/components/docs-viewer/content/sections/headers.tsx +0 -43
  295. package/renderer/components/docs-viewer/content/sections/overview.tsx +0 -40
  296. package/renderer/components/docs-viewer/content/sections/parameters.tsx +0 -43
  297. package/renderer/components/docs-viewer/content/sections/responses.tsx +0 -87
  298. package/renderer/components/docs-viewer/global-auth-modal.tsx +0 -352
  299. package/renderer/components/docs-viewer/index.tsx +0 -1662
  300. package/renderer/components/docs-viewer/playground/auth-editor.tsx +0 -280
  301. package/renderer/components/docs-viewer/playground/body-editor.tsx +0 -221
  302. package/renderer/components/docs-viewer/playground/code-editor.tsx +0 -224
  303. package/renderer/components/docs-viewer/playground/code-snippet.tsx +0 -387
  304. package/renderer/components/docs-viewer/playground/graphql-playground.tsx +0 -745
  305. package/renderer/components/docs-viewer/playground/index.tsx +0 -671
  306. package/renderer/components/docs-viewer/playground/key-value-editor.tsx +0 -261
  307. package/renderer/components/docs-viewer/playground/method-selector.tsx +0 -60
  308. package/renderer/components/docs-viewer/playground/request-builder.tsx +0 -179
  309. package/renderer/components/docs-viewer/playground/request-tabs.tsx +0 -237
  310. package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +0 -21
  311. package/renderer/components/docs-viewer/playground/response-cards/index.tsx +0 -93
  312. package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +0 -16
  313. package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +0 -23
  314. package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +0 -268
  315. package/renderer/components/docs-viewer/playground/response-cards/types.ts +0 -82
  316. package/renderer/components/docs-viewer/playground/response-viewer.tsx +0 -43
  317. package/renderer/components/docs-viewer/search/index.ts +0 -2
  318. package/renderer/components/docs-viewer/search/search-dialog.tsx +0 -331
  319. package/renderer/components/docs-viewer/search/use-search.ts +0 -117
  320. package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +0 -431
  321. package/renderer/components/docs-viewer/shared/method-badge.tsx +0 -41
  322. package/renderer/components/docs-viewer/shared/schema-viewer.tsx +0 -349
  323. package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +0 -259
  324. package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +0 -316
  325. package/renderer/components/docs-viewer/sidebar/index.tsx +0 -282
  326. package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +0 -202
  327. package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +0 -118
  328. package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +0 -212
  329. package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +0 -38
  330. package/renderer/components/theme-provider.tsx +0 -11
  331. package/renderer/components/theme-toggle.tsx +0 -76
  332. package/renderer/components/ui/badge.tsx +0 -46
  333. package/renderer/components/ui/button.tsx +0 -59
  334. package/renderer/components/ui/dialog.tsx +0 -118
  335. package/renderer/components/ui/dropdown-menu.tsx +0 -257
  336. package/renderer/components/ui/input.tsx +0 -21
  337. package/renderer/components/ui/label.tsx +0 -24
  338. package/renderer/components/ui/navigation-menu.tsx +0 -168
  339. package/renderer/components/ui/select.tsx +0 -190
  340. package/renderer/components/ui/spinner.tsx +0 -114
  341. package/renderer/components/ui/tabs.tsx +0 -66
  342. package/renderer/components/ui/tooltip.tsx +0 -61
  343. package/renderer/hooks/use-code-copy.ts +0 -88
  344. package/renderer/hooks/use-openapi-title.ts +0 -44
  345. package/renderer/lib/api-docs/agent/index.ts +0 -6
  346. package/renderer/lib/api-docs/agent/indexer.ts +0 -323
  347. package/renderer/lib/api-docs/agent/spec-summary.ts +0 -335
  348. package/renderer/lib/api-docs/agent/types.ts +0 -116
  349. package/renderer/lib/api-docs/auth/auth-context.tsx +0 -225
  350. package/renderer/lib/api-docs/auth/auth-storage.ts +0 -87
  351. package/renderer/lib/api-docs/auth/crypto.ts +0 -89
  352. package/renderer/lib/api-docs/auth/index.ts +0 -4
  353. package/renderer/lib/api-docs/code-editor/db.ts +0 -164
  354. package/renderer/lib/api-docs/code-editor/hooks.ts +0 -266
  355. package/renderer/lib/api-docs/code-editor/mode-context.tsx +0 -207
  356. package/renderer/lib/api-docs/code-editor/types.ts +0 -105
  357. package/renderer/lib/api-docs/codegen/definitions.ts +0 -297
  358. package/renderer/lib/api-docs/codegen/har.ts +0 -251
  359. package/renderer/lib/api-docs/codegen/index.ts +0 -159
  360. package/renderer/lib/api-docs/factories.ts +0 -170
  361. package/renderer/lib/api-docs/mobile-context.tsx +0 -112
  362. package/renderer/lib/api-docs/navigation-context.tsx +0 -88
  363. package/renderer/lib/api-docs/parsers/graphql/README.md +0 -129
  364. package/renderer/lib/api-docs/parsers/graphql/index.ts +0 -91
  365. package/renderer/lib/api-docs/parsers/graphql/parser.ts +0 -491
  366. package/renderer/lib/api-docs/parsers/graphql/transformer.ts +0 -246
  367. package/renderer/lib/api-docs/parsers/graphql/types.ts +0 -283
  368. package/renderer/lib/api-docs/parsers/openapi/README.md +0 -32
  369. package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +0 -60
  370. package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +0 -574
  371. package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +0 -403
  372. package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +0 -232
  373. package/renderer/lib/api-docs/parsers/openapi/index.ts +0 -171
  374. package/renderer/lib/api-docs/parsers/openapi/transformer.ts +0 -278
  375. package/renderer/lib/api-docs/parsers/openapi/validator.ts +0 -31
  376. package/renderer/lib/api-docs/playground/context.tsx +0 -107
  377. package/renderer/lib/api-docs/playground/navigation-context.tsx +0 -124
  378. package/renderer/lib/api-docs/playground/request-builder.ts +0 -223
  379. package/renderer/lib/api-docs/playground/request-runner.ts +0 -282
  380. package/renderer/lib/api-docs/playground/types.ts +0 -35
  381. package/renderer/lib/api-docs/types.ts +0 -269
  382. package/renderer/lib/api-docs/utils.ts +0 -311
  383. package/renderer/lib/cache.ts +0 -193
  384. package/renderer/lib/docs/config/domain-schema.ts +0 -260
  385. package/renderer/lib/docs/config/index.ts +0 -43
  386. package/renderer/lib/docs/config/loader.ts +0 -142
  387. package/renderer/lib/docs/config/schema.ts +0 -308
  388. package/renderer/lib/docs/index.ts +0 -12
  389. package/renderer/lib/docs/mdx/compiler.ts +0 -176
  390. package/renderer/lib/docs/mdx/frontmatter.ts +0 -91
  391. package/renderer/lib/docs/mdx/index.ts +0 -26
  392. package/renderer/lib/docs/navigation/generator.ts +0 -348
  393. package/renderer/lib/docs/navigation/index.ts +0 -12
  394. package/renderer/lib/docs/navigation/types.ts +0 -123
  395. package/renderer/lib/docs-navigation-context.tsx +0 -80
  396. package/renderer/lib/multi-tenant/context.ts +0 -105
  397. package/renderer/lib/storage/blob.ts +0 -1083
  398. package/renderer/lib/utils/icons.ts +0 -48
  399. package/renderer/lib/utils.ts +0 -6
  400. package/renderer/next.config.ts +0 -76
@@ -0,0 +1,227 @@
1
+ /**
2
+ * Recursively collect all requests from a collection
3
+ */ function collectAllRequests(collection) {
4
+ const requests = [
5
+ ...collection.requests
6
+ ];
7
+ for (const folder of collection.folders){
8
+ requests.push(...collectAllRequests(folder));
9
+ }
10
+ return requests;
11
+ }
12
+ /**
13
+ * Extract base URL from endpoints
14
+ */ function extractBaseUrl(requests) {
15
+ if (requests.length === 0) return null;
16
+ const firstEndpoint = requests[0].endpoint;
17
+ try {
18
+ const url = new URL(firstEndpoint);
19
+ return `${url.protocol}//${url.host}`;
20
+ } catch {
21
+ // Try to extract from path pattern
22
+ const match = firstEndpoint.match(/^(https?:\/\/[^\/]+)/);
23
+ return match ? match[1] : null;
24
+ }
25
+ }
26
+ /**
27
+ * Analyze authentication patterns across endpoints
28
+ */ function analyzeAuth(requests) {
29
+ const authTypes = new Set();
30
+ const authCounts = {};
31
+ for (const req of requests){
32
+ const type = req.auth.authType;
33
+ if (type !== 'none' && type !== 'inherit') {
34
+ authTypes.add(type);
35
+ authCounts[type] = (authCounts[type] || 0) + 1;
36
+ }
37
+ }
38
+ // Find primary auth method (most common)
39
+ let primaryMethod = null;
40
+ let maxCount = 0;
41
+ for (const [type, count] of Object.entries(authCounts)){
42
+ if (count > maxCount) {
43
+ maxCount = count;
44
+ primaryMethod = type;
45
+ }
46
+ }
47
+ return {
48
+ types: Array.from(authTypes),
49
+ primaryMethod
50
+ };
51
+ }
52
+ /**
53
+ * Group endpoints by resource/tag
54
+ */ function groupByResource(requests) {
55
+ const groups = new Map();
56
+ for (const req of requests){
57
+ // Use first tag, or extract from path, or use "Other"
58
+ let groupName = req.tags[0] || extractResourceFromPath(req.endpoint) || 'Other';
59
+ groupName = groupName.charAt(0).toUpperCase() + groupName.slice(1);
60
+ const summary = {
61
+ name: req.name,
62
+ method: req.method,
63
+ path: req.endpoint,
64
+ description: req.description,
65
+ tags: req.tags,
66
+ hasAuth: req.auth.authType !== 'none',
67
+ hasBody: req.body.body !== null,
68
+ paramCount: req.params.filter((p)=>p.active).length
69
+ };
70
+ if (!groups.has(groupName)) {
71
+ groups.set(groupName, []);
72
+ }
73
+ groups.get(groupName).push(summary);
74
+ }
75
+ // Convert to array and calculate operations
76
+ return Array.from(groups.entries()).map(([name, endpoints])=>({
77
+ name,
78
+ endpoints,
79
+ operations: {
80
+ create: endpoints.filter((e)=>e.method === 'POST').length,
81
+ read: endpoints.filter((e)=>e.method === 'GET').length,
82
+ update: endpoints.filter((e)=>e.method === 'PUT' || e.method === 'PATCH').length,
83
+ delete: endpoints.filter((e)=>e.method === 'DELETE').length
84
+ }
85
+ }));
86
+ }
87
+ /**
88
+ * Extract resource name from path
89
+ */ function extractResourceFromPath(path) {
90
+ // Remove base URL and extract first path segment
91
+ const pathMatch = path.match(/\/api\/v?\d*\/?([^\/]+)/) || path.match(/\/([^\/]+)/);
92
+ if (pathMatch) {
93
+ return pathMatch[1].replace(/[-_]/g, ' ');
94
+ }
95
+ return null;
96
+ }
97
+ /**
98
+ * Detect common API patterns
99
+ */ function detectPatterns(groups) {
100
+ const patterns = [];
101
+ // Check for RESTful CRUD patterns
102
+ const hasCRUD = groups.some((g)=>g.operations.create > 0 && g.operations.read > 0 && (g.operations.update > 0 || g.operations.delete > 0));
103
+ if (hasCRUD) {
104
+ patterns.push('RESTful CRUD operations');
105
+ }
106
+ // Check for authentication endpoints
107
+ const hasAuthEndpoints = groups.some((g)=>g.name.toLowerCase().includes('auth') || g.endpoints.some((e)=>e.name.toLowerCase().includes('login') || e.name.toLowerCase().includes('token') || e.name.toLowerCase().includes('session')));
108
+ if (hasAuthEndpoints) {
109
+ patterns.push('Authentication/Authorization flows');
110
+ }
111
+ // Check for search/filter patterns
112
+ const hasSearch = groups.some((g)=>g.endpoints.some((e)=>e.name.toLowerCase().includes('search') || e.name.toLowerCase().includes('filter') || e.paramCount > 2));
113
+ if (hasSearch) {
114
+ patterns.push('Search and filtering capabilities');
115
+ }
116
+ // Check for batch operations
117
+ const hasBatch = groups.some((g)=>g.endpoints.some((e)=>e.name.toLowerCase().includes('batch') || e.name.toLowerCase().includes('bulk')));
118
+ if (hasBatch) {
119
+ patterns.push('Batch/bulk operations');
120
+ }
121
+ // Check for webhooks
122
+ const hasWebhooks = groups.some((g)=>g.name.toLowerCase().includes('webhook') || g.endpoints.some((e)=>e.name.toLowerCase().includes('webhook')));
123
+ if (hasWebhooks) {
124
+ patterns.push('Webhook integrations');
125
+ }
126
+ return patterns;
127
+ }
128
+ /**
129
+ * Generate suggested use cases based on resources
130
+ */ function generateUseCases(groups, _apiName) {
131
+ const useCases = [];
132
+ for (const group of groups.slice(0, 5)){
133
+ if (group.operations.create > 0) {
134
+ useCases.push(`Create and manage ${group.name.toLowerCase()}`);
135
+ }
136
+ if (group.operations.read > 0 && group.endpoints.length > 1) {
137
+ useCases.push(`List and retrieve ${group.name.toLowerCase()} data`);
138
+ }
139
+ }
140
+ // Add generic use cases based on patterns
141
+ const hasAuth = groups.some((g)=>g.name.toLowerCase().includes('auth'));
142
+ if (hasAuth) {
143
+ useCases.push('Implement user authentication flow');
144
+ }
145
+ return useCases.slice(0, 6) // Limit to 6 use cases
146
+ ;
147
+ }
148
+ /**
149
+ * Generate a comprehensive API summary from a collection
150
+ */ export function generateAPISummary(collection) {
151
+ const requests = collectAllRequests(collection);
152
+ const groups = groupByResource(requests);
153
+ const patterns = detectPatterns(groups);
154
+ return {
155
+ name: collection.name,
156
+ description: collection.description,
157
+ baseUrl: extractBaseUrl(requests),
158
+ totalEndpoints: requests.length,
159
+ authSummary: analyzeAuth(requests),
160
+ resourceGroups: groups,
161
+ commonPatterns: patterns,
162
+ suggestedUseCases: generateUseCases(groups, collection.name)
163
+ };
164
+ }
165
+ /**
166
+ * Format API summary as a string for the LLM system prompt
167
+ */ export function formatAPISummaryForPrompt(summary) {
168
+ const sections = [];
169
+ // Header
170
+ sections.push(`# API Overview: ${summary.name}`);
171
+ if (summary.description) {
172
+ sections.push(summary.description);
173
+ }
174
+ // Stats
175
+ sections.push(`\n## Quick Stats`);
176
+ sections.push(`- Total Endpoints: ${summary.totalEndpoints}`);
177
+ if (summary.baseUrl) {
178
+ sections.push(`- Base URL: ${summary.baseUrl}`);
179
+ }
180
+ // Auth
181
+ if (summary.authSummary.types.length > 0) {
182
+ sections.push(`- Authentication: ${summary.authSummary.types.join(', ')}`);
183
+ if (summary.authSummary.primaryMethod) {
184
+ sections.push(`- Primary Auth Method: ${summary.authSummary.primaryMethod}`);
185
+ }
186
+ }
187
+ // Resources
188
+ sections.push(`\n## API Resources (${summary.resourceGroups.length} groups)`);
189
+ for (const group of summary.resourceGroups){
190
+ const ops = [];
191
+ if (group.operations.create) ops.push(`${group.operations.create} create`);
192
+ if (group.operations.read) ops.push(`${group.operations.read} read`);
193
+ if (group.operations.update) ops.push(`${group.operations.update} update`);
194
+ if (group.operations.delete) ops.push(`${group.operations.delete} delete`);
195
+ sections.push(`\n### ${group.name} (${group.endpoints.length} endpoints)`);
196
+ if (ops.length > 0) {
197
+ sections.push(`Operations: ${ops.join(', ')}`);
198
+ }
199
+ // List key endpoints (up to 5 per group)
200
+ const keyEndpoints = group.endpoints.slice(0, 5);
201
+ for (const ep of keyEndpoints){
202
+ const authIndicator = ep.hasAuth ? '🔐' : '';
203
+ sections.push(`- [${ep.method}] ${ep.name} ${authIndicator}`);
204
+ if (ep.description) {
205
+ sections.push(` ${ep.description.slice(0, 100)}${ep.description.length > 100 ? '...' : ''}`);
206
+ }
207
+ }
208
+ if (group.endpoints.length > 5) {
209
+ sections.push(` ... and ${group.endpoints.length - 5} more`);
210
+ }
211
+ }
212
+ // Patterns
213
+ if (summary.commonPatterns.length > 0) {
214
+ sections.push(`\n## Common Patterns`);
215
+ for (const pattern of summary.commonPatterns){
216
+ sections.push(`- ${pattern}`);
217
+ }
218
+ }
219
+ // Use cases
220
+ if (summary.suggestedUseCases.length > 0) {
221
+ sections.push(`\n## Suggested Use Cases`);
222
+ for (const useCase of summary.suggestedUseCases){
223
+ sections.push(`- ${useCase}`);
224
+ }
225
+ }
226
+ return sections.join('\n');
227
+ }
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Agent Mode Types
3
+ * Types for the AI-powered API documentation assistant
4
+ */ // Navigation action
5
+ export { };
@@ -0,0 +1,157 @@
1
+ 'use client';
2
+ import { jsx as _jsx } from "react/jsx-runtime";
3
+ import { createContext, useContext, useState, useEffect, useCallback, useRef } from 'react';
4
+ import { authStorage, fetchDeviceId } from './auth-storage';
5
+ const AuthContext = /*#__PURE__*/ createContext(null);
6
+ const DEFAULT_AUTH = {
7
+ authType: 'none',
8
+ authActive: true
9
+ };
10
+ export function AuthProvider({ children }) {
11
+ const [globalAuth, setGlobalAuthState] = useState(null);
12
+ const [isInitialized, setIsInitialized] = useState(false);
13
+ const [specSchemes, setSpecSchemes] = useState([]);
14
+ const [specDefaultAuth, setSpecDefaultAuth] = useState(null);
15
+ const deviceIdRef = useRef(null);
16
+ // Load auth from localStorage on mount (async with encryption)
17
+ useEffect(()=>{
18
+ async function initializeAuth() {
19
+ try {
20
+ // Fetch device ID from backend (required for encryption)
21
+ const deviceId = await fetchDeviceId();
22
+ deviceIdRef.current = deviceId;
23
+ // Load encrypted auth from storage
24
+ const stored = await authStorage.load(deviceId);
25
+ if (stored) {
26
+ setGlobalAuthState(stored);
27
+ }
28
+ // Fetch security schemes from the spec
29
+ const schemesResponse = await fetch('/api/auth-schemes');
30
+ if (schemesResponse.ok) {
31
+ const data = await schemesResponse.json();
32
+ setSpecSchemes(data.schemes || []);
33
+ setSpecDefaultAuth(data.defaultAuth || null);
34
+ }
35
+ } catch (error) {
36
+ console.error('[AuthProvider] Failed to initialize auth:', error);
37
+ // Clear any potentially corrupted data
38
+ authStorage.clear();
39
+ } finally{
40
+ setIsInitialized(true);
41
+ }
42
+ }
43
+ initializeAuth();
44
+ }, []);
45
+ const setGlobalAuth = useCallback(async (auth)=>{
46
+ if (!deviceIdRef.current) {
47
+ console.error('[AuthProvider] Cannot save auth: no device ID');
48
+ return;
49
+ }
50
+ setGlobalAuthState(auth);
51
+ await authStorage.save(auth, deviceIdRef.current);
52
+ }, []);
53
+ const clearAuth = useCallback(()=>{
54
+ setGlobalAuthState(null);
55
+ authStorage.clear();
56
+ }, []);
57
+ /**
58
+ * Get effective auth for an endpoint
59
+ * - If endpoint has 'inherit' auth type, use global auth
60
+ * - If endpoint has 'none' auth type and global is set, use global
61
+ * - Otherwise use endpoint's own auth
62
+ */ const getEffectiveAuth = useCallback((endpointAuth)=>{
63
+ // If endpoint explicitly wants to inherit, use global
64
+ if (endpointAuth.authType === 'inherit') {
65
+ return globalAuth || DEFAULT_AUTH;
66
+ }
67
+ // If endpoint has no auth but global is configured, suggest global
68
+ // but don't override - this allows explicit "no auth" endpoints
69
+ if (endpointAuth.authType === 'none' && globalAuth && globalAuth.authType !== 'none') {
70
+ // Return global auth but the UI should indicate it's using global
71
+ return globalAuth;
72
+ }
73
+ // Use endpoint's own auth
74
+ return endpointAuth;
75
+ }, [
76
+ globalAuth
77
+ ]);
78
+ /**
79
+ * Check if an endpoint is using global auth
80
+ */ const isUsingGlobalAuth = useCallback((endpointAuth)=>{
81
+ if (!globalAuth || globalAuth.authType === 'none') {
82
+ return false;
83
+ }
84
+ // Using global if inheriting or if endpoint has no auth
85
+ return endpointAuth.authType === 'inherit' || endpointAuth.authType === 'none';
86
+ }, [
87
+ globalAuth
88
+ ]);
89
+ const isAuthenticated = globalAuth !== null && globalAuth.authType !== 'none';
90
+ const value = {
91
+ globalAuth,
92
+ isAuthenticated,
93
+ setGlobalAuth,
94
+ clearAuth,
95
+ getEffectiveAuth,
96
+ isUsingGlobalAuth,
97
+ specSchemes,
98
+ specDefaultAuth
99
+ };
100
+ // Don't render until we've checked sessionStorage
101
+ if (!isInitialized) {
102
+ return null;
103
+ }
104
+ return /*#__PURE__*/ _jsx(AuthContext.Provider, {
105
+ value: value,
106
+ children: children
107
+ });
108
+ }
109
+ export function useAuth() {
110
+ const context = useContext(AuthContext);
111
+ if (!context) {
112
+ throw new Error('useAuth must be used within an AuthProvider');
113
+ }
114
+ return context;
115
+ }
116
+ /**
117
+ * Get a human-readable label for auth type
118
+ */ export function getAuthTypeLabel(auth) {
119
+ if (!auth || !auth.authType) {
120
+ return 'Not configured';
121
+ }
122
+ switch(auth.authType){
123
+ case 'none':
124
+ return 'No Auth';
125
+ case 'inherit':
126
+ return 'Inherited';
127
+ case 'basic':
128
+ return 'Basic Auth';
129
+ case 'bearer':
130
+ return 'Bearer Token';
131
+ case 'api-key':
132
+ return `API Key (${auth.key || 'key'})`;
133
+ case 'oauth-2':
134
+ return 'OAuth 2.0';
135
+ default:
136
+ return 'Not configured';
137
+ }
138
+ }
139
+ /**
140
+ * Get a masked preview of auth value
141
+ */ export function getAuthPreview(auth) {
142
+ switch(auth.authType){
143
+ case 'none':
144
+ case 'inherit':
145
+ return '';
146
+ case 'basic':
147
+ return auth.username ? `${auth.username}:****` : '';
148
+ case 'bearer':
149
+ return auth.token ? `${auth.token.slice(0, 8)}••••` : '';
150
+ case 'api-key':
151
+ return auth.value ? `${auth.value.slice(0, 4)}••••` : '';
152
+ case 'oauth-2':
153
+ return auth.grantTypeInfo.token ? `${auth.grantTypeInfo.token.slice(0, 8)}••••` : '';
154
+ default:
155
+ return '';
156
+ }
157
+ }
@@ -0,0 +1,66 @@
1
+ 'use client';
2
+ import { encrypt, decrypt } from './crypto';
3
+ const AUTH_STORAGE_KEY = 'brainfish-api-global-auth';
4
+ /**
5
+ * Fetch device ID from backend API
6
+ * The device ID is stored in an httpOnly cookie
7
+ * Throws error if device ID cannot be fetched (no fallback)
8
+ */ export async function fetchDeviceId() {
9
+ const response = await fetch('/api/device', {
10
+ credentials: 'include'
11
+ });
12
+ if (!response.ok) {
13
+ throw new Error('Failed to fetch device ID');
14
+ }
15
+ const data = await response.json();
16
+ if (!data.deviceId) {
17
+ throw new Error('No device ID returned from server');
18
+ }
19
+ return data.deviceId;
20
+ }
21
+ /**
22
+ * Encrypted localStorage wrapper for global authentication
23
+ * Uses device ID from backend as encryption key
24
+ * No fallbacks - encryption is required
25
+ */ export const authStorage = {
26
+ /**
27
+ * Save auth configuration to localStorage (encrypted)
28
+ * Requires valid device ID - throws if missing
29
+ */ async save (auth, deviceId) {
30
+ if (!deviceId) {
31
+ throw new Error('Device ID required for secure storage');
32
+ }
33
+ const data = {
34
+ globalAuth: auth,
35
+ timestamp: Date.now()
36
+ };
37
+ const encrypted = await encrypt(JSON.stringify(data), deviceId);
38
+ localStorage.setItem(AUTH_STORAGE_KEY, encrypted);
39
+ },
40
+ /**
41
+ * Load auth configuration from localStorage (decrypted)
42
+ * Returns null if no data or decryption fails
43
+ */ async load (deviceId) {
44
+ if (!deviceId) return null;
45
+ const stored = localStorage.getItem(AUTH_STORAGE_KEY);
46
+ if (!stored) return null;
47
+ try {
48
+ const jsonData = await decrypt(stored, deviceId);
49
+ const data = JSON.parse(jsonData);
50
+ if (!data.globalAuth?.authType) {
51
+ localStorage.removeItem(AUTH_STORAGE_KEY);
52
+ return null;
53
+ }
54
+ return data.globalAuth;
55
+ } catch {
56
+ // Decryption failed - clear corrupted/invalid data
57
+ localStorage.removeItem(AUTH_STORAGE_KEY);
58
+ return null;
59
+ }
60
+ },
61
+ /**
62
+ * Clear stored auth configuration
63
+ */ clear () {
64
+ localStorage.removeItem(AUTH_STORAGE_KEY);
65
+ }
66
+ };
@@ -0,0 +1,64 @@
1
+ 'use client';
2
+ /**
3
+ * Web Crypto API utilities for encrypting/decrypting auth credentials
4
+ * Uses AES-GCM with a key derived from device ID
5
+ */ const ALGORITHM = 'AES-GCM';
6
+ const KEY_LENGTH = 256;
7
+ const IV_LENGTH = 12 // 96 bits for AES-GCM
8
+ ;
9
+ /**
10
+ * Derive a crypto key from a device ID using PBKDF2
11
+ */ async function deriveKey(deviceId) {
12
+ const encoder = new TextEncoder();
13
+ const keyMaterial = await crypto.subtle.importKey('raw', encoder.encode(deviceId), 'PBKDF2', false, [
14
+ 'deriveKey'
15
+ ]);
16
+ // Use a fixed salt (could be made dynamic per-user if needed)
17
+ const salt = encoder.encode('brainfish-api-docs-v1');
18
+ return crypto.subtle.deriveKey({
19
+ name: 'PBKDF2',
20
+ salt,
21
+ iterations: 100000,
22
+ hash: 'SHA-256'
23
+ }, keyMaterial, {
24
+ name: ALGORITHM,
25
+ length: KEY_LENGTH
26
+ }, false, [
27
+ 'encrypt',
28
+ 'decrypt'
29
+ ]);
30
+ }
31
+ /**
32
+ * Encrypt a string using AES-GCM with device ID as key
33
+ */ export async function encrypt(plaintext, deviceId) {
34
+ const key = await deriveKey(deviceId);
35
+ const encoder = new TextEncoder();
36
+ const data = encoder.encode(plaintext);
37
+ // Generate random IV
38
+ const iv = crypto.getRandomValues(new Uint8Array(IV_LENGTH));
39
+ const encrypted = await crypto.subtle.encrypt({
40
+ name: ALGORITHM,
41
+ iv
42
+ }, key, data);
43
+ // Combine IV + encrypted data and encode as base64
44
+ const combined = new Uint8Array(iv.length + encrypted.byteLength);
45
+ combined.set(iv);
46
+ combined.set(new Uint8Array(encrypted), iv.length);
47
+ return btoa(String.fromCharCode(...combined));
48
+ }
49
+ /**
50
+ * Decrypt a string using AES-GCM with device ID as key
51
+ */ export async function decrypt(ciphertext, deviceId) {
52
+ const key = await deriveKey(deviceId);
53
+ // Decode base64
54
+ const combined = Uint8Array.from(atob(ciphertext), (c)=>c.charCodeAt(0));
55
+ // Extract IV and encrypted data
56
+ const iv = combined.slice(0, IV_LENGTH);
57
+ const encrypted = combined.slice(IV_LENGTH);
58
+ const decrypted = await crypto.subtle.decrypt({
59
+ name: ALGORITHM,
60
+ iv
61
+ }, key, encrypted);
62
+ const decoder = new TextDecoder();
63
+ return decoder.decode(decrypted);
64
+ }
@@ -0,0 +1,3 @@
1
+ export { AuthProvider, useAuth, getAuthTypeLabel, getAuthPreview } from './auth-context';
2
+ export { authStorage, fetchDeviceId } from './auth-storage';
3
+ export { encrypt, decrypt } from './crypto';
@@ -0,0 +1,145 @@
1
+ function _define_property(obj, key, value) {
2
+ if (key in obj) {
3
+ Object.defineProperty(obj, key, {
4
+ value: value,
5
+ enumerable: true,
6
+ configurable: true,
7
+ writable: true
8
+ });
9
+ } else {
10
+ obj[key] = value;
11
+ }
12
+ return obj;
13
+ }
14
+ import Dexie from 'dexie';
15
+ /**
16
+ * Simplified database for Notes workspace
17
+ * - One workspace per API (not multiple projects)
18
+ * - Notes are files for brainstorming (code, diagrams, markdown)
19
+ */ class NotesDatabase extends Dexie {
20
+ constructor(){
21
+ super('brainfish-notes'), _define_property(this, "workspaces", void 0), _define_property(this, "notes", void 0), _define_property(this, "editorStates", void 0);
22
+ this.version(1).stores({
23
+ // Single workspace per API
24
+ workspaces: 'id, apiSpecUrl, updatedAt',
25
+ // Notes/files in workspace
26
+ notes: 'id, workspaceId, [workspaceId+path], updatedAt',
27
+ // Editor state per workspace
28
+ editorStates: 'workspaceId'
29
+ });
30
+ }
31
+ }
32
+ // Singleton database instance
33
+ export const db = new NotesDatabase();
34
+ /**
35
+ * Generate a workspace ID from API spec URL
36
+ */ export function getWorkspaceId(apiSpecUrl) {
37
+ // Simple hash for consistent ID
38
+ let hash = 0;
39
+ for(let i = 0; i < apiSpecUrl.length; i++){
40
+ const char = apiSpecUrl.charCodeAt(i);
41
+ hash = (hash << 5) - hash + char;
42
+ hash = hash & hash; // Convert to 32-bit integer
43
+ }
44
+ return `ws-${Math.abs(hash).toString(16)}`;
45
+ }
46
+ /**
47
+ * Database operations - simplified for notes workspace
48
+ */ export const dbOperations = {
49
+ // Workspace operations (auto-created, one per API)
50
+ async getOrCreateWorkspace (apiSpecUrl, name) {
51
+ const id = getWorkspaceId(apiSpecUrl);
52
+ let workspace = await db.workspaces.get(id);
53
+ if (!workspace) {
54
+ workspace = {
55
+ id,
56
+ name,
57
+ apiSpecUrl,
58
+ createdAt: new Date(),
59
+ updatedAt: new Date()
60
+ };
61
+ await db.workspaces.add(workspace);
62
+ }
63
+ return workspace;
64
+ },
65
+ async getWorkspace (id) {
66
+ return db.workspaces.get(id);
67
+ },
68
+ async updateWorkspace (id, updates) {
69
+ await db.workspaces.update(id, {
70
+ ...updates,
71
+ updatedAt: new Date()
72
+ });
73
+ },
74
+ // Note/file operations
75
+ async createNote (workspaceId, path, content = '') {
76
+ const note = {
77
+ id: crypto.randomUUID(),
78
+ workspaceId,
79
+ path,
80
+ content,
81
+ createdAt: new Date(),
82
+ updatedAt: new Date()
83
+ };
84
+ await db.notes.add(note);
85
+ // Update workspace timestamp
86
+ await db.workspaces.update(workspaceId, {
87
+ updatedAt: new Date()
88
+ });
89
+ return note;
90
+ },
91
+ async getNote (workspaceId, path) {
92
+ return db.notes.where('[workspaceId+path]').equals([
93
+ workspaceId,
94
+ path
95
+ ]).first();
96
+ },
97
+ async listNotes (workspaceId) {
98
+ return db.notes.where('workspaceId').equals(workspaceId).toArray();
99
+ },
100
+ async updateNote (workspaceId, path, content) {
101
+ await db.notes.where('[workspaceId+path]').equals([
102
+ workspaceId,
103
+ path
104
+ ]).modify({
105
+ content,
106
+ updatedAt: new Date()
107
+ });
108
+ await db.workspaces.update(workspaceId, {
109
+ updatedAt: new Date()
110
+ });
111
+ },
112
+ async deleteNote (workspaceId, path) {
113
+ await db.notes.where('[workspaceId+path]').equals([
114
+ workspaceId,
115
+ path
116
+ ]).delete();
117
+ await db.workspaces.update(workspaceId, {
118
+ updatedAt: new Date()
119
+ });
120
+ },
121
+ async renameNote (workspaceId, oldPath, newPath) {
122
+ await db.notes.where('[workspaceId+path]').equals([
123
+ workspaceId,
124
+ oldPath
125
+ ]).modify({
126
+ path: newPath,
127
+ updatedAt: new Date()
128
+ });
129
+ await db.workspaces.update(workspaceId, {
130
+ updatedAt: new Date()
131
+ });
132
+ },
133
+ // Editor state operations
134
+ async getEditorState (workspaceId) {
135
+ return db.editorStates.get(workspaceId);
136
+ },
137
+ async saveEditorState (state) {
138
+ await db.editorStates.put(state);
139
+ },
140
+ // Clear all notes in workspace (reset)
141
+ async clearWorkspace (workspaceId) {
142
+ await db.notes.where('workspaceId').equals(workspaceId).delete();
143
+ await db.editorStates.delete(workspaceId);
144
+ }
145
+ };