@scalar/workspace-store 0.17.1 → 0.18.1

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 (202) hide show
  1. package/CHANGELOG.md +53 -0
  2. package/dist/client.d.ts +31 -19
  3. package/dist/client.d.ts.map +1 -1
  4. package/dist/client.js +267 -88
  5. package/dist/client.js.map +3 -3
  6. package/dist/events/bus.d.ts +96 -0
  7. package/dist/events/bus.d.ts.map +1 -0
  8. package/dist/events/bus.js +60 -0
  9. package/dist/events/bus.js.map +7 -0
  10. package/dist/events/definitions/analytics.d.ts +27 -0
  11. package/dist/events/definitions/analytics.d.ts.map +1 -0
  12. package/dist/events/definitions/analytics.js +1 -0
  13. package/dist/events/definitions/analytics.js.map +7 -0
  14. package/dist/events/definitions/auth.d.ts +30 -0
  15. package/dist/events/definitions/auth.d.ts.map +1 -0
  16. package/dist/events/definitions/auth.js +1 -0
  17. package/dist/events/definitions/auth.js.map +7 -0
  18. package/dist/events/definitions/common.d.ts +5 -0
  19. package/dist/events/definitions/common.d.ts.map +1 -0
  20. package/dist/events/definitions/common.js +1 -0
  21. package/dist/events/definitions/common.js.map +7 -0
  22. package/dist/events/definitions/document.d.ts +14 -0
  23. package/dist/events/definitions/document.d.ts.map +1 -0
  24. package/dist/events/definitions/document.js +1 -0
  25. package/dist/events/definitions/document.js.map +7 -0
  26. package/dist/events/definitions/environment.d.ts +37 -0
  27. package/dist/events/definitions/environment.d.ts.map +1 -0
  28. package/dist/events/definitions/environment.js +1 -0
  29. package/dist/events/definitions/environment.js.map +7 -0
  30. package/dist/events/definitions/index.d.ts +12 -0
  31. package/dist/events/definitions/index.d.ts.map +1 -0
  32. package/dist/events/definitions/index.js +1 -0
  33. package/dist/events/definitions/index.js.map +7 -0
  34. package/dist/events/definitions/meta.d.ts +17 -0
  35. package/dist/events/definitions/meta.d.ts.map +1 -0
  36. package/dist/events/definitions/meta.js +1 -0
  37. package/dist/events/definitions/meta.js.map +7 -0
  38. package/dist/events/definitions/operation.d.ts +11 -0
  39. package/dist/events/definitions/operation.d.ts.map +1 -0
  40. package/dist/events/definitions/operation.js +1 -0
  41. package/dist/events/definitions/operation.js.map +7 -0
  42. package/dist/events/definitions/server.d.ts +50 -0
  43. package/dist/events/definitions/server.d.ts.map +1 -0
  44. package/dist/events/definitions/server.js +1 -0
  45. package/dist/events/definitions/server.js.map +7 -0
  46. package/dist/events/definitions/ui.d.ts +52 -0
  47. package/dist/events/definitions/ui.d.ts.map +1 -0
  48. package/dist/events/definitions/ui.js +1 -0
  49. package/dist/events/definitions/ui.js.map +7 -0
  50. package/dist/events/index.d.ts +3 -1
  51. package/dist/events/index.d.ts.map +1 -1
  52. package/dist/events/index.js +3 -1
  53. package/dist/events/index.js.map +2 -2
  54. package/dist/events/listeners.d.ts +2 -1
  55. package/dist/events/listeners.d.ts.map +1 -1
  56. package/dist/events/listeners.js.map +2 -2
  57. package/dist/events/{definitions.d.ts → old-definitions.d.ts} +35 -11
  58. package/dist/events/old-definitions.d.ts.map +1 -0
  59. package/dist/events/{definitions.js → old-definitions.js} +1 -1
  60. package/dist/events/old-definitions.js.map +7 -0
  61. package/dist/helpers/debounce.d.ts +28 -0
  62. package/dist/helpers/debounce.d.ts.map +1 -0
  63. package/dist/helpers/debounce.js +31 -0
  64. package/dist/helpers/debounce.js.map +7 -0
  65. package/dist/helpers/detect-changes-proxy.d.ts +47 -0
  66. package/dist/helpers/detect-changes-proxy.d.ts.map +1 -0
  67. package/dist/helpers/detect-changes-proxy.js +59 -0
  68. package/dist/helpers/detect-changes-proxy.js.map +7 -0
  69. package/dist/helpers/overrides-proxy.d.ts +17 -6
  70. package/dist/helpers/overrides-proxy.d.ts.map +1 -1
  71. package/dist/helpers/overrides-proxy.js +33 -18
  72. package/dist/helpers/overrides-proxy.js.map +3 -3
  73. package/dist/helpers/unpack-proxy.d.ts +6 -0
  74. package/dist/helpers/unpack-proxy.d.ts.map +1 -0
  75. package/dist/helpers/unpack-proxy.js +9 -0
  76. package/dist/helpers/unpack-proxy.js.map +7 -0
  77. package/dist/mutators/environment.d.ts +21 -13
  78. package/dist/mutators/environment.d.ts.map +1 -1
  79. package/dist/mutators/environment.js +46 -25
  80. package/dist/mutators/environment.js.map +2 -2
  81. package/dist/mutators/index.d.ts +4 -12
  82. package/dist/mutators/index.d.ts.map +1 -1
  83. package/dist/mutators/index.js +7 -4
  84. package/dist/mutators/index.js.map +2 -2
  85. package/dist/navigation/get-navigation-options.d.ts +1 -1
  86. package/dist/navigation/get-navigation-options.d.ts.map +1 -1
  87. package/dist/navigation/get-navigation-options.js +66 -54
  88. package/dist/navigation/get-navigation-options.js.map +2 -2
  89. package/dist/navigation/helpers/get-tag.d.ts +7 -2
  90. package/dist/navigation/helpers/get-tag.d.ts.map +1 -1
  91. package/dist/navigation/helpers/get-tag.js +16 -2
  92. package/dist/navigation/helpers/get-tag.js.map +2 -2
  93. package/dist/navigation/helpers/traverse-description.d.ts +7 -2
  94. package/dist/navigation/helpers/traverse-description.d.ts.map +1 -1
  95. package/dist/navigation/helpers/traverse-description.js +24 -6
  96. package/dist/navigation/helpers/traverse-description.js.map +2 -2
  97. package/dist/navigation/helpers/traverse-document.d.ts +5 -2
  98. package/dist/navigation/helpers/traverse-document.d.ts.map +1 -1
  99. package/dist/navigation/helpers/traverse-document.js +53 -15
  100. package/dist/navigation/helpers/traverse-document.js.map +2 -2
  101. package/dist/navigation/helpers/traverse-paths.d.ts +9 -3
  102. package/dist/navigation/helpers/traverse-paths.d.ts.map +1 -1
  103. package/dist/navigation/helpers/traverse-paths.js +63 -9
  104. package/dist/navigation/helpers/traverse-paths.js.map +2 -2
  105. package/dist/navigation/helpers/traverse-schemas.d.ts +7 -8
  106. package/dist/navigation/helpers/traverse-schemas.d.ts.map +1 -1
  107. package/dist/navigation/helpers/traverse-schemas.js +35 -7
  108. package/dist/navigation/helpers/traverse-schemas.js.map +2 -2
  109. package/dist/navigation/helpers/traverse-tags.d.ts +8 -4
  110. package/dist/navigation/helpers/traverse-tags.d.ts.map +1 -1
  111. package/dist/navigation/helpers/traverse-tags.js +65 -17
  112. package/dist/navigation/helpers/traverse-tags.js.map +2 -2
  113. package/dist/navigation/helpers/traverse-webhooks.d.ts +9 -3
  114. package/dist/navigation/helpers/traverse-webhooks.d.ts.map +1 -1
  115. package/dist/navigation/helpers/traverse-webhooks.js +43 -17
  116. package/dist/navigation/helpers/traverse-webhooks.js.map +2 -2
  117. package/dist/navigation/types.d.ts +5 -24
  118. package/dist/navigation/types.d.ts.map +1 -1
  119. package/dist/persistence/index.d.ts +86 -0
  120. package/dist/persistence/index.d.ts.map +1 -0
  121. package/dist/persistence/index.js +196 -0
  122. package/dist/persistence/index.js.map +7 -0
  123. package/dist/persistence/indexdb.d.ts +87 -0
  124. package/dist/persistence/indexdb.d.ts.map +1 -0
  125. package/dist/persistence/indexdb.js +125 -0
  126. package/dist/persistence/indexdb.js.map +7 -0
  127. package/dist/{plugins.d.ts → plugins/bundler/index.d.ts} +1 -1
  128. package/dist/plugins/bundler/index.d.ts.map +1 -0
  129. package/dist/{plugins.js → plugins/bundler/index.js} +2 -2
  130. package/dist/{plugins.js.map → plugins/bundler/index.js.map} +1 -1
  131. package/dist/plugins/client/index.d.ts +3 -0
  132. package/dist/plugins/client/index.d.ts.map +1 -0
  133. package/dist/plugins/client/index.js +5 -0
  134. package/dist/plugins/client/index.js.map +7 -0
  135. package/dist/plugins/client/persistence.d.ts +13 -0
  136. package/dist/plugins/client/persistence.d.ts.map +1 -0
  137. package/dist/plugins/client/persistence.js +57 -0
  138. package/dist/plugins/client/persistence.js.map +7 -0
  139. package/dist/schemas/extensions/document/x-scalar-environments.d.ts +52 -0
  140. package/dist/schemas/extensions/document/x-scalar-environments.d.ts.map +1 -0
  141. package/dist/schemas/extensions/document/x-scalar-environments.js +25 -0
  142. package/dist/schemas/extensions/document/x-scalar-environments.js.map +7 -0
  143. package/dist/schemas/extensions/workspace/x-scalar-active-environment.d.ts +8 -0
  144. package/dist/schemas/extensions/workspace/x-scalar-active-environment.d.ts.map +1 -0
  145. package/dist/schemas/extensions/workspace/x-scalar-active-environment.js +8 -0
  146. package/dist/schemas/extensions/workspace/x-scalar-active-environment.js.map +7 -0
  147. package/dist/schemas/extensions.d.ts +1 -0
  148. package/dist/schemas/extensions.d.ts.map +1 -1
  149. package/dist/schemas/extensions.js +1 -0
  150. package/dist/schemas/extensions.js.map +2 -2
  151. package/dist/schemas/inmemory-workspace.d.ts +95 -28
  152. package/dist/schemas/inmemory-workspace.d.ts.map +1 -1
  153. package/dist/schemas/inmemory-workspace.js +1 -9
  154. package/dist/schemas/inmemory-workspace.js.map +2 -2
  155. package/dist/schemas/navigation.d.ts +128 -1
  156. package/dist/schemas/navigation.d.ts.map +1 -1
  157. package/dist/schemas/navigation.js +20 -1
  158. package/dist/schemas/navigation.js.map +2 -2
  159. package/dist/schemas/reference-config/index.d.ts +243 -206
  160. package/dist/schemas/reference-config/index.d.ts.map +1 -1
  161. package/dist/schemas/reference-config/index.js.map +2 -2
  162. package/dist/schemas/reference-config/settings.d.ts +45 -9
  163. package/dist/schemas/reference-config/settings.d.ts.map +1 -1
  164. package/dist/schemas/reference-config/settings.js +1 -1
  165. package/dist/schemas/reference-config/settings.js.map +1 -1
  166. package/dist/schemas/v3.1/strict/openapi-document.d.ts +1540 -311
  167. package/dist/schemas/v3.1/strict/openapi-document.d.ts.map +1 -1
  168. package/dist/schemas/v3.1/strict/openapi-document.js +20 -13
  169. package/dist/schemas/v3.1/strict/openapi-document.js.map +2 -2
  170. package/dist/schemas/v3.1/strict/parameter.d.ts +23 -1
  171. package/dist/schemas/v3.1/strict/parameter.d.ts.map +1 -1
  172. package/dist/schemas/v3.1/strict/parameter.js +5 -1
  173. package/dist/schemas/v3.1/strict/parameter.js.map +2 -2
  174. package/dist/schemas/v3.1/strict/ref-definitions.d.ts +2 -0
  175. package/dist/schemas/v3.1/strict/ref-definitions.d.ts.map +1 -1
  176. package/dist/schemas/v3.1/strict/ref-definitions.js +4 -1
  177. package/dist/schemas/v3.1/strict/ref-definitions.js.map +2 -2
  178. package/dist/schemas/workspace-specification/config.d.ts +46 -15
  179. package/dist/schemas/workspace-specification/config.d.ts.map +1 -1
  180. package/dist/schemas/workspace-specification/config.js.map +1 -1
  181. package/dist/schemas/workspace-specification/index.d.ts +48 -11
  182. package/dist/schemas/workspace-specification/index.d.ts.map +1 -1
  183. package/dist/schemas/workspace.d.ts +353 -85
  184. package/dist/schemas/workspace.d.ts.map +1 -1
  185. package/dist/schemas/workspace.js +19 -11
  186. package/dist/schemas/workspace.js.map +2 -2
  187. package/dist/server.d.ts +4 -3
  188. package/dist/server.d.ts.map +1 -1
  189. package/dist/server.js +7 -6
  190. package/dist/server.js.map +2 -2
  191. package/dist/workspace-plugin.d.ts +49 -0
  192. package/dist/workspace-plugin.d.ts.map +1 -0
  193. package/dist/workspace-plugin.js +1 -0
  194. package/dist/workspace-plugin.js.map +7 -0
  195. package/package.json +26 -9
  196. package/dist/events/definitions.d.ts.map +0 -1
  197. package/dist/events/definitions.js.map +0 -7
  198. package/dist/plugins.d.ts.map +0 -1
  199. package/dist/schemas/v3.1/strict/client-config-extensions/x-scalar-client-config-environments.d.ts +0 -33
  200. package/dist/schemas/v3.1/strict/client-config-extensions/x-scalar-client-config-environments.d.ts.map +0 -1
  201. package/dist/schemas/v3.1/strict/client-config-extensions/x-scalar-client-config-environments.js +0 -23
  202. package/dist/schemas/v3.1/strict/client-config-extensions/x-scalar-client-config-environments.js.map +0 -7
