@brainfish-ai/devdoc 0.1.41 → 0.1.43
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/ai-agents/.claude/skills/bootstrap-docs/SKILL.md +710 -79
- package/ai-agents/.claude/skills/check-docs/SKILL.md +83 -8
- package/ai-agents/.claude/skills/create-doc/SKILL.md +267 -55
- package/ai-agents/.claude/skills/update-doc/SKILL.md +162 -63
- package/ai-agents/.cursor/rules/devdoc-bootstrap.mdc +145 -15
- package/ai-agents/.cursor/rules/devdoc-create.mdc +108 -57
- package/ai-agents/.cursor/rules/devdoc-update.mdc +93 -70
- package/ai-agents/.cursor/rules/devdoc.mdc +21 -0
- package/ai-agents/schemas/docs.schema.json +332 -0
- package/ai-agents/schemas/theme.schema.json +243 -0
- package/dist/cli/commands/create.js +4 -9
- package/dist/cli/commands/deploy.js +50 -25
- package/dist/cli/commands/dev.js +19 -10
- package/package.json +3 -2
- package/renderer/app/api/assets/[...path]/route.js +108 -0
- package/renderer/app/api/assets/route.js +114 -0
- package/renderer/app/api/assets/upload/route.js +163 -0
- package/renderer/app/api/auth-schemes/route.js +58 -0
- package/renderer/app/api/chat/route.js +759 -0
- package/renderer/app/api/codegen/route.js +52 -0
- package/renderer/app/api/collections/route.js +675 -0
- package/renderer/app/api/debug/route.js +47 -0
- package/renderer/app/api/deploy/route.js +199 -0
- package/renderer/app/api/device/route.js +36 -0
- package/renderer/app/api/docs/route.js +205 -0
- package/renderer/app/api/domains/add/route.js +121 -0
- package/renderer/app/api/domains/lookup/route.js +43 -0
- package/renderer/app/api/domains/remove/route.js +89 -0
- package/renderer/app/api/domains/status/route.js +140 -0
- package/renderer/app/api/domains/verify/route.js +168 -0
- package/renderer/app/api/keys/regenerate/route.js +71 -0
- package/renderer/app/api/local-assets/[...path]/route.js +108 -0
- package/renderer/app/api/openapi-spec/route.js +73 -0
- package/renderer/app/api/projects/[slug]/route.js +129 -0
- package/renderer/app/api/projects/[slug]/stats/route.js +80 -0
- package/renderer/app/api/projects/register/route.js +176 -0
- package/renderer/app/api/proxy/route.js +139 -0
- package/renderer/app/api/proxy-stream/route.js +156 -0
- package/renderer/app/api/redirects/route.js +35 -0
- package/renderer/app/api/schema/route.js +85 -0
- package/renderer/app/api/subdomains/check/route.js +158 -0
- package/renderer/app/api/suggestions/route.js +175 -0
- package/renderer/app/layout.js +47 -0
- package/renderer/app/llms-full.txt/route.js +257 -0
- package/renderer/app/llms.txt/route.js +219 -0
- package/renderer/app/page.js +12 -0
- package/renderer/app/robots.txt/route.js +66 -0
- package/renderer/app/sitemap.xml/route.js +145 -0
- package/renderer/components/docs/index.js +8 -0
- package/renderer/components/docs/mdx/accordion.js +113 -0
- package/renderer/components/docs/mdx/badge.js +72 -0
- package/renderer/components/docs/mdx/callouts.js +137 -0
- package/renderer/components/docs/mdx/cards.js +175 -0
- package/renderer/components/docs/mdx/changelog.js +100 -0
- package/renderer/components/docs/mdx/code-block.js +147 -0
- package/renderer/components/docs/mdx/code-group.js +287 -0
- package/renderer/components/docs/mdx/file-embeds.js +82 -0
- package/renderer/components/docs/mdx/frame.js +59 -0
- package/renderer/components/docs/mdx/highlight.js +90 -0
- package/renderer/components/docs/mdx/iframe.js +69 -0
- package/renderer/components/docs/mdx/image.js +135 -0
- package/renderer/components/docs/mdx/index.js +134 -0
- package/renderer/components/docs/mdx/landing.js +315 -0
- package/renderer/components/docs/mdx/mermaid.js +212 -0
- package/renderer/components/docs/mdx/param-field.js +112 -0
- package/renderer/components/docs/mdx/steps.js +74 -0
- package/renderer/components/docs/mdx/tabs.js +50 -0
- package/renderer/components/docs/mdx-renderer.js +77 -0
- package/renderer/components/docs/navigation/breadcrumbs.js +64 -0
- package/renderer/components/docs/navigation/index.js +6 -0
- package/renderer/components/docs/navigation/page-nav.js +57 -0
- package/renderer/components/docs/navigation/sidebar.js +375 -0
- package/renderer/components/docs/navigation/toc.js +89 -0
- package/renderer/components/docs/notice.js +77 -0
- package/renderer/components/docs-header.js +202 -0
- package/renderer/components/docs-viewer/agent/agent-chat.js +1930 -0
- package/renderer/components/docs-viewer/agent/cards/debug-context-card.js +107 -0
- package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.js +57 -0
- package/renderer/components/docs-viewer/agent/cards/index.js +45 -0
- package/renderer/components/docs-viewer/agent/cards/response-options-card.js +154 -0
- package/renderer/components/docs-viewer/agent/cards/types.js +22 -0
- package/renderer/components/docs-viewer/agent/chat-message.js +2 -0
- package/renderer/components/docs-viewer/agent/index.js +4 -0
- package/renderer/components/docs-viewer/agent/messages/assistant-message.js +108 -0
- package/renderer/components/docs-viewer/agent/messages/chat-message.js +34 -0
- package/renderer/components/docs-viewer/agent/messages/index.js +6 -0
- package/renderer/components/docs-viewer/agent/messages/tool-call-display.js +1065 -0
- package/renderer/components/docs-viewer/agent/messages/types.js +2 -0
- package/renderer/components/docs-viewer/agent/messages/typing-indicator.js +26 -0
- package/renderer/components/docs-viewer/agent/messages/user-message.js +37 -0
- package/renderer/components/docs-viewer/code-editor/{index.tsx → index.js} +1 -1
- package/renderer/components/docs-viewer/code-editor/notes-mode.js +1338 -0
- package/renderer/components/docs-viewer/content/changelog-page.js +297 -0
- package/renderer/components/docs-viewer/content/doc-page.js +264 -0
- package/renderer/components/docs-viewer/content/documentation-viewer.js +14 -0
- package/renderer/components/docs-viewer/content/index.js +29 -0
- package/renderer/components/docs-viewer/content/not-found-page.js +300 -0
- package/renderer/components/docs-viewer/content/request-details.js +528 -0
- package/renderer/components/docs-viewer/content/sections/auth.js +108 -0
- package/renderer/components/docs-viewer/content/sections/body.js +80 -0
- package/renderer/components/docs-viewer/content/sections/headers.js +64 -0
- package/renderer/components/docs-viewer/content/sections/overview.js +56 -0
- package/renderer/components/docs-viewer/content/sections/parameters.js +64 -0
- package/renderer/components/docs-viewer/content/sections/responses.js +91 -0
- package/renderer/components/docs-viewer/global-auth-modal.js +427 -0
- package/renderer/components/docs-viewer/index.js +1552 -0
- package/renderer/components/docs-viewer/playground/auth-editor.js +418 -0
- package/renderer/components/docs-viewer/playground/body-editor.js +240 -0
- package/renderer/components/docs-viewer/playground/code-editor.js +135 -0
- package/renderer/components/docs-viewer/playground/code-snippet.js +393 -0
- package/renderer/components/docs-viewer/playground/graphql-playground.js +734 -0
- package/renderer/components/docs-viewer/playground/index.js +682 -0
- package/renderer/components/docs-viewer/playground/key-value-editor.js +317 -0
- package/renderer/components/docs-viewer/playground/method-selector.js +65 -0
- package/renderer/components/docs-viewer/playground/request-builder.js +181 -0
- package/renderer/components/docs-viewer/playground/request-tabs.js +240 -0
- package/renderer/components/docs-viewer/playground/response-cards/idle-card.js +42 -0
- package/renderer/components/docs-viewer/playground/response-cards/index.js +72 -0
- package/renderer/components/docs-viewer/playground/response-cards/loading-card.js +24 -0
- package/renderer/components/docs-viewer/playground/response-cards/network-error-card.js +28 -0
- package/renderer/components/docs-viewer/playground/response-cards/response-body-card.js +308 -0
- package/renderer/components/docs-viewer/playground/response-cards/types.js +9 -0
- package/renderer/components/docs-viewer/playground/response-viewer.js +18 -0
- package/renderer/components/docs-viewer/search/index.js +2 -0
- package/renderer/components/docs-viewer/search/search-dialog.js +367 -0
- package/renderer/components/docs-viewer/search/use-search.js +89 -0
- package/renderer/components/docs-viewer/shared/markdown-renderer.js +423 -0
- package/renderer/components/docs-viewer/shared/method-badge.js +23 -0
- package/renderer/components/docs-viewer/shared/schema-viewer.js +321 -0
- package/renderer/components/docs-viewer/sidebar/collection-tree.js +222 -0
- package/renderer/components/docs-viewer/sidebar/endpoint-options.js +512 -0
- package/renderer/components/docs-viewer/sidebar/index.js +196 -0
- package/renderer/components/docs-viewer/sidebar/right-sidebar.js +163 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-group.js +87 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-item.js +172 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-section.js +31 -0
- package/renderer/components/theme-provider.js +10 -0
- package/renderer/components/theme-toggle.js +106 -0
- package/renderer/components/ui/badge.js +29 -0
- package/renderer/components/ui/button.js +40 -0
- package/renderer/components/ui/dialog.js +50 -0
- package/renderer/components/ui/dropdown-menu.js +143 -0
- package/renderer/components/ui/input.js +12 -0
- package/renderer/components/ui/label.js +13 -0
- package/renderer/components/ui/navigation-menu.js +83 -0
- package/renderer/components/ui/select.js +116 -0
- package/renderer/components/ui/spinner.js +92 -0
- package/renderer/components/ui/tabs.js +34 -0
- package/renderer/components/ui/tooltip.js +43 -0
- package/renderer/hooks/use-code-copy.js +76 -0
- package/renderer/hooks/use-openapi-title.js +33 -0
- package/renderer/lib/api-docs/agent/index.js +4 -0
- package/renderer/lib/api-docs/agent/indexer.js +254 -0
- package/renderer/lib/api-docs/agent/spec-summary.js +227 -0
- package/renderer/lib/api-docs/agent/types.js +5 -0
- package/renderer/lib/api-docs/auth/auth-context.js +157 -0
- package/renderer/lib/api-docs/auth/auth-storage.js +66 -0
- package/renderer/lib/api-docs/auth/crypto.js +64 -0
- package/renderer/lib/api-docs/auth/index.js +3 -0
- package/renderer/lib/api-docs/code-editor/db.js +145 -0
- package/renderer/lib/api-docs/code-editor/hooks.js +254 -0
- package/renderer/lib/api-docs/code-editor/{index.ts → index.js} +3 -4
- package/renderer/lib/api-docs/code-editor/mode-context.js +154 -0
- package/renderer/lib/api-docs/code-editor/types.js +53 -0
- package/renderer/lib/api-docs/codegen/definitions.js +258 -0
- package/renderer/lib/api-docs/codegen/har.js +171 -0
- package/renderer/lib/api-docs/codegen/index.js +118 -0
- package/renderer/lib/api-docs/factories.js +136 -0
- package/renderer/lib/api-docs/{index.ts → index.js} +5 -10
- package/renderer/lib/api-docs/mobile-context.js +79 -0
- package/renderer/lib/api-docs/navigation-context.js +62 -0
- package/renderer/lib/api-docs/parsers/graphql/index.js +50 -0
- package/renderer/lib/api-docs/parsers/graphql/parser.js +350 -0
- package/renderer/lib/api-docs/parsers/graphql/transformer.js +215 -0
- package/renderer/lib/api-docs/parsers/graphql/types.js +46 -0
- package/renderer/lib/api-docs/parsers/openapi/dereferencer.js +43 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/auth.js +486 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/body.js +295 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/index.js +132 -0
- package/renderer/lib/api-docs/parsers/openapi/index.js +127 -0
- package/renderer/lib/api-docs/parsers/openapi/transformer.js +192 -0
- package/renderer/lib/api-docs/parsers/openapi/validator.js +24 -0
- package/renderer/lib/api-docs/playground/context.js +65 -0
- package/renderer/lib/api-docs/playground/navigation-context.js +74 -0
- package/renderer/lib/api-docs/playground/request-builder.js +163 -0
- package/renderer/lib/api-docs/playground/request-runner.js +224 -0
- package/renderer/lib/api-docs/playground/types.js +5 -0
- package/renderer/lib/api-docs/types.js +23 -0
- package/renderer/lib/api-docs/utils.js +212 -0
- package/renderer/lib/cache.js +157 -0
- package/renderer/lib/docs/config/domain-schema.js +161 -0
- package/renderer/lib/docs/config/index.js +5 -0
- package/renderer/lib/docs/config/loader.js +113 -0
- package/renderer/lib/docs/config/schema.js +269 -0
- package/renderer/lib/docs/index.js +8 -0
- package/renderer/lib/docs/mdx/compiler.js +128 -0
- package/renderer/lib/docs/mdx/frontmatter.js +73 -0
- package/renderer/lib/docs/mdx/index.js +8 -0
- package/renderer/lib/docs/navigation/generator.js +269 -0
- package/renderer/lib/docs/navigation/index.js +4 -0
- package/renderer/lib/docs/navigation/types.js +9 -0
- package/renderer/lib/docs-navigation-context.js +40 -0
- package/renderer/lib/multi-tenant/context.js +80 -0
- package/renderer/lib/storage/blob.js +767 -0
- package/renderer/lib/utils/icons.js +30 -0
- package/renderer/lib/utils.js +5 -0
- package/renderer/next.config.js +62 -0
- package/renderer/tsconfig.json +23 -5
- package/renderer/app/api/assets/[...path]/route.ts +0 -123
- package/renderer/app/api/assets/route.ts +0 -124
- package/renderer/app/api/assets/upload/route.ts +0 -177
- package/renderer/app/api/auth-schemes/route.ts +0 -77
- package/renderer/app/api/chat/route.ts +0 -858
- package/renderer/app/api/codegen/route.ts +0 -72
- package/renderer/app/api/collections/route.ts +0 -1002
- package/renderer/app/api/debug/route.ts +0 -53
- package/renderer/app/api/deploy/route.ts +0 -234
- package/renderer/app/api/device/route.ts +0 -42
- package/renderer/app/api/docs/route.ts +0 -201
- package/renderer/app/api/domains/add/route.ts +0 -132
- package/renderer/app/api/domains/lookup/route.ts +0 -43
- package/renderer/app/api/domains/remove/route.ts +0 -100
- package/renderer/app/api/domains/status/route.ts +0 -158
- package/renderer/app/api/domains/verify/route.ts +0 -181
- package/renderer/app/api/keys/regenerate/route.ts +0 -80
- package/renderer/app/api/local-assets/[...path]/route.ts +0 -122
- package/renderer/app/api/openapi-spec/route.ts +0 -151
- package/renderer/app/api/projects/[slug]/route.ts +0 -153
- package/renderer/app/api/projects/[slug]/stats/route.ts +0 -96
- package/renderer/app/api/projects/register/route.ts +0 -152
- package/renderer/app/api/proxy/route.ts +0 -149
- package/renderer/app/api/proxy-stream/route.ts +0 -168
- package/renderer/app/api/redirects/route.ts +0 -47
- package/renderer/app/api/schema/route.ts +0 -73
- package/renderer/app/api/subdomains/check/route.ts +0 -172
- package/renderer/app/api/suggestions/route.ts +0 -144
- package/renderer/app/layout.tsx +0 -54
- package/renderer/app/llms-full.txt/route.ts +0 -346
- package/renderer/app/llms.txt/route.ts +0 -279
- package/renderer/app/page.tsx +0 -14
- package/renderer/app/robots.txt/route.ts +0 -84
- package/renderer/app/sitemap.xml/route.ts +0 -199
- package/renderer/components/docs/index.ts +0 -12
- package/renderer/components/docs/mdx/accordion.tsx +0 -169
- package/renderer/components/docs/mdx/badge.tsx +0 -132
- package/renderer/components/docs/mdx/callouts.tsx +0 -154
- package/renderer/components/docs/mdx/cards.tsx +0 -241
- package/renderer/components/docs/mdx/changelog.tsx +0 -120
- package/renderer/components/docs/mdx/code-block.tsx +0 -186
- package/renderer/components/docs/mdx/code-group.tsx +0 -421
- package/renderer/components/docs/mdx/file-embeds.tsx +0 -105
- package/renderer/components/docs/mdx/frame.tsx +0 -112
- package/renderer/components/docs/mdx/highlight.tsx +0 -151
- package/renderer/components/docs/mdx/iframe.tsx +0 -134
- package/renderer/components/docs/mdx/image.tsx +0 -235
- package/renderer/components/docs/mdx/index.ts +0 -237
- package/renderer/components/docs/mdx/landing.tsx +0 -684
- package/renderer/components/docs/mdx/mermaid.tsx +0 -240
- package/renderer/components/docs/mdx/param-field.tsx +0 -200
- package/renderer/components/docs/mdx/steps.tsx +0 -113
- package/renderer/components/docs/mdx/tabs.tsx +0 -86
- package/renderer/components/docs/mdx-renderer.tsx +0 -100
- package/renderer/components/docs/navigation/breadcrumbs.tsx +0 -76
- package/renderer/components/docs/navigation/index.ts +0 -8
- package/renderer/components/docs/navigation/page-nav.tsx +0 -64
- package/renderer/components/docs/navigation/sidebar.tsx +0 -515
- package/renderer/components/docs/navigation/toc.tsx +0 -113
- package/renderer/components/docs/notice.tsx +0 -105
- package/renderer/components/docs-header.tsx +0 -278
- package/renderer/components/docs-viewer/agent/agent-chat.tsx +0 -2076
- package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +0 -90
- package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +0 -49
- package/renderer/components/docs-viewer/agent/cards/index.tsx +0 -50
- package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +0 -212
- package/renderer/components/docs-viewer/agent/cards/types.ts +0 -84
- package/renderer/components/docs-viewer/agent/chat-message.tsx +0 -17
- package/renderer/components/docs-viewer/agent/index.tsx +0 -6
- package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +0 -119
- package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +0 -46
- package/renderer/components/docs-viewer/agent/messages/index.ts +0 -17
- package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +0 -721
- package/renderer/components/docs-viewer/agent/messages/types.ts +0 -61
- package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +0 -24
- package/renderer/components/docs-viewer/agent/messages/user-message.tsx +0 -51
- package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +0 -1283
- package/renderer/components/docs-viewer/content/changelog-page.tsx +0 -331
- package/renderer/components/docs-viewer/content/doc-page.tsx +0 -367
- package/renderer/components/docs-viewer/content/documentation-viewer.tsx +0 -17
- package/renderer/components/docs-viewer/content/index.tsx +0 -29
- package/renderer/components/docs-viewer/content/not-found-page.tsx +0 -330
- package/renderer/components/docs-viewer/content/request-details.tsx +0 -330
- package/renderer/components/docs-viewer/content/sections/auth.tsx +0 -69
- package/renderer/components/docs-viewer/content/sections/body.tsx +0 -66
- package/renderer/components/docs-viewer/content/sections/headers.tsx +0 -43
- package/renderer/components/docs-viewer/content/sections/overview.tsx +0 -40
- package/renderer/components/docs-viewer/content/sections/parameters.tsx +0 -43
- package/renderer/components/docs-viewer/content/sections/responses.tsx +0 -87
- package/renderer/components/docs-viewer/global-auth-modal.tsx +0 -352
- package/renderer/components/docs-viewer/index.tsx +0 -1662
- package/renderer/components/docs-viewer/playground/auth-editor.tsx +0 -280
- package/renderer/components/docs-viewer/playground/body-editor.tsx +0 -221
- package/renderer/components/docs-viewer/playground/code-editor.tsx +0 -224
- package/renderer/components/docs-viewer/playground/code-snippet.tsx +0 -387
- package/renderer/components/docs-viewer/playground/graphql-playground.tsx +0 -745
- package/renderer/components/docs-viewer/playground/index.tsx +0 -671
- package/renderer/components/docs-viewer/playground/key-value-editor.tsx +0 -261
- package/renderer/components/docs-viewer/playground/method-selector.tsx +0 -60
- package/renderer/components/docs-viewer/playground/request-builder.tsx +0 -179
- package/renderer/components/docs-viewer/playground/request-tabs.tsx +0 -237
- package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +0 -21
- package/renderer/components/docs-viewer/playground/response-cards/index.tsx +0 -93
- package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +0 -16
- package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +0 -23
- package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +0 -268
- package/renderer/components/docs-viewer/playground/response-cards/types.ts +0 -82
- package/renderer/components/docs-viewer/playground/response-viewer.tsx +0 -43
- package/renderer/components/docs-viewer/search/index.ts +0 -2
- package/renderer/components/docs-viewer/search/search-dialog.tsx +0 -331
- package/renderer/components/docs-viewer/search/use-search.ts +0 -117
- package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +0 -431
- package/renderer/components/docs-viewer/shared/method-badge.tsx +0 -41
- package/renderer/components/docs-viewer/shared/schema-viewer.tsx +0 -349
- package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +0 -259
- package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +0 -316
- package/renderer/components/docs-viewer/sidebar/index.tsx +0 -282
- package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +0 -202
- package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +0 -118
- package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +0 -212
- package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +0 -38
- package/renderer/components/theme-provider.tsx +0 -11
- package/renderer/components/theme-toggle.tsx +0 -76
- package/renderer/components/ui/badge.tsx +0 -46
- package/renderer/components/ui/button.tsx +0 -59
- package/renderer/components/ui/dialog.tsx +0 -118
- package/renderer/components/ui/dropdown-menu.tsx +0 -257
- package/renderer/components/ui/input.tsx +0 -21
- package/renderer/components/ui/label.tsx +0 -24
- package/renderer/components/ui/navigation-menu.tsx +0 -168
- package/renderer/components/ui/select.tsx +0 -190
- package/renderer/components/ui/spinner.tsx +0 -114
- package/renderer/components/ui/tabs.tsx +0 -66
- package/renderer/components/ui/tooltip.tsx +0 -61
- package/renderer/hooks/use-code-copy.ts +0 -88
- package/renderer/hooks/use-openapi-title.ts +0 -44
- package/renderer/lib/api-docs/agent/index.ts +0 -6
- package/renderer/lib/api-docs/agent/indexer.ts +0 -323
- package/renderer/lib/api-docs/agent/spec-summary.ts +0 -335
- package/renderer/lib/api-docs/agent/types.ts +0 -116
- package/renderer/lib/api-docs/auth/auth-context.tsx +0 -225
- package/renderer/lib/api-docs/auth/auth-storage.ts +0 -87
- package/renderer/lib/api-docs/auth/crypto.ts +0 -89
- package/renderer/lib/api-docs/auth/index.ts +0 -4
- package/renderer/lib/api-docs/code-editor/db.ts +0 -164
- package/renderer/lib/api-docs/code-editor/hooks.ts +0 -266
- package/renderer/lib/api-docs/code-editor/mode-context.tsx +0 -207
- package/renderer/lib/api-docs/code-editor/types.ts +0 -105
- package/renderer/lib/api-docs/codegen/definitions.ts +0 -297
- package/renderer/lib/api-docs/codegen/har.ts +0 -251
- package/renderer/lib/api-docs/codegen/index.ts +0 -159
- package/renderer/lib/api-docs/factories.ts +0 -170
- package/renderer/lib/api-docs/mobile-context.tsx +0 -112
- package/renderer/lib/api-docs/navigation-context.tsx +0 -88
- package/renderer/lib/api-docs/parsers/graphql/README.md +0 -129
- package/renderer/lib/api-docs/parsers/graphql/index.ts +0 -91
- package/renderer/lib/api-docs/parsers/graphql/parser.ts +0 -491
- package/renderer/lib/api-docs/parsers/graphql/transformer.ts +0 -246
- package/renderer/lib/api-docs/parsers/graphql/types.ts +0 -283
- package/renderer/lib/api-docs/parsers/openapi/README.md +0 -32
- package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +0 -60
- package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +0 -574
- package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +0 -403
- package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +0 -232
- package/renderer/lib/api-docs/parsers/openapi/index.ts +0 -171
- package/renderer/lib/api-docs/parsers/openapi/transformer.ts +0 -278
- package/renderer/lib/api-docs/parsers/openapi/validator.ts +0 -31
- package/renderer/lib/api-docs/playground/context.tsx +0 -107
- package/renderer/lib/api-docs/playground/navigation-context.tsx +0 -124
- package/renderer/lib/api-docs/playground/request-builder.ts +0 -223
- package/renderer/lib/api-docs/playground/request-runner.ts +0 -282
- package/renderer/lib/api-docs/playground/types.ts +0 -35
- package/renderer/lib/api-docs/types.ts +0 -269
- package/renderer/lib/api-docs/utils.ts +0 -311
- package/renderer/lib/cache.ts +0 -193
- package/renderer/lib/docs/config/domain-schema.ts +0 -260
- package/renderer/lib/docs/config/index.ts +0 -43
- package/renderer/lib/docs/config/loader.ts +0 -142
- package/renderer/lib/docs/config/schema.ts +0 -308
- package/renderer/lib/docs/index.ts +0 -12
- package/renderer/lib/docs/mdx/compiler.ts +0 -176
- package/renderer/lib/docs/mdx/frontmatter.ts +0 -91
- package/renderer/lib/docs/mdx/index.ts +0 -26
- package/renderer/lib/docs/navigation/generator.ts +0 -348
- package/renderer/lib/docs/navigation/index.ts +0 -12
- package/renderer/lib/docs/navigation/types.ts +0 -123
- package/renderer/lib/docs-navigation-context.tsx +0 -80
- package/renderer/lib/multi-tenant/context.ts +0 -105
- package/renderer/lib/storage/blob.ts +0 -1083
- package/renderer/lib/utils/icons.ts +0 -48
- package/renderer/lib/utils.ts +0 -6
- package/renderer/next.config.ts +0 -76
|
@@ -0,0 +1,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,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 
|
|
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;
|