@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,224 @@
1
+ /**
2
+ * Request Runner
3
+ *
4
+ * Executes HTTP requests via backend proxy to bypass CORS
5
+ * and ensure auth headers are sent properly.
6
+ * Supports streaming responses for SSE endpoints.
7
+ */ import { buildRequest } from './request-builder';
8
+ /**
9
+ * Executes a REST request via the backend proxy
10
+ */ export async function executeRequest(request, options = {}) {
11
+ try {
12
+ // Build the request
13
+ const builtRequest = buildRequest(request, options);
14
+ // Send request through our backend proxy
15
+ const proxyResponse = await fetch('/api/proxy', {
16
+ method: 'POST',
17
+ headers: {
18
+ 'Content-Type': 'application/json'
19
+ },
20
+ body: JSON.stringify({
21
+ url: builtRequest.url,
22
+ method: builtRequest.method,
23
+ headers: builtRequest.headers,
24
+ requestBody: builtRequest.body instanceof FormData ? undefined // FormData not supported through JSON proxy
25
+ : builtRequest.body
26
+ }),
27
+ signal: options.signal
28
+ });
29
+ if (!proxyResponse.ok) {
30
+ const errorData = await proxyResponse.json().catch(()=>({}));
31
+ return {
32
+ status: proxyResponse.status,
33
+ statusText: proxyResponse.statusText,
34
+ headers: {},
35
+ body: JSON.stringify(errorData),
36
+ responseTime: 0,
37
+ size: 0,
38
+ error: errorData.error || 'Proxy request failed'
39
+ };
40
+ }
41
+ const result = await proxyResponse.json();
42
+ return {
43
+ status: result.status,
44
+ statusText: result.statusText,
45
+ headers: result.headers || {},
46
+ body: result.body,
47
+ responseTime: result.responseTime || 0,
48
+ size: result.size || 0,
49
+ error: result.error
50
+ };
51
+ } catch (error) {
52
+ if (error instanceof Error && error.name === 'AbortError') {
53
+ throw new Error('Request cancelled');
54
+ }
55
+ return {
56
+ status: 0,
57
+ statusText: 'Error',
58
+ headers: {},
59
+ body: null,
60
+ responseTime: 0,
61
+ size: 0,
62
+ error: error instanceof Error ? error.message : 'Unknown error'
63
+ };
64
+ }
65
+ }
66
+ /**
67
+ * Executes a streaming request (for SSE endpoints)
68
+ * Falls back to regular request if the endpoint isn't actually SSE
69
+ */ export async function executeStreamingRequest(request, options = {}) {
70
+ const startTime = Date.now();
71
+ try {
72
+ const builtRequest = buildRequest(request, options);
73
+ // Use streaming proxy
74
+ const response = await fetch('/api/proxy-stream', {
75
+ method: 'POST',
76
+ headers: {
77
+ 'Content-Type': 'application/json'
78
+ },
79
+ body: JSON.stringify({
80
+ url: builtRequest.url,
81
+ method: builtRequest.method,
82
+ headers: builtRequest.headers,
83
+ requestBody: builtRequest.body instanceof FormData ? undefined : builtRequest.body
84
+ }),
85
+ signal: options.signal
86
+ });
87
+ if (!response.ok) {
88
+ const errorData = await response.json().catch(()=>({}));
89
+ return {
90
+ status: response.status,
91
+ statusText: response.statusText,
92
+ headers: {},
93
+ body: JSON.stringify(errorData),
94
+ responseTime: Date.now() - startTime,
95
+ size: 0,
96
+ error: errorData.error || 'Proxy request failed'
97
+ };
98
+ }
99
+ const contentType = response.headers.get('content-type') || '';
100
+ // Check if streaming response (our proxy wraps SSE responses as SSE)
101
+ const isProxyStreaming = contentType.includes('text/event-stream');
102
+ if (isProxyStreaming && response.body) {
103
+ const reader = response.body.getReader();
104
+ const decoder = new TextDecoder();
105
+ let accumulated = '';
106
+ let metadata = null;
107
+ let totalSize = 0;
108
+ let buffer = '';
109
+ try {
110
+ while(true){
111
+ const { done, value } = await reader.read();
112
+ if (done) break;
113
+ buffer += decoder.decode(value, {
114
+ stream: true
115
+ });
116
+ // Process complete SSE events (ending with \n\n)
117
+ const events = buffer.split('\n\n');
118
+ buffer = events.pop() || ''; // Keep incomplete event in buffer
119
+ for (const eventStr of events){
120
+ const lines = eventStr.split('\n');
121
+ for (const line of lines){
122
+ if (line.startsWith('data: ')) {
123
+ try {
124
+ const event = JSON.parse(line.slice(6));
125
+ if (event.type === 'metadata') {
126
+ metadata = {
127
+ status: event.status,
128
+ statusText: event.statusText,
129
+ headers: event.headers
130
+ };
131
+ options.onStreamStart?.(metadata);
132
+ } else if (event.type === 'chunk') {
133
+ // Accumulate raw SSE data - formatting happens in response viewer
134
+ accumulated += event.data;
135
+ totalSize += event.size || 0;
136
+ options.onChunk?.(event.data, accumulated);
137
+ } else if (event.type === 'done') {
138
+ totalSize = event.totalSize || totalSize;
139
+ } else if (event.type === 'error') {
140
+ return {
141
+ status: metadata?.status || 0,
142
+ statusText: metadata?.statusText || 'Error',
143
+ headers: metadata?.headers || {},
144
+ body: accumulated,
145
+ responseTime: Date.now() - startTime,
146
+ size: totalSize,
147
+ error: event.error,
148
+ isStreaming: true
149
+ };
150
+ }
151
+ } catch {
152
+ // Ignore parse errors for partial data
153
+ }
154
+ }
155
+ }
156
+ }
157
+ }
158
+ } catch (readError) {
159
+ // Handle read errors gracefully
160
+ if (readError instanceof Error && readError.name === 'AbortError') {
161
+ throw readError;
162
+ }
163
+ }
164
+ return {
165
+ status: metadata?.status || 200,
166
+ statusText: metadata?.statusText || 'OK',
167
+ headers: metadata?.headers || {},
168
+ body: accumulated,
169
+ responseTime: Date.now() - startTime,
170
+ size: totalSize,
171
+ isStreaming: true
172
+ };
173
+ }
174
+ // Non-streaming response from streaming proxy (endpoint wasn't SSE)
175
+ const result = await response.json();
176
+ return {
177
+ status: result.status,
178
+ statusText: result.statusText,
179
+ headers: result.headers || {},
180
+ body: result.body,
181
+ responseTime: Date.now() - startTime,
182
+ size: result.size || 0,
183
+ error: result.error,
184
+ isStreaming: false
185
+ };
186
+ } catch (error) {
187
+ if (error instanceof Error && error.name === 'AbortError') {
188
+ throw new Error('Request cancelled');
189
+ }
190
+ return {
191
+ status: 0,
192
+ statusText: 'Error',
193
+ headers: {},
194
+ body: null,
195
+ responseTime: Date.now() - startTime,
196
+ size: 0,
197
+ error: error instanceof Error ? error.message : 'Unknown error'
198
+ };
199
+ }
200
+ }
201
+ /**
202
+ * Creates a cancelable request execution
203
+ */ export function createCancelableRequest(request, options = {}) {
204
+ const abortController = new AbortController();
205
+ return {
206
+ execute: ()=>executeRequest(request, {
207
+ ...options,
208
+ signal: abortController.signal
209
+ }),
210
+ cancel: ()=>abortController.abort()
211
+ };
212
+ }
213
+ /**
214
+ * Creates a cancelable streaming request execution
215
+ */ export function createCancelableStreamingRequest(request, options = {}) {
216
+ const abortController = new AbortController();
217
+ return {
218
+ execute: ()=>executeStreamingRequest(request, {
219
+ ...options,
220
+ signal: abortController.signal
221
+ }),
222
+ cancel: ()=>abortController.abort()
223
+ };
224
+ }
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Playground Types
3
+ *
4
+ * Types for the API playground (Try It Out functionality)
5
+ */ export { };
@@ -0,0 +1,23 @@
1
+ /**
2
+ * Brainfish API Documentation Types
3
+ *
4
+ * These types are inspired by Hoppscotch's data structures but adapted for Brainfish.
5
+ * Ported from Hoppscotch's HoppCollection and HoppRESTRequest types.
6
+ */ // ============================================================================
7
+ // Core Request Types
8
+ // ============================================================================
9
+ // ============================================================================
10
+ // Helper Type Guards
11
+ // ============================================================================
12
+ export function isBrainfishRESTAuthBasic(auth) {
13
+ return auth.authType === 'basic';
14
+ }
15
+ export function isBrainfishRESTAuthBearer(auth) {
16
+ return auth.authType === 'bearer';
17
+ }
18
+ export function isBrainfishRESTAuthAPIKey(auth) {
19
+ return auth.authType === 'api-key';
20
+ }
21
+ export function isBrainfishRESTAuthOAuth2(auth) {
22
+ return auth.authType === 'oauth-2';
23
+ }
@@ -0,0 +1,212 @@
1
+ /**
2
+ * Utility Functions for Brainfish API Documentation
3
+ */ /**
4
+ * Checks if a value is a valid HTTP method
5
+ */ export function isValidHTTPMethod(value) {
6
+ const validMethods = [
7
+ 'GET',
8
+ 'POST',
9
+ 'PUT',
10
+ 'DELETE',
11
+ 'PATCH',
12
+ 'HEAD',
13
+ 'OPTIONS'
14
+ ];
15
+ return validMethods.includes(value);
16
+ }
17
+ /**
18
+ * Checks if a number is numeric
19
+ */ export function isNumeric(value) {
20
+ if (typeof value === 'number') return !isNaN(value);
21
+ if (typeof value === 'string') {
22
+ const num = Number(value);
23
+ return !isNaN(num) && isFinite(num);
24
+ }
25
+ return false;
26
+ }
27
+ /**
28
+ * Gets status code reason phrase
29
+ */ export function getStatusCodeReasonPhrase(code, defaultPhrase) {
30
+ const statusCodes = {
31
+ 200: 'OK',
32
+ 201: 'Created',
33
+ 202: 'Accepted',
34
+ 204: 'No Content',
35
+ 400: 'Bad Request',
36
+ 401: 'Unauthorized',
37
+ 403: 'Forbidden',
38
+ 404: 'Not Found',
39
+ 500: 'Internal Server Error',
40
+ 502: 'Bad Gateway',
41
+ 503: 'Service Unavailable'
42
+ };
43
+ return statusCodes[code] || defaultPhrase || 'Unknown';
44
+ }
45
+ /**
46
+ * Checks if an object has a specific property
47
+ */ export function objectHasProperty(obj, propName) {
48
+ return !!obj && typeof obj === 'object' && Object.prototype.hasOwnProperty.call(obj, propName);
49
+ }
50
+ /**
51
+ * Deep clone an object
52
+ */ export function cloneDeep(obj) {
53
+ if (obj === null || typeof obj !== 'object') {
54
+ return obj;
55
+ }
56
+ if (obj instanceof Date) {
57
+ return new Date(obj.getTime());
58
+ }
59
+ if (Array.isArray(obj)) {
60
+ return obj.map((item)=>cloneDeep(item));
61
+ }
62
+ const cloned = {};
63
+ for(const key in obj){
64
+ if (Object.prototype.hasOwnProperty.call(obj, key)) {
65
+ cloned[key] = cloneDeep(obj[key]);
66
+ }
67
+ }
68
+ return cloned;
69
+ }
70
+ /**
71
+ * Strips markdown syntax from text and returns plain text
72
+ * Useful for displaying markdown content in compact UI areas like sidebars
73
+ */ export function stripMarkdown(markdown) {
74
+ if (!markdown) return '';
75
+ let text = markdown;
76
+ // Remove code blocks (```...```)
77
+ text = text.replace(/```[\s\S]*?```/g, '');
78
+ // Remove inline code (`...`)
79
+ text = text.replace(/`([^`]+)`/g, '$1');
80
+ // Remove headers (# ## ###)
81
+ text = text.replace(/^#{1,6}\s+(.+)$/gm, '$1');
82
+ // Remove bold (**text** or __text__)
83
+ text = text.replace(/\*\*([^*]+)\*\*/g, '$1');
84
+ text = text.replace(/__([^_]+)__/g, '$1');
85
+ // Remove italic (*text* or _text_)
86
+ text = text.replace(/\*([^*]+)\*/g, '$1');
87
+ text = text.replace(/_([^_]+)_/g, '$1');
88
+ // Remove links [text](url)
89
+ text = text.replace(/\[([^\]]+)\]\([^\)]+\)/g, '$1');
90
+ // Remove images ![alt](url)
91
+ text = text.replace(/!\[([^\]]*)\]\([^\)]+\)/g, '$1');
92
+ // Remove horizontal rules (--- or ***)
93
+ text = text.replace(/^[-*]{3,}$/gm, '');
94
+ // Remove list markers (-, *, +, 1.)
95
+ text = text.replace(/^[\s]*[-*+]\s+/gm, '');
96
+ text = text.replace(/^[\s]*\d+\.\s+/gm, '');
97
+ // Remove blockquotes (>)
98
+ text = text.replace(/^>\s+/gm, '');
99
+ // Clean up multiple newlines
100
+ text = text.replace(/\n{3,}/g, '\n\n');
101
+ // Trim whitespace
102
+ text = text.trim();
103
+ return text;
104
+ }
105
+ /**
106
+ * Gets a preview of text (strips markdown and truncates)
107
+ */ export function getTextPreview(text, maxLength = 150) {
108
+ if (!text) return '';
109
+ const stripped = stripMarkdown(text);
110
+ if (stripped.length <= maxLength) {
111
+ return stripped;
112
+ }
113
+ // Truncate at word boundary
114
+ const truncated = stripped.substring(0, maxLength);
115
+ const lastSpace = truncated.lastIndexOf(' ');
116
+ if (lastSpace > maxLength * 0.7) {
117
+ return truncated.substring(0, lastSpace) + '...';
118
+ }
119
+ return truncated + '...';
120
+ }
121
+ /**
122
+ * Get all headings from markdown text
123
+ * Properly handles code blocks to avoid extracting code as headings
124
+ */ function getAllHeadings(markdown) {
125
+ const headings = [];
126
+ const lines = markdown.split('\n');
127
+ let inCodeBlock = false;
128
+ for (const line of lines){
129
+ // Check for code block delimiters (``` or ~~~)
130
+ if (line.trim().startsWith('```') || line.trim().startsWith('~~~')) {
131
+ inCodeBlock = !inCodeBlock;
132
+ continue;
133
+ }
134
+ // Skip lines inside code blocks
135
+ if (inCodeBlock) {
136
+ continue;
137
+ }
138
+ // Match markdown headings (# ## ### etc.)
139
+ // Must start with # followed by space, then have actual text content
140
+ const match = line.match(/^(#{1,6})\s+(.+)$/);
141
+ if (match) {
142
+ const text = match[2].trim();
143
+ // Skip if text looks like JSON or code (starts with special chars)
144
+ if (text.startsWith('{') || text.startsWith('"') || text.startsWith('[') || text.startsWith('}') || text.startsWith(']') || text === '{' || text === '}') {
145
+ continue;
146
+ }
147
+ headings.push({
148
+ level: match[1].length,
149
+ text: text
150
+ });
151
+ }
152
+ }
153
+ return headings;
154
+ }
155
+ /**
156
+ * Get the lowest heading level in the document
157
+ */ function getLowestHeadingLevel(headings) {
158
+ if (headings.length === 0) return 1;
159
+ return Math.min(...headings.map((h)=>h.level));
160
+ }
161
+ /**
162
+ * Generate a URL-friendly slug from text
163
+ */ function generateSlug(text, idCounts) {
164
+ let baseId = text.toLowerCase().replace(/[^a-z0-9]+/g, '-').replace(/^-+|-+$/g, '');
165
+ // Ensure uniqueness by appending a counter if needed
166
+ if (idCounts[baseId] !== undefined) {
167
+ idCounts[baseId]++;
168
+ baseId = `${baseId}-${idCounts[baseId]}`;
169
+ } else {
170
+ idCounts[baseId] = 0;
171
+ }
172
+ return baseId;
173
+ }
174
+ export function extractMarkdownHeadings(markdown) {
175
+ if (!markdown?.trim()) return [];
176
+ const allHeadings = getAllHeadings(markdown);
177
+ const lowestLevel = getLowestHeadingLevel(allHeadings);
178
+ const idCounts = {};
179
+ const entries = [];
180
+ let currentParent = null;
181
+ // Add "Introduction" if description doesn't start with a heading
182
+ if (!markdown.trim().startsWith('#')) {
183
+ entries.push({
184
+ level: lowestLevel,
185
+ text: 'Introduction',
186
+ id: generateSlug('Introduction', idCounts),
187
+ children: []
188
+ });
189
+ }
190
+ // Process headings - only include lowest level and one level deeper
191
+ for (const heading of allHeadings){
192
+ // Skip headings that are not at the lowest two levels
193
+ if (heading.level !== lowestLevel && heading.level !== lowestLevel + 1) {
194
+ continue;
195
+ }
196
+ const entry = {
197
+ level: heading.level,
198
+ text: heading.text,
199
+ id: generateSlug(heading.text, idCounts)
200
+ };
201
+ if (heading.level === lowestLevel) {
202
+ // This is a top-level heading
203
+ entry.children = [];
204
+ entries.push(entry);
205
+ currentParent = entry;
206
+ } else if (currentParent && currentParent.children) {
207
+ // This is a child heading
208
+ currentParent.children.push(entry);
209
+ }
210
+ }
211
+ return entries;
212
+ }
@@ -0,0 +1,157 @@
1
+ /**
2
+ * Universal caching utility that works in both local and production environments
3
+ *
4
+ * - Local development: Uses node-cache (in-memory)
5
+ * - Production (Vercel): Uses Vercel KV
6
+ *
7
+ * Environment-based cache selection:
8
+ * 1. Explicit override: CACHE_TYPE=local or CACHE_TYPE=kv
9
+ * 2. Auto-detect: KV_REST_API_URL or KV_URL present → Uses Vercel KV
10
+ * 3. Fallback: Uses local node-cache
11
+ *
12
+ * Example usage:
13
+ * ```typescript
14
+ * import { CacheUtils } from '@/utils/cache';
15
+ *
16
+ * // Get from cache
17
+ * const cached = await CacheUtils.get<MyType>('my-key', { logHit: true });
18
+ *
19
+ * // Set with TTL
20
+ * await CacheUtils.set('my-key', data, 3600, { log: true }); // 1 hour TTL
21
+ *
22
+ * // Delete
23
+ * await CacheUtils.del('my-key', { log: true });
24
+ * ```
25
+ */ import NodeCache from 'node-cache';
26
+ import { kv } from '@vercel/kv';
27
+ /**
28
+ * Cache environment configuration
29
+ */ const CACHE_CONFIG = {
30
+ // Environment variables that indicate KV should be used
31
+ KV_INDICATORS: [
32
+ 'KV_REST_API_URL',
33
+ 'KV_URL'
34
+ ],
35
+ // Allow explicit override via CACHE_TYPE env var
36
+ CACHE_TYPE: process.env.CACHE_TYPE
37
+ };
38
+ /**
39
+ * Determine cache type based on environment variables
40
+ */ function getCacheType() {
41
+ // Explicit override takes precedence
42
+ if (CACHE_CONFIG.CACHE_TYPE === 'local') {
43
+ return 'local';
44
+ }
45
+ if (CACHE_CONFIG.CACHE_TYPE === 'kv') {
46
+ return 'kv';
47
+ }
48
+ // Auto-detect based on KV environment variables
49
+ const hasKVConfig = CACHE_CONFIG.KV_INDICATORS.some((envVar)=>!!process.env[envVar]);
50
+ return hasKVConfig ? 'kv' : 'local';
51
+ }
52
+ // Determine cache type at module load time
53
+ const cacheType = getCacheType();
54
+ // Initialize cache backends based on environment
55
+ let localCache = null;
56
+ if (cacheType === 'kv') {
57
+ console.log('[Cache] Using Vercel KV cache');
58
+ } else {
59
+ console.log('[Cache] Using local node-cache');
60
+ localCache = new NodeCache({
61
+ stdTTL: 60 * 60 * 24,
62
+ checkperiod: 60 * 10,
63
+ useClones: false
64
+ });
65
+ }
66
+ /**
67
+ * Universal cache utility functions
68
+ */ export class CacheUtils {
69
+ /**
70
+ * Get a value from cache with optional logging
71
+ */ static async get(key, options) {
72
+ try {
73
+ let value;
74
+ if (cacheType === 'kv') {
75
+ value = await kv.get(key);
76
+ } else {
77
+ value = localCache?.get(key);
78
+ }
79
+ // Fix: Only return null for missing keys (undefined), not for falsy values
80
+ const result = value !== undefined ? value : null;
81
+ if (options?.logHit) {
82
+ if (result !== null) {
83
+ console.log(`[Cache] HIT: ${key}`);
84
+ } else {
85
+ console.log(`[Cache] MISS: ${key}`);
86
+ }
87
+ }
88
+ return result;
89
+ } catch (error) {
90
+ console.warn(`[Cache] Get error for key "${key}":`, error);
91
+ return null;
92
+ }
93
+ }
94
+ /**
95
+ * Set a value in cache with TTL and logging
96
+ */ static async set(key, value, ttlSeconds, options) {
97
+ try {
98
+ if (cacheType === 'kv') {
99
+ if (ttlSeconds) {
100
+ await kv.set(key, value, {
101
+ ex: ttlSeconds
102
+ });
103
+ } else {
104
+ await kv.set(key, value);
105
+ }
106
+ } else {
107
+ const ttl = ttlSeconds || 60 * 60 * 24; // Default to 24 hours
108
+ localCache?.set(key, value, ttl);
109
+ }
110
+ if (options?.log) {
111
+ const ttlInfo = ttlSeconds ? ` (TTL: ${ttlSeconds}s)` : '';
112
+ console.log(`[Cache] SET: ${key}${ttlInfo}`);
113
+ }
114
+ } catch (error) {
115
+ console.warn(`[Cache] Set error for key "${key}":`, error);
116
+ }
117
+ }
118
+ /**
119
+ * Delete a key from cache
120
+ */ static async del(key, options) {
121
+ try {
122
+ if (cacheType === 'kv') {
123
+ await kv.del(key);
124
+ } else {
125
+ localCache?.del(key);
126
+ }
127
+ if (options?.log) {
128
+ console.log(`[Cache] DEL: ${key}`);
129
+ }
130
+ } catch (error) {
131
+ console.warn(`[Cache] Delete error for key "${key}":`, error);
132
+ }
133
+ }
134
+ /**
135
+ * Clear all cache entries (use with caution!)
136
+ */ static async clear() {
137
+ try {
138
+ if (cacheType === 'kv') {
139
+ await kv.flushdb();
140
+ } else {
141
+ localCache?.flushAll();
142
+ }
143
+ console.log('[Cache] Cleared all entries');
144
+ } catch (error) {
145
+ console.warn('[Cache] Clear error:', error);
146
+ }
147
+ }
148
+ /**
149
+ * Get current cache type for debugging/monitoring
150
+ */ static getCacheType() {
151
+ return cacheType;
152
+ }
153
+ }
154
+ // Legacy exports for backward compatibility
155
+ export const cache = CacheUtils;
156
+ export { cacheType };
157
+ export default CacheUtils;