@brainfish-ai/devdoc 0.1.21

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (268) hide show
  1. package/LICENSE +33 -0
  2. package/README.md +415 -0
  3. package/bin/devdoc.js +13 -0
  4. package/dist/cli/commands/build.d.ts +5 -0
  5. package/dist/cli/commands/build.js +87 -0
  6. package/dist/cli/commands/check.d.ts +1 -0
  7. package/dist/cli/commands/check.js +143 -0
  8. package/dist/cli/commands/create.d.ts +24 -0
  9. package/dist/cli/commands/create.js +387 -0
  10. package/dist/cli/commands/deploy.d.ts +9 -0
  11. package/dist/cli/commands/deploy.js +433 -0
  12. package/dist/cli/commands/dev.d.ts +6 -0
  13. package/dist/cli/commands/dev.js +139 -0
  14. package/dist/cli/commands/init.d.ts +11 -0
  15. package/dist/cli/commands/init.js +238 -0
  16. package/dist/cli/commands/keys.d.ts +12 -0
  17. package/dist/cli/commands/keys.js +165 -0
  18. package/dist/cli/commands/start.d.ts +5 -0
  19. package/dist/cli/commands/start.js +56 -0
  20. package/dist/cli/commands/upload.d.ts +13 -0
  21. package/dist/cli/commands/upload.js +238 -0
  22. package/dist/cli/commands/whoami.d.ts +8 -0
  23. package/dist/cli/commands/whoami.js +91 -0
  24. package/dist/cli/index.d.ts +1 -0
  25. package/dist/cli/index.js +106 -0
  26. package/dist/config/index.d.ts +80 -0
  27. package/dist/config/index.js +133 -0
  28. package/dist/constants.d.ts +9 -0
  29. package/dist/constants.js +13 -0
  30. package/dist/index.d.ts +7 -0
  31. package/dist/index.js +12 -0
  32. package/dist/utils/logger.d.ts +16 -0
  33. package/dist/utils/logger.js +61 -0
  34. package/dist/utils/paths.d.ts +16 -0
  35. package/dist/utils/paths.js +50 -0
  36. package/package.json +51 -0
  37. package/renderer/app/api/assets/[...path]/route.ts +123 -0
  38. package/renderer/app/api/assets/route.ts +124 -0
  39. package/renderer/app/api/assets/upload/route.ts +177 -0
  40. package/renderer/app/api/auth-schemes/route.ts +77 -0
  41. package/renderer/app/api/chat/route.ts +858 -0
  42. package/renderer/app/api/codegen/route.ts +72 -0
  43. package/renderer/app/api/collections/route.ts +1016 -0
  44. package/renderer/app/api/debug/route.ts +53 -0
  45. package/renderer/app/api/deploy/route.ts +234 -0
  46. package/renderer/app/api/device/route.ts +42 -0
  47. package/renderer/app/api/docs/route.ts +187 -0
  48. package/renderer/app/api/keys/regenerate/route.ts +80 -0
  49. package/renderer/app/api/openapi-spec/route.ts +151 -0
  50. package/renderer/app/api/projects/[slug]/route.ts +153 -0
  51. package/renderer/app/api/projects/[slug]/stats/route.ts +96 -0
  52. package/renderer/app/api/projects/register/route.ts +152 -0
  53. package/renderer/app/api/proxy/route.ts +149 -0
  54. package/renderer/app/api/proxy-stream/route.ts +168 -0
  55. package/renderer/app/api/redirects/route.ts +47 -0
  56. package/renderer/app/api/schema/route.ts +65 -0
  57. package/renderer/app/api/subdomains/check/route.ts +172 -0
  58. package/renderer/app/api/suggestions/route.ts +144 -0
  59. package/renderer/app/favicon.ico +0 -0
  60. package/renderer/app/globals.css +1103 -0
  61. package/renderer/app/layout.tsx +47 -0
  62. package/renderer/app/llms-full.txt/route.ts +346 -0
  63. package/renderer/app/llms.txt/route.ts +279 -0
  64. package/renderer/app/page.tsx +14 -0
  65. package/renderer/app/robots.txt/route.ts +84 -0
  66. package/renderer/app/sitemap.xml/route.ts +199 -0
  67. package/renderer/components/docs/index.ts +12 -0
  68. package/renderer/components/docs/mdx/accordion.tsx +169 -0
  69. package/renderer/components/docs/mdx/badge.tsx +132 -0
  70. package/renderer/components/docs/mdx/callouts.tsx +154 -0
  71. package/renderer/components/docs/mdx/cards.tsx +213 -0
  72. package/renderer/components/docs/mdx/changelog.tsx +120 -0
  73. package/renderer/components/docs/mdx/code-block.tsx +186 -0
  74. package/renderer/components/docs/mdx/code-group.tsx +421 -0
  75. package/renderer/components/docs/mdx/file-embeds.tsx +105 -0
  76. package/renderer/components/docs/mdx/frame.tsx +112 -0
  77. package/renderer/components/docs/mdx/highlight.tsx +151 -0
  78. package/renderer/components/docs/mdx/iframe.tsx +134 -0
  79. package/renderer/components/docs/mdx/image.tsx +235 -0
  80. package/renderer/components/docs/mdx/index.ts +204 -0
  81. package/renderer/components/docs/mdx/mermaid.tsx +240 -0
  82. package/renderer/components/docs/mdx/param-field.tsx +200 -0
  83. package/renderer/components/docs/mdx/steps.tsx +113 -0
  84. package/renderer/components/docs/mdx/tabs.tsx +86 -0
  85. package/renderer/components/docs/mdx-renderer.tsx +100 -0
  86. package/renderer/components/docs/navigation/breadcrumbs.tsx +76 -0
  87. package/renderer/components/docs/navigation/index.ts +8 -0
  88. package/renderer/components/docs/navigation/page-nav.tsx +64 -0
  89. package/renderer/components/docs/navigation/sidebar.tsx +515 -0
  90. package/renderer/components/docs/navigation/toc.tsx +113 -0
  91. package/renderer/components/docs/notice.tsx +105 -0
  92. package/renderer/components/docs-header.tsx +274 -0
  93. package/renderer/components/docs-viewer/agent/agent-chat.tsx +2076 -0
  94. package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +90 -0
  95. package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +49 -0
  96. package/renderer/components/docs-viewer/agent/cards/index.tsx +50 -0
  97. package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +212 -0
  98. package/renderer/components/docs-viewer/agent/cards/types.ts +84 -0
  99. package/renderer/components/docs-viewer/agent/chat-message.tsx +17 -0
  100. package/renderer/components/docs-viewer/agent/index.tsx +6 -0
  101. package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +119 -0
  102. package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +46 -0
  103. package/renderer/components/docs-viewer/agent/messages/index.ts +17 -0
  104. package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +721 -0
  105. package/renderer/components/docs-viewer/agent/messages/types.ts +61 -0
  106. package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +24 -0
  107. package/renderer/components/docs-viewer/agent/messages/user-message.tsx +51 -0
  108. package/renderer/components/docs-viewer/code-editor/index.tsx +2 -0
  109. package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +1283 -0
  110. package/renderer/components/docs-viewer/content/changelog-page.tsx +331 -0
  111. package/renderer/components/docs-viewer/content/doc-page.tsx +285 -0
  112. package/renderer/components/docs-viewer/content/documentation-viewer.tsx +17 -0
  113. package/renderer/components/docs-viewer/content/index.tsx +29 -0
  114. package/renderer/components/docs-viewer/content/introduction.tsx +21 -0
  115. package/renderer/components/docs-viewer/content/request-details.tsx +330 -0
  116. package/renderer/components/docs-viewer/content/sections/auth.tsx +69 -0
  117. package/renderer/components/docs-viewer/content/sections/body.tsx +66 -0
  118. package/renderer/components/docs-viewer/content/sections/headers.tsx +43 -0
  119. package/renderer/components/docs-viewer/content/sections/overview.tsx +40 -0
  120. package/renderer/components/docs-viewer/content/sections/parameters.tsx +43 -0
  121. package/renderer/components/docs-viewer/content/sections/responses.tsx +87 -0
  122. package/renderer/components/docs-viewer/global-auth-modal.tsx +352 -0
  123. package/renderer/components/docs-viewer/index.tsx +1466 -0
  124. package/renderer/components/docs-viewer/playground/auth-editor.tsx +280 -0
  125. package/renderer/components/docs-viewer/playground/body-editor.tsx +221 -0
  126. package/renderer/components/docs-viewer/playground/code-editor.tsx +224 -0
  127. package/renderer/components/docs-viewer/playground/code-snippet.tsx +387 -0
  128. package/renderer/components/docs-viewer/playground/graphql-playground.tsx +745 -0
  129. package/renderer/components/docs-viewer/playground/index.tsx +671 -0
  130. package/renderer/components/docs-viewer/playground/key-value-editor.tsx +261 -0
  131. package/renderer/components/docs-viewer/playground/method-selector.tsx +60 -0
  132. package/renderer/components/docs-viewer/playground/request-builder.tsx +179 -0
  133. package/renderer/components/docs-viewer/playground/request-tabs.tsx +237 -0
  134. package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +21 -0
  135. package/renderer/components/docs-viewer/playground/response-cards/index.tsx +93 -0
  136. package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +16 -0
  137. package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +23 -0
  138. package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +268 -0
  139. package/renderer/components/docs-viewer/playground/response-cards/types.ts +82 -0
  140. package/renderer/components/docs-viewer/playground/response-viewer.tsx +43 -0
  141. package/renderer/components/docs-viewer/search/index.ts +2 -0
  142. package/renderer/components/docs-viewer/search/search-dialog.tsx +331 -0
  143. package/renderer/components/docs-viewer/search/use-search.ts +117 -0
  144. package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +431 -0
  145. package/renderer/components/docs-viewer/shared/method-badge.tsx +41 -0
  146. package/renderer/components/docs-viewer/shared/schema-viewer.tsx +349 -0
  147. package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +239 -0
  148. package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +316 -0
  149. package/renderer/components/docs-viewer/sidebar/index.tsx +343 -0
  150. package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +202 -0
  151. package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +118 -0
  152. package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +226 -0
  153. package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +52 -0
  154. package/renderer/components/theme-provider.tsx +11 -0
  155. package/renderer/components/theme-toggle.tsx +76 -0
  156. package/renderer/components/ui/badge.tsx +46 -0
  157. package/renderer/components/ui/button.tsx +59 -0
  158. package/renderer/components/ui/dialog.tsx +118 -0
  159. package/renderer/components/ui/dropdown-menu.tsx +257 -0
  160. package/renderer/components/ui/input.tsx +21 -0
  161. package/renderer/components/ui/label.tsx +24 -0
  162. package/renderer/components/ui/navigation-menu.tsx +168 -0
  163. package/renderer/components/ui/select.tsx +190 -0
  164. package/renderer/components/ui/spinner.tsx +114 -0
  165. package/renderer/components/ui/tabs.tsx +66 -0
  166. package/renderer/components/ui/tooltip.tsx +61 -0
  167. package/renderer/hooks/use-code-copy.ts +88 -0
  168. package/renderer/hooks/use-openapi-title.ts +44 -0
  169. package/renderer/lib/api-docs/agent/index.ts +6 -0
  170. package/renderer/lib/api-docs/agent/indexer.ts +323 -0
  171. package/renderer/lib/api-docs/agent/spec-summary.ts +335 -0
  172. package/renderer/lib/api-docs/agent/types.ts +116 -0
  173. package/renderer/lib/api-docs/auth/auth-context.tsx +225 -0
  174. package/renderer/lib/api-docs/auth/auth-storage.ts +87 -0
  175. package/renderer/lib/api-docs/auth/crypto.ts +89 -0
  176. package/renderer/lib/api-docs/auth/index.ts +4 -0
  177. package/renderer/lib/api-docs/code-editor/db.ts +164 -0
  178. package/renderer/lib/api-docs/code-editor/hooks.ts +266 -0
  179. package/renderer/lib/api-docs/code-editor/index.ts +6 -0
  180. package/renderer/lib/api-docs/code-editor/mode-context.tsx +207 -0
  181. package/renderer/lib/api-docs/code-editor/types.ts +105 -0
  182. package/renderer/lib/api-docs/codegen/definitions.ts +297 -0
  183. package/renderer/lib/api-docs/codegen/har.ts +251 -0
  184. package/renderer/lib/api-docs/codegen/index.ts +159 -0
  185. package/renderer/lib/api-docs/factories.ts +151 -0
  186. package/renderer/lib/api-docs/index.ts +17 -0
  187. package/renderer/lib/api-docs/mobile-context.tsx +112 -0
  188. package/renderer/lib/api-docs/navigation-context.tsx +88 -0
  189. package/renderer/lib/api-docs/parsers/graphql/README.md +129 -0
  190. package/renderer/lib/api-docs/parsers/graphql/index.ts +91 -0
  191. package/renderer/lib/api-docs/parsers/graphql/parser.ts +491 -0
  192. package/renderer/lib/api-docs/parsers/graphql/transformer.ts +246 -0
  193. package/renderer/lib/api-docs/parsers/graphql/types.ts +283 -0
  194. package/renderer/lib/api-docs/parsers/openapi/README.md +32 -0
  195. package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +60 -0
  196. package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +574 -0
  197. package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +403 -0
  198. package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +232 -0
  199. package/renderer/lib/api-docs/parsers/openapi/index.ts +171 -0
  200. package/renderer/lib/api-docs/parsers/openapi/transformer.ts +277 -0
  201. package/renderer/lib/api-docs/parsers/openapi/validator.ts +31 -0
  202. package/renderer/lib/api-docs/playground/context.tsx +107 -0
  203. package/renderer/lib/api-docs/playground/navigation-context.tsx +124 -0
  204. package/renderer/lib/api-docs/playground/request-builder.ts +223 -0
  205. package/renderer/lib/api-docs/playground/request-runner.ts +282 -0
  206. package/renderer/lib/api-docs/playground/types.ts +35 -0
  207. package/renderer/lib/api-docs/types.ts +269 -0
  208. package/renderer/lib/api-docs/utils.ts +311 -0
  209. package/renderer/lib/cache.ts +193 -0
  210. package/renderer/lib/docs/config/index.ts +29 -0
  211. package/renderer/lib/docs/config/loader.ts +142 -0
  212. package/renderer/lib/docs/config/schema.ts +298 -0
  213. package/renderer/lib/docs/index.ts +12 -0
  214. package/renderer/lib/docs/mdx/compiler.ts +176 -0
  215. package/renderer/lib/docs/mdx/frontmatter.ts +80 -0
  216. package/renderer/lib/docs/mdx/index.ts +26 -0
  217. package/renderer/lib/docs/navigation/generator.ts +348 -0
  218. package/renderer/lib/docs/navigation/index.ts +12 -0
  219. package/renderer/lib/docs/navigation/types.ts +123 -0
  220. package/renderer/lib/docs-navigation-context.tsx +80 -0
  221. package/renderer/lib/multi-tenant/context.ts +105 -0
  222. package/renderer/lib/storage/blob.ts +845 -0
  223. package/renderer/lib/utils.ts +6 -0
  224. package/renderer/next.config.ts +76 -0
  225. package/renderer/package.json +66 -0
  226. package/renderer/postcss.config.mjs +5 -0
  227. package/renderer/public/assets/images/screenshot.png +0 -0
  228. package/renderer/public/assets/logo/dark.svg +9 -0
  229. package/renderer/public/assets/logo/light.svg +9 -0
  230. package/renderer/public/assets/logo.svg +9 -0
  231. package/renderer/public/file.svg +1 -0
  232. package/renderer/public/globe.svg +1 -0
  233. package/renderer/public/icon.png +0 -0
  234. package/renderer/public/logo.svg +9 -0
  235. package/renderer/public/window.svg +1 -0
  236. package/renderer/tsconfig.json +28 -0
  237. package/templates/basic/README.md +139 -0
  238. package/templates/basic/assets/favicon.svg +4 -0
  239. package/templates/basic/assets/logo.svg +9 -0
  240. package/templates/basic/docs.json +47 -0
  241. package/templates/basic/guides/configuration.mdx +149 -0
  242. package/templates/basic/guides/overview.mdx +96 -0
  243. package/templates/basic/index.mdx +39 -0
  244. package/templates/basic/package.json +14 -0
  245. package/templates/basic/quickstart.mdx +92 -0
  246. package/templates/basic/vercel.json +6 -0
  247. package/templates/graphql/README.md +139 -0
  248. package/templates/graphql/api-reference/schema.graphql +305 -0
  249. package/templates/graphql/assets/favicon.svg +4 -0
  250. package/templates/graphql/assets/logo.svg +9 -0
  251. package/templates/graphql/docs.json +54 -0
  252. package/templates/graphql/guides/configuration.mdx +149 -0
  253. package/templates/graphql/guides/overview.mdx +96 -0
  254. package/templates/graphql/index.mdx +39 -0
  255. package/templates/graphql/package.json +14 -0
  256. package/templates/graphql/quickstart.mdx +92 -0
  257. package/templates/graphql/vercel.json +6 -0
  258. package/templates/openapi/README.md +139 -0
  259. package/templates/openapi/api-reference/openapi.json +419 -0
  260. package/templates/openapi/assets/favicon.svg +4 -0
  261. package/templates/openapi/assets/logo.svg +9 -0
  262. package/templates/openapi/docs.json +61 -0
  263. package/templates/openapi/guides/configuration.mdx +149 -0
  264. package/templates/openapi/guides/overview.mdx +96 -0
  265. package/templates/openapi/index.mdx +39 -0
  266. package/templates/openapi/package.json +14 -0
  267. package/templates/openapi/quickstart.mdx +92 -0
  268. package/templates/openapi/vercel.json +6 -0
