@redocly/openapi-core 1.0.0 → 1.0.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 (275) hide show
  1. package/CHANGELOG.md +7 -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
package/src/utils.ts ADDED
@@ -0,0 +1,276 @@
1
+ import * as fs from 'fs';
2
+ import { extname } from 'path';
3
+ import * as minimatch from 'minimatch';
4
+ import fetch from 'node-fetch';
5
+ import * as pluralize from 'pluralize';
6
+ import { parseYaml } from './js-yaml';
7
+ import { UserContext } from './walk';
8
+ import { HttpResolveConfig } from './config';
9
+ import { env } from './env';
10
+ import { logger, colorize } from './logger';
11
+
12
+ export { parseYaml, stringifyYaml } from './js-yaml';
13
+
14
+ export type StackFrame<T> = {
15
+ prev: StackFrame<T> | null;
16
+ value: T;
17
+ };
18
+
19
+ export type Stack<T> = StackFrame<T> | null;
20
+ export type StackNonEmpty<T> = StackFrame<T>;
21
+ export function pushStack<T, P extends Stack<T> = Stack<T>>(head: P, value: T) {
22
+ return { prev: head, value };
23
+ }
24
+
25
+ export function popStack<T, P extends Stack<T>>(head: P) {
26
+ return head?.prev ?? null;
27
+ }
28
+
29
+ export type BundleOutputFormat = 'json' | 'yml' | 'yaml';
30
+
31
+ export async function loadYaml<T>(filename: string): Promise<T> {
32
+ const contents = await fs.promises.readFile(filename, 'utf-8');
33
+ return parseYaml(contents) as T;
34
+ }
35
+
36
+ export function isDefined<T>(x: T | undefined): x is T {
37
+ return x !== undefined;
38
+ }
39
+
40
+ export function isPlainObject(value: any): value is object {
41
+ return value !== null && typeof value === 'object' && !Array.isArray(value);
42
+ }
43
+
44
+ export function isEmptyObject(value: any): value is object {
45
+ return isPlainObject(value) && Object.keys(value).length === 0;
46
+ }
47
+
48
+ export function isEmptyArray(value: any) {
49
+ return Array.isArray(value) && value.length === 0;
50
+ }
51
+
52
+ export async function readFileFromUrl(url: string, config: HttpResolveConfig) {
53
+ const headers: Record<string, string> = {};
54
+ for (const header of config.headers) {
55
+ if (match(url, header.matches)) {
56
+ headers[header.name] =
57
+ header.envVariable !== undefined ? env[header.envVariable] || '' : header.value;
58
+ }
59
+ }
60
+
61
+ const req = await (config.customFetch || fetch)(url, {
62
+ headers: headers,
63
+ });
64
+
65
+ if (!req.ok) {
66
+ throw new Error(`Failed to load ${url}: ${req.status} ${req.statusText}`);
67
+ }
68
+
69
+ return { body: await req.text(), mimeType: req.headers.get('content-type') };
70
+ }
71
+
72
+ function match(url: string, pattern: string) {
73
+ if (!pattern.match(/^https?:\/\//)) {
74
+ // if pattern doesn't specify protocol directly, do not match against it
75
+ url = url.replace(/^https?:\/\//, '');
76
+ }
77
+ return minimatch(url, pattern);
78
+ }
79
+
80
+ export function pickObjectProps<T extends Record<string, unknown>>(
81
+ object: T,
82
+ keys: Array<string>
83
+ ): T {
84
+ return Object.fromEntries(
85
+ keys.filter((key: string) => key in object).map((key: string) => [key, object[key]])
86
+ ) as T;
87
+ }
88
+
89
+ export function omitObjectProps<T extends Record<string, unknown>>(
90
+ object: T,
91
+ keys: Array<string>
92
+ ): T {
93
+ return Object.fromEntries(Object.entries(object).filter(([key]) => !keys.includes(key))) as T;
94
+ }
95
+
96
+ export function splitCamelCaseIntoWords(str: string) {
97
+ const camel = str
98
+ .split(/(?:[-._])|([A-Z][a-z]+)/)
99
+ .filter(isTruthy)
100
+ .map((item) => item.toLocaleLowerCase());
101
+ const caps = str
102
+ .split(/([A-Z]{2,})/)
103
+ .filter((e: string) => e && e === e.toUpperCase())
104
+ .map((item) => item.toLocaleLowerCase());
105
+ return new Set([...camel, ...caps]);
106
+ }
107
+
108
+ export function validateMimeType(
109
+ { type, value }: any,
110
+ { report, location }: UserContext,
111
+ allowedValues: string[]
112
+ ) {
113
+ const ruleType = type === 'consumes' ? 'request' : 'response';
114
+ if (!allowedValues)
115
+ throw new Error(`Parameter "allowedValues" is not provided for "${ruleType}-mime-type" rule`);
116
+ if (!value[type]) return;
117
+
118
+ for (const mime of value[type]) {
119
+ if (!allowedValues.includes(mime)) {
120
+ report({
121
+ message: `Mime type "${mime}" is not allowed`,
122
+ location: location.child(value[type].indexOf(mime)).key(),
123
+ });
124
+ }
125
+ }
126
+ }
127
+
128
+ export function validateMimeTypeOAS3(
129
+ { type, value }: any,
130
+ { report, location }: UserContext,
131
+ allowedValues: string[]
132
+ ) {
133
+ const ruleType = type === 'consumes' ? 'request' : 'response';
134
+ if (!allowedValues)
135
+ throw new Error(`Parameter "allowedValues" is not provided for "${ruleType}-mime-type" rule`);
136
+ if (!value.content) return;
137
+
138
+ for (const mime of Object.keys(value.content)) {
139
+ if (!allowedValues.includes(mime)) {
140
+ report({
141
+ message: `Mime type "${mime}" is not allowed`,
142
+ location: location.child('content').child(mime).key(),
143
+ });
144
+ }
145
+ }
146
+ }
147
+
148
+ export function isSingular(path: string) {
149
+ return pluralize.isSingular(path);
150
+ }
151
+
152
+ export function readFileAsStringSync(filePath: string) {
153
+ return fs.readFileSync(filePath, 'utf-8');
154
+ }
155
+
156
+ export function yamlAndJsonSyncReader<T>(filePath: string): T {
157
+ const content = fs.readFileSync(filePath, 'utf-8');
158
+ return parseYaml(content) as T;
159
+ }
160
+
161
+ export function isPathParameter(pathSegment: string) {
162
+ return pathSegment.startsWith('{') && pathSegment.endsWith('}');
163
+ }
164
+
165
+ /**
166
+ * Convert Windows backslash paths to slash paths: foo\\bar ➔ foo/bar
167
+ */
168
+ export function slash(path: string): string {
169
+ const isExtendedLengthPath = /^\\\\\?\\/.test(path);
170
+ if (isExtendedLengthPath) {
171
+ return path;
172
+ }
173
+
174
+ return path.replace(/\\/g, '/');
175
+ }
176
+
177
+ export function isNotEmptyObject(obj: any) {
178
+ return !!obj && Object.keys(obj).length > 0;
179
+ }
180
+
181
+ // TODO: use it everywhere
182
+ export function isString(value: unknown): value is string {
183
+ return typeof value === 'string';
184
+ }
185
+
186
+ export function isNotString<T>(value: string | T): value is T {
187
+ return !isString(value);
188
+ }
189
+
190
+ export function assignExisting<T>(target: Record<string, T>, obj: Record<string, T>) {
191
+ for (const k of Object.keys(obj)) {
192
+ if (target.hasOwnProperty(k)) {
193
+ target[k] = obj[k];
194
+ }
195
+ }
196
+ }
197
+
198
+ export function getMatchingStatusCodeRange(code: number | string): string {
199
+ return `${code}`.replace(/^(\d)\d\d$/, (_, firstDigit) => `${firstDigit}XX`);
200
+ }
201
+
202
+ export function isCustomRuleId(id: string) {
203
+ return id.includes('/');
204
+ }
205
+
206
+ export function doesYamlFileExist(filePath: string): boolean {
207
+ return (
208
+ (extname(filePath) === '.yaml' || extname(filePath) === '.yml') &&
209
+ fs.hasOwnProperty('existsSync') &&
210
+ fs.existsSync(filePath)
211
+ );
212
+ }
213
+
214
+ export function showWarningForDeprecatedField(
215
+ deprecatedField: string,
216
+ updatedField?: string,
217
+ updatedObject?: string
218
+ ) {
219
+ logger.warn(
220
+ `The '${colorize.red(deprecatedField)}' field is deprecated. ${
221
+ updatedField
222
+ ? `Use ${colorize.green(getUpdatedFieldName(updatedField, updatedObject))} instead. `
223
+ : ''
224
+ }Read more about this change: https://redocly.com/docs/api-registry/guides/migration-guide-config-file/#changed-properties\n`
225
+ );
226
+ }
227
+
228
+ export function showErrorForDeprecatedField(
229
+ deprecatedField: string,
230
+ updatedField?: string,
231
+ updatedObject?: string
232
+ ) {
233
+ throw new Error(
234
+ `Do not use '${deprecatedField}' field. ${
235
+ updatedField ? `Use '${getUpdatedFieldName(updatedField, updatedObject)}' instead. ` : ''
236
+ }\n`
237
+ );
238
+ }
239
+
240
+ export type Falsy = undefined | null | false | '' | 0;
241
+
242
+ export function isTruthy<Truthy>(value: Truthy | Falsy): value is Truthy {
243
+ return !!value;
244
+ }
245
+
246
+ export function identity<T>(value: T): T {
247
+ return value;
248
+ }
249
+
250
+ export function keysOf<T>(obj: T) {
251
+ if (!obj) return [];
252
+ return Object.keys(obj) as (keyof T)[];
253
+ }
254
+
255
+ export function pickDefined<T extends Record<string, unknown>>(
256
+ obj?: T
257
+ ): Record<string, unknown> | undefined {
258
+ if (!obj) return undefined;
259
+ const res: Record<string, unknown> = {};
260
+ for (const key in obj) {
261
+ if (obj[key] !== undefined) {
262
+ res[key] = obj[key];
263
+ }
264
+ }
265
+ return res;
266
+ }
267
+
268
+ export function nextTick() {
269
+ new Promise((resolve) => {
270
+ setTimeout(resolve);
271
+ });
272
+ }
273
+
274
+ function getUpdatedFieldName(updatedField: string, updatedObject?: string) {
275
+ return `${typeof updatedObject !== 'undefined' ? `${updatedObject}.` : ''}${updatedField}`;
276
+ }