@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
@@ -0,0 +1,316 @@
1
+ import { outdent } from 'outdent';
2
+ import { bundleDocument } from '../../bundle';
3
+ import { BaseResolver } from '../../resolve';
4
+ import { parseYamlToDocument, yamlSerializer, makeConfig } from '../../../__tests__/utils';
5
+
6
+ describe('oas3 remove-x-internal', () => {
7
+ expect.addSnapshotSerializer(yamlSerializer);
8
+ const testDocument = parseYamlToDocument(
9
+ outdent`
10
+ openapi: 3.0.0
11
+ paths:
12
+ /pet:
13
+ removeit: true
14
+ get:
15
+ parameters:
16
+ - $ref: '#/components/parameters/x'
17
+ components:
18
+ parameters:
19
+ x:
20
+ name: x
21
+ `
22
+ );
23
+
24
+ it('should use `internalFlagProperty` option to remove internal paths', async () => {
25
+ const { bundle: res } = await bundleDocument({
26
+ document: testDocument,
27
+ externalRefResolver: new BaseResolver(),
28
+ config: await makeConfig({}, { 'remove-x-internal': { internalFlagProperty: 'removeit' } }),
29
+ });
30
+ expect(res.parsed).toMatchInlineSnapshot(`
31
+ openapi: 3.0.0
32
+ components:
33
+ parameters:
34
+ x:
35
+ name: x
36
+
37
+ `);
38
+ });
39
+
40
+ it('should clean types: Server, Operation, Parameter, PathItem, Example', async () => {
41
+ const testDoc = parseYamlToDocument(
42
+ outdent`
43
+ openapi: 3.1.0
44
+ servers:
45
+ - url: //petstore.swagger.io/v2
46
+ description: Default server
47
+ x-internal: true
48
+ paths:
49
+ /pet:
50
+ get:
51
+ x-internal: true
52
+ operationId: getPet
53
+ parameters:
54
+ - $ref: '#/components/parameters/x'
55
+ put:
56
+ parameters:
57
+ - name: Accept-Language
58
+ x-internal: true
59
+ in: header
60
+ example: en-US
61
+ required: false
62
+ - name: cookieParam
63
+ x-internal: true
64
+ in: cookie
65
+ description: Some cookie
66
+ required: true
67
+ /admin:
68
+ x-internal: true
69
+ post:
70
+ parameters:
71
+ - $ref: '#/components/parameters/y'
72
+ /store/order:
73
+ post:
74
+ operationId: placeOrder
75
+ responses:
76
+ '200':
77
+ description: successful operation
78
+ content:
79
+ application/json:
80
+ examples:
81
+ response:
82
+ x-internal: true
83
+ value: OK
84
+ components:
85
+ parameters:
86
+ x:
87
+ name: x
88
+ y:
89
+ name: y
90
+ `
91
+ );
92
+ const { bundle: res } = await bundleDocument({
93
+ document: testDoc,
94
+ externalRefResolver: new BaseResolver(),
95
+ config: await makeConfig({}, { 'remove-x-internal': 'on' }),
96
+ });
97
+ expect(res.parsed).toMatchInlineSnapshot(`
98
+ openapi: 3.1.0
99
+ paths:
100
+ /pet:
101
+ put: {}
102
+ /store/order:
103
+ post:
104
+ operationId: placeOrder
105
+ responses:
106
+ '200':
107
+ description: successful operation
108
+ content:
109
+ application/json: {}
110
+ components:
111
+ parameters:
112
+ x:
113
+ name: x
114
+ 'y':
115
+ name: 'y'
116
+
117
+ `);
118
+ });
119
+
120
+ it('should clean types: Schema, Response, RequestBody, MediaType, Callback', async () => {
121
+ const testDoc = parseYamlToDocument(
122
+ outdent`
123
+ openapi: 3.1.0
124
+ paths:
125
+ /pet:
126
+ post:
127
+ summary: test
128
+ requestBody:
129
+ x-internal: true
130
+ content:
131
+ application/x-www-form-urlencoded:
132
+ schema:
133
+ type: object
134
+ /store/order:
135
+ post:
136
+ operationId: storeOrder
137
+ parameters:
138
+ - name: api_key
139
+ schema:
140
+ x-internal: true
141
+ type: string
142
+ responses:
143
+ '200':
144
+ x-internal: true
145
+ content:
146
+ application/json:
147
+ examples:
148
+ response:
149
+ value: OK
150
+ requestBody:
151
+ content:
152
+ application/x-www-form-urlencoded:
153
+ x-internal: true
154
+ schema:
155
+ type: object
156
+ callbacks:
157
+ orderInProgress:
158
+ x-internal: true
159
+ '{$request.body#/callbackUrl}?event={$request.body#/eventName}':
160
+ servers:
161
+ - url: //callback-url.path-level/v1
162
+ description: Path level server
163
+ `
164
+ );
165
+ const { bundle: res } = await bundleDocument({
166
+ document: testDoc,
167
+ externalRefResolver: new BaseResolver(),
168
+ config: await makeConfig({}, { 'remove-x-internal': 'on' }),
169
+ });
170
+
171
+ expect(res.parsed).toMatchInlineSnapshot(`
172
+ openapi: 3.1.0
173
+ paths:
174
+ /pet:
175
+ post:
176
+ summary: test
177
+ /store/order:
178
+ post:
179
+ operationId: storeOrder
180
+ parameters:
181
+ - name: api_key
182
+ requestBody: {}
183
+ components: {}
184
+
185
+ `);
186
+ });
187
+
188
+ it('should remove refs', async () => {
189
+ const testDoc = parseYamlToDocument(
190
+ outdent`
191
+ openapi: 3.0.1
192
+ info:
193
+ version: 1.0.0
194
+ title: Test API
195
+ paths:
196
+ /test1:
197
+ get:
198
+ parameters:
199
+ - $ref: '#/components/parameters/Internal'
200
+ - in: query
201
+ name: inline
202
+ schema:
203
+ type: string
204
+ - in: query
205
+ name: inline-internal
206
+ schema:
207
+ type: string
208
+ x-internal: true
209
+ /test2:
210
+ get:
211
+ parameters:
212
+ - $ref: '#/components/parameters/Public'
213
+ - $ref: '#/components/parameters/Internal'
214
+ requestBody:
215
+ $ref: '#/components/requestBodies/Public'
216
+ components:
217
+ requestBodies:
218
+ Public:
219
+ content:
220
+ application/json:
221
+ schema:
222
+ type: string
223
+ required: true
224
+ x-internal: true
225
+ parameters:
226
+ Public:
227
+ in: path
228
+ name: product_id
229
+ schema:
230
+ type: string
231
+ Internal:
232
+ in: header
233
+ name: X-Vendor
234
+ schema:
235
+ type: string
236
+ x-internal: true
237
+ `
238
+ );
239
+ const { bundle: res } = await bundleDocument({
240
+ document: testDoc,
241
+ externalRefResolver: new BaseResolver(),
242
+ config: await makeConfig({}, { 'remove-x-internal': 'on' }),
243
+ });
244
+ expect(res.parsed).toMatchInlineSnapshot(`
245
+ openapi: 3.0.1
246
+ info:
247
+ version: 1.0.0
248
+ title: Test API
249
+ paths:
250
+ /test1:
251
+ get:
252
+ parameters:
253
+ - in: query
254
+ name: inline
255
+ schema:
256
+ type: string
257
+ /test2:
258
+ get:
259
+ parameters:
260
+ - $ref: '#/components/parameters/Public'
261
+ components:
262
+ parameters:
263
+ Public:
264
+ in: path
265
+ name: product_id
266
+ schema:
267
+ type: string
268
+
269
+ `);
270
+ });
271
+ });
272
+
273
+ describe('oas2 remove-x-internal', () => {
274
+ it('should clean types - base test', async () => {
275
+ const testDoc = parseYamlToDocument(
276
+ outdent`
277
+ swagger: '2.0'
278
+ host: api.instagram.com
279
+ paths:
280
+ '/geographies/{geo-id}/media/recent':
281
+ get:
282
+ parameters:
283
+ - description: The geography ID.
284
+ x-internal: true
285
+ in: path
286
+ name: geo-id
287
+ required: true
288
+ type: string
289
+ - description: Max number of media to return.
290
+ x-internal: true
291
+ format: int32
292
+ in: query
293
+ name: count
294
+ required: false
295
+ type: integer
296
+ responses:
297
+ '200':
298
+ x-internal: true
299
+ description: List of recent media entries.
300
+ `
301
+ );
302
+ const { bundle: res } = await bundleDocument({
303
+ document: testDoc,
304
+ externalRefResolver: new BaseResolver(),
305
+ config: await makeConfig({}, { 'remove-x-internal': 'on' }),
306
+ });
307
+ expect(res.parsed).toMatchInlineSnapshot(`
308
+ swagger: '2.0'
309
+ host: api.instagram.com
310
+ paths:
311
+ /geographies/{geo-id}/media/recent:
312
+ get: {}
313
+
314
+ `);
315
+ });
316
+ });
@@ -0,0 +1,3 @@
1
+ def:
2
+ value:
3
+ b: from external file
@@ -0,0 +1,3 @@
1
+ def:
2
+ value:
3
+ name: test response name
@@ -0,0 +1,72 @@
1
+ import { UserContext } from '../../../walk';
2
+ import { isRef } from '../../../ref-utils';
3
+ import { isEmptyArray, isEmptyObject, isPlainObject } from '../../../utils';
4
+
5
+ export function filter(node: any, ctx: UserContext, criteria: (item: any) => boolean) {
6
+ const { parent, key } = ctx;
7
+ let didDelete = false;
8
+ if (Array.isArray(node)) {
9
+ for (let i = 0; i < node.length; i++) {
10
+ if (isRef(node[i])) {
11
+ const resolved = ctx.resolve(node[i]);
12
+ if (criteria(resolved.node)) {
13
+ node.splice(i, 1);
14
+ didDelete = true;
15
+ i--;
16
+ }
17
+ }
18
+ if (criteria(node[i])) {
19
+ node.splice(i, 1);
20
+ didDelete = true;
21
+ i--;
22
+ }
23
+ }
24
+ } else if (isPlainObject(node)) {
25
+ for (const key of Object.keys(node)) {
26
+ node = node as any;
27
+ if (isRef(node[key])) {
28
+ const resolved = ctx.resolve(node[key]);
29
+ if (criteria(resolved.node)) {
30
+ delete node[key];
31
+ didDelete = true;
32
+ }
33
+ }
34
+ if (criteria(node[key])) {
35
+ delete node[key];
36
+ didDelete = true;
37
+ }
38
+ }
39
+ }
40
+ if (didDelete && (isEmptyObject(node) || isEmptyArray(node))) {
41
+ delete parent[key];
42
+ }
43
+ }
44
+
45
+ export function checkIfMatchByStrategy(
46
+ nodeValue: any,
47
+ decoratorValue: any,
48
+ strategy: 'all' | 'any'
49
+ ): boolean {
50
+ if (nodeValue === undefined || decoratorValue === undefined) {
51
+ return false;
52
+ }
53
+
54
+ if (!Array.isArray(decoratorValue) && !Array.isArray(nodeValue)) {
55
+ return nodeValue === decoratorValue;
56
+ }
57
+
58
+ decoratorValue = toArrayIfNeeded<string>(decoratorValue);
59
+ nodeValue = toArrayIfNeeded<string>(nodeValue);
60
+
61
+ if (strategy === 'any') {
62
+ return decoratorValue.some((item: string) => nodeValue.includes(item));
63
+ }
64
+ if (strategy === 'all') {
65
+ return decoratorValue.every((item: string) => nodeValue.includes(item));
66
+ }
67
+ return false;
68
+ }
69
+
70
+ function toArrayIfNeeded<T>(value: T | T[]): T[] {
71
+ return Array.isArray(value) ? value : [value];
72
+ }
@@ -0,0 +1,18 @@
1
+ import { Oas2Decorator, Oas3Decorator } from '../../../visitors';
2
+ import { checkIfMatchByStrategy, filter } from './filter-helper';
3
+
4
+ const DEFAULT_STRATEGY = 'any';
5
+
6
+ export const FilterIn: Oas3Decorator | Oas2Decorator = ({ property, value, matchStrategy }) => {
7
+ const strategy = matchStrategy || DEFAULT_STRATEGY;
8
+ const filterInCriteria = (item: any) =>
9
+ item?.[property] && !checkIfMatchByStrategy(item?.[property], value, strategy);
10
+
11
+ return {
12
+ any: {
13
+ enter: (node, ctx) => {
14
+ filter(node, ctx, filterInCriteria);
15
+ },
16
+ },
17
+ };
18
+ };
@@ -0,0 +1,18 @@
1
+ import { Oas2Decorator, Oas3Decorator } from '../../../visitors';
2
+ import { checkIfMatchByStrategy, filter } from './filter-helper';
3
+
4
+ const DEFAULT_STRATEGY = 'any';
5
+
6
+ export const FilterOut: Oas3Decorator | Oas2Decorator = ({ property, value, matchStrategy }) => {
7
+ const strategy = matchStrategy || DEFAULT_STRATEGY;
8
+ const filterOutCriteria = (item: any) =>
9
+ checkIfMatchByStrategy(item?.[property], value, strategy);
10
+
11
+ return {
12
+ any: {
13
+ enter: (node, ctx) => {
14
+ filter(node, ctx, filterOutCriteria);
15
+ },
16
+ },
17
+ };
18
+ };
@@ -0,0 +1,24 @@
1
+ import { Oas3Decorator, Oas2Decorator } from '../../visitors';
2
+ import { readFileAsStringSync } from '../../utils';
3
+ import { UserContext } from '../../walk';
4
+
5
+ export const InfoDescriptionOverride: Oas3Decorator | Oas2Decorator = ({ filePath }) => {
6
+ return {
7
+ Info: {
8
+ leave(info, { report, location }: UserContext) {
9
+ if (!filePath)
10
+ throw new Error(
11
+ `Parameter "filePath" is not provided for "info-description-override" rule`
12
+ );
13
+ try {
14
+ info.description = readFileAsStringSync(filePath);
15
+ } catch (e) {
16
+ report({
17
+ message: `Failed to read markdown override file for "info.description".\n${e.message}`,
18
+ location: location.child('description'),
19
+ });
20
+ }
21
+ },
22
+ },
23
+ };
24
+ };
@@ -0,0 +1,15 @@
1
+ import { Oas3Decorator, Oas2Decorator } from '../../visitors';
2
+
3
+ export const InfoOverride: Oas3Decorator | Oas2Decorator = (newInfo) => {
4
+ return {
5
+ Info: {
6
+ leave(info) {
7
+ if (typeof newInfo !== 'object' || Array.isArray(newInfo) || newInfo === null) {
8
+ throw new Error(`"info-override" decorator should be called with an object`);
9
+ }
10
+ const { severity: _, ...rest } = newInfo;
11
+ Object.assign(info, rest);
12
+ },
13
+ },
14
+ };
15
+ };
@@ -0,0 +1,79 @@
1
+ import { Oas3Decorator } from '../../visitors';
2
+ import { Oas3Operation, Oas3RequestBody, Oas3Response } from '../../typings/openapi';
3
+ import { yamlAndJsonSyncReader } from '../../utils';
4
+ import { isRef } from '../../ref-utils';
5
+ import { ResolveFn, UserContext } from '../../walk';
6
+
7
+ export const MediaTypeExamplesOverride: Oas3Decorator = ({ operationIds }) => {
8
+ return {
9
+ Operation: {
10
+ enter(operation: Oas3Operation, ctx: UserContext) {
11
+ const operationId = operation.operationId;
12
+
13
+ if (!operationId) {
14
+ return;
15
+ }
16
+
17
+ const properties = operationIds[operationId];
18
+
19
+ if (!properties) {
20
+ return;
21
+ }
22
+
23
+ if (properties.responses && operation.responses) {
24
+ for (const responseCode of Object.keys(properties.responses)) {
25
+ const resolvedResponse = checkAndResolveRef<Oas3Response>(
26
+ operation.responses[responseCode],
27
+ ctx.resolve
28
+ );
29
+
30
+ if (!resolvedResponse) {
31
+ continue;
32
+ }
33
+
34
+ resolvedResponse.content = resolvedResponse.content ? resolvedResponse.content : {};
35
+
36
+ Object.keys(properties.responses[responseCode]).forEach((mimeType) => {
37
+ resolvedResponse.content![mimeType] = {
38
+ ...resolvedResponse.content![mimeType],
39
+ examples: yamlAndJsonSyncReader(properties.responses[responseCode][mimeType]),
40
+ };
41
+ });
42
+
43
+ operation.responses[responseCode] = resolvedResponse;
44
+ }
45
+ }
46
+
47
+ if (properties.request && operation.requestBody) {
48
+ const resolvedRequest = checkAndResolveRef<Oas3RequestBody>(
49
+ operation.requestBody,
50
+ ctx.resolve
51
+ );
52
+
53
+ if (!resolvedRequest) {
54
+ return;
55
+ }
56
+
57
+ resolvedRequest.content = resolvedRequest.content ? resolvedRequest.content : {};
58
+
59
+ Object.keys(properties.request).forEach((mimeType) => {
60
+ resolvedRequest.content[mimeType] = {
61
+ ...resolvedRequest.content[mimeType],
62
+ examples: yamlAndJsonSyncReader(properties.request[mimeType]),
63
+ };
64
+ });
65
+ operation.requestBody = resolvedRequest;
66
+ }
67
+ },
68
+ },
69
+ };
70
+ };
71
+
72
+ function checkAndResolveRef<T>(node: any, resolver: ResolveFn): T | undefined {
73
+ if (!isRef(node)) {
74
+ return node;
75
+ }
76
+
77
+ const resolved = resolver<T>(node);
78
+ return resolved.error ? undefined : JSON.parse(JSON.stringify(resolved.node));
79
+ }
@@ -0,0 +1,30 @@
1
+ import { Oas3Decorator, Oas2Decorator } from '../../visitors';
2
+ import { Oas2Operation } from '../../typings/swagger';
3
+ import { Oas3Operation } from '../../typings/openapi';
4
+ import { readFileAsStringSync } from '../../utils';
5
+ import { UserContext } from '../../walk';
6
+
7
+ export const OperationDescriptionOverride: Oas3Decorator | Oas2Decorator = ({ operationIds }) => {
8
+ return {
9
+ Operation: {
10
+ leave(operation: Oas2Operation | Oas3Operation, { report, location }: UserContext) {
11
+ if (!operation.operationId) return;
12
+ if (!operationIds)
13
+ throw new Error(
14
+ `Parameter "operationIds" is not provided for "operation-description-override" rule`
15
+ );
16
+ const operationId = operation.operationId;
17
+ if (operationIds[operationId]) {
18
+ try {
19
+ operation.description = readFileAsStringSync(operationIds[operationId]);
20
+ } catch (e) {
21
+ report({
22
+ message: `Failed to read markdown override file for operation "${operationId}".\n${e.message}`,
23
+ location: location.child('operationId').key(),
24
+ });
25
+ }
26
+ }
27
+ },
28
+ },
29
+ };
30
+ };
@@ -0,0 +1,25 @@
1
+ import { UserContext } from '../../walk';
2
+ import { isRedoclyRegistryURL } from '../../redocly';
3
+
4
+ import { Oas3Decorator, Oas2Decorator } from '../../visitors';
5
+
6
+ export const RegistryDependencies: Oas3Decorator | Oas2Decorator = () => {
7
+ const registryDependencies = new Set<string>();
8
+
9
+ return {
10
+ Root: {
11
+ leave(_: any, ctx: UserContext) {
12
+ const data = ctx.getVisitorData();
13
+ data.links = Array.from(registryDependencies);
14
+ },
15
+ },
16
+ ref(node) {
17
+ if (node.$ref) {
18
+ const link = node.$ref.split('#/')[0];
19
+ if (isRedoclyRegistryURL(link)) {
20
+ registryDependencies.add(link);
21
+ }
22
+ }
23
+ },
24
+ };
25
+ };
@@ -0,0 +1,59 @@
1
+ import { Oas3Decorator, Oas2Decorator } from '../../visitors';
2
+ import { isEmptyArray, isEmptyObject, isPlainObject } from '../../utils';
3
+ import { UserContext } from '../../walk';
4
+ import { isRef } from '../../ref-utils';
5
+
6
+ const DEFAULT_INTERNAL_PROPERTY_NAME = 'x-internal';
7
+
8
+ export const RemoveXInternal: Oas3Decorator | Oas2Decorator = ({ internalFlagProperty }) => {
9
+ const hiddenTag = internalFlagProperty || DEFAULT_INTERNAL_PROPERTY_NAME;
10
+
11
+ function removeInternal(node: any, ctx: UserContext) {
12
+ const { parent, key } = ctx;
13
+ let didDelete = false;
14
+ if (Array.isArray(node)) {
15
+ for (let i = 0; i < node.length; i++) {
16
+ if (isRef(node[i])) {
17
+ const resolved = ctx.resolve(node[i]);
18
+ if (resolved.node?.[hiddenTag]) {
19
+ node.splice(i, 1);
20
+ didDelete = true;
21
+ i--;
22
+ }
23
+ }
24
+ if (node[i]?.[hiddenTag]) {
25
+ node.splice(i, 1);
26
+ didDelete = true;
27
+ i--;
28
+ }
29
+ }
30
+ } else if (isPlainObject(node)) {
31
+ for (const key of Object.keys(node)) {
32
+ node = node as any;
33
+ if (isRef(node[key])) {
34
+ const resolved = ctx.resolve(node[key]);
35
+ if (resolved.node?.[hiddenTag]) {
36
+ delete node[key];
37
+ didDelete = true;
38
+ }
39
+ }
40
+ if (node[key]?.[hiddenTag]) {
41
+ delete node[key];
42
+ didDelete = true;
43
+ }
44
+ }
45
+ }
46
+
47
+ if (didDelete && (isEmptyObject(node) || isEmptyArray(node))) {
48
+ delete parent[key];
49
+ }
50
+ }
51
+
52
+ return {
53
+ any: {
54
+ enter: (node, ctx) => {
55
+ removeInternal(node, ctx);
56
+ },
57
+ },
58
+ };
59
+ };
@@ -0,0 +1,25 @@
1
+ import { Oas3Decorator, Oas2Decorator } from '../../visitors';
2
+ import { readFileAsStringSync } from '../../utils';
3
+ import { UserContext } from '../../walk';
4
+
5
+ export const TagDescriptionOverride: Oas3Decorator | Oas2Decorator = ({ tagNames }) => {
6
+ return {
7
+ Tag: {
8
+ leave(tag, { report }: UserContext) {
9
+ if (!tagNames)
10
+ throw new Error(
11
+ `Parameter "tagNames" is not provided for "tag-description-override" rule`
12
+ );
13
+ if (tagNames[tag.name]) {
14
+ try {
15
+ tag.description = readFileAsStringSync(tagNames[tag.name]);
16
+ } catch (e) {
17
+ report({
18
+ message: `Failed to read markdown override file for tag "${tag.name}".\n${e.message}`,
19
+ });
20
+ }
21
+ }
22
+ },
23
+ },
24
+ };
25
+ };