@redocly/openapi-core 1.0.0 → 1.0.2

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 (275) hide show
  1. package/CHANGELOG.md +9 -0
  2. package/__tests__/utils.ts +88 -0
  3. package/lib/config/all.js +0 -1
  4. package/lib/config/minimal.js +0 -1
  5. package/lib/config/recommended.js +0 -1
  6. package/package.json +1 -1
  7. package/src/__tests__/__snapshots__/bundle.test.ts.snap +437 -0
  8. package/src/__tests__/bundle.test.ts +236 -0
  9. package/src/__tests__/codeframes.test.ts +530 -0
  10. package/src/__tests__/fixtures/.redocly.lint-ignore.yaml +5 -0
  11. package/src/__tests__/fixtures/extension.js +24 -0
  12. package/src/__tests__/fixtures/refs/definitions.yaml +3 -0
  13. package/src/__tests__/fixtures/refs/examples.yaml +8 -0
  14. package/src/__tests__/fixtures/refs/external-request-body.yaml +13 -0
  15. package/src/__tests__/fixtures/refs/externalref.yaml +35 -0
  16. package/src/__tests__/fixtures/refs/hosted.yaml +35 -0
  17. package/src/__tests__/fixtures/refs/openapi-with-external-refs-conflicting-names.yaml +21 -0
  18. package/src/__tests__/fixtures/refs/openapi-with-external-refs.yaml +33 -0
  19. package/src/__tests__/fixtures/refs/openapi-with-url-refs.yaml +18 -0
  20. package/src/__tests__/fixtures/refs/param-b.yaml +1 -0
  21. package/src/__tests__/fixtures/refs/param-c.yaml +1 -0
  22. package/src/__tests__/fixtures/refs/rename.yaml +1 -0
  23. package/src/__tests__/fixtures/refs/requestBody.yaml +9 -0
  24. package/src/__tests__/fixtures/refs/schema-a.yaml +1 -0
  25. package/src/__tests__/fixtures/refs/simple.yaml +1 -0
  26. package/src/__tests__/fixtures/refs/vendor.schema.yaml +20 -0
  27. package/src/__tests__/fixtures/resolve/External.yaml +10 -0
  28. package/src/__tests__/fixtures/resolve/External2.yaml +4 -0
  29. package/src/__tests__/fixtures/resolve/description.md +3 -0
  30. package/src/__tests__/fixtures/resolve/externalInfo.yaml +4 -0
  31. package/src/__tests__/fixtures/resolve/externalLicense.yaml +1 -0
  32. package/src/__tests__/fixtures/resolve/openapi-with-back.yaml +13 -0
  33. package/src/__tests__/fixtures/resolve/openapi-with-md-description.yaml +5 -0
  34. package/src/__tests__/fixtures/resolve/openapi.yaml +28 -0
  35. package/src/__tests__/fixtures/resolve/schemas/type-a.yaml +10 -0
  36. package/src/__tests__/fixtures/resolve/schemas/type-b.yaml +6 -0
  37. package/src/__tests__/fixtures/resolve/transitive/a.yaml +1 -0
  38. package/src/__tests__/fixtures/resolve/transitive/components.yaml +5 -0
  39. package/src/__tests__/fixtures/resolve/transitive/schemas.yaml +3 -0
  40. package/src/__tests__/format.test.ts +76 -0
  41. package/src/__tests__/js-yaml.test.ts +73 -0
  42. package/src/__tests__/lint.test.ts +392 -0
  43. package/src/__tests__/logger-browser.test.ts +53 -0
  44. package/src/__tests__/logger.test.ts +47 -0
  45. package/src/__tests__/login.test.ts +17 -0
  46. package/src/__tests__/normalizeVisitors.test.ts +151 -0
  47. package/src/__tests__/output-browser.test.ts +18 -0
  48. package/src/__tests__/output.test.ts +15 -0
  49. package/src/__tests__/ref-utils.test.ts +120 -0
  50. package/src/__tests__/resolve-http.test.ts +77 -0
  51. package/src/__tests__/resolve.test.ts +431 -0
  52. package/src/__tests__/utils-browser.test.ts +11 -0
  53. package/src/__tests__/utils.test.ts +144 -0
  54. package/src/__tests__/walk.test.ts +1545 -0
  55. package/src/benchmark/benches/lint-with-many-rules.bench.ts +35 -0
  56. package/src/benchmark/benches/lint-with-nested-rule.bench.ts +39 -0
  57. package/src/benchmark/benches/lint-with-no-rules.bench.ts +20 -0
  58. package/src/benchmark/benches/lint-with-top-level-rule-report.bench.ts +35 -0
  59. package/src/benchmark/benches/lint-with-top-level-rule.bench.ts +32 -0
  60. package/src/benchmark/benches/rebilly.yaml +32275 -0
  61. package/src/benchmark/benches/recommended-oas3.bench.ts +22 -0
  62. package/src/benchmark/benches/resolve-with-no-external.bench.ts +23 -0
  63. package/src/benchmark/benchmark.js +311 -0
  64. package/src/benchmark/colors.js +29 -0
  65. package/src/benchmark/fork.js +83 -0
  66. package/src/benchmark/utils.ts +36 -0
  67. package/src/bundle.ts +417 -0
  68. package/src/config/__tests__/__snapshots__/config-resolvers.test.ts.snap +164 -0
  69. package/src/config/__tests__/__snapshots__/config.test.ts.snap +144 -0
  70. package/src/config/__tests__/config-resolvers.test.ts +491 -0
  71. package/src/config/__tests__/config.test.ts +312 -0
  72. package/src/config/__tests__/fixtures/ingore-file.ts +8 -0
  73. package/src/config/__tests__/fixtures/load-redocly.yaml +2 -0
  74. package/src/config/__tests__/fixtures/plugin-config.yaml +2 -0
  75. package/src/config/__tests__/fixtures/plugin.js +56 -0
  76. package/src/config/__tests__/fixtures/resolve-config/api/nested-config.yaml +11 -0
  77. package/src/config/__tests__/fixtures/resolve-config/api/plugin.js +69 -0
  78. package/src/config/__tests__/fixtures/resolve-config/local-config-with-circular.yaml +7 -0
  79. package/src/config/__tests__/fixtures/resolve-config/local-config-with-custom-function.yaml +17 -0
  80. package/src/config/__tests__/fixtures/resolve-config/local-config-with-file.yaml +18 -0
  81. package/src/config/__tests__/fixtures/resolve-config/local-config-with-wrong-custom-function.yaml +15 -0
  82. package/src/config/__tests__/fixtures/resolve-config/local-config.yaml +9 -0
  83. package/src/config/__tests__/fixtures/resolve-config/plugin.js +80 -0
  84. package/src/config/__tests__/fixtures/resolve-remote-configs/nested-remote-config.yaml +3 -0
  85. package/src/config/__tests__/fixtures/resolve-remote-configs/remote-config.yaml +4 -0
  86. package/src/config/__tests__/load.test.ts +167 -0
  87. package/src/config/__tests__/resolve-plugins.test.ts +27 -0
  88. package/src/config/__tests__/utils.test.ts +204 -0
  89. package/src/config/all.ts +74 -0
  90. package/src/config/builtIn.ts +37 -0
  91. package/src/config/config-resolvers.ts +474 -0
  92. package/src/config/config.ts +332 -0
  93. package/src/config/index.ts +7 -0
  94. package/src/config/load.ts +144 -0
  95. package/src/config/minimal.ts +61 -0
  96. package/src/config/recommended.ts +61 -0
  97. package/src/config/rules.ts +54 -0
  98. package/src/config/types.ts +231 -0
  99. package/src/config/utils.ts +349 -0
  100. package/src/decorators/__tests__/filter-in.test.ts +310 -0
  101. package/src/decorators/__tests__/filter-out.test.ts +335 -0
  102. package/src/decorators/__tests__/media-type-examples-override.test.ts +665 -0
  103. package/src/decorators/__tests__/remove-x-internal.test.ts +316 -0
  104. package/src/decorators/__tests__/resources/request.yaml +3 -0
  105. package/src/decorators/__tests__/resources/response.yaml +3 -0
  106. package/src/decorators/common/filters/filter-helper.ts +72 -0
  107. package/src/decorators/common/filters/filter-in.ts +18 -0
  108. package/src/decorators/common/filters/filter-out.ts +18 -0
  109. package/src/decorators/common/info-description-override.ts +24 -0
  110. package/src/decorators/common/info-override.ts +15 -0
  111. package/src/decorators/common/media-type-examples-override.ts +79 -0
  112. package/src/decorators/common/operation-description-override.ts +30 -0
  113. package/src/decorators/common/registry-dependencies.ts +25 -0
  114. package/src/decorators/common/remove-x-internal.ts +59 -0
  115. package/src/decorators/common/tag-description-override.ts +25 -0
  116. package/src/decorators/oas2/index.ts +20 -0
  117. package/src/decorators/oas3/index.ts +22 -0
  118. package/src/env.ts +5 -0
  119. package/src/format/codeframes.ts +216 -0
  120. package/src/format/format.ts +375 -0
  121. package/src/index.ts +71 -0
  122. package/src/js-yaml/index.ts +14 -0
  123. package/src/lint.ts +148 -0
  124. package/src/logger.ts +34 -0
  125. package/src/oas-types.ts +57 -0
  126. package/src/output.ts +7 -0
  127. package/src/redocly/__tests__/redocly-client.test.ts +146 -0
  128. package/src/redocly/index.ts +187 -0
  129. package/src/redocly/redocly-client-types.ts +10 -0
  130. package/src/redocly/registry-api-types.ts +32 -0
  131. package/src/redocly/registry-api.ts +150 -0
  132. package/src/ref-utils.ts +85 -0
  133. package/src/resolve.ts +417 -0
  134. package/src/rules/__tests__/fixtures/code-sample.php +9 -0
  135. package/src/rules/__tests__/fixtures/invalid-yaml.yaml +1 -0
  136. package/src/rules/__tests__/fixtures/ref.yaml +1 -0
  137. package/src/rules/__tests__/no-unresolved-refs.test.ts +257 -0
  138. package/src/rules/__tests__/utils.test.ts +160 -0
  139. package/src/rules/ajv.ts +102 -0
  140. package/src/rules/common/__tests__/info-license.test.ts +62 -0
  141. package/src/rules/common/__tests__/license-url.test.ts +63 -0
  142. package/src/rules/common/__tests__/no-ambiguous-paths.test.ts +96 -0
  143. package/src/rules/common/__tests__/no-enum-type-mismatch.test.ts +210 -0
  144. package/src/rules/common/__tests__/no-identical-paths.test.ts +58 -0
  145. package/src/rules/common/__tests__/no-path-trailing-slash.test.ts +85 -0
  146. package/src/rules/common/__tests__/operation-2xx-response.test.ts +192 -0
  147. package/src/rules/common/__tests__/operation-4xx-response.test.ts +231 -0
  148. package/src/rules/common/__tests__/operation-operationId-unique.test.ts +76 -0
  149. package/src/rules/common/__tests__/operation-operationId-url-safe.test.ts +45 -0
  150. package/src/rules/common/__tests__/operation-parameters-unique.test.ts +167 -0
  151. package/src/rules/common/__tests__/operation-singular-tag.test.ts +72 -0
  152. package/src/rules/common/__tests__/path-http-verbs-order.test.ts +95 -0
  153. package/src/rules/common/__tests__/path-not-include-query.test.ts +64 -0
  154. package/src/rules/common/__tests__/path-params-defined.test.ts +202 -0
  155. package/src/rules/common/__tests__/paths-kebab-case.test.ts +108 -0
  156. package/src/rules/common/__tests__/scalar-property-missing-example.test.ts +264 -0
  157. package/src/rules/common/__tests__/security-defined.test.ts +175 -0
  158. package/src/rules/common/__tests__/spec-strict-refs.test.ts +69 -0
  159. package/src/rules/common/__tests__/spec.test.ts +610 -0
  160. package/src/rules/common/__tests__/tag-description.test.ts +65 -0
  161. package/src/rules/common/__tests__/tags-alphabetical.test.ts +64 -0
  162. package/src/rules/common/assertions/__tests__/asserts.test.ts +869 -0
  163. package/src/rules/common/assertions/__tests__/index.test.ts +100 -0
  164. package/src/rules/common/assertions/__tests__/utils.test.ts +236 -0
  165. package/src/rules/common/assertions/asserts.ts +357 -0
  166. package/src/rules/common/assertions/index.ts +53 -0
  167. package/src/rules/common/assertions/utils.ts +331 -0
  168. package/src/rules/common/info-contact.ts +15 -0
  169. package/src/rules/common/info-license-url.ts +10 -0
  170. package/src/rules/common/info-license.ts +15 -0
  171. package/src/rules/common/no-ambiguous-paths.ts +50 -0
  172. package/src/rules/common/no-enum-type-mismatch.ts +52 -0
  173. package/src/rules/common/no-http-verbs-in-paths.ts +36 -0
  174. package/src/rules/common/no-identical-paths.ts +24 -0
  175. package/src/rules/common/no-invalid-parameter-examples.ts +36 -0
  176. package/src/rules/common/no-invalid-schema-examples.ts +27 -0
  177. package/src/rules/common/no-path-trailing-slash.ts +15 -0
  178. package/src/rules/common/operation-2xx-response.ts +24 -0
  179. package/src/rules/common/operation-4xx-response.ts +24 -0
  180. package/src/rules/common/operation-description.ts +13 -0
  181. package/src/rules/common/operation-operationId-unique.ts +21 -0
  182. package/src/rules/common/operation-operationId-url-safe.ts +19 -0
  183. package/src/rules/common/operation-operationId.ts +17 -0
  184. package/src/rules/common/operation-parameters-unique.ts +48 -0
  185. package/src/rules/common/operation-singular-tag.ts +17 -0
  186. package/src/rules/common/operation-summary.ts +13 -0
  187. package/src/rules/common/operation-tag-defined.ts +26 -0
  188. package/src/rules/common/parameter-description.ts +22 -0
  189. package/src/rules/common/path-declaration-must-exist.ts +15 -0
  190. package/src/rules/common/path-excludes-patterns.ts +23 -0
  191. package/src/rules/common/path-http-verbs-order.ts +30 -0
  192. package/src/rules/common/path-not-include-query.ts +17 -0
  193. package/src/rules/common/path-params-defined.ts +65 -0
  194. package/src/rules/common/path-segment-plural.ts +31 -0
  195. package/src/rules/common/paths-kebab-case.ts +19 -0
  196. package/src/rules/common/required-string-property-missing-min-length.ts +44 -0
  197. package/src/rules/common/response-contains-header.ts +35 -0
  198. package/src/rules/common/scalar-property-missing-example.ts +58 -0
  199. package/src/rules/common/security-defined.ts +65 -0
  200. package/src/rules/common/spec-strict-refs.ts +30 -0
  201. package/src/rules/common/spec.ts +175 -0
  202. package/src/rules/common/tag-description.ts +10 -0
  203. package/src/rules/common/tags-alphabetical.ts +20 -0
  204. package/src/rules/no-unresolved-refs.ts +51 -0
  205. package/src/rules/oas2/__tests__/boolean-parameter-prefixes.test.ts +110 -0
  206. package/src/rules/oas2/__tests__/response-contains-header.test.ts +174 -0
  207. package/src/rules/oas2/__tests__/response-contains-property.test.ts +155 -0
  208. package/src/rules/oas2/__tests__/spec/fixtures/description.md +1 -0
  209. package/src/rules/oas2/__tests__/spec/info.test.ts +355 -0
  210. package/src/rules/oas2/__tests__/spec/operation.test.ts +123 -0
  211. package/src/rules/oas2/__tests__/spec/paths.test.ts +245 -0
  212. package/src/rules/oas2/__tests__/spec/referenceableScalars.test.ts +35 -0
  213. package/src/rules/oas2/__tests__/spec/utils.ts +32 -0
  214. package/src/rules/oas2/boolean-parameter-prefixes.ts +26 -0
  215. package/src/rules/oas2/index.ts +91 -0
  216. package/src/rules/oas2/remove-unused-components.ts +81 -0
  217. package/src/rules/oas2/request-mime-type.ts +16 -0
  218. package/src/rules/oas2/response-contains-property.ts +36 -0
  219. package/src/rules/oas2/response-mime-type.ts +16 -0
  220. package/src/rules/oas3/__tests__/boolean-parameter-prefixes.test.ts +111 -0
  221. package/src/rules/oas3/__tests__/component-name-unique.test.ts +823 -0
  222. package/src/rules/oas3/__tests__/fixtures/common.yaml +11 -0
  223. package/src/rules/oas3/__tests__/no-empty-enum-servers.com.test.ts +205 -0
  224. package/src/rules/oas3/__tests__/no-example-value-and-externalValue.test.ts +65 -0
  225. package/src/rules/oas3/__tests__/no-invalid-media-type-examples.test.ts +473 -0
  226. package/src/rules/oas3/__tests__/no-server-example.com.test.ts +60 -0
  227. package/src/rules/oas3/__tests__/no-server-trailing-slash.test.ts +79 -0
  228. package/src/rules/oas3/__tests__/no-unused-components.test.ts +131 -0
  229. package/src/rules/oas3/__tests__/operation-4xx-problem-details-rfc7807.test.ts +145 -0
  230. package/src/rules/oas3/__tests__/response-contains-header.test.ts +389 -0
  231. package/src/rules/oas3/__tests__/response-contains-property.test.ts +403 -0
  232. package/src/rules/oas3/__tests__/spec/callbacks.test.ts +41 -0
  233. package/src/rules/oas3/__tests__/spec/fixtures/description.md +1 -0
  234. package/src/rules/oas3/__tests__/spec/info.test.ts +391 -0
  235. package/src/rules/oas3/__tests__/spec/operation.test.ts +253 -0
  236. package/src/rules/oas3/__tests__/spec/paths.test.ts +284 -0
  237. package/src/rules/oas3/__tests__/spec/referenceableScalars.test.ts +77 -0
  238. package/src/rules/oas3/__tests__/spec/servers.test.ts +505 -0
  239. package/src/rules/oas3/__tests__/spec/spec.test.ts +298 -0
  240. package/src/rules/oas3/__tests__/spec/utils.ts +32 -0
  241. package/src/rules/oas3/__tests__/spec-components-invalid-map-name.test.ts +276 -0
  242. package/src/rules/oas3/__tests__/utils/lint-document-for-test.ts +23 -0
  243. package/src/rules/oas3/boolean-parameter-prefixes.ts +28 -0
  244. package/src/rules/oas3/component-name-unique.ts +158 -0
  245. package/src/rules/oas3/index.ts +113 -0
  246. package/src/rules/oas3/no-empty-servers.ts +22 -0
  247. package/src/rules/oas3/no-example-value-and-externalValue.ts +14 -0
  248. package/src/rules/oas3/no-invalid-media-type-examples.ts +49 -0
  249. package/src/rules/oas3/no-server-example.com.ts +14 -0
  250. package/src/rules/oas3/no-server-trailing-slash.ts +15 -0
  251. package/src/rules/oas3/no-server-variables-empty-enum.ts +66 -0
  252. package/src/rules/oas3/no-undefined-server-variable.ts +30 -0
  253. package/src/rules/oas3/no-unused-components.ts +75 -0
  254. package/src/rules/oas3/operation-4xx-problem-details-rfc7807.ts +35 -0
  255. package/src/rules/oas3/remove-unused-components.ts +95 -0
  256. package/src/rules/oas3/request-mime-type.ts +30 -0
  257. package/src/rules/oas3/response-contains-property.ts +38 -0
  258. package/src/rules/oas3/response-mime-type.ts +30 -0
  259. package/src/rules/oas3/spec-components-invalid-map-name.ts +69 -0
  260. package/src/rules/other/stats.ts +73 -0
  261. package/src/rules/utils.ts +193 -0
  262. package/src/types/config-external-schemas.ts +917 -0
  263. package/src/types/index.ts +149 -0
  264. package/src/types/oas2.ts +478 -0
  265. package/src/types/oas3.ts +597 -0
  266. package/src/types/oas3_1.ts +258 -0
  267. package/src/types/redocly-yaml.ts +1040 -0
  268. package/src/typings/common.ts +17 -0
  269. package/src/typings/openapi.ts +298 -0
  270. package/src/typings/swagger.ts +236 -0
  271. package/src/utils.ts +276 -0
  272. package/src/visitors.ts +491 -0
  273. package/src/walk.ts +439 -0
  274. package/tsconfig.json +8 -0
  275. package/tsconfig.tsbuildinfo +1 -0
