zudoku 0.26.1 → 0.28.0

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 (264) hide show
  1. package/dist/app/main.d.ts +1 -1
  2. package/dist/app/main.js +19 -7
  3. package/dist/app/main.js.map +1 -1
  4. package/dist/config/loader.js +1 -1
  5. package/dist/config/loader.js.map +1 -1
  6. package/dist/config/validators/InputSidebarSchema.d.ts +2 -2
  7. package/dist/config/validators/common.d.ts +67 -0
  8. package/dist/config/validators/common.js +5 -0
  9. package/dist/config/validators/common.js.map +1 -1
  10. package/dist/config/validators/validate.d.ts +29 -0
  11. package/dist/lib/components/AnchorLink.js +5 -2
  12. package/dist/lib/components/AnchorLink.js.map +1 -1
  13. package/dist/lib/components/Header.js +1 -1
  14. package/dist/lib/components/Header.js.map +1 -1
  15. package/dist/lib/components/Heading.d.ts +1 -1
  16. package/dist/lib/components/Markdown.d.ts +2 -2
  17. package/dist/lib/components/Markdown.js +3 -1
  18. package/dist/lib/components/Markdown.js.map +1 -1
  19. package/dist/lib/components/StatusPage.d.ts +7 -0
  20. package/dist/lib/components/StatusPage.js +71 -0
  21. package/dist/lib/components/StatusPage.js.map +1 -0
  22. package/dist/lib/components/SyntaxHighlight.d.ts +2 -1
  23. package/dist/lib/components/SyntaxHighlight.js +2 -2
  24. package/dist/lib/components/SyntaxHighlight.js.map +1 -1
  25. package/dist/lib/components/ThemeSwitch.js +4 -4
  26. package/dist/lib/components/ThemeSwitch.js.map +1 -1
  27. package/dist/lib/components/cache.d.ts +6 -0
  28. package/dist/lib/components/cache.js +13 -0
  29. package/dist/lib/components/cache.js.map +1 -0
  30. package/dist/lib/components/context/ViewportAnchorContext.js +16 -4
  31. package/dist/lib/components/context/ViewportAnchorContext.js.map +1 -1
  32. package/dist/lib/components/context/ZudokuContext.js +2 -1
  33. package/dist/lib/components/context/ZudokuContext.js.map +1 -1
  34. package/dist/lib/components/index.d.ts +9 -2
  35. package/dist/lib/components/index.js +3 -0
  36. package/dist/lib/components/index.js.map +1 -1
  37. package/dist/lib/components/navigation/SidebarCategory.js +3 -3
  38. package/dist/lib/components/navigation/SidebarCategory.js.map +1 -1
  39. package/dist/lib/core/RouteGuard.d.ts +1 -0
  40. package/dist/lib/core/RouteGuard.js +28 -0
  41. package/dist/lib/core/RouteGuard.js.map +1 -0
  42. package/dist/lib/core/ZudokuContext.d.ts +4 -2
  43. package/dist/lib/core/ZudokuContext.js +9 -7
  44. package/dist/lib/core/ZudokuContext.js.map +1 -1
  45. package/dist/lib/oas/graphql/circular.d.ts +3 -0
  46. package/dist/lib/oas/graphql/circular.js +27 -0
  47. package/dist/lib/oas/graphql/circular.js.map +1 -0
  48. package/dist/lib/oas/graphql/index.d.ts +1 -0
  49. package/dist/lib/oas/graphql/index.js +46 -29
  50. package/dist/lib/oas/graphql/index.js.map +1 -1
  51. package/dist/lib/oas/parser/dereference/index.d.ts +0 -1
  52. package/dist/lib/oas/parser/dereference/index.js +1 -1
  53. package/dist/lib/oas/parser/dereference/index.js.map +1 -1
  54. package/dist/lib/plugins/openapi/Endpoint.js +2 -2
  55. package/dist/lib/plugins/openapi/Endpoint.js.map +1 -1
  56. package/dist/lib/plugins/openapi/{Route.d.ts → OpenApiRoute.d.ts} +2 -1
  57. package/dist/lib/plugins/openapi/{Route.js → OpenApiRoute.js} +3 -4
  58. package/dist/lib/plugins/openapi/OpenApiRoute.js.map +1 -0
  59. package/dist/lib/plugins/openapi/OperationList.d.ts +4 -1
  60. package/dist/lib/plugins/openapi/OperationList.js +20 -14
  61. package/dist/lib/plugins/openapi/OperationList.js.map +1 -1
  62. package/dist/lib/plugins/openapi/OperationListItem.js +1 -1
  63. package/dist/lib/plugins/openapi/OperationListItem.js.map +1 -1
  64. package/dist/lib/plugins/openapi/ParameterListItem.js +1 -1
  65. package/dist/lib/plugins/openapi/ParameterListItem.js.map +1 -1
  66. package/dist/lib/plugins/openapi/RequestBodySidecarBox.d.ts +1 -1
  67. package/dist/lib/plugins/openapi/RequestBodySidecarBox.js +2 -0
  68. package/dist/lib/plugins/openapi/RequestBodySidecarBox.js.map +1 -1
  69. package/dist/lib/plugins/openapi/Sidecar.js +3 -3
  70. package/dist/lib/plugins/openapi/Sidecar.js.map +1 -1
  71. package/dist/lib/plugins/openapi/SidecarExamples.js +17 -14
  72. package/dist/lib/plugins/openapi/SidecarExamples.js.map +1 -1
  73. package/dist/lib/plugins/openapi/graphql/gql.d.ts +6 -2
  74. package/dist/lib/plugins/openapi/graphql/gql.js +3 -2
  75. package/dist/lib/plugins/openapi/graphql/gql.js.map +1 -1
  76. package/dist/lib/plugins/openapi/graphql/graphql.d.ts +47 -26
  77. package/dist/lib/plugins/openapi/graphql/graphql.js +20 -16
  78. package/dist/lib/plugins/openapi/graphql/graphql.js.map +1 -1
  79. package/dist/lib/plugins/openapi/index.js +101 -65
  80. package/dist/lib/plugins/openapi/index.js.map +1 -1
  81. package/dist/lib/plugins/openapi/interfaces.d.ts +8 -2
  82. package/dist/lib/plugins/openapi/playground/ExamplesDropdown.js +5 -5
  83. package/dist/lib/plugins/openapi/playground/ExamplesDropdown.js.map +1 -1
  84. package/dist/lib/plugins/openapi/playground/Headers.js +17 -16
  85. package/dist/lib/plugins/openapi/playground/Headers.js.map +1 -1
  86. package/dist/lib/plugins/openapi/playground/ParamsGrid.d.ts +5 -0
  87. package/dist/lib/plugins/openapi/playground/ParamsGrid.js +4 -0
  88. package/dist/lib/plugins/openapi/playground/ParamsGrid.js.map +1 -0
  89. package/dist/lib/plugins/openapi/playground/PathParams.js +4 -12
  90. package/dist/lib/plugins/openapi/playground/PathParams.js.map +1 -1
  91. package/dist/lib/plugins/openapi/playground/Playground.d.ts +13 -0
  92. package/dist/lib/plugins/openapi/playground/Playground.js +19 -31
  93. package/dist/lib/plugins/openapi/playground/Playground.js.map +1 -1
  94. package/dist/lib/plugins/openapi/playground/PlaygroundDialog.js +1 -1
  95. package/dist/lib/plugins/openapi/playground/PlaygroundDialog.js.map +1 -1
  96. package/dist/lib/plugins/openapi/playground/QueryParams.js +4 -3
  97. package/dist/lib/plugins/openapi/playground/QueryParams.js.map +1 -1
  98. package/dist/lib/plugins/openapi/playground/SubmitButton.d.ts +7 -0
  99. package/dist/lib/plugins/openapi/playground/SubmitButton.js +22 -0
  100. package/dist/lib/plugins/openapi/playground/SubmitButton.js.map +1 -0
  101. package/dist/lib/plugins/openapi/playground/result-panel/RequestTab.d.ts +7 -0
  102. package/dist/lib/plugins/openapi/playground/result-panel/RequestTab.js +11 -0
  103. package/dist/lib/plugins/openapi/playground/result-panel/RequestTab.js.map +1 -0
  104. package/dist/lib/plugins/openapi/playground/result-panel/ResponseTab.d.ts +8 -0
  105. package/dist/lib/plugins/openapi/playground/result-panel/ResponseTab.js +95 -0
  106. package/dist/lib/plugins/openapi/playground/result-panel/ResponseTab.js.map +1 -0
  107. package/dist/lib/plugins/openapi/playground/result-panel/ResultPanel.d.ts +7 -0
  108. package/dist/lib/plugins/openapi/playground/result-panel/ResultPanel.js +16 -0
  109. package/dist/lib/plugins/openapi/playground/result-panel/ResultPanel.js.map +1 -0
  110. package/dist/lib/plugins/openapi/playground/result-panel/convertToTypes.d.ts +10 -0
  111. package/dist/lib/plugins/openapi/playground/result-panel/convertToTypes.js +32 -0
  112. package/dist/lib/plugins/openapi/playground/result-panel/convertToTypes.js.map +1 -0
  113. package/dist/lib/plugins/openapi/playground/result-panel/convertToTypes.test.d.ts +1 -0
  114. package/dist/lib/plugins/openapi/playground/result-panel/convertToTypes.test.js +56 -0
  115. package/dist/lib/plugins/openapi/playground/result-panel/convertToTypes.test.js.map +1 -0
  116. package/dist/lib/plugins/openapi/schema/SchemaComponents.js +1 -1
  117. package/dist/lib/plugins/openapi/schema/SchemaComponents.js.map +1 -1
  118. package/dist/lib/ui/Command.js +1 -1
  119. package/dist/lib/ui/Command.js.map +1 -1
  120. package/dist/lib/ui/Select.js +2 -2
  121. package/dist/lib/ui/Select.js.map +1 -1
  122. package/dist/lib/util/MdxComponents.js +2 -2
  123. package/dist/lib/util/MdxComponents.js.map +1 -1
  124. package/dist/lib/util/joinUrl.js +1 -1
  125. package/dist/lib/util/joinUrl.js.map +1 -1
  126. package/dist/lib/util/joinUrl.test.d.ts +1 -0
  127. package/dist/lib/util/joinUrl.test.js +43 -0
  128. package/dist/lib/util/joinUrl.test.js.map +1 -0
  129. package/dist/lib/util/useScrollToAnchor.d.ts +1 -0
  130. package/dist/lib/util/useScrollToAnchor.js +26 -15
  131. package/dist/lib/util/useScrollToAnchor.js.map +1 -1
  132. package/dist/vite/plugin-api.js +15 -3
  133. package/dist/vite/plugin-api.js.map +1 -1
  134. package/dist/vite/prerender.js +1 -0
  135. package/dist/vite/prerender.js.map +1 -1
  136. package/dist/zuplo/enrich-with-zuplo.js +1 -1
  137. package/dist/zuplo/enrich-with-zuplo.js.map +1 -1
  138. package/dist/zuplo/with-zuplo.d.ts +2 -1
  139. package/dist/zuplo/with-zuplo.js +3 -1
  140. package/dist/zuplo/with-zuplo.js.map +1 -1
  141. package/lib/{AuthenticationPlugin-C9SwOxkc.js → AuthenticationPlugin-Du8cLBSr.js} +3 -3
  142. package/lib/{AuthenticationPlugin-C9SwOxkc.js.map → AuthenticationPlugin-Du8cLBSr.js.map} +1 -1
  143. package/lib/{Markdown-DFN6p0J-.js → Markdown-Cyrx_JrO.js} +1195 -1185
  144. package/lib/{Markdown-DFN6p0J-.js.map → Markdown-Cyrx_JrO.js.map} +1 -1
  145. package/lib/{MdxPage-D9c4z09Q.js → MdxPage-DewragjB.js} +6 -6
  146. package/lib/{MdxPage-D9c4z09Q.js.map → MdxPage-DewragjB.js.map} +1 -1
  147. package/lib/OpenApiRoute-UrC_t0e5.js +36 -0
  148. package/lib/OpenApiRoute-UrC_t0e5.js.map +1 -0
  149. package/lib/{OperationList-DGJWDx1G.js → OperationList-D_ejrepA.js} +1970 -1957
  150. package/lib/OperationList-D_ejrepA.js.map +1 -0
  151. package/lib/{Select-D3O7wISy.js → Select-CnCZ4WhS.js} +61 -61
  152. package/lib/Select-CnCZ4WhS.js.map +1 -0
  153. package/lib/{SlotletProvider-_3zzX_g_.js → SlotletProvider-mQiPDQIH.js} +4 -4
  154. package/lib/{SlotletProvider-_3zzX_g_.js.map → SlotletProvider-mQiPDQIH.js.map} +1 -1
  155. package/lib/{SyntaxHighlight-CJCSPG1F.js → SyntaxHighlight-B0L4SC_N.js} +309 -298
  156. package/lib/SyntaxHighlight-B0L4SC_N.js.map +1 -0
  157. package/lib/{ZudokuContext-DeQZEp-x.js → ZudokuContext-BTUJPpQl.js} +257 -246
  158. package/lib/ZudokuContext-BTUJPpQl.js.map +1 -0
  159. package/lib/{chunk-SYFQ2XB5-BF5IDYrB.js → chunk-SYFQ2XB5-BPvC-soB.js} +5 -5
  160. package/lib/{chunk-SYFQ2XB5-BF5IDYrB.js.map → chunk-SYFQ2XB5-BPvC-soB.js.map} +1 -1
  161. package/lib/circular-Dgpd6AN-.js +15397 -0
  162. package/lib/circular-Dgpd6AN-.js.map +1 -0
  163. package/lib/{createServer-BcaswoFO.js → createServer-BydbkTsd.js} +3487 -5601
  164. package/lib/createServer-BydbkTsd.js.map +1 -0
  165. package/lib/{hook-BRQEDRbn.js → hook-FT3SJLe_.js} +2 -2
  166. package/lib/{hook-BRQEDRbn.js.map → hook-FT3SJLe_.js.map} +1 -1
  167. package/lib/{index-LNp6rxyU.js → index-CjJS0l4l.js} +2 -2
  168. package/lib/{index-LNp6rxyU.js.map → index-CjJS0l4l.js.map} +1 -1
  169. package/lib/index-DGugJOLc.js +1974 -0
  170. package/lib/index-DGugJOLc.js.map +1 -0
  171. package/lib/{joinUrl-BTy9bvoK.js → joinUrl-nLx9pD-Z.js} +2 -2
  172. package/lib/joinUrl-nLx9pD-Z.js.map +1 -0
  173. package/lib/ui/Command.js +27 -27
  174. package/lib/ui/Command.js.map +1 -1
  175. package/lib/ui/Select.js +2 -2
  176. package/lib/ui/Select.js.map +1 -1
  177. package/lib/{useExposedProps-CetwhZpP.js → useExposedProps-BLKFBylA.js} +2 -2
  178. package/lib/{useExposedProps-CetwhZpP.js.map → useExposedProps-BLKFBylA.js.map} +1 -1
  179. package/lib/useScrollToAnchor-eRM9tVvD.js +289 -0
  180. package/lib/useScrollToAnchor-eRM9tVvD.js.map +1 -0
  181. package/lib/zudoku.auth-clerk.js +1 -1
  182. package/lib/zudoku.auth-openid.js +4 -4
  183. package/lib/zudoku.components.js +740 -979
  184. package/lib/zudoku.components.js.map +1 -1
  185. package/lib/zudoku.plugin-api-catalog.js +4 -4
  186. package/lib/zudoku.plugin-api-keys.js +5 -5
  187. package/lib/zudoku.plugin-custom-pages.js +2 -2
  188. package/lib/zudoku.plugin-markdown.js +1 -1
  189. package/lib/zudoku.plugin-openapi.js +6 -5
  190. package/lib/zudoku.plugin-openapi.js.map +1 -1
  191. package/lib/zudoku.plugin-redirect.js +1 -1
  192. package/package.json +2 -2
  193. package/src/app/main.tsx +26 -7
  194. package/src/lib/components/AnchorLink.tsx +5 -2
  195. package/src/lib/components/Header.tsx +1 -1
  196. package/src/lib/components/Markdown.tsx +14 -15
  197. package/src/lib/components/StatusPage.tsx +91 -0
  198. package/src/lib/components/SyntaxHighlight.tsx +14 -0
  199. package/src/lib/components/ThemeSwitch.tsx +14 -15
  200. package/src/lib/components/cache.ts +15 -0
  201. package/src/lib/components/context/ViewportAnchorContext.tsx +20 -6
  202. package/src/lib/components/context/ZudokuContext.ts +3 -1
  203. package/src/lib/components/index.ts +7 -0
  204. package/src/lib/components/navigation/SidebarCategory.tsx +3 -2
  205. package/src/lib/core/RouteGuard.tsx +35 -0
  206. package/src/lib/core/ZudokuContext.ts +9 -8
  207. package/src/lib/oas/graphql/circular.ts +29 -0
  208. package/src/lib/oas/graphql/index.ts +72 -44
  209. package/src/lib/oas/parser/dereference/index.ts +1 -2
  210. package/src/lib/plugins/openapi/Endpoint.tsx +2 -2
  211. package/src/lib/plugins/openapi/{Route.tsx → OpenApiRoute.tsx} +3 -3
  212. package/src/lib/plugins/openapi/OperationList.tsx +34 -12
  213. package/src/lib/plugins/openapi/OperationListItem.tsx +0 -2
  214. package/src/lib/plugins/openapi/ParameterListItem.tsx +1 -0
  215. package/src/lib/plugins/openapi/RequestBodySidecarBox.tsx +2 -0
  216. package/src/lib/plugins/openapi/Sidecar.tsx +4 -3
  217. package/src/lib/plugins/openapi/SidecarExamples.tsx +24 -24
  218. package/src/lib/plugins/openapi/graphql/gql.ts +12 -4
  219. package/src/lib/plugins/openapi/graphql/graphql.ts +66 -43
  220. package/src/lib/plugins/openapi/index.tsx +134 -82
  221. package/src/lib/plugins/openapi/interfaces.ts +11 -2
  222. package/src/lib/plugins/openapi/playground/ExamplesDropdown.tsx +30 -27
  223. package/src/lib/plugins/openapi/playground/Headers.tsx +65 -65
  224. package/src/lib/plugins/openapi/playground/ParamsGrid.tsx +8 -0
  225. package/src/lib/plugins/openapi/playground/PathParams.tsx +34 -74
  226. package/src/lib/plugins/openapi/playground/Playground.tsx +64 -116
  227. package/src/lib/plugins/openapi/playground/PlaygroundDialog.tsx +1 -1
  228. package/src/lib/plugins/openapi/playground/QueryParams.tsx +46 -45
  229. package/src/lib/plugins/openapi/playground/SubmitButton.tsx +75 -0
  230. package/src/lib/plugins/openapi/playground/result-panel/RequestTab.tsx +73 -0
  231. package/src/lib/plugins/openapi/playground/result-panel/ResponseTab.tsx +210 -0
  232. package/src/lib/plugins/openapi/playground/result-panel/ResultPanel.tsx +101 -0
  233. package/src/lib/plugins/openapi/playground/result-panel/convertToTypes.test.ts +64 -0
  234. package/src/lib/plugins/openapi/playground/result-panel/convertToTypes.ts +36 -0
  235. package/src/lib/plugins/openapi/schema/SchemaComponents.tsx +1 -1
  236. package/src/lib/ui/Command.tsx +1 -1
  237. package/src/lib/ui/Select.tsx +1 -1
  238. package/src/lib/util/MdxComponents.tsx +2 -1
  239. package/src/lib/util/joinUrl.test.ts +62 -0
  240. package/src/lib/util/joinUrl.ts +1 -1
  241. package/src/lib/util/useScrollToAnchor.ts +32 -15
  242. package/dist/lib/plugins/openapi/Route.js.map +0 -1
  243. package/dist/lib/plugins/openapi/playground/ResponseTab.d.ts +0 -4
  244. package/dist/lib/plugins/openapi/playground/ResponseTab.js +0 -42
  245. package/dist/lib/plugins/openapi/playground/ResponseTab.js.map +0 -1
  246. package/lib/AnchorLink-bObQitZv.js +0 -34
  247. package/lib/AnchorLink-bObQitZv.js.map +0 -1
  248. package/lib/OperationList-DGJWDx1G.js.map +0 -1
  249. package/lib/Route-VdmEyOD0.js +0 -35
  250. package/lib/Route-VdmEyOD0.js.map +0 -1
  251. package/lib/Select-D3O7wISy.js.map +0 -1
  252. package/lib/StaggeredRender-DgsamH_G.js +0 -17
  253. package/lib/StaggeredRender-DgsamH_G.js.map +0 -1
  254. package/lib/SyntaxHighlight-CJCSPG1F.js.map +0 -1
  255. package/lib/ZudokuContext-DeQZEp-x.js.map +0 -1
  256. package/lib/createServer-BcaswoFO.js.map +0 -1
  257. package/lib/index-Bn6Lc9tq.js +0 -9
  258. package/lib/index-Bn6Lc9tq.js.map +0 -1
  259. package/lib/index-CXRrqOIl.js +0 -1750
  260. package/lib/index-CXRrqOIl.js.map +0 -1
  261. package/lib/index-TaRXY2w1.js +0 -43
  262. package/lib/index-TaRXY2w1.js.map +0 -1
  263. package/lib/joinUrl-BTy9bvoK.js.map +0 -1
  264. package/src/lib/plugins/openapi/playground/ResponseTab.tsx +0 -76