@@ -8,13 +8,32 @@ import { getTag } from "./get-tag.js";
8
8
  const isDeprecatedOperation = (operation) => {
9
9
  return operation.deprecated || operation["x-scalar-stability"] === XScalarStabilityValues.Deprecated;
10
10
  };
11
- const createOperationEntry = (ref, operation, method, path = "Unknown", tag, getOperationId) => {
12
- const id = getOperationId({ ...operation, method, path }, tag);
11
+ const createOperationEntry = ({
12
+ ref,
13
+ operation,
14
+ method,
15
+ path,
16
+ generateId,
17
+ parentId,
18
+ parentTag
19
+ }) => {
20
+ const id = generateId({
21
+ type: "operation",
22
+ operation,
23
+ parentTag,
24
+ method,
25
+ path,
26
+ parentId
27
+ });
13
28
  const title = operation.summary?.trim() ? operation.summary : path;
14
29
  const isDeprecated = isDeprecatedOperation(operation);
15
30
  const examples = traverseOperationExamples(operation).map((example) => ({
16
31
  type: "example",
17
- id: `${id}/example-${example}`,
32
+ id: generateId({
33
+ type: "example",
34
+ parentId: id,
35
+ name: example
36
+ }),
18
37
  title: example,
19
38
  name: example
20
39
  }));
@@ -30,8 +49,13 @@ const createOperationEntry = (ref, operation, method, path = "Unknown", tag, get
30
49
  };
31
50
  return entry;
32
51
  };
33
- const traversePaths = (content, tagsMap, getOperationId) => {
34
- Object.entries(content.paths ?? {}).forEach(([path, pathItemObject]) => {
52
+ const traversePaths = ({
53
+ document,
54
+ tagsMap,
55
+ generateId,
56
+ documentId
57
+ }) => {
58
+ Object.entries(document.paths ?? {}).forEach(([path, pathItemObject]) => {
35
59
  const pathKeys = objectKeys(pathItemObject ?? {}).filter((key) => isHttpMethod(key));
36
60
  pathKeys.forEach((method) => {
37
61
  const _operation = pathItemObject?.[method];
@@ -45,12 +69,42 @@ const traversePaths = (content, tagsMap, getOperationId) => {
45
69
  const ref = `#/paths/${escapeJsonPointer(path)}/${method}`;
46
70
  if (operation.tags?.length) {
47
71
  operation.tags.forEach((tagName) => {
48
- const { tag } = getTag(tagsMap, tagName);
49
- tagsMap.get(tagName)?.entries.push(createOperationEntry(ref, operation, method, path, tag, getOperationId));
72
+ const { tag, id: tagId } = getTag({
73
+ tagsMap,
74
+ name: tagName,
75
+ documentId,
76
+ generateId
77
+ });
78
+ tagsMap.get(tagName)?.entries.push(
79
+ createOperationEntry({
80
+ ref,
81
+ operation,
82
+ method,
83
+ path,
84
+ parentTag: { tag, id: tagId },
85
+ generateId,
86
+ parentId: tagId
87
+ })
88
+ );
50
89
  });
51
90
  } else {
52
- const { tag } = getTag(tagsMap, "default");
53
- tagsMap.get("default")?.entries.push(createOperationEntry(ref, operation, method, path, tag, getOperationId));
91
+ const { tag, id: tagId } = getTag({
92
+ tagsMap,
93
+ name: "default",
94
+ documentId,
95
+ generateId
96
+ });
97
+ tagsMap.get("default")?.entries.push(
98
+ createOperationEntry({
99
+ ref,
100
+ operation,
101
+ method,
102
+ path,
103
+ parentTag: { tag, id: tagId },
104
+ generateId,
105
+ parentId: tagId
106
+ })
107
+ );
54
108
  }
55
109
  });
56
110
  });
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/navigation/helpers/traverse-paths.ts"],
4
- "sourcesContent": ["import type { HttpMethod } from '@scalar/helpers/http/http-methods'\nimport { isHttpMethod } from '@scalar/helpers/http/is-http-method'\nimport { objectKeys } from '@scalar/helpers/object/object-keys'\nimport { escapeJsonPointer } from '@scalar/json-magic/helpers/escape-json-pointer'\n\nimport { getResolvedRef } from '@/helpers/get-resolved-ref'\nimport { traverseOperationExamples } from '@/navigation/helpers/traverse-examples'\nimport type { TagsMap, TraverseSpecOptions } from '@/navigation/types'\nimport { XScalarStabilityValues } from '@/schemas/extensions/operation/x-scalar-stability'\nimport type { TraversedExample, TraversedOperation } from '@/schemas/navigation'\nimport type { OpenApiDocument, OperationObject, TagObject } from '@/schemas/v3.1/strict/openapi-document'\n\nimport { getTag } from './get-tag'\n\nexport const isDeprecatedOperation = (operation: OperationObject) => {\n return operation.deprecated || operation['x-scalar-stability'] === XScalarStabilityValues.Deprecated\n}\n\n/**\n * Creates a traversed operation entry from an OpenAPI operation object.\n *\n * @param ref - JSON pointer reference to the operation in the OpenAPI document\n * @param operation - The OpenAPI operation object\n * @param method - HTTP method of the operation\n * @param path - API path of the operation, defaults to 'Unknown'\n * @param tag - Tag object associated with the operation\n * @param entitiesMap - Map to store operation IDs and titles for mobile header navigation\n * @param getOperationId - Function to generate unique IDs for operations\n * @returns A traversed operation entry with ID, title, path, method and reference\n */\nconst createOperationEntry = (\n ref: string,\n operation: OperationObject,\n method: HttpMethod,\n path = 'Unknown',\n tag: TagObject,\n getOperationId: TraverseSpecOptions['getOperationId'],\n): TraversedOperation => {\n const id = getOperationId({ ...operation, method, path }, tag)\n const title = operation.summary?.trim() ? operation.summary : path\n\n const isDeprecated = isDeprecatedOperation(operation)\n\n const examples: TraversedExample[] = traverseOperationExamples(operation).map((example) => ({\n type: 'example',\n id: `${id}/example-${example}`,\n title: example,\n name: example,\n }))\n\n const entry = {\n id,\n title,\n path,\n method,\n ref,\n type: 'operation',\n isDeprecated,\n children: examples.length ? examples : undefined,\n } satisfies TraversedOperation\n\n return entry\n}\n\n/**\n * Traverses the paths in an OpenAPI document to build a map of operations organized by tags.\n *\n * This function processes each path and its operations to:\n * - Filter out internal operations (marked with x-internal) and operations to ignore (marked with x-scalar-ignore)\n * - Group operations by their tags\n * - Create a default tag group for untagged operations\n * - Generate unique references and IDs for each operation\n *\n * TODO: filter out internal and scalar-ignore tags\n *\n * @param content - The OpenAPI document to traverse\n * @param tagsDict - Dictionary mapping tag names to their OpenAPI tag objects\n * @param entitiesMap - Map to store operation IDs and titles for mobile header navigation\n * @param getOperationId - Function to generate unique IDs for operations\n * @returns Map of tag names to arrays of traversed operations\n */\nexport const traversePaths = (\n content: OpenApiDocument,\n /** Map of tags and their entries */\n tagsMap: TagsMap,\n getOperationId: TraverseSpecOptions['getOperationId'],\n) => {\n // Traverse paths\n Object.entries(content.paths ?? {}).forEach(([path, pathItemObject]) => {\n const pathKeys = objectKeys(pathItemObject ?? {}).filter((key) => isHttpMethod(key))\n\n pathKeys.forEach((method) => {\n const _operation = pathItemObject?.[method]\n const operation = getResolvedRef(_operation)\n if (!operation) {\n return\n }\n\n // Skip if the operation is internal or scalar-ignore\n if (operation['x-internal'] || operation['x-scalar-ignore'] || !isHttpMethod(method)) {\n return\n }\n\n const ref = `#/paths/${escapeJsonPointer(path)}/${method}`\n\n // Traverse tags\n if (operation.tags?.length) {\n operation.tags.forEach((tagName: string) => {\n const { tag } = getTag(tagsMap, tagName)\n tagsMap.get(tagName)?.entries.push(createOperationEntry(ref, operation, method, path, tag, getOperationId))\n })\n }\n // Add to default tag\n else {\n const { tag } = getTag(tagsMap, 'default')\n tagsMap.get('default')?.entries.push(createOperationEntry(ref, operation, method, path, tag, getOperationId))\n }\n })\n })\n}\n"],
5
- "mappings": "AACA,SAAS,oBAAoB;AAC7B,SAAS,kBAAkB;AAC3B,SAAS,yBAAyB;AAElC,SAAS,sBAAsB;AAC/B,SAAS,iCAAiC;AAE1C,SAAS,8BAA8B;AAIvC,SAAS,cAAc;AAEhB,MAAM,wBAAwB,CAAC,cAA+B;AACnE,SAAO,UAAU,cAAc,UAAU,oBAAoB,MAAM,uBAAuB;AAC5F;AAcA,MAAM,uBAAuB,CAC3B,KACA,WACA,QACA,OAAO,WACP,KACA,mBACuB;AACvB,QAAM,KAAK,eAAe,EAAE,GAAG,WAAW,QAAQ,KAAK,GAAG,GAAG;AAC7D,QAAM,QAAQ,UAAU,SAAS,KAAK,IAAI,UAAU,UAAU;AAE9D,QAAM,eAAe,sBAAsB,SAAS;AAEpD,QAAM,WAA+B,0BAA0B,SAAS,EAAE,IAAI,CAAC,aAAa;AAAA,IAC1F,MAAM;AAAA,IACN,IAAI,GAAG,EAAE,YAAY,OAAO;AAAA,IAC5B,OAAO;AAAA,IACP,MAAM;AAAA,EACR,EAAE;AAEF,QAAM,QAAQ;AAAA,IACZ;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA,MAAM;AAAA,IACN;AAAA,IACA,UAAU,SAAS,SAAS,WAAW;AAAA,EACzC;AAEA,SAAO;AACT;AAmBO,MAAM,gBAAgB,CAC3B,SAEA,SACA,mBACG;AAEH,SAAO,QAAQ,QAAQ,SAAS,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAC,MAAM,cAAc,MAAM;AACtE,UAAM,WAAW,WAAW,kBAAkB,CAAC,CAAC,EAAE,OAAO,CAAC,QAAQ,aAAa,GAAG,CAAC;AAEnF,aAAS,QAAQ,CAAC,WAAW;AAC3B,YAAM,aAAa,iBAAiB,MAAM;AAC1C,YAAM,YAAY,eAAe,UAAU;AAC3C,UAAI,CAAC,WAAW;AACd;AAAA,MACF;AAGA,UAAI,UAAU,YAAY,KAAK,UAAU,iBAAiB,KAAK,CAAC,aAAa,MAAM,GAAG;AACpF;AAAA,MACF;AAEA,YAAM,MAAM,WAAW,kBAAkB,IAAI,CAAC,IAAI,MAAM;AAGxD,UAAI,UAAU,MAAM,QAAQ;AAC1B,kBAAU,KAAK,QAAQ,CAAC,YAAoB;AAC1C,gBAAM,EAAE,IAAI,IAAI,OAAO,SAAS,OAAO;AACvC,kBAAQ,IAAI,OAAO,GAAG,QAAQ,KAAK,qBAAqB,KAAK,WAAW,QAAQ,MAAM,KAAK,cAAc,CAAC;AAAA,QAC5G,CAAC;AAAA,MACH,OAEK;AACH,cAAM,EAAE,IAAI,IAAI,OAAO,SAAS,SAAS;AACzC,gBAAQ,IAAI,SAAS,GAAG,QAAQ,KAAK,qBAAqB,KAAK,WAAW,QAAQ,MAAM,KAAK,cAAc,CAAC;AAAA,MAC9G;AAAA,IACF,CAAC;AAAA,EACH,CAAC;AACH;",
4
+ "sourcesContent": ["import type { HttpMethod } from '@scalar/helpers/http/http-methods'\nimport { isHttpMethod } from '@scalar/helpers/http/is-http-method'\nimport { objectKeys } from '@scalar/helpers/object/object-keys'\nimport { escapeJsonPointer } from '@scalar/json-magic/helpers/escape-json-pointer'\n\nimport { getResolvedRef } from '@/helpers/get-resolved-ref'\nimport { traverseOperationExamples } from '@/navigation/helpers/traverse-examples'\nimport type { TagsMap, TraverseSpecOptions } from '@/navigation/types'\nimport { XScalarStabilityValues } from '@/schemas/extensions/operation/x-scalar-stability'\nimport type { ParentTag, TraversedExample, TraversedOperation } from '@/schemas/navigation'\nimport type { OpenApiDocument, OperationObject } from '@/schemas/v3.1/strict/openapi-document'\n\nimport { getTag } from './get-tag'\n\nexport const isDeprecatedOperation = (operation: OperationObject) => {\n return operation.deprecated || operation['x-scalar-stability'] === XScalarStabilityValues.Deprecated\n}\n\n/**\n * Creates a traversed operation entry from an OpenAPI operation object.\n *\n * @param ref - JSON pointer reference to the operation in the OpenAPI document\n * @param operation - The OpenAPI operation object\n * @param method - HTTP method of the operation\n * @param path - API path of the operation, defaults to 'Unknown'\n * @param tag - Tag object associated with the operation\n * @param entitiesMap - Map to store operation IDs and titles for mobile header navigation\n * @param getOperationId - Function to generate unique IDs for operations\n * @returns A traversed operation entry with ID, title, path, method and reference\n */\nconst createOperationEntry = ({\n ref,\n operation,\n method,\n path,\n generateId,\n parentId,\n parentTag,\n}: {\n ref: string\n operation: OperationObject\n method: HttpMethod\n path: string\n parentTag: ParentTag\n generateId: TraverseSpecOptions['generateId']\n parentId: string\n}): TraversedOperation => {\n const id = generateId({\n type: 'operation',\n operation,\n parentTag,\n method: method,\n path: path,\n parentId: parentId,\n })\n const title = operation.summary?.trim() ? operation.summary : path\n\n const isDeprecated = isDeprecatedOperation(operation)\n\n const examples: TraversedExample[] = traverseOperationExamples(operation).map((example) => ({\n type: 'example',\n id: generateId({\n type: 'example',\n parentId: id,\n name: example,\n }),\n title: example,\n name: example,\n }))\n\n const entry = {\n id,\n title,\n path,\n method,\n ref,\n type: 'operation',\n isDeprecated,\n children: examples.length ? examples : undefined,\n } satisfies TraversedOperation\n\n return entry\n}\n\n/**\n * Traverses the paths in an OpenAPI document to build a map of operations organized by tags.\n *\n * This function processes each path and its operations to:\n * - Filter out internal operations (marked with x-internal) and operations to ignore (marked with x-scalar-ignore)\n * - Group operations by their tags\n * - Create a default tag group for untagged operations\n * - Generate unique references and IDs for each operation\n *\n * TODO: filter out internal and scalar-ignore tags\n *\n * @param content - The OpenAPI document to traverse\n * @param tagsDict - Dictionary mapping tag names to their OpenAPI tag objects\n * @param entitiesMap - Map to store operation IDs and titles for mobile header navigation\n * @param getOperationId - Function to generate unique IDs for operations\n * @returns Map of tag names to arrays of traversed operations\n */\nexport const traversePaths = ({\n document,\n tagsMap,\n generateId,\n documentId,\n}: {\n document: OpenApiDocument\n /** Map of tags and their entries */\n tagsMap: TagsMap\n /** Function used to generate unique IDs for operations */\n generateId: TraverseSpecOptions['generateId']\n /** Document ID */\n documentId: string\n}) => {\n // Traverse paths\n Object.entries(document.paths ?? {}).forEach(([path, pathItemObject]) => {\n const pathKeys = objectKeys(pathItemObject ?? {}).filter((key) => isHttpMethod(key))\n\n pathKeys.forEach((method) => {\n const _operation = pathItemObject?.[method]\n const operation = getResolvedRef(_operation)\n if (!operation) {\n return\n }\n\n // Skip if the operation is internal or scalar-ignore\n if (operation['x-internal'] || operation['x-scalar-ignore'] || !isHttpMethod(method)) {\n return\n }\n\n const ref = `#/paths/${escapeJsonPointer(path)}/${method}`\n\n // Traverse tags\n if (operation.tags?.length) {\n operation.tags.forEach((tagName: string) => {\n const { tag, id: tagId } = getTag({\n tagsMap,\n name: tagName,\n documentId,\n generateId,\n })\n tagsMap.get(tagName)?.entries.push(\n createOperationEntry({\n ref,\n operation,\n method,\n path,\n parentTag: { tag, id: tagId },\n generateId,\n parentId: tagId,\n }),\n )\n })\n }\n // Add to default tag\n else {\n const { tag, id: tagId } = getTag({\n tagsMap,\n name: 'default',\n documentId,\n generateId,\n })\n tagsMap.get('default')?.entries.push(\n createOperationEntry({\n ref,\n operation,\n method,\n path,\n parentTag: { tag, id: tagId },\n generateId,\n parentId: tagId,\n }),\n )\n }\n })\n })\n}\n"],
5
+ "mappings": "AACA,SAAS,oBAAoB;AAC7B,SAAS,kBAAkB;AAC3B,SAAS,yBAAyB;AAElC,SAAS,sBAAsB;AAC/B,SAAS,iCAAiC;AAE1C,SAAS,8BAA8B;AAIvC,SAAS,cAAc;AAEhB,MAAM,wBAAwB,CAAC,cAA+B;AACnE,SAAO,UAAU,cAAc,UAAU,oBAAoB,MAAM,uBAAuB;AAC5F;AAcA,MAAM,uBAAuB,CAAC;AAAA,EAC5B;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF,MAQ0B;AACxB,QAAM,KAAK,WAAW;AAAA,IACpB,MAAM;AAAA,IACN;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EACF,CAAC;AACD,QAAM,QAAQ,UAAU,SAAS,KAAK,IAAI,UAAU,UAAU;AAE9D,QAAM,eAAe,sBAAsB,SAAS;AAEpD,QAAM,WAA+B,0BAA0B,SAAS,EAAE,IAAI,CAAC,aAAa;AAAA,IAC1F,MAAM;AAAA,IACN,IAAI,WAAW;AAAA,MACb,MAAM;AAAA,MACN,UAAU;AAAA,MACV,MAAM;AAAA,IACR,CAAC;AAAA,IACD,OAAO;AAAA,IACP,MAAM;AAAA,EACR,EAAE;AAEF,QAAM,QAAQ;AAAA,IACZ;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA,MAAM;AAAA,IACN;AAAA,IACA,UAAU,SAAS,SAAS,WAAW;AAAA,EACzC;AAEA,SAAO;AACT;AAmBO,MAAM,gBAAgB,CAAC;AAAA,EAC5B;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF,MAQM;AAEJ,SAAO,QAAQ,SAAS,SAAS,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAC,MAAM,cAAc,MAAM;AACvE,UAAM,WAAW,WAAW,kBAAkB,CAAC,CAAC,EAAE,OAAO,CAAC,QAAQ,aAAa,GAAG,CAAC;AAEnF,aAAS,QAAQ,CAAC,WAAW;AAC3B,YAAM,aAAa,iBAAiB,MAAM;AAC1C,YAAM,YAAY,eAAe,UAAU;AAC3C,UAAI,CAAC,WAAW;AACd;AAAA,MACF;AAGA,UAAI,UAAU,YAAY,KAAK,UAAU,iBAAiB,KAAK,CAAC,aAAa,MAAM,GAAG;AACpF;AAAA,MACF;AAEA,YAAM,MAAM,WAAW,kBAAkB,IAAI,CAAC,IAAI,MAAM;AAGxD,UAAI,UAAU,MAAM,QAAQ;AAC1B,kBAAU,KAAK,QAAQ,CAAC,YAAoB;AAC1C,gBAAM,EAAE,KAAK,IAAI,MAAM,IAAI,OAAO;AAAA,YAChC;AAAA,YACA,MAAM;AAAA,YACN;AAAA,YACA;AAAA,UACF,CAAC;AACD,kBAAQ,IAAI,OAAO,GAAG,QAAQ;AAAA,YAC5B,qBAAqB;AAAA,cACnB;AAAA,cACA;AAAA,cACA;AAAA,cACA;AAAA,cACA,WAAW,EAAE,KAAK,IAAI,MAAM;AAAA,cAC5B;AAAA,cACA,UAAU;AAAA,YACZ,CAAC;AAAA,UACH;AAAA,QACF,CAAC;AAAA,MACH,OAEK;AACH,cAAM,EAAE,KAAK,IAAI,MAAM,IAAI,OAAO;AAAA,UAChC;AAAA,UACA,MAAM;AAAA,UACN;AAAA,UACA;AAAA,QACF,CAAC;AACD,gBAAQ,IAAI,SAAS,GAAG,QAAQ;AAAA,UAC9B,qBAAqB;AAAA,YACnB;AAAA,YACA;AAAA,YACA;AAAA,YACA;AAAA,YACA,WAAW,EAAE,KAAK,IAAI,MAAM;AAAA,YAC5B;AAAA,YACA,UAAU;AAAA,UACZ,CAAC;AAAA,QACH;AAAA,MACF;AAAA,IACF,CAAC;AAAA,EACH,CAAC;AACH;",
6
6
  "names": []
7
7
  }
@@ -7,13 +7,12 @@ import type { OpenApiDocument } from '../../schemas/v3.1/strict/openapi-document
7
7
  * - Filter out internal schemas (marked with x-internal) and schemas to ignore (marked with x-scalar-ignore)
8
8
  * - Create model entries with unique references and IDs
9
9
  * - Store model IDs and titles for mobile header navigation
10
- *
11
- * @param content - The OpenAPI document to traverse
12
- * @param entitiesMap - Map to store schema IDs and titles for mobile header navigation
13
- * @param getModelId - Function to generate unique IDs for schemas
14
- * @returns Array of traversed schema entries
15
10
  */
16
- export declare const traverseSchemas: (content: OpenApiDocument,
17
- /** Map of tagNames and their entries */
18
- tagsMap: TagsMap, getModelId: TraverseSpecOptions["getModelId"]) => TraversedSchema[];
11
+ export declare const traverseSchemas: ({ document, tagsMap, generateId, documentId, }: {
12
+ document: OpenApiDocument;
13
+ /** Map of tagNames and their entries */
14
+ tagsMap: TagsMap;
15
+ generateId: TraverseSpecOptions["generateId"];
16
+ documentId: string;
17
+ }) => TraversedSchema[];
19
18
  //# sourceMappingURL=traverse-schemas.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"traverse-schemas.d.ts","sourceRoot":"","sources":["../../../src/navigation/helpers/traverse-schemas.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,OAAO,EAAE,mBAAmB,EAAE,MAAM,oBAAoB,CAAA;AACtE,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,sBAAsB,CAAA;AAC3D,OAAO,KAAK,EAAE,eAAe,EAA2B,MAAM,wCAAwC,CAAA;AAmCtG;;;;;;;;;;;GAWG;AACH,eAAO,MAAM,eAAe,GAC1B,SAAS,eAAe;AACxB,wCAAwC;AACxC,SAAS,OAAO,EAChB,YAAY,mBAAmB,CAAC,YAAY,CAAC,KAC5C,eAAe,EA4BjB,CAAA"}
1
+ {"version":3,"file":"traverse-schemas.d.ts","sourceRoot":"","sources":["../../../src/navigation/helpers/traverse-schemas.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,OAAO,EAAE,mBAAmB,EAAE,MAAM,oBAAoB,CAAA;AACtE,OAAO,KAAK,EAAa,eAAe,EAAE,MAAM,sBAAsB,CAAA;AACtE,OAAO,KAAK,EAAE,eAAe,EAAgB,MAAM,wCAAwC,CAAA;AA2C3F;;;;;;GAMG;AACH,eAAO,MAAM,eAAe,GAAI,gDAK7B;IACD,QAAQ,EAAE,eAAe,CAAA;IACzB,wCAAwC;IACxC,OAAO,EAAE,OAAO,CAAA;IAChB,UAAU,EAAE,mBAAmB,CAAC,YAAY,CAAC,CAAA;IAC7C,UAAU,EAAE,MAAM,CAAA;CACnB,KAAG,eAAe,EA4ClB,CAAA"}
@@ -1,7 +1,14 @@
1
1
  import { getResolvedRef } from "../../helpers/get-resolved-ref.js";
2
2
  import { getTag } from "../../navigation/helpers/get-tag.js";
3
- const createSchemaEntry = (ref, name = "Unknown", getModelId, tag, _schema) => {
4
- const id = getModelId({ name }, tag);
3
+ const createSchemaEntry = ({
4
+ ref,
5
+ name,
6
+ generateId,
7
+ parentTag,
8
+ _schema,
9
+ parentId
10
+ }) => {
11
+ const id = generateId({ name, type: "model", parentTag, parentId, schema: _schema });
5
12
  const schema = getResolvedRef(_schema);
6
13
  const title = schema && "title" in schema && schema.title || name;
7
14
  const entry = {
@@ -13,8 +20,13 @@ const createSchemaEntry = (ref, name = "Unknown", getModelId, tag, _schema) => {
13
20
  };
14
21
  return entry;
15
22
  };
16
- const traverseSchemas = (content, tagsMap, getModelId) => {
17
- const schemas = content.components?.schemas ?? {};
23
+ const traverseSchemas = ({
24
+ document,
25
+ tagsMap,
26
+ generateId,
27
+ documentId
28
+ }) => {
29
+ const schemas = document.components?.schemas ?? {};
18
30
  const untagged = [];
19
31
  for (const name in schemas) {
20
32
  const schema = getResolvedRef(schemas[name]);
@@ -24,11 +36,27 @@ const traverseSchemas = (content, tagsMap, getModelId) => {
24
36
  const ref = `#/components/schemas/${name}`;
25
37
  if (schema?.["x-tags"]) {
26
38
  schema["x-tags"].forEach((tagName) => {
27
- const { tag } = getTag(tagsMap, tagName);
28
- tagsMap.get(tagName)?.entries.push(createSchemaEntry(ref, name, getModelId, tag));
39
+ const { tag, id: tagId } = getTag({ tagsMap, name: tagName, documentId, generateId });
40
+ tagsMap.get(tagName)?.entries.push(
41
+ createSchemaEntry({
42
+ ref,
43
+ name,
44
+ generateId,
45
+ parentTag: { tag, id: tagId },
46
+ parentId: documentId
47
+ })
48
+ );
29
49
  });
30
50
  } else {
31
- untagged.push(createSchemaEntry(ref, name, getModelId, void 0, getResolvedRef(schemas[name])));
51
+ untagged.push(
52
+ createSchemaEntry({
53
+ ref,
54
+ name,
55
+ generateId,
56
+ _schema: getResolvedRef(schemas[name]),
57
+ parentId: documentId
58
+ })
59
+ );
32
60
  }
33
61
  }
34
62
  return untagged;
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/navigation/helpers/traverse-schemas.ts"],
4
- "sourcesContent": ["import { getResolvedRef } from '@/helpers/get-resolved-ref'\nimport { getTag } from '@/navigation/helpers/get-tag'\nimport type { TagsMap, TraverseSpecOptions } from '@/navigation/types'\nimport type { TraversedSchema } from '@/schemas/navigation'\nimport type { OpenApiDocument, SchemaObject, TagObject } from '@/schemas/v3.1/strict/openapi-document'\n\n/** Creates a traversed schema entry from an OpenAPI schema object.\n *\n * @param ref - JSON pointer reference to the schema in the OpenAPI document\n * @param name - Name of the schema, defaults to 'Unknown'\n * @param entriesMap - Map to store schema IDs and titles for mobile header navigation\n * @param getModelId - Function to generate unique IDs for schemas\n * @returns A traversed schema entry with ID, title, name and reference\n */\nconst createSchemaEntry = (\n ref: string,\n name = 'Unknown',\n getModelId: TraverseSpecOptions['getModelId'],\n tag?: TagObject,\n _schema?: SchemaObject,\n): TraversedSchema => {\n const id = getModelId({ name }, tag)\n const schema = getResolvedRef(_schema)\n\n // Use schema.title if available, otherwise fall back to name\n // @see https://json-schema.org/draft/2020-12/json-schema-core#section-4.3.5\n const title = (schema && 'title' in schema && (schema.title as string)) || name\n\n const entry = {\n id,\n title,\n name,\n ref,\n type: 'model',\n } satisfies TraversedSchema\n\n return entry\n}\n\n/** Traverses the schemas in an OpenAPI document to build an array of model entries.\n *\n * This function processes each schema in components.schemas to:\n * - Filter out internal schemas (marked with x-internal) and schemas to ignore (marked with x-scalar-ignore)\n * - Create model entries with unique references and IDs\n * - Store model IDs and titles for mobile header navigation\n *\n * @param content - The OpenAPI document to traverse\n * @param entitiesMap - Map to store schema IDs and titles for mobile header navigation\n * @param getModelId - Function to generate unique IDs for schemas\n * @returns Array of traversed schema entries\n */\nexport const traverseSchemas = (\n content: OpenApiDocument,\n /** Map of tagNames and their entries */\n tagsMap: TagsMap,\n getModelId: TraverseSpecOptions['getModelId'],\n): TraversedSchema[] => {\n const schemas = content.components?.schemas ?? {}\n const untagged: TraversedSchema[] = []\n\n // biome-ignore lint/suspicious/useGuardForIn: we do have an if statement after de-ref\n for (const name in schemas) {\n const schema = getResolvedRef(schemas[name])\n\n if (schema?.['x-internal'] || schema?.['x-scalar-ignore'] || !Object.hasOwn(schemas, name)) {\n continue\n }\n\n const ref = `#/components/schemas/${name}`\n\n // Add to tags\n if (schema?.['x-tags']) {\n schema['x-tags'].forEach((tagName: string) => {\n const { tag } = getTag(tagsMap, tagName)\n tagsMap.get(tagName)?.entries.push(createSchemaEntry(ref, name, getModelId, tag))\n })\n }\n // Add to untagged\n else {\n untagged.push(createSchemaEntry(ref, name, getModelId, undefined, getResolvedRef(schemas[name])))\n }\n }\n\n return untagged\n}\n"],
5
- "mappings": "AAAA,SAAS,sBAAsB;AAC/B,SAAS,cAAc;AAavB,MAAM,oBAAoB,CACxB,KACA,OAAO,WACP,YACA,KACA,YACoB;AACpB,QAAM,KAAK,WAAW,EAAE,KAAK,GAAG,GAAG;AACnC,QAAM,SAAS,eAAe,OAAO;AAIrC,QAAM,QAAS,UAAU,WAAW,UAAW,OAAO,SAAqB;AAE3E,QAAM,QAAQ;AAAA,IACZ;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA,MAAM;AAAA,EACR;AAEA,SAAO;AACT;AAcO,MAAM,kBAAkB,CAC7B,SAEA,SACA,eACsB;AACtB,QAAM,UAAU,QAAQ,YAAY,WAAW,CAAC;AAChD,QAAM,WAA8B,CAAC;AAGrC,aAAW,QAAQ,SAAS;AAC1B,UAAM,SAAS,eAAe,QAAQ,IAAI,CAAC;AAE3C,QAAI,SAAS,YAAY,KAAK,SAAS,iBAAiB,KAAK,CAAC,OAAO,OAAO,SAAS,IAAI,GAAG;AAC1F;AAAA,IACF;AAEA,UAAM,MAAM,wBAAwB,IAAI;AAGxC,QAAI,SAAS,QAAQ,GAAG;AACtB,aAAO,QAAQ,EAAE,QAAQ,CAAC,YAAoB;AAC5C,cAAM,EAAE,IAAI,IAAI,OAAO,SAAS,OAAO;AACvC,gBAAQ,IAAI,OAAO,GAAG,QAAQ,KAAK,kBAAkB,KAAK,MAAM,YAAY,GAAG,CAAC;AAAA,MAClF,CAAC;AAAA,IACH,OAEK;AACH,eAAS,KAAK,kBAAkB,KAAK,MAAM,YAAY,QAAW,eAAe,QAAQ,IAAI,CAAC,CAAC,CAAC;AAAA,IAClG;AAAA,EACF;AAEA,SAAO;AACT;",
4
+ "sourcesContent": ["import { getResolvedRef } from '@/helpers/get-resolved-ref'\nimport { getTag } from '@/navigation/helpers/get-tag'\nimport type { TagsMap, TraverseSpecOptions } from '@/navigation/types'\nimport type { ParentTag, TraversedSchema } from '@/schemas/navigation'\nimport type { OpenApiDocument, SchemaObject } from '@/schemas/v3.1/strict/openapi-document'\n\n/** Creates a traversed schema entry from an OpenAPI schema object.\n *\n * @param ref - JSON pointer reference to the schema in the OpenAPI document\n * @param name - Name of the schema, defaults to 'Unknown'\n * @param entriesMap - Map to store schema IDs and titles for mobile header navigation\n * @param getModelId - Function to generate unique IDs for schemas\n * @returns A traversed schema entry with ID, title, name and reference\n */\nconst createSchemaEntry = ({\n ref,\n name,\n generateId,\n parentTag,\n _schema,\n parentId,\n}: {\n ref: string\n name: string\n generateId: TraverseSpecOptions['generateId']\n parentTag?: ParentTag\n _schema?: SchemaObject\n parentId: string\n}): TraversedSchema => {\n const id = generateId({ name, type: 'model', parentTag, parentId, schema: _schema })\n const schema = getResolvedRef(_schema)\n\n // Use schema.title if available, otherwise fall back to name\n // @see https://json-schema.org/draft/2020-12/json-schema-core#section-4.3.5\n const title = (schema && 'title' in schema && (schema.title as string)) || name\n\n const entry = {\n id,\n title,\n name,\n ref,\n type: 'model',\n } satisfies TraversedSchema\n\n return entry\n}\n\n/** Traverses the schemas in an OpenAPI document to build an array of model entries.\n *\n * This function processes each schema in components.schemas to:\n * - Filter out internal schemas (marked with x-internal) and schemas to ignore (marked with x-scalar-ignore)\n * - Create model entries with unique references and IDs\n * - Store model IDs and titles for mobile header navigation\n */\nexport const traverseSchemas = ({\n document,\n tagsMap,\n generateId,\n documentId,\n}: {\n document: OpenApiDocument\n /** Map of tagNames and their entries */\n tagsMap: TagsMap\n generateId: TraverseSpecOptions['generateId']\n documentId: string\n}): TraversedSchema[] => {\n const schemas = document.components?.schemas ?? {}\n const untagged: TraversedSchema[] = []\n\n // biome-ignore lint/suspicious/useGuardForIn: we do have an if statement after de-ref\n for (const name in schemas) {\n const schema = getResolvedRef(schemas[name])\n\n if (schema?.['x-internal'] || schema?.['x-scalar-ignore'] || !Object.hasOwn(schemas, name)) {\n continue\n }\n\n const ref = `#/components/schemas/${name}`\n\n // Add to tags\n if (schema?.['x-tags']) {\n schema['x-tags'].forEach((tagName: string) => {\n const { tag, id: tagId } = getTag({ tagsMap, name: tagName, documentId, generateId })\n tagsMap.get(tagName)?.entries.push(\n createSchemaEntry({\n ref,\n name,\n generateId,\n parentTag: { tag, id: tagId },\n parentId: documentId,\n }),\n )\n })\n }\n // Add to untagged\n else {\n untagged.push(\n createSchemaEntry({\n ref,\n name,\n generateId,\n _schema: getResolvedRef(schemas[name]),\n parentId: documentId,\n }),\n )\n }\n }\n\n return untagged\n}\n"],
5
+ "mappings": "AAAA,SAAS,sBAAsB;AAC/B,SAAS,cAAc;AAavB,MAAM,oBAAoB,CAAC;AAAA,EACzB;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF,MAOuB;AACrB,QAAM,KAAK,WAAW,EAAE,MAAM,MAAM,SAAS,WAAW,UAAU,QAAQ,QAAQ,CAAC;AACnF,QAAM,SAAS,eAAe,OAAO;AAIrC,QAAM,QAAS,UAAU,WAAW,UAAW,OAAO,SAAqB;AAE3E,QAAM,QAAQ;AAAA,IACZ;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA,MAAM;AAAA,EACR;AAEA,SAAO;AACT;AASO,MAAM,kBAAkB,CAAC;AAAA,EAC9B;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF,MAMyB;AACvB,QAAM,UAAU,SAAS,YAAY,WAAW,CAAC;AACjD,QAAM,WAA8B,CAAC;AAGrC,aAAW,QAAQ,SAAS;AAC1B,UAAM,SAAS,eAAe,QAAQ,IAAI,CAAC;AAE3C,QAAI,SAAS,YAAY,KAAK,SAAS,iBAAiB,KAAK,CAAC,OAAO,OAAO,SAAS,IAAI,GAAG;AAC1F;AAAA,IACF;AAEA,UAAM,MAAM,wBAAwB,IAAI;AAGxC,QAAI,SAAS,QAAQ,GAAG;AACtB,aAAO,QAAQ,EAAE,QAAQ,CAAC,YAAoB;AAC5C,cAAM,EAAE,KAAK,IAAI,MAAM,IAAI,OAAO,EAAE,SAAS,MAAM,SAAS,YAAY,WAAW,CAAC;AACpF,gBAAQ,IAAI,OAAO,GAAG,QAAQ;AAAA,UAC5B,kBAAkB;AAAA,YAChB;AAAA,YACA;AAAA,YACA;AAAA,YACA,WAAW,EAAE,KAAK,IAAI,MAAM;AAAA,YAC5B,UAAU;AAAA,UACZ,CAAC;AAAA,QACH;AAAA,MACF,CAAC;AAAA,IACH,OAEK;AACH,eAAS;AAAA,QACP,kBAAkB;AAAA,UAChB;AAAA,UACA;AAAA,UACA;AAAA,UACA,SAAS,eAAe,QAAQ,IAAI,CAAC;AAAA,UACrC,UAAU;AAAA,QACZ,CAAC;AAAA,MACH;AAAA,IACF;AAAA,EACF;AAEA,SAAO;AACT;",
6
6
  "names": []
7
7
  }
@@ -1,7 +1,7 @@
1
1
  import type { TagsMap, TraverseSpecOptions } from '../../navigation/types.js';
2
2
  import type { TraversedEntry } from '../../schemas/navigation.js';
3
3
  import type { OpenApiDocument } from '../../schemas/v3.1/strict/openapi-document.js';
4
- type Options = Pick<TraverseSpecOptions, 'getTagId' | 'tagsSorter' | 'operationsSorter'>;
4
+ type Options = Pick<TraverseSpecOptions, 'tagsSorter' | 'operationsSorter' | 'generateId'>;
5
5
  /**
6
6
  * Traverses the tags map to create navigation entries, handling both grouped and ungrouped tags.
7
7
  *
@@ -11,8 +11,12 @@ type Options = Pick<TraverseSpecOptions, 'getTagId' | 'tagsSorter' | 'operations
11
11
  * - Create navigation entries for each tag or tag group
12
12
  * - Flatten default tag entries if it's the only tag present
13
13
  */
14
- export declare const traverseTags: (content: OpenApiDocument,
15
- /** Map of tags and their entries */
16
- tagsMap: TagsMap, { getTagId, tagsSorter, operationsSorter }: Options) => TraversedEntry[];
14
+ export declare const traverseTags: ({ document, tagsMap, documentId, options: { generateId, tagsSorter, operationsSorter }, }: {
15
+ document: OpenApiDocument;
16
+ /** Map of tags and their entries */
17
+ tagsMap: TagsMap;
18
+ documentId: string;
19
+ options: Options;
20
+ }) => TraversedEntry[];
17
21
  export {};
18
22
  //# sourceMappingURL=traverse-tags.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"traverse-tags.d.ts","sourceRoot":"","sources":["../../../src/navigation/helpers/traverse-tags.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,OAAO,EAAE,mBAAmB,EAAE,MAAM,oBAAoB,CAAA;AACtE,OAAO,KAAK,EAAE,cAAc,EAAgB,MAAM,sBAAsB,CAAA;AACxE,OAAO,KAAK,EAAE,eAAe,EAAa,MAAM,wCAAwC,CAAA;AAIxF,KAAK,OAAO,GAAG,IAAI,CAAC,mBAAmB,EAAE,UAAU,GAAG,YAAY,GAAG,kBAAkB,CAAC,CAAA;AA4HxF;;;;;;;;GAQG;AACH,eAAO,MAAM,YAAY,GACvB,SAAS,eAAe;AACxB,oCAAoC;AACpC,SAAS,OAAO,EAChB,4CAA4C,OAAO,KAClD,cAAc,EAyBhB,CAAA"}
1
+ {"version":3,"file":"traverse-tags.d.ts","sourceRoot":"","sources":["../../../src/navigation/helpers/traverse-tags.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,OAAO,EAAE,mBAAmB,EAAE,MAAM,oBAAoB,CAAA;AACtE,OAAO,KAAK,EAAE,cAAc,EAAgB,MAAM,sBAAsB,CAAA;AACxE,OAAO,KAAK,EAAE,eAAe,EAAa,MAAM,wCAAwC,CAAA;AAIxF,KAAK,OAAO,GAAG,IAAI,CAAC,mBAAmB,EAAE,YAAY,GAAG,kBAAkB,GAAG,YAAY,CAAC,CAAA;AAkK1F;;;;;;;;GAQG;AACH,eAAO,MAAM,YAAY,GAAI,2FAK1B;IACD,QAAQ,EAAE,eAAe,CAAA;IACzB,oCAAoC;IACpC,OAAO,EAAE,OAAO,CAAA;IAChB,UAAU,EAAE,MAAM,CAAA;IAClB,OAAO,EAAE,OAAO,CAAA;CACjB,KAAG,cAAc,EAuCjB,CAAA"}
@@ -1,7 +1,17 @@
1
1
  import { getXKeysFromObject } from "../../navigation/helpers/get-x-keys.js";
2
2
  import { getTag } from "./get-tag.js";
3
- const createTagEntry = (tag, getTagId, children, isGroup = false) => {
4
- const id = getTagId(tag);
3
+ const createTagEntry = ({
4
+ tag,
5
+ generateId,
6
+ children,
7
+ isGroup = false,
8
+ parentId
9
+ }) => {
10
+ const id = generateId({
11
+ type: "tag",
12
+ tag,
13
+ parentId
14
+ });
5
15
  const title = tag["x-displayName"] ?? tag.name ?? "Untitled Tag";
6
16
  const entry = {
7
17
  id,
@@ -16,23 +26,38 @@ const createTagEntry = (tag, getTagId, children, isGroup = false) => {
16
26
  };
17
27
  return entry;
18
28
  };
19
- const getSortedTagEntries = (_keys, tagsMap, { getTagId, tagsSorter, operationsSorter }) => {
29
+ const getSortedTagEntries = ({
30
+ _keys,
31
+ tagsMap,
32
+ options: { tagsSorter, operationsSorter, generateId },
33
+ documentId
34
+ }) => {
20
35
  const hasDefault = _keys.includes("default");
21
36
  const keys = hasDefault ? _keys.filter((key) => key !== "default") : _keys;
22
37
  if (tagsSorter === "alpha") {
23
38
  keys.sort((a, b) => {
24
- const nameA = getTag(tagsMap, a).tag["x-displayName"] || a || "Untitled Tag";
25
- const nameB = getTag(tagsMap, b).tag["x-displayName"] || b || "Untitled Tag";
39
+ const nameA = getTag({
40
+ tagsMap,
41
+ name: a,
42
+ documentId,
43
+ generateId
44
+ }).tag["x-displayName"] || a || "Untitled Tag";
45
+ const nameB = getTag({ tagsMap, name: b, documentId, generateId }).tag["x-displayName"] || b || "Untitled Tag";
26
46
  return nameA.localeCompare(nameB);
27
47
  });
28
48
  } else if (typeof tagsSorter === "function") {
29
- keys.sort((a, b) => tagsSorter(getTag(tagsMap, a).tag, getTag(tagsMap, b).tag));
49
+ keys.sort(
50
+ (a, b) => tagsSorter(
51
+ getTag({ tagsMap, name: a, documentId, generateId }).tag,
52
+ getTag({ tagsMap, name: b, documentId, generateId }).tag
53
+ )
54
+ );
30
55
  }
31
56
  if (hasDefault) {
32
57
  keys.push("default");
33
58
  }
34
59
  return keys.flatMap((key) => {
35
- const { tag, entries } = getTag(tagsMap, key);
60
+ const { tag, entries } = getTag({ tagsMap, name: key, documentId, generateId });
36
61
  if (tag["x-internal"] || tag["x-scalar-ignore"]) {
37
62
  return [];
38
63
  }
@@ -53,23 +78,46 @@ const getSortedTagEntries = (_keys, tagsMap, { getTagId, tagsSorter, operationsS
53
78
  );
54
79
  });
55
80
  }
56
- return entries.length ? createTagEntry(tag, getTagId, entries) : [];
81
+ return entries.length ? createTagEntry({
82
+ tag,
83
+ generateId,
84
+ children: entries,
85
+ parentId: documentId,
86
+ isGroup: false
87
+ }) : [];
57
88
  });
58
89
  };
59
- const traverseTags = (content, tagsMap, { getTagId, tagsSorter, operationsSorter }) => {
60
- if (content["x-tagGroups"]) {
61
- const tagGroups = content["x-tagGroups"];
90
+ const traverseTags = ({
91
+ document,
92
+ tagsMap,
93
+ documentId,
94
+ options: { generateId, tagsSorter, operationsSorter }
95
+ }) => {
96
+ if (document["x-tagGroups"]) {
97
+ const tagGroups = document["x-tagGroups"];
62
98
  return tagGroups.flatMap((tagGroup) => {
63
- const entries = getSortedTagEntries(tagGroup.tags ?? [], tagsMap, {
64
- getTagId,
65
- tagsSorter,
66
- operationsSorter
99
+ const entries = getSortedTagEntries({
100
+ _keys: tagGroup.tags,
101
+ tagsMap,
102
+ options: { tagsSorter, operationsSorter, generateId },
103
+ documentId
67
104
  });
68
- return entries.length ? createTagEntry(tagGroup, getTagId, entries, true) : [];
105
+ return entries.length ? createTagEntry({
106
+ tag: tagGroup,
107
+ generateId,
108
+ children: entries,
109
+ parentId: documentId,
110
+ isGroup: true
111
+ }) : [];
69
112
  });
70
113
  }
71
114
  const keys = Array.from(tagsMap.keys());
72
- const tags = getSortedTagEntries(keys, tagsMap, { getTagId, tagsSorter, operationsSorter });
115
+ const tags = getSortedTagEntries({
116
+ _keys: keys,
117
+ tagsMap,
118
+ options: { generateId, tagsSorter, operationsSorter },
119
+ documentId
120
+ });
73
121
  if (tags.length === 1 && tags[0]?.title === "default") {
74
122
  return tags[0]?.children ?? [];
75
123
  }
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/navigation/helpers/traverse-tags.ts"],
4
- "sourcesContent": ["import { getXKeysFromObject } from '@/navigation/helpers/get-x-keys'\nimport type { TagsMap, TraverseSpecOptions } from '@/navigation/types'\nimport type { TraversedEntry, TraversedTag } from '@/schemas/navigation'\nimport type { OpenApiDocument, TagObject } from '@/schemas/v3.1/strict/openapi-document'\n\nimport { getTag } from './get-tag'\n\ntype Options = Pick<TraverseSpecOptions, 'getTagId' | 'tagsSorter' | 'operationsSorter'>\n\n/** Creates a traversed tag entry from an OpenAPI tag object.\n *\n * @param tag - The OpenAPI tag object\n * @param entriesMap - Map to store tag IDs and titles for mobile header navigation\n * @param getTagId - Function to generate unique IDs for tags\n * @param children - Array of child entries (operations, webhooks, etc.)\n * @param isGroup - Whether this tag represents a group of tags\n * @returns A traversed tag entry with ID, title, name and children\n */\nconst createTagEntry = (\n tag: TagObject,\n getTagId: TraverseSpecOptions['getTagId'],\n children: TraversedEntry[],\n isGroup = false,\n): TraversedTag => {\n const id = getTagId(tag)\n const title = tag['x-displayName'] ?? tag.name ?? 'Untitled Tag'\n\n const entry = {\n id,\n title,\n name: tag.name || title,\n description: tag.description,\n children,\n isGroup,\n isWebhooks: false,\n type: 'tag',\n xKeys: getXKeysFromObject(tag),\n } satisfies TraversedTag\n\n return entry\n}\n\n/** Sorts and processes tags to create a hierarchical structure of tag entries.\n *\n * This function handles:\n * - Sorting tags alphabetically or using a custom sort function\n * - Ensuring the default tag appears last\n * - Sorting operations within tags by title, method, or custom function\n * - Filtering out internal and ignored tags\n * - Creating tag entries with their associated operations\n *\n * @param _keys - Array of tag keys to process\n * @param tagsMap - Map of tags and their entries\n * @param tagsDict - Dictionary of OpenAPI tags by name\n * @param titlesMap - Map of titles for the mobile header\n * @param options - Sorting and ID generation options\n * @returns Array of processed and sorted tag entries\n */\n/** Sorts tags and returns entries */\nconst getSortedTagEntries = (\n _keys: string[],\n /** Map of tags and their entries */\n tagsMap: TagsMap,\n { getTagId, tagsSorter, operationsSorter }: Options,\n) => {\n // Ensure that default is last if it exists\n const hasDefault = _keys.includes('default')\n const keys = hasDefault ? _keys.filter((key) => key !== 'default') : _keys\n\n // Alpha sort\n if (tagsSorter === 'alpha') {\n keys.sort((a, b) => {\n const nameA = getTag(tagsMap, a).tag['x-displayName'] || a || 'Untitled Tag'\n const nameB = getTag(tagsMap, b).tag['x-displayName'] || b || 'Untitled Tag'\n return nameA.localeCompare(nameB)\n })\n }\n // Custom sort\n else if (typeof tagsSorter === 'function') {\n keys.sort((a, b) => tagsSorter(getTag(tagsMap, a).tag, getTag(tagsMap, b).tag))\n }\n\n if (hasDefault) {\n keys.push('default')\n }\n\n /**\n * Process each tag and its entries:\n * - Skip internal and ignored tags\n * - Sort operations within tags\n * - Create tag entries with sorted operations\n */\n return keys.flatMap((key) => {\n const { tag, entries } = getTag(tagsMap, key)\n\n // Skip if the tag is internal or scalar-ignore\n if (tag['x-internal'] || tag['x-scalar-ignore']) {\n return []\n }\n\n // Alpha sort\n if (operationsSorter === 'alpha') {\n entries.sort((a, b) => (a.type === 'operation' && b.type === 'operation' ? a.title.localeCompare(b.title) : 0))\n }\n // Method sort\n else if (operationsSorter === 'method') {\n entries.sort((a, b) => (a.type === 'operation' && b.type === 'operation' ? a.method.localeCompare(b.method) : 0))\n }\n // Custom sort\n else if (typeof operationsSorter === 'function') {\n entries.sort((a, b) => {\n // Guard against tags\n if (!(a.type === 'operation' || a.type === 'webhook') || !(b.type === 'operation' || b.type === 'webhook')) {\n return 0\n }\n\n // Handle webhooks as well as operations\n const pathA = a.type === 'operation' ? a.path : a.name\n const pathB = b.type === 'operation' ? b.path : b.name\n\n return operationsSorter(\n { method: a.method, path: pathA, ref: a.ref, httpVerb: a.method },\n { method: b.method, path: pathB, ref: b.ref, httpVerb: b.method },\n )\n })\n }\n\n return entries.length ? createTagEntry(tag, getTagId, entries) : []\n })\n}\n\n/**\n * Traverses the tags map to create navigation entries, handling both grouped and ungrouped tags.\n *\n * This function processes the OpenAPI document's tags to:\n * - Handle tag groups if specified via x-tagGroups\n * - Sort tags and their operations according to provided sorters\n * - Create navigation entries for each tag or tag group\n * - Flatten default tag entries if it's the only tag present\n */\nexport const traverseTags = (\n content: OpenApiDocument,\n /** Map of tags and their entries */\n tagsMap: TagsMap,\n { getTagId, tagsSorter, operationsSorter }: Options,\n): TraversedEntry[] => {\n // x-tagGroups\n if (content['x-tagGroups']) {\n const tagGroups = content['x-tagGroups']\n\n return tagGroups.flatMap((tagGroup) => {\n const entries = getSortedTagEntries(tagGroup.tags ?? [], tagsMap, {\n getTagId,\n tagsSorter,\n operationsSorter,\n })\n return entries.length ? createTagEntry(tagGroup, getTagId, entries, true) : []\n })\n }\n\n // Ungrouped regular tags\n const keys = Array.from(tagsMap.keys())\n const tags = getSortedTagEntries(keys, tagsMap, { getTagId, tagsSorter, operationsSorter })\n\n // Flatten if we only have default tag\n if (tags.length === 1 && tags[0]?.title === 'default') {\n return tags[0]?.children ?? []\n }\n\n return tags\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["import { getXKeysFromObject } from '@/navigation/helpers/get-x-keys'\nimport type { TagsMap, TraverseSpecOptions } from '@/navigation/types'\nimport type { TraversedEntry, TraversedTag } from '@/schemas/navigation'\nimport type { OpenApiDocument, TagObject } from '@/schemas/v3.1/strict/openapi-document'\n\nimport { getTag } from './get-tag'\n\ntype Options = Pick<TraverseSpecOptions, 'tagsSorter' | 'operationsSorter' | 'generateId'>\n\n/** Creates a traversed tag entry from an OpenAPI tag object.\n *\n * @param tag - The OpenAPI tag object\n * @param entriesMap - Map to store tag IDs and titles for mobile header navigation\n * @param getTagId - Function to generate unique IDs for tags\n * @param children - Array of child entries (operations, webhooks, etc.)\n * @param isGroup - Whether this tag represents a group of tags\n * @returns A traversed tag entry with ID, title, name and children\n */\nconst createTagEntry = ({\n tag,\n generateId,\n children,\n isGroup = false,\n parentId,\n}: {\n tag: TagObject\n generateId: TraverseSpecOptions['generateId']\n children: TraversedEntry[]\n isGroup: boolean\n parentId: string\n}): TraversedTag => {\n const id = generateId({\n type: 'tag',\n tag,\n parentId,\n })\n const title = tag['x-displayName'] ?? tag.name ?? 'Untitled Tag'\n\n const entry = {\n id,\n title,\n name: tag.name || title,\n description: tag.description,\n children,\n isGroup,\n isWebhooks: false,\n type: 'tag',\n xKeys: getXKeysFromObject(tag),\n } satisfies TraversedTag\n\n return entry\n}\n\n/** Sorts and processes tags to create a hierarchical structure of tag entries.\n *\n * This function handles:\n * - Sorting tags alphabetically or using a custom sort function\n * - Ensuring the default tag appears last\n * - Sorting operations within tags by title, method, or custom function\n * - Filtering out internal and ignored tags\n * - Creating tag entries with their associated operations\n *\n * @param _keys - Array of tag keys to process\n * @param tagsMap - Map of tags and their entries\n * @param tagsDict - Dictionary of OpenAPI tags by name\n * @param titlesMap - Map of titles for the mobile header\n * @param options - Sorting and ID generation options\n * @returns Array of processed and sorted tag entries\n */\n/** Sorts tags and returns entries */\nconst getSortedTagEntries = ({\n _keys,\n tagsMap,\n options: { tagsSorter, operationsSorter, generateId },\n documentId,\n}: {\n _keys: string[]\n /** Map of tags and their entries */\n tagsMap: TagsMap\n options: Options\n documentId: string\n}) => {\n // Ensure that default is last if it exists\n const hasDefault = _keys.includes('default')\n const keys = hasDefault ? _keys.filter((key) => key !== 'default') : _keys\n\n // Alpha sort\n if (tagsSorter === 'alpha') {\n keys.sort((a, b) => {\n const nameA =\n getTag({\n tagsMap,\n name: a,\n documentId,\n generateId,\n }).tag['x-displayName'] ||\n a ||\n 'Untitled Tag'\n const nameB = getTag({ tagsMap, name: b, documentId, generateId }).tag['x-displayName'] || b || 'Untitled Tag'\n return nameA.localeCompare(nameB)\n })\n }\n // Custom sort\n else if (typeof tagsSorter === 'function') {\n keys.sort((a, b) =>\n tagsSorter(\n getTag({ tagsMap, name: a, documentId, generateId }).tag,\n getTag({ tagsMap, name: b, documentId, generateId }).tag,\n ),\n )\n }\n\n if (hasDefault) {\n keys.push('default')\n }\n\n /**\n * Process each tag and its entries:\n * - Skip internal and ignored tags\n * - Sort operations within tags\n * - Create tag entries with sorted operations\n */\n return keys.flatMap((key) => {\n const { tag, entries } = getTag({ tagsMap, name: key, documentId, generateId })\n\n // Skip if the tag is internal or scalar-ignore\n if (tag['x-internal'] || tag['x-scalar-ignore']) {\n return []\n }\n\n // Alpha sort\n if (operationsSorter === 'alpha') {\n entries.sort((a, b) => (a.type === 'operation' && b.type === 'operation' ? a.title.localeCompare(b.title) : 0))\n }\n // Method sort\n else if (operationsSorter === 'method') {\n entries.sort((a, b) => (a.type === 'operation' && b.type === 'operation' ? a.method.localeCompare(b.method) : 0))\n }\n // Custom sort\n else if (typeof operationsSorter === 'function') {\n entries.sort((a, b) => {\n // Guard against tags\n if (!(a.type === 'operation' || a.type === 'webhook') || !(b.type === 'operation' || b.type === 'webhook')) {\n return 0\n }\n\n // Handle webhooks as well as operations\n const pathA = a.type === 'operation' ? a.path : a.name\n const pathB = b.type === 'operation' ? b.path : b.name\n\n return operationsSorter(\n { method: a.method, path: pathA, ref: a.ref, httpVerb: a.method },\n { method: b.method, path: pathB, ref: b.ref, httpVerb: b.method },\n )\n })\n }\n\n return entries.length\n ? createTagEntry({\n tag,\n generateId,\n children: entries,\n parentId: documentId,\n isGroup: false,\n })\n : []\n })\n}\n\n/**\n * Traverses the tags map to create navigation entries, handling both grouped and ungrouped tags.\n *\n * This function processes the OpenAPI document's tags to:\n * - Handle tag groups if specified via x-tagGroups\n * - Sort tags and their operations according to provided sorters\n * - Create navigation entries for each tag or tag group\n * - Flatten default tag entries if it's the only tag present\n */\nexport const traverseTags = ({\n document,\n tagsMap,\n documentId,\n options: { generateId, tagsSorter, operationsSorter },\n}: {\n document: OpenApiDocument\n /** Map of tags and their entries */\n tagsMap: TagsMap\n documentId: string\n options: Options\n}): TraversedEntry[] => {\n // x-tagGroups\n if (document['x-tagGroups']) {\n const tagGroups = document['x-tagGroups']\n\n return tagGroups.flatMap((tagGroup) => {\n const entries = getSortedTagEntries({\n _keys: tagGroup.tags,\n tagsMap,\n options: { tagsSorter, operationsSorter, generateId },\n documentId: documentId,\n })\n return entries.length\n ? createTagEntry({\n tag: tagGroup,\n generateId,\n children: entries,\n parentId: documentId,\n isGroup: true,\n })\n : []\n })\n }\n\n // Ungrouped regular tags\n const keys = Array.from(tagsMap.keys())\n const tags = getSortedTagEntries({\n _keys: keys,\n tagsMap,\n options: { generateId, tagsSorter, operationsSorter },\n documentId: documentId,\n })\n\n // Flatten if we only have default tag\n if (tags.length === 1 && tags[0]?.title === 'default') {\n return tags[0]?.children ?? []\n }\n\n return tags\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -16,7 +16,13 @@ import type { OpenApiDocument } from '../../schemas/v3.1/strict/openapi-document
16
16
  * @param getWebhookId - Function to generate unique IDs for webhooks
17
17
  * @returns Array of untagged webhook entries
18
18
  */
19
- export declare const traverseWebhooks: (content: OpenApiDocument,
20
- /** The tag map from from traversing paths */
21
- tagsMap: TagsMap, getWebhookId: TraverseSpecOptions["getWebhookId"]) => TraversedWebhook[];
19
+ export declare const traverseWebhooks: ({ document, tagsMap, generateId, untaggedWebhooksParentId, documentId, }: {
20
+ /** Openapi document */
21
+ document: OpenApiDocument;
22
+ /** The tag map from from traversing paths */
23
+ tagsMap: TagsMap;
24
+ generateId: TraverseSpecOptions["generateId"];
25
+ untaggedWebhooksParentId: string;
26
+ documentId: string;
27
+ }) => TraversedWebhook[];
22
28
  //# sourceMappingURL=traverse-webhooks.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"traverse-webhooks.d.ts","sourceRoot":"","sources":["../../../src/navigation/helpers/traverse-webhooks.ts"],"names":[],"mappings":"AAMA,OAAO,KAAK,EAAE,OAAO,EAAE,mBAAmB,EAAE,MAAM,oBAAoB,CAAA;AACtE,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,sBAAsB,CAAA;AAC5D,OAAO,KAAK,EAAE,eAAe,EAAa,MAAM,wCAAwC,CAAA;AAuCxF;;;;;;;;;;;;;;GAcG;AACH,eAAO,MAAM,gBAAgB,GAC3B,SAAS,eAAe;AACxB,6CAA6C;AAC7C,SAAS,OAAO,EAChB,cAAc,mBAAmB,CAAC,cAAc,CAAC,KAChD,gBAAgB,EAyDlB,CAAA"}
1
+ {"version":3,"file":"traverse-webhooks.d.ts","sourceRoot":"","sources":["../../../src/navigation/helpers/traverse-webhooks.ts"],"names":[],"mappings":"AAMA,OAAO,KAAK,EAAE,OAAO,EAAE,mBAAmB,EAAE,MAAM,oBAAoB,CAAA;AACtE,OAAO,KAAK,EAAa,gBAAgB,EAAE,MAAM,sBAAsB,CAAA;AACvE,OAAO,KAAK,EAAE,eAAe,EAA8B,MAAM,wCAAwC,CAAA;AA2DzG;;;;;;;;;;;;;;GAcG;AACH,eAAO,MAAM,gBAAgB,GAAI,0EAM9B;IACD,uBAAuB;IACvB,QAAQ,EAAE,eAAe,CAAA;IACzB,6CAA6C;IAC7C,OAAO,EAAE,OAAO,CAAA;IAChB,UAAU,EAAE,mBAAmB,CAAC,YAAY,CAAC,CAAA;IAC7C,wBAAwB,EAAE,MAAM,CAAA;IAChC,UAAU,EAAE,MAAM,CAAA;CACnB,KAAG,gBAAgB,EA0DnB,CAAA"}
@@ -3,8 +3,25 @@ import { objectKeys } from "@scalar/helpers/object/object-keys";
3
3
  import { getResolvedRef } from "../../helpers/get-resolved-ref.js";
4
4
  import { isDeprecatedOperation } from "../../navigation/helpers/traverse-paths.js";
5
5
  import { getTag } from "./get-tag.js";
6
- const createWebhookEntry = (ref, method, name = "Unknown", title = "Unknown", getWebhookId, tag, isDeprecated) => {
7
- const id = getWebhookId({ name, method }, tag);
6
+ const createWebhookEntry = ({
7
+ ref,
8
+ method,
9
+ name,
10
+ title,
11
+ generateId,
12
+ parentTag,
13
+ webhook,
14
+ isDeprecated,
15
+ parentId
16
+ }) => {
17
+ const id = generateId({
18
+ type: "webhook",
19
+ name,
20
+ method,
21
+ webhook,
22
+ parentTag,
23
+ parentId
24
+ });
8
25
  const entry = {
9
26
  id,
10
27
  title,
@@ -16,9 +33,15 @@ const createWebhookEntry = (ref, method, name = "Unknown", title = "Unknown", ge
16
33
  };
17
34
  return entry;
18
35
  };
19
- const traverseWebhooks = (content, tagsMap, getWebhookId) => {
36
+ const traverseWebhooks = ({
37
+ document,
38
+ tagsMap,
39
+ generateId,
40
+ untaggedWebhooksParentId,
41
+ documentId
42
+ }) => {
20
43
  const untagged = [];
21
- Object.entries(content.webhooks ?? {}).forEach(([name, pathItemObject]) => {
44
+ Object.entries(document.webhooks ?? {}).forEach(([name, pathItemObject]) => {
22
45
  const pathKeys = objectKeys(pathItemObject ?? {}).filter((key) => isHttpMethod(key));
23
46
  pathKeys.forEach((method) => {
24
47
  const _operation = pathItemObject?.[method];
@@ -32,30 +55,33 @@ const traverseWebhooks = (content, tagsMap, getWebhookId) => {
32
55
  const ref = `#/webhooks/${name}/${method}`;
33
56
  if (operation.tags?.length) {
34
57
  operation.tags.forEach((tagName) => {
35
- const { tag } = getTag(tagsMap, tagName);
58
+ const { tag, id: tagId } = getTag({ tagsMap, name: tagName, documentId, generateId });
36
59
  tagsMap.get(tagName)?.entries.push(
37
- createWebhookEntry(
60
+ createWebhookEntry({
38
61
  ref,
39
62
  method,
40
63
  name,
41
- operation.summary ?? name,
42
- getWebhookId,
43
- tag,
44
- isDeprecatedOperation(operation)
45
- )
64
+ title: operation.summary ?? name,
65
+ webhook: operation,
66
+ generateId,
67
+ parentTag: { tag, id: tagId },
68
+ parentId: tagId,
69
+ isDeprecated: isDeprecatedOperation(operation)
70
+ })
46
71
  );
47
72
  });
48
73
  } else {
49
74
  untagged.push(
50
- createWebhookEntry(
75
+ createWebhookEntry({
51
76
  ref,
52
77
  method,
53
78
  name,
54
- operation.summary ?? name,
55
- getWebhookId,
56
- void 0,
57
- isDeprecatedOperation(operation)
58
- )
79
+ title: operation.summary ?? name,
80
+ generateId,
81
+ isDeprecated: isDeprecatedOperation(operation),
82
+ webhook: operation,
83
+ parentId: untaggedWebhooksParentId
84
+ })
59
85
  );
60
86
  }
61
87
  });