@@ -0,0 +1,90 @@
1
+ 'use client'
2
+
3
+ import { Bug, ArrowsClockwise } from '@phosphor-icons/react'
4
+ import { Badge } from '@/components/ui/badge'
5
+ import { MethodBadge } from '../../shared/method-badge'
6
+ import type { DebugContext } from '../../playground/response-viewer'
7
+ import type { HTTPMethod } from '@/lib/api-docs/types'
8
+ import { cn } from '@/lib/utils'
9
+
10
+ interface DebugContextCardProps {
11
+ context: DebugContext
12
+ }
13
+
14
+ export function DebugContextCard({ context }: DebugContextCardProps) {
15
+ const isError = context.status >= 400
16
+
17
+ // Try to format the response body
18
+ let formattedBody = context.responseBody
19
+ try {
20
+ const parsed = JSON.parse(context.responseBody)
21
+ formattedBody = JSON.stringify(parsed, null, 2)
22
+ } catch {
23
+ // Keep original
24
+ }
25
+
26
+ // Truncate if too long
27
+ const truncatedBody = formattedBody.length > 300
28
+ ? formattedBody.slice(0, 300) + '...'
29
+ : formattedBody
30
+
31
+ return (
32
+ <div
33
+ className={cn(
34
+ "fade-in slide-in-from-bottom-1 animate-in duration-150",
35
+ "mx-auto w-full max-w-[var(--thread-max-width)] px-2 py-3"
36
+ )}
37
+ >
38
+ <div className="rounded-lg border border-border bg-muted/30 overflow-hidden">
39
+ {/* Header */}
40
+ <div className="flex items-center gap-2 px-3 py-2 bg-muted/50 border-b border-border">
41
+ <Bug className="h-4 w-4 text-amber-500" weight="bold" />
42
+ <span className="text-xs font-medium text-muted-foreground">Debug Context</span>
43
+ </div>
44
+
45
+ {/* Endpoint info */}
46
+ {context.endpointName && (
47
+ <div className="px-3 py-2 border-b border-border">
48
+ <div className="flex items-center gap-2">
49
+ {context.endpointMethod && (
50
+ <MethodBadge method={context.endpointMethod as HTTPMethod} size="sm" />
51
+ )}
52
+ <span className="text-sm font-medium text-foreground">{context.endpointName}</span>
53
+ </div>
54
+ {context.endpointPath && (
55
+ <p className="text-xs text-muted-foreground mt-1 font-mono">{context.endpointPath}</p>
56
+ )}
57
+ </div>
58
+ )}
59
+
60
+ {/* Response status */}
61
+ <div className="px-3 py-2 border-b border-border">
62
+ <div className="flex items-center gap-2">
63
+ <ArrowsClockwise className="h-3.5 w-3.5 text-muted-foreground" weight="bold" />
64
+ <span className="text-xs text-muted-foreground">Response</span>
65
+ <Badge variant={isError ? 'destructive' : 'default'} className="text-xs h-5">
66
+ {context.status} {context.statusText}
67
+ </Badge>
68
+ </div>
69
+ </div>
70
+
71
+ {/* Response body preview */}
72
+ <div className="px-3 py-2">
73
+ <pre className="text-xs font-mono text-muted-foreground bg-background/50 rounded p-2 overflow-x-auto max-h-32 overflow-y-auto">
74
+ {truncatedBody}
75
+ </pre>
76
+ </div>
77
+
78
+ {/* Request body if present */}
79
+ {context.requestBody && (
80
+ <div className="px-3 py-2 border-t border-border">
81
+ <p className="text-xs text-muted-foreground mb-1">Request Body</p>
82
+ <pre className="text-xs font-mono text-muted-foreground bg-background/50 rounded p-2 overflow-x-auto max-h-20 overflow-y-auto">
83
+ {context.requestBody.length > 200 ? context.requestBody.slice(0, 200) + '...' : context.requestBody}
84
+ </pre>
85
+ </div>
86
+ )}
87
+ </div>
88
+ </div>
89
+ )
90
+ }
@@ -0,0 +1,49 @@
1
+ 'use client'
2
+
3
+ import { Crosshair } from '@phosphor-icons/react'
4
+ import { MethodBadge } from '../../shared/method-badge'
5
+ import type { EndpointContextCardData } from './types'
6
+ import { cn } from '@/lib/utils'
7
+
8
+ interface EndpointContextCardProps {
9
+ endpoint: EndpointContextCardData['endpoint']
10
+ }
11
+
12
+ export function EndpointContextCard({ endpoint }: EndpointContextCardProps) {
13
+ return (
14
+ <div
15
+ className={cn(
16
+ "fade-in slide-in-from-bottom-1 animate-in duration-150",
17
+ "mx-auto w-full max-w-[var(--thread-max-width)] px-2 py-2"
18
+ )}
19
+ >
20
+ <div className="rounded-lg border border-border/60 bg-muted/20 overflow-hidden">
21
+ {/* Header */}
22
+ <div className="flex items-center gap-2 px-3 py-2 bg-muted/30 border-b border-border/40">
23
+ <Crosshair className="h-3.5 w-3.5 text-muted-foreground" weight="bold" />
24
+ <span className="text-xs font-medium text-muted-foreground">Asking about</span>
25
+ </div>
26
+
27
+ {/* Content */}
28
+ <div className="p-3">
29
+ <div className="flex items-start gap-3">
30
+ <MethodBadge method={endpoint.method} size="sm" />
31
+ <div className="flex-1 min-w-0">
32
+ <p className="text-sm font-medium text-foreground truncate">
33
+ {endpoint.name}
34
+ </p>
35
+ <code className="text-xs text-muted-foreground font-mono block truncate mt-0.5">
36
+ {endpoint.path}
37
+ </code>
38
+ {endpoint.description && (
39
+ <p className="text-xs text-muted-foreground mt-1.5 line-clamp-2">
40
+ {endpoint.description}
41
+ </p>
42
+ )}
43
+ </div>
44
+ </div>
45
+ </div>
46
+ </div>
47
+ </div>
48
+ )
49
+ }
@@ -0,0 +1,50 @@
1
+ 'use client'
2
+
3
+ import { DebugContextCard } from './debug-context-card'
4
+ import { ResponseOptionsCard } from './response-options-card'
5
+ import { EndpointContextCard } from './endpoint-context-card'
6
+ import { decodeCardMessage, type CardData, type CardActions } from './types'
7
+
8
+ // Export types and utilities
9
+ export * from './types'
10
+ export { DebugContextCard } from './debug-context-card'
11
+ export { ResponseOptionsCard, detectErrorType, type ErrorType } from './response-options-card'
12
+ export { EndpointContextCard } from './endpoint-context-card'
13
+
14
+ interface CardRendererProps {
15
+ content: string
16
+ actions?: CardActions
17
+ }
18
+
19
+ /**
20
+ * Renders a card based on the encoded message content.
21
+ * Returns null if the content is not a valid card message.
22
+ */
23
+ export function CardRenderer({ content, actions }: CardRendererProps) {
24
+ const cardData = decodeCardMessage(content)
25
+
26
+ if (!cardData) {
27
+ return null
28
+ }
29
+
30
+ return <CardByType data={cardData} actions={actions} />
31
+ }
32
+
33
+ /**
34
+ * Renders the appropriate card component based on the card type.
35
+ */
36
+ function CardByType({ data, actions }: { data: CardData; actions?: CardActions }) {
37
+ switch (data.type) {
38
+ case 'debug_context':
39
+ return <DebugContextCard context={data.context} />
40
+
41
+ case 'response_options':
42
+ return <ResponseOptionsCard errorType={data.errorType} actions={actions} />
43
+
44
+ case 'endpoint_context':
45
+ return <EndpointContextCard endpoint={data.endpoint} />
46
+
47
+ default:
48
+ return null
49
+ }
50
+ }
@@ -0,0 +1,212 @@
1
+ 'use client'
2
+
3
+ import { ShieldCheck, CaretRight, Sliders, Code, ListBullets, type Icon } from '@phosphor-icons/react'
4
+ import type { CardActions } from './types'
5
+ import { cn } from '@/lib/utils'
6
+
7
+ // Extended error type with more specific categories
8
+ export type ErrorType =
9
+ | 'auth'
10
+ | 'missing_params'
11
+ | 'invalid_body'
12
+ | 'missing_headers'
13
+ | 'validation'
14
+ | 'not_found'
15
+ | 'server'
16
+ | 'generic'
17
+
18
+ interface ResponseOptionsCardProps {
19
+ errorType: ErrorType
20
+ actions?: CardActions
21
+ }
22
+
23
+ interface ActionButton {
24
+ id: string
25
+ icon: Icon
26
+ iconColor: string
27
+ bgColor: string
28
+ hoverBorder: string
29
+ title: string
30
+ description: string
31
+ onClick?: () => void
32
+ }
33
+
34
+ export function ResponseOptionsCard({ errorType, actions }: ResponseOptionsCardProps) {
35
+ // Build list of actions based on error type
36
+ const actionButtons: ActionButton[] = []
37
+
38
+ // Auth errors
39
+ if (errorType === 'auth') {
40
+ if (actions?.onOpenGlobalAuth) {
41
+ actionButtons.push({
42
+ id: 'global-auth',
43
+ icon: ShieldCheck,
44
+ iconColor: 'text-green-600',
45
+ bgColor: 'bg-green-500/10',
46
+ hoverBorder: 'hover:border-green-500/30',
47
+ title: 'Set Global Auth',
48
+ description: 'Configure API credentials for all requests',
49
+ onClick: actions.onOpenGlobalAuth,
50
+ })
51
+ }
52
+ if (actions?.onNavigateToAuthTab) {
53
+ actionButtons.push({
54
+ id: 'auth-tab',
55
+ icon: ShieldCheck,
56
+ iconColor: 'text-blue-600',
57
+ bgColor: 'bg-blue-500/10',
58
+ hoverBorder: 'hover:border-blue-500/30',
59
+ title: 'Go to Authorization Tab',
60
+ description: 'Set auth for this specific request',
61
+ onClick: actions.onNavigateToAuthTab,
62
+ })
63
+ }
64
+ }
65
+
66
+ // Missing params errors
67
+ if (errorType === 'missing_params' || errorType === 'validation') {
68
+ if (actions?.onNavigateToParamsTab) {
69
+ actionButtons.push({
70
+ id: 'params-tab',
71
+ icon: Sliders,
72
+ iconColor: 'text-purple-600',
73
+ bgColor: 'bg-purple-500/10',
74
+ hoverBorder: 'hover:border-purple-500/30',
75
+ title: 'Go to Parameters Tab',
76
+ description: 'Check and fill in required parameters',
77
+ onClick: actions.onNavigateToParamsTab,
78
+ })
79
+ }
80
+ }
81
+
82
+ // Invalid body errors
83
+ if (errorType === 'invalid_body' || errorType === 'validation') {
84
+ if (actions?.onNavigateToBodyTab) {
85
+ actionButtons.push({
86
+ id: 'body-tab',
87
+ icon: Code,
88
+ iconColor: 'text-amber-600',
89
+ bgColor: 'bg-amber-500/10',
90
+ hoverBorder: 'hover:border-amber-500/30',
91
+ title: 'Go to Body Tab',
92
+ description: 'Review and fix the request body',
93
+ onClick: actions.onNavigateToBodyTab,
94
+ })
95
+ }
96
+ }
97
+
98
+ // Missing headers errors
99
+ if (errorType === 'missing_headers') {
100
+ if (actions?.onNavigateToHeadersTab) {
101
+ actionButtons.push({
102
+ id: 'headers-tab',
103
+ icon: ListBullets,
104
+ iconColor: 'text-cyan-600',
105
+ bgColor: 'bg-cyan-500/10',
106
+ hoverBorder: 'hover:border-cyan-500/30',
107
+ title: 'Go to Headers Tab',
108
+ description: 'Add required request headers',
109
+ onClick: actions.onNavigateToHeadersTab,
110
+ })
111
+ }
112
+ }
113
+
114
+ // Don't render if no actions
115
+ if (actionButtons.length === 0) {
116
+ return null
117
+ }
118
+
119
+ return (
120
+ <div
121
+ className={cn(
122
+ "fade-in slide-in-from-bottom-1 animate-in duration-150",
123
+ "mx-auto w-full max-w-[var(--thread-max-width)] px-2 py-2"
124
+ )}
125
+ >
126
+ <div className="space-y-2">
127
+ <p className="text-xs font-medium text-muted-foreground px-1 mb-3">Suggested Actions</p>
128
+ {actionButtons.map((action) => (
129
+ <button
130
+ key={action.id}
131
+ onClick={action.onClick}
132
+ className={cn(
133
+ "w-full flex items-center gap-3 p-3 rounded-lg border border-border",
134
+ "bg-muted/30 hover:bg-muted/60 transition-all group text-left",
135
+ action.hoverBorder
136
+ )}
137
+ >
138
+ <div className={cn("shrink-0 w-8 h-8 rounded-md flex items-center justify-center", action.bgColor)}>
139
+ <action.icon className={cn("h-4 w-4", action.iconColor)} weight="bold" />
140
+ </div>
141
+ <div className="flex-1 min-w-0">
142
+ <p className="text-sm font-medium text-foreground">{action.title}</p>
143
+ <p className="text-xs text-muted-foreground truncate">{action.description}</p>
144
+ </div>
145
+ <CaretRight className="shrink-0 h-4 w-4 text-muted-foreground group-hover:text-foreground transition-colors" weight="bold" />
146
+ </button>
147
+ ))}
148
+ </div>
149
+ </div>
150
+ )
151
+ }
152
+
153
+ // Helper to determine error type from status and response body
154
+ export function detectErrorType(status: number, responseBody: string): ErrorType {
155
+ const lowerBody = responseBody.toLowerCase()
156
+
157
+ // Auth errors (401, 403)
158
+ if (status === 401 || status === 403) return 'auth'
159
+ if (
160
+ lowerBody.includes('unauthorized') ||
161
+ lowerBody.includes('authentication') ||
162
+ lowerBody.includes('api key') ||
163
+ lowerBody.includes('api_key') ||
164
+ lowerBody.includes('token') ||
165
+ lowerBody.includes('credentials') ||
166
+ lowerBody.includes('forbidden')
167
+ ) {
168
+ return 'auth'
169
+ }
170
+
171
+ // Missing parameters
172
+ if (
173
+ lowerBody.includes('missing') && (lowerBody.includes('param') || lowerBody.includes('query')) ||
174
+ lowerBody.includes('required parameter') ||
175
+ lowerBody.includes('required field') ||
176
+ lowerBody.includes('parameter is required')
177
+ ) {
178
+ return 'missing_params'
179
+ }
180
+
181
+ // Invalid body
182
+ if (
183
+ lowerBody.includes('invalid json') ||
184
+ lowerBody.includes('parse error') ||
185
+ lowerBody.includes('body') && (lowerBody.includes('invalid') || lowerBody.includes('malformed')) ||
186
+ lowerBody.includes('request body') ||
187
+ lowerBody.includes('payload')
188
+ ) {
189
+ return 'invalid_body'
190
+ }
191
+
192
+ // Missing headers
193
+ if (
194
+ lowerBody.includes('missing header') ||
195
+ lowerBody.includes('required header') ||
196
+ lowerBody.includes('header') && lowerBody.includes('required') ||
197
+ lowerBody.includes('content-type')
198
+ ) {
199
+ return 'missing_headers'
200
+ }
201
+
202
+ // Generic validation errors (400, 422)
203
+ if (status === 400 || status === 422) return 'validation'
204
+
205
+ // Not found
206
+ if (status === 404) return 'not_found'
207
+
208
+ // Server errors
209
+ if (status >= 500) return 'server'
210
+
211
+ return 'generic'
212
+ }
@@ -0,0 +1,84 @@
1
+ import type { DebugContext } from '../../playground/response-viewer'
2
+ import type { HTTPMethod } from '@/lib/api-docs/types'
3
+
4
+ // Card type identifiers
5
+ export type CardType = 'debug_context' | 'response_options' | 'endpoint_context'
6
+
7
+ // Error types for response options
8
+ export type ErrorType =
9
+ | 'auth'
10
+ | 'missing_params'
11
+ | 'invalid_body'
12
+ | 'missing_headers'
13
+ | 'validation'
14
+ | 'not_found'
15
+ | 'server'
16
+ | 'generic'
17
+
18
+ // Base interface for all cards
19
+ export interface BaseCardData {
20
+ type: CardType
21
+ }
22
+
23
+ // Debug context card data
24
+ export interface DebugContextCardData extends BaseCardData {
25
+ type: 'debug_context'
26
+ context: DebugContext
27
+ }
28
+
29
+ // Response options card data
30
+ export interface ResponseOptionsCardData extends BaseCardData {
31
+ type: 'response_options'
32
+ errorType: ErrorType
33
+ }
34
+
35
+ // Endpoint context card data
36
+ export interface EndpointContextCardData extends BaseCardData {
37
+ type: 'endpoint_context'
38
+ endpoint: {
39
+ id: string
40
+ name: string
41
+ method: HTTPMethod
42
+ path: string
43
+ description?: string | null
44
+ }
45
+ }
46
+
47
+ // Union type for all card data
48
+ export type CardData = DebugContextCardData | ResponseOptionsCardData | EndpointContextCardData
49
+
50
+ // Card action callbacks
51
+ export interface CardActions {
52
+ onOpenGlobalAuth?: () => void
53
+ onNavigateToAuthTab?: () => void
54
+ onNavigateToParamsTab?: () => void
55
+ onNavigateToBodyTab?: () => void
56
+ onNavigateToHeadersTab?: () => void
57
+ }
58
+
59
+ // Prefix used to identify card messages
60
+ export const CARD_MESSAGE_PREFIX = '__CARD__'
61
+
62
+ // Helper to encode card data into a message
63
+ export function encodeCardMessage(data: CardData): string {
64
+ return `${CARD_MESSAGE_PREFIX}${JSON.stringify(data)}`
65
+ }
66
+
67
+ // Helper to decode card data from a message
68
+ export function decodeCardMessage(content: string): CardData | null {
69
+ if (!content?.startsWith(CARD_MESSAGE_PREFIX)) {
70
+ return null
71
+ }
72
+
73
+ try {
74
+ const json = content.slice(CARD_MESSAGE_PREFIX.length)
75
+ return JSON.parse(json) as CardData
76
+ } catch {
77
+ return null
78
+ }
79
+ }
80
+
81
+ // Check if a message is a card message
82
+ export function isCardMessage(content: string): boolean {
83
+ return content?.startsWith(CARD_MESSAGE_PREFIX) ?? false
84
+ }
@@ -0,0 +1,17 @@
1
+ // Re-export from messages folder for backwards compatibility
2
+ export {
3
+ ChatMessage,
4
+ TypingIndicator,
5
+ UserMessage,
6
+ AssistantMessage,
7
+ ToolCallDisplay,
8
+ } from './messages'
9
+
10
+ export type {
11
+ ToolInvocation,
12
+ MessageImage,
13
+ ChatMessageProps,
14
+ UserMessageProps,
15
+ AssistantMessageProps,
16
+ ToolCallDisplayProps,
17
+ } from './messages'
@@ -0,0 +1,6 @@
1
+ /**
2
+ * Agent Components Exports
3
+ */
4
+
5
+ export { AgentChat } from './agent-chat'
6
+ export { ChatMessage, TypingIndicator } from './chat-message'
@@ -0,0 +1,119 @@
1
+ 'use client'
2
+
3
+ import { useState, useCallback } from 'react'
4
+ import { Copy, Check, Trash } from '@phosphor-icons/react'
5
+ import { cn } from '@/lib/utils'
6
+ import { MarkdownRenderer } from '../../shared/markdown-renderer'
7
+ import {
8
+ Tooltip,
9
+ TooltipContent,
10
+ TooltipTrigger,
11
+ } from '@/components/ui/tooltip'
12
+ import { ToolCallDisplay } from './tool-call-display'
13
+ import type { AssistantMessageProps, AssistantActionBarProps } from './types'
14
+
15
+ export function AssistantMessage({
16
+ content,
17
+ messageId,
18
+ toolInvocations,
19
+ isStreaming,
20
+ onDeleteMessage,
21
+ onNavigate,
22
+ onNavigateToDocPage,
23
+ onOpenFile
24
+ }: AssistantMessageProps) {
25
+ return (
26
+ <div
27
+ className={cn(
28
+ "aui-assistant-message-root",
29
+ "fade-in slide-in-from-bottom-1 animate-in duration-150",
30
+ "relative mx-auto w-full max-w-[var(--thread-max-width)] py-3"
31
+ )}
32
+ data-role="assistant"
33
+ >
34
+ {/* Text content */}
35
+ {content && (
36
+ <div className="aui-assistant-message-content wrap-break-word px-2 text-foreground leading-relaxed">
37
+ <div className="prose prose-sm dark:prose-invert max-w-none">
38
+ <MarkdownRenderer content={content} />
39
+ </div>
40
+ </div>
41
+ )}
42
+
43
+ {/* Tool invocations */}
44
+ {toolInvocations && toolInvocations.length > 0 && (
45
+ <div className="mt-2 px-2 space-y-1">
46
+ {toolInvocations.map((tool) => (
47
+ <ToolCallDisplay key={tool.toolCallId} tool={tool} onNavigate={onNavigate} onNavigateToDocPage={onNavigateToDocPage} onOpenFile={onOpenFile} />
48
+ ))}
49
+ </div>
50
+ )}
51
+
52
+ {/* Action bar - only show when streaming is finished */}
53
+ {!isStreaming && (
54
+ <div className="aui-assistant-message-footer mt-1 px-2 flex justify-end">
55
+ <AssistantActionBar content={content} messageId={messageId} onDeleteMessage={onDeleteMessage} />
56
+ </div>
57
+ )}
58
+ </div>
59
+ )
60
+ }
61
+
62
+ function AssistantActionBar({
63
+ content,
64
+ messageId,
65
+ onDeleteMessage
66
+ }: AssistantActionBarProps) {
67
+ const [copied, setCopied] = useState(false)
68
+
69
+ const handleCopy = useCallback(async () => {
70
+ try {
71
+ await navigator.clipboard.writeText(content)
72
+ setCopied(true)
73
+ setTimeout(() => setCopied(false), 2000)
74
+ } catch (err) {
75
+ console.error('Failed to copy:', err)
76
+ }
77
+ }, [content])
78
+
79
+ const handleDelete = useCallback(() => {
80
+ if (messageId && onDeleteMessage) {
81
+ onDeleteMessage(messageId)
82
+ }
83
+ }, [messageId, onDeleteMessage])
84
+
85
+ return (
86
+ <div className={cn(
87
+ "aui-assistant-action-bar-root",
88
+ "flex gap-1 text-muted-foreground"
89
+ )}>
90
+ <Tooltip>
91
+ <TooltipTrigger asChild>
92
+ <button
93
+ onClick={handleCopy}
94
+ className="aui-button-icon size-6 p-1 rounded hover:bg-accent hover:text-foreground transition-colors"
95
+ aria-label={copied ? "Copied" : "Copy"}
96
+ >
97
+ {copied ? <Check className="size-4" weight="bold" /> : <Copy className="size-4" />}
98
+ </button>
99
+ </TooltipTrigger>
100
+ <TooltipContent>{copied ? "Copied!" : "Copy"}</TooltipContent>
101
+ </Tooltip>
102
+
103
+ {messageId && onDeleteMessage && (
104
+ <Tooltip>
105
+ <TooltipTrigger asChild>
106
+ <button
107
+ onClick={handleDelete}
108
+ className="aui-button-icon size-6 p-1 rounded hover:bg-accent hover:text-foreground transition-colors"
109
+ aria-label="Delete message"
110
+ >
111
+ <Trash className="size-4" />
112
+ </button>
113
+ </TooltipTrigger>
114
+ <TooltipContent>Delete message</TooltipContent>
115
+ </Tooltip>
116
+ )}
117
+ </div>
118
+ )
119
+ }
@@ -0,0 +1,46 @@
1
+ 'use client'
2
+
3
+ import { CardRenderer, isCardMessage } from '../cards'
4
+ import { UserMessage } from './user-message'
5
+ import { AssistantMessage } from './assistant-message'
6
+ import type { ChatMessageProps } from './types'
7
+
8
+ export function ChatMessage({
9
+ role,
10
+ content,
11
+ messageId,
12
+ images,
13
+ toolInvocations,
14
+ isStreaming,
15
+ onDeleteMessage,
16
+ onNavigate,
17
+ onNavigateToDocPage,
18
+ onOpenFile,
19
+ cardActions
20
+ }: ChatMessageProps) {
21
+ if (role === 'user') {
22
+ return <UserMessage content={content} images={images} cardActions={cardActions} />
23
+ }
24
+
25
+ if (role === 'assistant') {
26
+ // Check if this is a card message (like response options)
27
+ if (isCardMessage(content)) {
28
+ return <CardRenderer content={content} actions={cardActions} />
29
+ }
30
+
31
+ return (
32
+ <AssistantMessage
33
+ content={content}
34
+ messageId={messageId}
35
+ toolInvocations={toolInvocations}
36
+ isStreaming={isStreaming}
37
+ onDeleteMessage={onDeleteMessage}
38
+ onNavigate={onNavigate}
39
+ onNavigateToDocPage={onNavigateToDocPage}
40
+ onOpenFile={onOpenFile}
41
+ />
42
+ )
43
+ }
44
+
45
+ return null
46
+ }