@@ -0,0 +1,491 @@
1
+ import type {
2
+ Oas3Definition,
3
+ Oas3ExternalDocs,
4
+ Oas3Info,
5
+ Oas3Contact,
6
+ Oas3Components,
7
+ Oas3License,
8
+ Oas3Schema,
9
+ Oas3Header,
10
+ Oas3Parameter,
11
+ Oas3Operation,
12
+ Oas3PathItem,
13
+ Oas3ServerVariable,
14
+ Oas3Server,
15
+ Oas3MediaType,
16
+ Oas3Response,
17
+ Oas3Example,
18
+ Oas3RequestBody,
19
+ Oas3Tag,
20
+ OasRef,
21
+ Oas3SecurityScheme,
22
+ Oas3SecurityRequirement,
23
+ Oas3Encoding,
24
+ Oas3Link,
25
+ Oas3Xml,
26
+ Oas3Discriminator,
27
+ Oas3Callback,
28
+ } from './typings/openapi';
29
+
30
+ import type {
31
+ Oas2Definition,
32
+ Oas2Tag,
33
+ Oas2ExternalDocs,
34
+ Oas2SecurityRequirement,
35
+ Oas2Info,
36
+ Oas2Contact,
37
+ Oas2License,
38
+ Oas2PathItem,
39
+ Oas2Operation,
40
+ Oas2Header,
41
+ Oas2Response,
42
+ Oas2Schema,
43
+ Oas2Xml,
44
+ Oas2Parameter,
45
+ Oas2SecurityScheme,
46
+ } from './typings/swagger';
47
+
48
+ import { NormalizedNodeType, SpecExtension } from './types';
49
+ import type { Stack } from './utils';
50
+ import type { UserContext, ResolveResult, ProblemSeverity } from './walk';
51
+ import type { Location } from './ref-utils';
52
+
53
+ export type SkipFunctionContext = Pick<
54
+ UserContext,
55
+ 'location' | 'rawNode' | 'resolve' | 'rawLocation'
56
+ >;
57
+
58
+ export type VisitFunction<T> = (
59
+ node: T,
60
+ ctx: UserContext & { ignoreNextVisitorsOnNode: () => void },
61
+ parents?: any,
62
+ context?: any
63
+ ) => void;
64
+
65
+ type VisitRefFunction = (node: OasRef, ctx: UserContext, resolved: ResolveResult<any>) => void;
66
+
67
+ type SkipFunction<T> = (node: T, key: string | number, ctx: SkipFunctionContext) => boolean;
68
+
69
+ type VisitObject<T> = {
70
+ enter?: VisitFunction<T>;
71
+ leave?: VisitFunction<T>;
72
+ skip?: SkipFunction<T>;
73
+ };
74
+
75
+ type NestedVisitObject<T, P> = VisitObject<T> & NestedVisitor<P>;
76
+
77
+ type VisitFunctionOrObject<T> = VisitFunction<T> | VisitObject<T>;
78
+
79
+ type VisitorNode<T> = {
80
+ ruleId: string;
81
+ severity: ProblemSeverity;
82
+ context: VisitorLevelContext | VisitorSkippedLevelContext;
83
+ depth: number;
84
+ visit: VisitFunction<T>;
85
+ skip?: SkipFunction<T>;
86
+ };
87
+
88
+ type VisitorRefNode = {
89
+ ruleId: string;
90
+ severity: ProblemSeverity;
91
+ context: VisitorLevelContext;
92
+ depth: number;
93
+ visit: VisitRefFunction;
94
+ };
95
+
96
+ export type VisitorLevelContext = {
97
+ isSkippedLevel: false;
98
+ type: NormalizedNodeType;
99
+ parent: VisitorLevelContext | null;
100
+ activatedOn: Stack<{
101
+ node?: any;
102
+ withParentNode?: any;
103
+ skipped: boolean;
104
+ nextLevelTypeActivated: Stack<NormalizedNodeType>;
105
+ location?: Location;
106
+ }>;
107
+ };
108
+
109
+ export type VisitorSkippedLevelContext = {
110
+ isSkippedLevel: true;
111
+ parent: VisitorLevelContext;
112
+ seen: Set<any>;
113
+ };
114
+
115
+ type NormalizeVisitor<Fn> = Fn extends VisitFunction<infer T> ? VisitorNode<T> : never;
116
+
117
+ export type BaseVisitor = {
118
+ any?:
119
+ | {
120
+ enter?: VisitFunction<any>;
121
+ leave?: VisitFunction<any>;
122
+ skip?: SkipFunction<any>;
123
+ }
124
+ | VisitFunction<any>;
125
+
126
+ ref?:
127
+ | {
128
+ enter?: VisitRefFunction;
129
+ leave?: VisitRefFunction;
130
+ }
131
+ | VisitRefFunction;
132
+ };
133
+
134
+ type Oas3FlatVisitor = {
135
+ Root?: VisitFunctionOrObject<Oas3Definition>;
136
+ Tag?: VisitFunctionOrObject<Oas3Tag>;
137
+ ExternalDocs?: VisitFunctionOrObject<Oas3ExternalDocs>;
138
+ Server?: VisitFunctionOrObject<Oas3Server>;
139
+ ServerVariable?: VisitFunctionOrObject<Oas3ServerVariable>;
140
+ SecurityRequirement?: VisitFunctionOrObject<Oas3SecurityRequirement>;
141
+ Info?: VisitFunctionOrObject<Oas3Info>;
142
+ Contact?: VisitFunctionOrObject<Oas3Contact>;
143
+ License?: VisitFunctionOrObject<Oas3License>;
144
+ Paths?: VisitFunctionOrObject<Record<string, Oas3PathItem>>;
145
+ PathItem?: VisitFunctionOrObject<Oas3PathItem>;
146
+ Callback?: VisitFunctionOrObject<Oas3Callback>;
147
+ CallbacksMap?: VisitFunctionOrObject<Record<string, Oas3Callback>>;
148
+ Parameter?: VisitFunctionOrObject<Oas3Parameter>;
149
+ Operation?: VisitFunctionOrObject<Oas3Operation>;
150
+ RequestBody?: VisitFunctionOrObject<Oas3RequestBody>;
151
+ MediaTypesMap?: VisitFunctionOrObject<Record<string, Oas3MediaType>>;
152
+ MediaType?: VisitFunctionOrObject<Oas3MediaType>;
153
+ Example?: VisitFunctionOrObject<Oas3Example>;
154
+ Encoding?: VisitFunctionOrObject<Oas3Encoding>;
155
+ Header?: VisitFunctionOrObject<Oas3Header>;
156
+ Responses?: VisitFunctionOrObject<Record<string, Oas3Response>>;
157
+ Response?: VisitFunctionOrObject<Oas3Response>;
158
+ Link?: VisitFunctionOrObject<Oas3Link>;
159
+ Schema?: VisitFunctionOrObject<Oas3Schema>;
160
+ Xml?: VisitFunctionOrObject<Oas3Xml>;
161
+ SchemaProperties?: VisitFunctionOrObject<Record<string, Oas3Schema>>;
162
+ DiscriminatorMapping?: VisitFunctionOrObject<Record<string, string>>;
163
+ Discriminator?: VisitFunctionOrObject<Oas3Discriminator>;
164
+ Components?: VisitFunctionOrObject<Oas3Components>;
165
+ NamedSchemas?: VisitFunctionOrObject<Record<string, Oas3Schema>>;
166
+ NamedResponses?: VisitFunctionOrObject<Record<string, Oas3Response>>;
167
+ NamedParameters?: VisitFunctionOrObject<Record<string, Oas3Parameter>>;
168
+ NamedExamples?: VisitFunctionOrObject<Record<string, Oas3Example>>;
169
+ NamedRequestBodies?: VisitFunctionOrObject<Record<string, Oas3RequestBody>>;
170
+ NamedHeaders?: VisitFunctionOrObject<Record<string, Oas3Header>>;
171
+ NamedSecuritySchemes?: VisitFunctionOrObject<Record<string, Oas3SecurityScheme>>;
172
+ NamedLinks?: VisitFunctionOrObject<Record<string, Oas3Link>>;
173
+ NamedCallbacks?: VisitFunctionOrObject<Record<string, Oas3Callback>>;
174
+ ImplicitFlow?: VisitFunctionOrObject<Oas3SecurityScheme['flows']['implicit']>;
175
+ PasswordFlow?: VisitFunctionOrObject<Oas3SecurityScheme['flows']['password']>;
176
+ ClientCredentials?: VisitFunctionOrObject<Oas3SecurityScheme['flows']['clientCredentials']>;
177
+ AuthorizationCode?: VisitFunctionOrObject<Oas3SecurityScheme['flows']['authorizationCode']>;
178
+ OAuth2Flows?: VisitFunctionOrObject<Oas3SecurityScheme['flows']>;
179
+ SecurityScheme?: VisitFunctionOrObject<Oas3SecurityScheme>;
180
+ SpecExtension?: VisitFunctionOrObject<unknown>;
181
+ };
182
+
183
+ type Oas2FlatVisitor = {
184
+ Root?: VisitFunctionOrObject<Oas2Definition>;
185
+ Tag?: VisitFunctionOrObject<Oas2Tag>;
186
+ ExternalDocs?: VisitFunctionOrObject<Oas2ExternalDocs>;
187
+ SecurityRequirement?: VisitFunctionOrObject<Oas2SecurityRequirement>;
188
+ Info?: VisitFunctionOrObject<Oas2Info>;
189
+ Contact?: VisitFunctionOrObject<Oas2Contact>;
190
+ License?: VisitFunctionOrObject<Oas2License>;
191
+ Paths?: VisitFunctionOrObject<Record<string, Oas2PathItem>>;
192
+ PathItem?: VisitFunctionOrObject<Oas2PathItem>;
193
+ Parameter?: VisitFunctionOrObject<any>;
194
+ Operation?: VisitFunctionOrObject<Oas2Operation>;
195
+ Examples?: VisitFunctionOrObject<Record<string, any>>;
196
+ Header?: VisitFunctionOrObject<Oas2Header>;
197
+ Responses?: VisitFunctionOrObject<Record<string, Oas2Response>>;
198
+ Response?: VisitFunctionOrObject<Oas2Response>;
199
+ Schema?: VisitFunctionOrObject<Oas2Schema>;
200
+ Xml?: VisitFunctionOrObject<Oas2Xml>;
201
+ SchemaProperties?: VisitFunctionOrObject<Record<string, Oas2Schema>>;
202
+ NamedSchemas?: VisitFunctionOrObject<Record<string, Oas2Schema>>;
203
+ NamedResponses?: VisitFunctionOrObject<Record<string, Oas2Response>>;
204
+ NamedParameters?: VisitFunctionOrObject<Record<string, Oas2Parameter>>;
205
+ SecurityScheme?: VisitFunctionOrObject<Oas2SecurityScheme>;
206
+ SpecExtension?: VisitFunctionOrObject<unknown>;
207
+ };
208
+
209
+ const legacyTypesMap = {
210
+ Root: 'DefinitionRoot',
211
+ ServerVariablesMap: 'ServerVariableMap',
212
+ Paths: ['PathMap', 'PathsMap'],
213
+ CallbacksMap: 'CallbackMap',
214
+ MediaTypesMap: 'MediaTypeMap',
215
+ ExamplesMap: 'ExampleMap',
216
+ EncodingMap: 'EncodingsMap',
217
+ HeadersMap: 'HeaderMap',
218
+ LinksMap: 'LinkMap',
219
+ OAuth2Flows: 'SecuritySchemeFlows',
220
+ Responses: 'ResponsesMap',
221
+ };
222
+
223
+ type Oas3NestedVisitor = {
224
+ [T in keyof Oas3FlatVisitor]: Oas3FlatVisitor[T] extends Function
225
+ ? Oas3FlatVisitor[T]
226
+ : Oas3FlatVisitor[T] & NestedVisitor<Oas3NestedVisitor>;
227
+ };
228
+
229
+ type Oas2NestedVisitor = {
230
+ [T in keyof Oas2FlatVisitor]: Oas2FlatVisitor[T] extends Function
231
+ ? Oas2FlatVisitor[T]
232
+ : Oas2FlatVisitor[T] & NestedVisitor<Oas2NestedVisitor>;
233
+ };
234
+
235
+ export type Oas3Visitor = BaseVisitor &
236
+ Oas3NestedVisitor &
237
+ Record<string, VisitFunction<any> | NestedVisitObject<any, Oas3NestedVisitor>>;
238
+
239
+ export type Oas2Visitor = BaseVisitor &
240
+ Oas2NestedVisitor &
241
+ Record<string, VisitFunction<any> | NestedVisitObject<any, Oas2NestedVisitor>>;
242
+
243
+ export type Oas3TransformVisitor = BaseVisitor &
244
+ Oas3FlatVisitor &
245
+ Record<string, VisitFunction<any> | VisitObject<any>>;
246
+
247
+ export type Oas2TransformVisitor = BaseVisitor &
248
+ Oas2FlatVisitor &
249
+ Record<string, VisitFunction<any> | VisitObject<any>>;
250
+
251
+ export type NestedVisitor<T> = Exclude<T, 'any' | 'ref' | 'Root'>;
252
+
253
+ export type NormalizedOasVisitors<T extends BaseVisitor> = {
254
+ [V in keyof T]-?: {
255
+ enter: Array<NormalizeVisitor<T[V]>>;
256
+ leave: Array<NormalizeVisitor<T[V]>>;
257
+ };
258
+ } & {
259
+ ref: {
260
+ enter: Array<VisitorRefNode>;
261
+ leave: Array<VisitorRefNode>;
262
+ };
263
+ [k: string]: {
264
+ // any internal types
265
+ enter: Array<VisitorNode<any>>;
266
+ leave: Array<VisitorNode<any>>;
267
+ };
268
+ };
269
+
270
+ export type Oas3Rule = (options: Record<string, any>) => Oas3Visitor | Oas3Visitor[];
271
+ export type Oas2Rule = (options: Record<string, any>) => Oas2Visitor | Oas2Visitor[];
272
+ export type Oas3Preprocessor = (options: Record<string, any>) => Oas3TransformVisitor;
273
+ export type Oas2Preprocessor = (options: Record<string, any>) => Oas2TransformVisitor;
274
+ export type Oas3Decorator = (options: Record<string, any>) => Oas3TransformVisitor;
275
+ export type Oas2Decorator = (options: Record<string, any>) => Oas2TransformVisitor;
276
+
277
+ // alias for the latest version supported
278
+ // every time we update it - consider semver
279
+ export type OasRule = Oas3Rule;
280
+ export type OasPreprocessor = Oas3Preprocessor;
281
+ export type OasDecorator = Oas3Decorator;
282
+
283
+ export type RuleInstanceConfig = {
284
+ ruleId: string;
285
+ severity: ProblemSeverity;
286
+ };
287
+
288
+ export function normalizeVisitors<T extends BaseVisitor>(
289
+ visitorsConfig: (RuleInstanceConfig & { visitor: NestedVisitObject<any, T> })[],
290
+ types: Record<keyof T, NormalizedNodeType>
291
+ ): NormalizedOasVisitors<T> {
292
+ const normalizedVisitors: NormalizedOasVisitors<T> = {} as any;
293
+
294
+ normalizedVisitors.any = {
295
+ enter: [],
296
+ leave: [],
297
+ };
298
+
299
+ for (const typeName of Object.keys(types) as Array<keyof T>) {
300
+ normalizedVisitors[typeName] = {
301
+ enter: [],
302
+ leave: [],
303
+ } as any;
304
+ }
305
+
306
+ normalizedVisitors.ref = {
307
+ enter: [],
308
+ leave: [],
309
+ };
310
+
311
+ for (const { ruleId, severity, visitor } of visitorsConfig) {
312
+ normalizeVisitorLevel({ ruleId, severity }, visitor, null);
313
+ }
314
+
315
+ for (const v of Object.keys(normalizedVisitors)) {
316
+ normalizedVisitors[v].enter.sort((a, b) => b.depth - a.depth);
317
+ normalizedVisitors[v].leave.sort((a, b) => a.depth - b.depth);
318
+ }
319
+
320
+ return normalizedVisitors;
321
+
322
+ function addWeakNodes(
323
+ ruleConf: RuleInstanceConfig,
324
+ from: NormalizedNodeType,
325
+ to: NormalizedNodeType,
326
+ parentContext: VisitorLevelContext,
327
+ stack: NormalizedNodeType[] = []
328
+ ) {
329
+ if (stack.includes(from)) return;
330
+
331
+ stack = [...stack, from];
332
+
333
+ const possibleChildren = new Set<NormalizedNodeType>();
334
+
335
+ for (const type of Object.values(from.properties)) {
336
+ if (type === to) {
337
+ addWeakFromStack(ruleConf, stack);
338
+ continue;
339
+ }
340
+ if (typeof type === 'object' && type !== null && type.name) {
341
+ possibleChildren.add(type);
342
+ }
343
+ }
344
+ if (from.additionalProperties && typeof from.additionalProperties !== 'function') {
345
+ if (from.additionalProperties === to) {
346
+ addWeakFromStack(ruleConf, stack);
347
+ } else if (from.additionalProperties.name !== undefined) {
348
+ possibleChildren.add(from.additionalProperties);
349
+ }
350
+ }
351
+ if (from.items) {
352
+ if (from.items === to) {
353
+ addWeakFromStack(ruleConf, stack);
354
+ } else if (from.items.name !== undefined) {
355
+ possibleChildren.add(from.items);
356
+ }
357
+ }
358
+
359
+ if (from.extensionsPrefix) {
360
+ possibleChildren.add(SpecExtension);
361
+ }
362
+
363
+ for (const fromType of Array.from(possibleChildren.values())) {
364
+ addWeakNodes(ruleConf, fromType, to, parentContext, stack);
365
+ }
366
+
367
+ function addWeakFromStack(ruleConf: RuleInstanceConfig, stack: NormalizedNodeType[]) {
368
+ for (const interType of stack.slice(1)) {
369
+ (normalizedVisitors as any)[interType.name] =
370
+ normalizedVisitors[interType.name] ||
371
+ ({
372
+ enter: [],
373
+ leave: [],
374
+ } as any);
375
+ normalizedVisitors[interType.name].enter.push({
376
+ ...ruleConf,
377
+ visit: () => undefined,
378
+ depth: 0,
379
+ context: {
380
+ isSkippedLevel: true,
381
+ seen: new Set(),
382
+ parent: parentContext,
383
+ },
384
+ });
385
+ }
386
+ }
387
+ }
388
+
389
+ function findLegacyVisitorNode<T>(
390
+ visitor: NestedVisitObject<any, T>,
391
+ typeName: keyof T | Array<keyof T>
392
+ ) {
393
+ if (Array.isArray(typeName)) {
394
+ const name = typeName.find((name) => visitor[name]) || undefined;
395
+ return name && visitor[name];
396
+ }
397
+
398
+ return visitor[typeName];
399
+ }
400
+
401
+ function normalizeVisitorLevel(
402
+ ruleConf: RuleInstanceConfig,
403
+ visitor: NestedVisitObject<any, T>,
404
+ parentContext: VisitorLevelContext | null,
405
+ depth = 0
406
+ ) {
407
+ const visitorKeys = Object.keys(types) as Array<keyof T | 'any'>;
408
+
409
+ if (depth === 0) {
410
+ visitorKeys.push('any');
411
+ visitorKeys.push('ref');
412
+ } else {
413
+ if (visitor.any) {
414
+ throw new Error('any() is allowed only on top level');
415
+ }
416
+ if (visitor.ref) {
417
+ throw new Error('ref() is allowed only on top level');
418
+ }
419
+ }
420
+
421
+ for (const typeName of visitorKeys as Array<keyof T>) {
422
+ const typeVisitor = (visitor[typeName] ||
423
+ findLegacyVisitorNode(
424
+ visitor,
425
+ legacyTypesMap[typeName as keyof typeof legacyTypesMap] as keyof T
426
+ )) as any as NestedVisitObject<any, T>;
427
+ const normalizedTypeVisitor = normalizedVisitors[typeName];
428
+
429
+ if (!typeVisitor) continue;
430
+
431
+ let visitorEnter: VisitFunction<any> | undefined;
432
+ let visitorLeave: VisitFunction<any> | undefined;
433
+ let visitorSkip: SkipFunction<any> | undefined;
434
+
435
+ const isObjectVisitor = typeof typeVisitor === 'object';
436
+
437
+ if (typeName === 'ref' && isObjectVisitor && typeVisitor.skip) {
438
+ throw new Error('ref() visitor does not support skip');
439
+ }
440
+
441
+ if (typeof typeVisitor === 'function') {
442
+ visitorEnter = typeVisitor as any;
443
+ } else if (isObjectVisitor) {
444
+ visitorEnter = typeVisitor.enter;
445
+ visitorLeave = typeVisitor.leave;
446
+ visitorSkip = typeVisitor.skip;
447
+ }
448
+
449
+ const context: VisitorLevelContext = {
450
+ activatedOn: null,
451
+ type: types[typeName],
452
+ parent: parentContext,
453
+ isSkippedLevel: false,
454
+ };
455
+
456
+ if (typeof typeVisitor === 'object') {
457
+ normalizeVisitorLevel(ruleConf, typeVisitor as any, context, depth + 1);
458
+ }
459
+
460
+ if (parentContext) {
461
+ addWeakNodes(ruleConf, parentContext.type, types[typeName], parentContext);
462
+ }
463
+
464
+ if (visitorEnter || isObjectVisitor) {
465
+ if (visitorEnter && typeof visitorEnter !== 'function') {
466
+ throw new Error('DEV: should be function');
467
+ }
468
+
469
+ normalizedTypeVisitor.enter.push({
470
+ ...ruleConf,
471
+ visit: visitorEnter || (() => undefined),
472
+ skip: visitorSkip,
473
+ depth,
474
+ context,
475
+ });
476
+ }
477
+
478
+ if (visitorLeave) {
479
+ if (typeof visitorLeave !== 'function') {
480
+ throw new Error('DEV: should be function');
481
+ }
482
+ normalizedTypeVisitor.leave.push({
483
+ ...ruleConf,
484
+ visit: visitorLeave,
485
+ depth,
486
+ context,
487
+ });
488
+ }
489
+ }
490
+ }
491
+ }