@@ -9,6 +9,7 @@ import { Checkbox } from "zudoku/ui/Checkbox.js";
9
9
  import { Autocomplete } from "../../../components/Autocomplete.js";
10
10
  import { Input } from "../../../ui/Input.js";
11
11
  import { InlineInput } from "./InlineInput.js";
12
+ import ParamsGrid from "./ParamsGrid.js";
12
13
  import { type PlaygroundForm, type QueryParam } from "./Playground.js";
13
14
 
14
15
  export const QueryParams = ({
@@ -34,51 +35,51 @@ export const QueryParams = ({
34
35
  (param) => param.name === field.name,
35
36
  );
36
37
  return (
37
- <div
38
- key={field.id}
39
- className="hover:bg-accent/40 grid grid-cols-[min-content_1fr_1fr] gap-2 items-center px-3"
40
- >
41
- <Controller
42
- control={control}
43
- name={`queryParams.${i}.active`}
44
- render={({ field }) => (
45
- <Checkbox
46
- variant="outline"
47
- id={`queryParams.${i}.active`}
48
- className="mr-2"
49
- checked={field.value}
50
- onCheckedChange={field.onChange}
51
- />
52
- )}
53
- />
54
- <Controller
55
- control={control}
56
- render={({ field }) =>
57
- !requiredFields[i] ? (
58
- <Autocomplete
59
- value={field.value}
60
- options={queryParams.map((param) => param.name)}
61
- onChange={(e) => {
62
- field.onChange(e);
63
- }}
64
- className="border-0 font-mono text-xs bg-transparent hover:bg-transparent"
38
+ <ParamsGrid key={field.id}>
39
+ <div className="flex items-center">
40
+ <Controller
41
+ control={control}
42
+ name={`queryParams.${i}.active`}
43
+ render={({ field }) => (
44
+ <Checkbox
45
+ variant="outline"
46
+ id={`queryParams.${i}.active`}
47
+ className="mr-2"
48
+ checked={field.value}
49
+ onCheckedChange={field.onChange}
65
50
  />
66
- ) : (
67
- <InlineInput asChild>
68
- <label
69
- className="flex items-center cursor-pointer gap-1"
70
- htmlFor={`queryParams.${i}.active`}
71
- title={requiredFields[i] ? "Required field" : undefined}
72
- >
73
- {field.value}
74
- {requiredFields[i] && <sup>&nbsp;*</sup>}
75
- </label>
76
- </InlineInput>
77
- )
78
- }
79
- name={`queryParams.${i}.name`}
80
- />
81
-
51
+ )}
52
+ />
53
+ <Controller
54
+ control={control}
55
+ render={({ field }) =>
56
+ !requiredFields[i] ? (
57
+ <Autocomplete
58
+ value={field.value}
59
+ options={queryParams.map((param) => param.name)}
60
+ onChange={(e) => {
61
+ field.onChange(e);
62
+ }}
63
+ className="border-0 font-mono text-xs bg-transparent hover:bg-transparent"
64
+ />
65
+ ) : (
66
+ <InlineInput asChild>
67
+ <label
68
+ className="flex items-center cursor-pointer gap-1"
69
+ htmlFor={`queryParams.${i}.active`}
70
+ title={
71
+ requiredFields[i] ? "Required field" : undefined
72
+ }
73
+ >
74
+ {field.value}
75
+ {requiredFields[i] && <sup>&nbsp;*</sup>}
76
+ </label>
77
+ </InlineInput>
78
+ )
79
+ }
80
+ name={`queryParams.${i}.name`}
81
+ />
82
+ </div>
82
83
  <div className="flex justify-between items-center">
83
84
  <Controller
84
85
  control={control}
@@ -117,7 +118,7 @@ export const QueryParams = ({
117
118
  name={`queryParams.${i}.value`}
118
119
  />
119
120
  </div>
120
- </div>
121
+ </ParamsGrid>
121
122
  );
122
123
  })}
123
124
  </div>
@@ -0,0 +1,75 @@
1
+ import { ChevronDownIcon } from "lucide-react";
2
+ import { useState } from "react";
3
+ import { useFormContext } from "react-hook-form";
4
+ import { Button } from "zudoku/ui/Button.js";
5
+ import {
6
+ DropdownMenu,
7
+ DropdownMenuContent,
8
+ DropdownMenuItem,
9
+ DropdownMenuTrigger,
10
+ } from "zudoku/ui/DropdownMenu.js";
11
+ import { RadioGroup, RadioGroupItem } from "zudoku/ui/RadioGroup.js";
12
+ import { ApiIdentity } from "../../../core/ZudokuContext.js";
13
+ import { NO_IDENTITY } from "./Playground.js";
14
+
15
+ const SubmitButton = ({
16
+ identities,
17
+ formRef,
18
+ disabled,
19
+ }: {
20
+ identities: ApiIdentity[];
21
+ formRef?: React.RefObject<HTMLFormElement | null>;
22
+ disabled?: boolean;
23
+ }) => {
24
+ const { setValue } = useFormContext();
25
+ const [dropdownValue, setDropdownValue] = useState<string | undefined>();
26
+ if (identities.length === 0) {
27
+ return <Button disabled={disabled}>Send</Button>;
28
+ }
29
+ return (
30
+ <div className="flex">
31
+ <Button
32
+ className="rounded-r-none inset-shadow-sm"
33
+ disabled={disabled}
34
+ onClick={() => formRef?.current?.requestSubmit()}
35
+ >
36
+ Send
37
+ </Button>
38
+ <DropdownMenu>
39
+ <DropdownMenuTrigger asChild>
40
+ <Button
41
+ disabled={disabled}
42
+ className="rounded-l-none border-l border-border/40 inset-shadow-sm w-6"
43
+ size="icon"
44
+ >
45
+ <ChevronDownIcon className="w-4 h-4" />
46
+ </Button>
47
+ </DropdownMenuTrigger>
48
+ <RadioGroup value={dropdownValue}>
49
+ <DropdownMenuContent className="w-56" align="end" alignOffset={-150}>
50
+ {[{ id: NO_IDENTITY, label: "None" }, ...identities].map(
51
+ (identity) => (
52
+ <DropdownMenuItem
53
+ key={identity.id}
54
+ onClick={() => {
55
+ setDropdownValue(identity.id);
56
+ setValue("identity", identity.id);
57
+ formRef?.current?.requestSubmit();
58
+ }}
59
+ onMouseEnter={() => setDropdownValue(identity.id)}
60
+ onMouseLeave={() => setDropdownValue(undefined)}
61
+ >
62
+ <RadioGroupItem value={identity.id} className="mr-2" />
63
+
64
+ {identity.label}
65
+ </DropdownMenuItem>
66
+ ),
67
+ )}
68
+ </DropdownMenuContent>
69
+ </RadioGroup>
70
+ </DropdownMenu>
71
+ </div>
72
+ );
73
+ };
74
+
75
+ export default SubmitButton;
@@ -0,0 +1,73 @@
1
+ import { ChevronRightIcon } from "lucide-react";
2
+ import { Fragment } from "react";
3
+ import {
4
+ Collapsible,
5
+ CollapsibleContent,
6
+ CollapsibleTrigger,
7
+ } from "../../../../ui/Collapsible.js";
8
+ import { cn } from "../../../../util/cn.js";
9
+ import { methodForColor } from "../../util/methodToColor.js";
10
+
11
+ export const RequestTab = ({
12
+ method,
13
+ url,
14
+ headers,
15
+ body,
16
+ }: {
17
+ method: string;
18
+ url: string;
19
+ headers: Array<[string, string]>;
20
+ body?: string;
21
+ }) => {
22
+ return (
23
+ <div className="flex flex-col gap-2 font-mono text-xs">
24
+ <div className="gap-2 p-2 bg-muted rounded-md">
25
+ <span className={cn(methodForColor(method), "font-semibold")}>
26
+ {method}
27
+ </span>
28
+ &nbsp;
29
+ <span className="break-all">{url}</span>&nbsp;
30
+ <span className="text-muted-foreground">HTTP/1.1</span>
31
+ </div>
32
+ <div className="mx-1.5 flex flex-col gap-3">
33
+ <Collapsible defaultOpen>
34
+ <CollapsibleTrigger className="flex items-center gap-2 hover:text-primary group">
35
+ <ChevronRightIcon className="h-4 w-4 transition-transform duration-200 group-data-[state=open]:rotate-[90deg]" />
36
+ <span className="font-semibold">Headers</span>
37
+ </CollapsibleTrigger>
38
+ <CollapsibleContent>
39
+ <div className="grid grid-cols-[auto,1fr] gap-x-8 gap-y-1 pl-1.5 pt-2">
40
+ {headers.map(([key, value]) => (
41
+ <Fragment key={key}>
42
+ <div className="text-primary">{key}</div>
43
+ <div className="break-words">{value}</div>
44
+ </Fragment>
45
+ ))}
46
+ </div>
47
+ </CollapsibleContent>
48
+ </Collapsible>
49
+
50
+ <Collapsible defaultOpen>
51
+ <CollapsibleTrigger className="flex items-center gap-2 hover:text-primary group">
52
+ <ChevronRightIcon className="h-4 w-4 transition-transform duration-200 group-data-[state=open]:rotate-[90deg]" />
53
+ <span className="font-semibold">Body</span>
54
+ </CollapsibleTrigger>
55
+ <CollapsibleContent>
56
+ <div className="pl-0 pt-2">
57
+ <div
58
+ className={cn(
59
+ "whitespace-pre-wrap break-all bg-muted p-2 rounded-md",
60
+ !body && "text-muted-foreground",
61
+ )}
62
+ >
63
+ {body ?? "Empty body"}
64
+ </div>
65
+ </div>
66
+ </CollapsibleContent>
67
+ </Collapsible>
68
+ </div>
69
+ </div>
70
+ );
71
+ };
72
+
73
+ export default RequestTab;
@@ -0,0 +1,210 @@
1
+ import { useQuery } from "@tanstack/react-query";
2
+ import { ChevronRightIcon } from "lucide-react";
3
+ import { Fragment, useState } from "react";
4
+ import {
5
+ Collapsible,
6
+ CollapsibleContent,
7
+ CollapsibleTrigger,
8
+ } from "zudoku/ui/Collapsible.js";
9
+ import {
10
+ Select,
11
+ SelectContent,
12
+ SelectItem,
13
+ SelectTrigger,
14
+ SelectValue,
15
+ } from "zudoku/ui/Select.js";
16
+ import { SyntaxHighlight } from "../../../../components/SyntaxHighlight.js";
17
+ import { Card } from "../../../../ui/Card.js";
18
+ import { convertToTypes } from "./convertToTypes.js";
19
+
20
+ const statusCodeMap: Record<number, string> = {
21
+ 200: "OK",
22
+ 201: "Created",
23
+ 202: "Accepted",
24
+ 204: "No Content",
25
+ 400: "Bad Request",
26
+ 401: "Unauthorized",
27
+ 403: "Forbidden",
28
+ 404: "Not Found",
29
+ 405: "Method Not Allowed",
30
+ 500: "Internal Server Error",
31
+ };
32
+
33
+ const mimeTypeToLanguage = (mimeType: string) => {
34
+ const mimeTypeMapping = {
35
+ "application/json": "json",
36
+ "text/json": "json",
37
+ "text/html": "html",
38
+ "text/css": "css",
39
+ "text/javascript": "javascript",
40
+ "application/xml": "xml",
41
+ "application/xhtml+xml": "xhtml",
42
+ };
43
+
44
+ return Object.entries(mimeTypeMapping).find(([mime]) =>
45
+ mimeType.includes(mime),
46
+ )?.[1];
47
+ };
48
+
49
+ const detectLanguage = (headers: Array<[string, string]>) => {
50
+ const contentType =
51
+ headers.find(([key, value]) => key === "Content-Type")?.[1] || "";
52
+ return mimeTypeToLanguage(contentType);
53
+ };
54
+
55
+ const tryParseJson = (body: string) => {
56
+ try {
57
+ return JSON.stringify(JSON.parse(body), null, 2);
58
+ } catch {
59
+ return null;
60
+ }
61
+ };
62
+
63
+ const sortHeadersByRelevance = (
64
+ headers: Array<[string, string]>,
65
+ ): Array<[string, string]> => {
66
+ const priorityOrder = [
67
+ "Content-Type",
68
+ "Content-Length",
69
+ "Authorization",
70
+ "X-RateLimit-Remaining",
71
+ "X-RateLimit-Limit",
72
+ "Cache-Control",
73
+ "ETag",
74
+ ].map((key) => key.toLowerCase());
75
+
76
+ return [...headers].sort(([keyA], [keyB]) => {
77
+ const indexA = priorityOrder.indexOf(keyA.toLowerCase());
78
+ const indexB = priorityOrder.indexOf(keyB.toLowerCase());
79
+ if (indexA === indexB) return 0;
80
+ if (indexA === -1) return 1;
81
+ if (indexB === -1) return -1;
82
+ return indexA - indexB;
83
+ });
84
+ };
85
+
86
+ export const ResponseTab = ({
87
+ body = "",
88
+ headers,
89
+ status,
90
+ time,
91
+ size,
92
+ url,
93
+ }: {
94
+ body?: string;
95
+ headers: Array<[string, string]>;
96
+ status: number;
97
+ time: number;
98
+ size: number;
99
+ url: string;
100
+ }) => {
101
+ const detectedLanguage = detectLanguage(headers);
102
+ const jsonContent = tryParseJson(body);
103
+ const beautifiedBody = jsonContent || body;
104
+ const [view, setView] = useState<"formatted" | "raw" | "types">(
105
+ jsonContent ? "formatted" : "raw",
106
+ );
107
+
108
+ const types = useQuery({
109
+ queryKey: ["types", beautifiedBody],
110
+ queryFn: async () => {
111
+ return convertToTypes(JSON.parse(beautifiedBody));
112
+ },
113
+ enabled: view === "types",
114
+ });
115
+
116
+ const sortedHeaders = sortHeadersByRelevance([...headers]);
117
+
118
+ return (
119
+ <div className="flex flex-col gap-2 h-full overflow-y-scroll max-h-[calc(100vh-220px)] py-4">
120
+ <Collapsible defaultOpen>
121
+ <CollapsibleTrigger className="flex items-center gap-2 hover:text-primary group">
122
+ <ChevronRightIcon className="h-4 w-4 transition-transform duration-200 group-data-[state=open]:rotate-[90deg]" />
123
+ <span className="font-semibold">Headers</span>
124
+ </CollapsibleTrigger>
125
+ <CollapsibleContent>
126
+ <div className="grid grid-cols-[auto,1fr] gap-x-8 gap-y-1 pl-1.5 pt-2 font-mono text-xs">
127
+ {sortedHeaders.slice(0, 5).map(([key, value]) => (
128
+ <Fragment key={key}>
129
+ <div className="text-primary whitespace-pre">{key}</div>
130
+ <div className="break-all">{value}</div>
131
+ </Fragment>
132
+ ))}
133
+ {sortedHeaders.length > 5 && (
134
+ <Collapsible className="col-span-full grid-cols-subgrid grid">
135
+ <CollapsibleTrigger className="col-span-2 text-xs text-muted-foreground hover:text-primary flex items-center gap-1 py-1">
136
+ <ChevronRightIcon className="h-3 w-3 transition-transform duration-200 group-data-[state=open]:rotate-[90deg]" />
137
+ Show {sortedHeaders.length - 5} more headers
138
+ </CollapsibleTrigger>
139
+ <CollapsibleContent className="col-span-full grid grid-cols-subgrid gap-x-8 gap-y-1 ">
140
+ {sortedHeaders.slice(5).map(([key, value]) => (
141
+ <Fragment key={key}>
142
+ <div className="text-primary whitespace-pre">{key}</div>
143
+ <div className="break-all">{value}</div>
144
+ </Fragment>
145
+ ))}
146
+ </CollapsibleContent>
147
+ </Collapsible>
148
+ )}
149
+ </div>
150
+ </CollapsibleContent>
151
+ </Collapsible>
152
+
153
+ <Card className="shadow-none">
154
+ <SyntaxHighlight
155
+ language={
156
+ view === "types"
157
+ ? "typescript"
158
+ : view === "raw"
159
+ ? jsonContent
160
+ ? "plain"
161
+ : detectedLanguage
162
+ : "json"
163
+ }
164
+ noBackground
165
+ // playground dialog has h-5/6 ≈ 83.333vh
166
+ className="overflow-x-auto p-4 text-xs max-h-[calc(83.333vh-180px)]"
167
+ code={
168
+ (view === "raw"
169
+ ? body
170
+ : view === "types"
171
+ ? types.data?.lines.join("\n")
172
+ : beautifiedBody) ?? ""
173
+ }
174
+ />
175
+ </Card>
176
+ <div className="flex gap-2 justify-between">
177
+ <div className="flex text-xs gap-2 border bg-muted rounded-md p-2 items-center h-8 font-mono divide-x">
178
+ <div>
179
+ <span className="text-muted-foreground">Status</span> {status}{" "}
180
+ {statusCodeMap[status] ?? ""}
181
+ </div>
182
+ <div>
183
+ <span className="text-muted-foreground">Time</span>{" "}
184
+ {time.toFixed(0)}ms
185
+ </div>
186
+ <div>
187
+ <span className="text-muted-foreground">Size</span> {size}B
188
+ </div>
189
+ </div>
190
+ {jsonContent && (
191
+ <div>
192
+ <Select
193
+ value={view}
194
+ onValueChange={(value) => setView(value as "formatted" | "raw")}
195
+ >
196
+ <SelectTrigger className="min-w-32">
197
+ <SelectValue placeholder="View" />
198
+ </SelectTrigger>
199
+ <SelectContent>
200
+ <SelectItem value="formatted">Formatted</SelectItem>
201
+ <SelectItem value="raw">Raw</SelectItem>
202
+ <SelectItem value="types">Types</SelectItem>
203
+ </SelectContent>
204
+ </Select>
205
+ </div>
206
+ )}
207
+ </div>
208
+ </div>
209
+ );
210
+ };
@@ -0,0 +1,101 @@
1
+ import { UseMutationResult } from "@tanstack/react-query";
2
+ import { Spinner } from "../../../../components/Spinner.js";
3
+ import { Callout } from "../../../../ui/Callout.js";
4
+ import {
5
+ Card,
6
+ CardContent,
7
+ CardHeader,
8
+ CardTitle,
9
+ } from "../../../../ui/Card.js";
10
+ import {
11
+ Tabs,
12
+ TabsContent,
13
+ TabsList,
14
+ TabsTrigger,
15
+ } from "../../../../ui/Tabs.js";
16
+ import { cn } from "../../../../util/cn.js";
17
+ import { PlaygroundResult } from "../Playground.js";
18
+ import { RequestTab } from "./RequestTab.js";
19
+ import { ResponseTab } from "./ResponseTab.js";
20
+
21
+ export const ResultPanel = ({
22
+ queryMutation,
23
+ showPathParamsWarning,
24
+ }: {
25
+ queryMutation: UseMutationResult<PlaygroundResult, Error, any, unknown>;
26
+ showPathParamsWarning: boolean;
27
+ }) => {
28
+ const status = ((queryMutation.data?.status ?? 0) / 100).toFixed(0);
29
+ return (
30
+ <div className="min-w-0 p-8 bg-muted/70 overflow-y-auto">
31
+ {queryMutation.error ? (
32
+ <div className="flex flex-col gap-2">
33
+ {showPathParamsWarning && (
34
+ <Callout type="caution">
35
+ Some path parameters are missing values. Please fill them in to
36
+ ensure the request is sent correctly.
37
+ </Callout>
38
+ )}
39
+ <Card>
40
+ <CardHeader>
41
+ <CardTitle>Request failed</CardTitle>
42
+ </CardHeader>
43
+ <CardContent>
44
+ Error:{" "}
45
+ {queryMutation.error.message ||
46
+ String(queryMutation.error) ||
47
+ "Unexpected error"}
48
+ </CardContent>
49
+ </Card>
50
+ </div>
51
+ ) : queryMutation.data ? (
52
+ <div className="flex flex-col gap-2">
53
+ <Tabs defaultValue="response">
54
+ <TabsList>
55
+ <TabsTrigger value="request">Request</TabsTrigger>
56
+ <TabsTrigger value="response">
57
+ Response
58
+ <span
59
+ className={cn(
60
+ "text-xs font-mono ml-1",
61
+ status === "2" && "text-green-500",
62
+ status === "3" && "text-blue-500",
63
+ status === "4" && "text-yellow-500",
64
+ status === "5" && "text-red-500",
65
+ )}
66
+ >
67
+ ({queryMutation.data.status})
68
+ </span>
69
+ </TabsTrigger>
70
+ </TabsList>
71
+ <TabsContent value="request">
72
+ <RequestTab {...queryMutation.data.request} />
73
+ </TabsContent>
74
+ <TabsContent value="response">
75
+ <ResponseTab
76
+ status={queryMutation.data.status}
77
+ time={queryMutation.data.time}
78
+ size={queryMutation.data.size}
79
+ headers={queryMutation.data.headers}
80
+ body={queryMutation.data.body}
81
+ url={queryMutation.data.request.url}
82
+ />
83
+ </TabsContent>
84
+ </Tabs>
85
+ </div>
86
+ ) : (
87
+ <div className="grid place-items-center h-full">
88
+ <span className="text-[16px] font-semibold text-muted-foreground">
89
+ {queryMutation.isPending ? (
90
+ <Spinner />
91
+ ) : (
92
+ "Send a request first to see the response here"
93
+ )}
94
+ </span>
95
+ </div>
96
+ )}
97
+ </div>
98
+ );
99
+ };
100
+
101
+ export default ResultPanel;
@@ -0,0 +1,64 @@
1
+ import { describe, expect, it } from "vitest";
2
+ import { generateInterface } from "./convertToTypes.js";
3
+
4
+ describe("generateInterface", () => {
5
+ it("should handle primitive types", () => {
6
+ const input = {
7
+ string: "hello",
8
+ number: 42,
9
+ boolean: true,
10
+ null: null,
11
+ };
12
+
13
+ const expected = [
14
+ "{",
15
+ " string: string;",
16
+ " number: number;",
17
+ " boolean: boolean;",
18
+ " null: null;",
19
+ "}",
20
+ ].join("\n");
21
+
22
+ expect(generateInterface(input)).toBe(expected);
23
+ });
24
+
25
+ it("should handle nested objects", () => {
26
+ const input = {
27
+ user: {
28
+ name: "John",
29
+ age: 30,
30
+ },
31
+ };
32
+
33
+ const expected = [
34
+ "{",
35
+ " user: {",
36
+ " name: string;",
37
+ " age: number;",
38
+ "};",
39
+ "}",
40
+ ].join("\n");
41
+
42
+ expect(generateInterface(input)).toBe(expected);
43
+ });
44
+
45
+ it("should handle arrays", () => {
46
+ const input = {
47
+ numbers: [1, 2, 3],
48
+ empty: [],
49
+ objects: [{ id: 1 }],
50
+ };
51
+
52
+ const expected = [
53
+ "{",
54
+ " numbers: number[];",
55
+ " empty: any[];",
56
+ " objects: {",
57
+ " id: number;",
58
+ "}[];",
59
+ "}",
60
+ ].join("\n");
61
+
62
+ expect(generateInterface(input)).toBe(expected);
63
+ });
64
+ });
@@ -0,0 +1,36 @@
1
+ type JsonValue = string | number | boolean | null | JsonObject | JsonArray;
2
+ type JsonObject = { [key: string]: JsonValue };
3
+ type JsonArray = JsonValue[];
4
+
5
+ function inferType(value: JsonValue): string {
6
+ if (value === null) return "null";
7
+ if (Array.isArray(value)) {
8
+ if (value.length === 0) return "any[]";
9
+ const firstValue = value[0];
10
+ if (firstValue === undefined) return "any[]";
11
+ const elementType = inferType(firstValue);
12
+ return `${elementType}[]`;
13
+ }
14
+ if (typeof value === "object") {
15
+ return generateInterface(value);
16
+ }
17
+ return typeof value;
18
+ }
19
+
20
+ export function generateInterface(obj: JsonObject, indentation = ""): string {
21
+ const lines: string[] = ["{"];
22
+
23
+ for (const [key, value] of Object.entries(obj)) {
24
+ const propertyType = inferType(value);
25
+ lines.push(` ${key}: ${propertyType};`);
26
+ }
27
+
28
+ lines.push("}");
29
+ return lines.join("\n");
30
+ }
31
+
32
+ export function convertToTypes(json: JsonValue): { lines: string[] } {
33
+ const typeDefinition = inferType(json);
34
+ const lines = [`type GeneratedType = ${typeDefinition};`];
35
+ return { lines };
36
+ }
@@ -3,7 +3,7 @@ import { ListPlusIcon, RefreshCcwDotIcon } from "lucide-react";
3
3
  import { useCallback, useState } from "react";
4
4
  import { Badge } from "zudoku/ui/Badge.js";
5
5
  import { Markdown, ProseClasses } from "../../../components/Markdown.js";
6
- import { CIRCULAR_REF } from "../../../oas/parser/dereference/index.js";
6
+ import { CIRCULAR_REF } from "../../../oas/graphql/circular.js";
7
7
  import type { SchemaObject } from "../../../oas/parser/index.js";
8
8
  import { Button } from "../../../ui/Button.js";
9
9
  import { cn } from "../../../util/cn.js";