@redocly/openapi-core 1.0.0-rc.3 → 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.
- package/CHANGELOG.md +7 -0
- package/__tests__/utils.ts +88 -0
- package/lib/config/all.js +0 -1
- package/lib/config/minimal.js +0 -1
- package/lib/config/recommended.js +0 -1
- package/package.json +1 -1
- package/src/__tests__/__snapshots__/bundle.test.ts.snap +437 -0
- package/src/__tests__/bundle.test.ts +236 -0
- package/src/__tests__/codeframes.test.ts +530 -0
- package/src/__tests__/fixtures/.redocly.lint-ignore.yaml +5 -0
- package/src/__tests__/fixtures/extension.js +24 -0
- package/src/__tests__/fixtures/refs/definitions.yaml +3 -0
- package/src/__tests__/fixtures/refs/examples.yaml +8 -0
- package/src/__tests__/fixtures/refs/external-request-body.yaml +13 -0
- package/src/__tests__/fixtures/refs/externalref.yaml +35 -0
- package/src/__tests__/fixtures/refs/hosted.yaml +35 -0
- package/src/__tests__/fixtures/refs/openapi-with-external-refs-conflicting-names.yaml +21 -0
- package/src/__tests__/fixtures/refs/openapi-with-external-refs.yaml +33 -0
- package/src/__tests__/fixtures/refs/openapi-with-url-refs.yaml +18 -0
- package/src/__tests__/fixtures/refs/param-b.yaml +1 -0
- package/src/__tests__/fixtures/refs/param-c.yaml +1 -0
- package/src/__tests__/fixtures/refs/rename.yaml +1 -0
- package/src/__tests__/fixtures/refs/requestBody.yaml +9 -0
- package/src/__tests__/fixtures/refs/schema-a.yaml +1 -0
- package/src/__tests__/fixtures/refs/simple.yaml +1 -0
- package/src/__tests__/fixtures/refs/vendor.schema.yaml +20 -0
- package/src/__tests__/fixtures/resolve/External.yaml +10 -0
- package/src/__tests__/fixtures/resolve/External2.yaml +4 -0
- package/src/__tests__/fixtures/resolve/description.md +3 -0
- package/src/__tests__/fixtures/resolve/externalInfo.yaml +4 -0
- package/src/__tests__/fixtures/resolve/externalLicense.yaml +1 -0
- package/src/__tests__/fixtures/resolve/openapi-with-back.yaml +13 -0
- package/src/__tests__/fixtures/resolve/openapi-with-md-description.yaml +5 -0
- package/src/__tests__/fixtures/resolve/openapi.yaml +28 -0
- package/src/__tests__/fixtures/resolve/schemas/type-a.yaml +10 -0
- package/src/__tests__/fixtures/resolve/schemas/type-b.yaml +6 -0
- package/src/__tests__/fixtures/resolve/transitive/a.yaml +1 -0
- package/src/__tests__/fixtures/resolve/transitive/components.yaml +5 -0
- package/src/__tests__/fixtures/resolve/transitive/schemas.yaml +3 -0
- package/src/__tests__/format.test.ts +76 -0
- package/src/__tests__/js-yaml.test.ts +73 -0
- package/src/__tests__/lint.test.ts +392 -0
- package/src/__tests__/logger-browser.test.ts +53 -0
- package/src/__tests__/logger.test.ts +47 -0
- package/src/__tests__/login.test.ts +17 -0
- package/src/__tests__/normalizeVisitors.test.ts +151 -0
- package/src/__tests__/output-browser.test.ts +18 -0
- package/src/__tests__/output.test.ts +15 -0
- package/src/__tests__/ref-utils.test.ts +120 -0
- package/src/__tests__/resolve-http.test.ts +77 -0
- package/src/__tests__/resolve.test.ts +431 -0
- package/src/__tests__/utils-browser.test.ts +11 -0
- package/src/__tests__/utils.test.ts +144 -0
- package/src/__tests__/walk.test.ts +1545 -0
- package/src/benchmark/benches/lint-with-many-rules.bench.ts +35 -0
- package/src/benchmark/benches/lint-with-nested-rule.bench.ts +39 -0
- package/src/benchmark/benches/lint-with-no-rules.bench.ts +20 -0
- package/src/benchmark/benches/lint-with-top-level-rule-report.bench.ts +35 -0
- package/src/benchmark/benches/lint-with-top-level-rule.bench.ts +32 -0
- package/src/benchmark/benches/rebilly.yaml +32275 -0
- package/src/benchmark/benches/recommended-oas3.bench.ts +22 -0
- package/src/benchmark/benches/resolve-with-no-external.bench.ts +23 -0
- package/src/benchmark/benchmark.js +311 -0
- package/src/benchmark/colors.js +29 -0
- package/src/benchmark/fork.js +83 -0
- package/src/benchmark/utils.ts +36 -0
- package/src/bundle.ts +417 -0
- package/src/config/__tests__/__snapshots__/config-resolvers.test.ts.snap +164 -0
- package/src/config/__tests__/__snapshots__/config.test.ts.snap +144 -0
- package/src/config/__tests__/config-resolvers.test.ts +491 -0
- package/src/config/__tests__/config.test.ts +312 -0
- package/src/config/__tests__/fixtures/ingore-file.ts +8 -0
- package/src/config/__tests__/fixtures/load-redocly.yaml +2 -0
- package/src/config/__tests__/fixtures/plugin-config.yaml +2 -0
- package/src/config/__tests__/fixtures/plugin.js +56 -0
- package/src/config/__tests__/fixtures/resolve-config/api/nested-config.yaml +11 -0
- package/src/config/__tests__/fixtures/resolve-config/api/plugin.js +69 -0
- package/src/config/__tests__/fixtures/resolve-config/local-config-with-circular.yaml +7 -0
- package/src/config/__tests__/fixtures/resolve-config/local-config-with-custom-function.yaml +17 -0
- package/src/config/__tests__/fixtures/resolve-config/local-config-with-file.yaml +18 -0
- package/src/config/__tests__/fixtures/resolve-config/local-config-with-wrong-custom-function.yaml +15 -0
- package/src/config/__tests__/fixtures/resolve-config/local-config.yaml +9 -0
- package/src/config/__tests__/fixtures/resolve-config/plugin.js +80 -0
- package/src/config/__tests__/fixtures/resolve-remote-configs/nested-remote-config.yaml +3 -0
- package/src/config/__tests__/fixtures/resolve-remote-configs/remote-config.yaml +4 -0
- package/src/config/__tests__/load.test.ts +167 -0
- package/src/config/__tests__/resolve-plugins.test.ts +27 -0
- package/src/config/__tests__/utils.test.ts +204 -0
- package/src/config/all.ts +74 -0
- package/src/config/builtIn.ts +37 -0
- package/src/config/config-resolvers.ts +474 -0
- package/src/config/config.ts +332 -0
- package/src/config/index.ts +7 -0
- package/src/config/load.ts +144 -0
- package/src/config/minimal.ts +61 -0
- package/src/config/recommended.ts +61 -0
- package/src/config/rules.ts +54 -0
- package/src/config/types.ts +231 -0
- package/src/config/utils.ts +349 -0
- package/src/decorators/__tests__/filter-in.test.ts +310 -0
- package/src/decorators/__tests__/filter-out.test.ts +335 -0
- package/src/decorators/__tests__/media-type-examples-override.test.ts +665 -0
- package/src/decorators/__tests__/remove-x-internal.test.ts +316 -0
- package/src/decorators/__tests__/resources/request.yaml +3 -0
- package/src/decorators/__tests__/resources/response.yaml +3 -0
- package/src/decorators/common/filters/filter-helper.ts +72 -0
- package/src/decorators/common/filters/filter-in.ts +18 -0
- package/src/decorators/common/filters/filter-out.ts +18 -0
- package/src/decorators/common/info-description-override.ts +24 -0
- package/src/decorators/common/info-override.ts +15 -0
- package/src/decorators/common/media-type-examples-override.ts +79 -0
- package/src/decorators/common/operation-description-override.ts +30 -0
- package/src/decorators/common/registry-dependencies.ts +25 -0
- package/src/decorators/common/remove-x-internal.ts +59 -0
- package/src/decorators/common/tag-description-override.ts +25 -0
- package/src/decorators/oas2/index.ts +20 -0
- package/src/decorators/oas3/index.ts +22 -0
- package/src/env.ts +5 -0
- package/src/format/codeframes.ts +216 -0
- package/src/format/format.ts +375 -0
- package/src/index.ts +71 -0
- package/src/js-yaml/index.ts +14 -0
- package/src/lint.ts +148 -0
- package/src/logger.ts +34 -0
- package/src/oas-types.ts +57 -0
- package/src/output.ts +7 -0
- package/src/redocly/__tests__/redocly-client.test.ts +146 -0
- package/src/redocly/index.ts +187 -0
- package/src/redocly/redocly-client-types.ts +10 -0
- package/src/redocly/registry-api-types.ts +32 -0
- package/src/redocly/registry-api.ts +150 -0
- package/src/ref-utils.ts +85 -0
- package/src/resolve.ts +417 -0
- package/src/rules/__tests__/fixtures/code-sample.php +9 -0
- package/src/rules/__tests__/fixtures/invalid-yaml.yaml +1 -0
- package/src/rules/__tests__/fixtures/ref.yaml +1 -0
- package/src/rules/__tests__/no-unresolved-refs.test.ts +257 -0
- package/src/rules/__tests__/utils.test.ts +160 -0
- package/src/rules/ajv.ts +102 -0
- package/src/rules/common/__tests__/info-license.test.ts +62 -0
- package/src/rules/common/__tests__/license-url.test.ts +63 -0
- package/src/rules/common/__tests__/no-ambiguous-paths.test.ts +96 -0
- package/src/rules/common/__tests__/no-enum-type-mismatch.test.ts +210 -0
- package/src/rules/common/__tests__/no-identical-paths.test.ts +58 -0
- package/src/rules/common/__tests__/no-path-trailing-slash.test.ts +85 -0
- package/src/rules/common/__tests__/operation-2xx-response.test.ts +192 -0
- package/src/rules/common/__tests__/operation-4xx-response.test.ts +231 -0
- package/src/rules/common/__tests__/operation-operationId-unique.test.ts +76 -0
- package/src/rules/common/__tests__/operation-operationId-url-safe.test.ts +45 -0
- package/src/rules/common/__tests__/operation-parameters-unique.test.ts +167 -0
- package/src/rules/common/__tests__/operation-singular-tag.test.ts +72 -0
- package/src/rules/common/__tests__/path-http-verbs-order.test.ts +95 -0
- package/src/rules/common/__tests__/path-not-include-query.test.ts +64 -0
- package/src/rules/common/__tests__/path-params-defined.test.ts +202 -0
- package/src/rules/common/__tests__/paths-kebab-case.test.ts +108 -0
- package/src/rules/common/__tests__/scalar-property-missing-example.test.ts +264 -0
- package/src/rules/common/__tests__/security-defined.test.ts +175 -0
- package/src/rules/common/__tests__/spec-strict-refs.test.ts +69 -0
- package/src/rules/common/__tests__/spec.test.ts +610 -0
- package/src/rules/common/__tests__/tag-description.test.ts +65 -0
- package/src/rules/common/__tests__/tags-alphabetical.test.ts +64 -0
- package/src/rules/common/assertions/__tests__/asserts.test.ts +869 -0
- package/src/rules/common/assertions/__tests__/index.test.ts +100 -0
- package/src/rules/common/assertions/__tests__/utils.test.ts +236 -0
- package/src/rules/common/assertions/asserts.ts +357 -0
- package/src/rules/common/assertions/index.ts +53 -0
- package/src/rules/common/assertions/utils.ts +331 -0
- package/src/rules/common/info-contact.ts +15 -0
- package/src/rules/common/info-license-url.ts +10 -0
- package/src/rules/common/info-license.ts +15 -0
- package/src/rules/common/no-ambiguous-paths.ts +50 -0
- package/src/rules/common/no-enum-type-mismatch.ts +52 -0
- package/src/rules/common/no-http-verbs-in-paths.ts +36 -0
- package/src/rules/common/no-identical-paths.ts +24 -0
- package/src/rules/common/no-invalid-parameter-examples.ts +36 -0
- package/src/rules/common/no-invalid-schema-examples.ts +27 -0
- package/src/rules/common/no-path-trailing-slash.ts +15 -0
- package/src/rules/common/operation-2xx-response.ts +24 -0
- package/src/rules/common/operation-4xx-response.ts +24 -0
- package/src/rules/common/operation-description.ts +13 -0
- package/src/rules/common/operation-operationId-unique.ts +21 -0
- package/src/rules/common/operation-operationId-url-safe.ts +19 -0
- package/src/rules/common/operation-operationId.ts +17 -0
- package/src/rules/common/operation-parameters-unique.ts +48 -0
- package/src/rules/common/operation-singular-tag.ts +17 -0
- package/src/rules/common/operation-summary.ts +13 -0
- package/src/rules/common/operation-tag-defined.ts +26 -0
- package/src/rules/common/parameter-description.ts +22 -0
- package/src/rules/common/path-declaration-must-exist.ts +15 -0
- package/src/rules/common/path-excludes-patterns.ts +23 -0
- package/src/rules/common/path-http-verbs-order.ts +30 -0
- package/src/rules/common/path-not-include-query.ts +17 -0
- package/src/rules/common/path-params-defined.ts +65 -0
- package/src/rules/common/path-segment-plural.ts +31 -0
- package/src/rules/common/paths-kebab-case.ts +19 -0
- package/src/rules/common/required-string-property-missing-min-length.ts +44 -0
- package/src/rules/common/response-contains-header.ts +35 -0
- package/src/rules/common/scalar-property-missing-example.ts +58 -0
- package/src/rules/common/security-defined.ts +65 -0
- package/src/rules/common/spec-strict-refs.ts +30 -0
- package/src/rules/common/spec.ts +175 -0
- package/src/rules/common/tag-description.ts +10 -0
- package/src/rules/common/tags-alphabetical.ts +20 -0
- package/src/rules/no-unresolved-refs.ts +51 -0
- package/src/rules/oas2/__tests__/boolean-parameter-prefixes.test.ts +110 -0
- package/src/rules/oas2/__tests__/response-contains-header.test.ts +174 -0
- package/src/rules/oas2/__tests__/response-contains-property.test.ts +155 -0
- package/src/rules/oas2/__tests__/spec/fixtures/description.md +1 -0
- package/src/rules/oas2/__tests__/spec/info.test.ts +355 -0
- package/src/rules/oas2/__tests__/spec/operation.test.ts +123 -0
- package/src/rules/oas2/__tests__/spec/paths.test.ts +245 -0
- package/src/rules/oas2/__tests__/spec/referenceableScalars.test.ts +35 -0
- package/src/rules/oas2/__tests__/spec/utils.ts +32 -0
- package/src/rules/oas2/boolean-parameter-prefixes.ts +26 -0
- package/src/rules/oas2/index.ts +91 -0
- package/src/rules/oas2/remove-unused-components.ts +81 -0
- package/src/rules/oas2/request-mime-type.ts +16 -0
- package/src/rules/oas2/response-contains-property.ts +36 -0
- package/src/rules/oas2/response-mime-type.ts +16 -0
- package/src/rules/oas3/__tests__/boolean-parameter-prefixes.test.ts +111 -0
- package/src/rules/oas3/__tests__/component-name-unique.test.ts +823 -0
- package/src/rules/oas3/__tests__/fixtures/common.yaml +11 -0
- package/src/rules/oas3/__tests__/no-empty-enum-servers.com.test.ts +205 -0
- package/src/rules/oas3/__tests__/no-example-value-and-externalValue.test.ts +65 -0
- package/src/rules/oas3/__tests__/no-invalid-media-type-examples.test.ts +473 -0
- package/src/rules/oas3/__tests__/no-server-example.com.test.ts +60 -0
- package/src/rules/oas3/__tests__/no-server-trailing-slash.test.ts +79 -0
- package/src/rules/oas3/__tests__/no-unused-components.test.ts +131 -0
- package/src/rules/oas3/__tests__/operation-4xx-problem-details-rfc7807.test.ts +145 -0
- package/src/rules/oas3/__tests__/response-contains-header.test.ts +389 -0
- package/src/rules/oas3/__tests__/response-contains-property.test.ts +403 -0
- package/src/rules/oas3/__tests__/spec/callbacks.test.ts +41 -0
- package/src/rules/oas3/__tests__/spec/fixtures/description.md +1 -0
- package/src/rules/oas3/__tests__/spec/info.test.ts +391 -0
- package/src/rules/oas3/__tests__/spec/operation.test.ts +253 -0
- package/src/rules/oas3/__tests__/spec/paths.test.ts +284 -0
- package/src/rules/oas3/__tests__/spec/referenceableScalars.test.ts +77 -0
- package/src/rules/oas3/__tests__/spec/servers.test.ts +505 -0
- package/src/rules/oas3/__tests__/spec/spec.test.ts +298 -0
- package/src/rules/oas3/__tests__/spec/utils.ts +32 -0
- package/src/rules/oas3/__tests__/spec-components-invalid-map-name.test.ts +276 -0
- package/src/rules/oas3/__tests__/utils/lint-document-for-test.ts +23 -0
- package/src/rules/oas3/boolean-parameter-prefixes.ts +28 -0
- package/src/rules/oas3/component-name-unique.ts +158 -0
- package/src/rules/oas3/index.ts +113 -0
- package/src/rules/oas3/no-empty-servers.ts +22 -0
- package/src/rules/oas3/no-example-value-and-externalValue.ts +14 -0
- package/src/rules/oas3/no-invalid-media-type-examples.ts +49 -0
- package/src/rules/oas3/no-server-example.com.ts +14 -0
- package/src/rules/oas3/no-server-trailing-slash.ts +15 -0
- package/src/rules/oas3/no-server-variables-empty-enum.ts +66 -0
- package/src/rules/oas3/no-undefined-server-variable.ts +30 -0
- package/src/rules/oas3/no-unused-components.ts +75 -0
- package/src/rules/oas3/operation-4xx-problem-details-rfc7807.ts +35 -0
- package/src/rules/oas3/remove-unused-components.ts +95 -0
- package/src/rules/oas3/request-mime-type.ts +30 -0
- package/src/rules/oas3/response-contains-property.ts +38 -0
- package/src/rules/oas3/response-mime-type.ts +30 -0
- package/src/rules/oas3/spec-components-invalid-map-name.ts +69 -0
- package/src/rules/other/stats.ts +73 -0
- package/src/rules/utils.ts +193 -0
- package/src/types/config-external-schemas.ts +917 -0
- package/src/types/index.ts +149 -0
- package/src/types/oas2.ts +478 -0
- package/src/types/oas3.ts +597 -0
- package/src/types/oas3_1.ts +258 -0
- package/src/types/redocly-yaml.ts +1040 -0
- package/src/typings/common.ts +17 -0
- package/src/typings/openapi.ts +298 -0
- package/src/typings/swagger.ts +236 -0
- package/src/utils.ts +276 -0
- package/src/visitors.ts +491 -0
- package/src/walk.ts +439 -0
- package/tsconfig.json +8 -0
- package/tsconfig.tsbuildinfo +1 -0
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
interface VersionParams {
|
|
2
|
+
organizationId: string;
|
|
3
|
+
name: string;
|
|
4
|
+
version: string;
|
|
5
|
+
}
|
|
6
|
+
|
|
7
|
+
export interface PrepareFileuploadParams extends VersionParams {
|
|
8
|
+
filesHash: string;
|
|
9
|
+
filename: string;
|
|
10
|
+
isUpsert?: boolean;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export interface PushApiParams extends VersionParams {
|
|
14
|
+
rootFilePath: string;
|
|
15
|
+
filePaths: string[];
|
|
16
|
+
branch?: string;
|
|
17
|
+
isUpsert?: boolean;
|
|
18
|
+
isPublic?: boolean;
|
|
19
|
+
batchId?: string;
|
|
20
|
+
batchSize?: number;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
export interface PrepareFileuploadOKResponse {
|
|
24
|
+
filePath: string;
|
|
25
|
+
signedUploadUrl: string;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
export interface NotFoundProblemResponse {
|
|
29
|
+
status: 404;
|
|
30
|
+
title: 'Not Found';
|
|
31
|
+
code: 'ORGANIZATION_NOT_FOUND' | 'API_VERSION_NOT_FOUND';
|
|
32
|
+
}
|
|
@@ -0,0 +1,150 @@
|
|
|
1
|
+
import fetch, { RequestInit, HeadersInit } from 'node-fetch';
|
|
2
|
+
import type {
|
|
3
|
+
NotFoundProblemResponse,
|
|
4
|
+
PrepareFileuploadOKResponse,
|
|
5
|
+
PrepareFileuploadParams,
|
|
6
|
+
PushApiParams,
|
|
7
|
+
} from './registry-api-types';
|
|
8
|
+
import type { AccessTokens, Region } from '../config/types';
|
|
9
|
+
import { DEFAULT_REGION, DOMAINS } from '../config/config';
|
|
10
|
+
import { isNotEmptyObject } from '../utils';
|
|
11
|
+
|
|
12
|
+
const version = require('../../package.json').version;
|
|
13
|
+
|
|
14
|
+
export class RegistryApi {
|
|
15
|
+
constructor(private accessTokens: AccessTokens, private region: Region) {}
|
|
16
|
+
|
|
17
|
+
get accessToken() {
|
|
18
|
+
return isNotEmptyObject(this.accessTokens) && this.accessTokens[this.region];
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
getBaseUrl(region: Region = DEFAULT_REGION) {
|
|
22
|
+
return `https://api.${DOMAINS[region]}/registry`;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
setAccessTokens(accessTokens: AccessTokens) {
|
|
26
|
+
this.accessTokens = accessTokens;
|
|
27
|
+
return this;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
private async request(path = '', options: RequestInit = {}, region?: Region) {
|
|
31
|
+
const currentCommand =
|
|
32
|
+
typeof process !== 'undefined' ? process.env?.REDOCLY_CLI_COMMAND || '' : '';
|
|
33
|
+
const redoclyEnv = typeof process !== 'undefined' ? process.env?.REDOCLY_ENVIRONMENT || '' : '';
|
|
34
|
+
|
|
35
|
+
const headers = Object.assign({}, options.headers || {}, {
|
|
36
|
+
'x-redocly-cli-version': version,
|
|
37
|
+
'user-agent': `redocly-cli / ${version} ${currentCommand} ${redoclyEnv}`,
|
|
38
|
+
});
|
|
39
|
+
|
|
40
|
+
if (!headers.hasOwnProperty('authorization')) {
|
|
41
|
+
throw new Error('Unauthorized');
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
const response = await fetch(
|
|
45
|
+
`${this.getBaseUrl(region)}${path}`,
|
|
46
|
+
Object.assign({}, options, { headers })
|
|
47
|
+
);
|
|
48
|
+
|
|
49
|
+
if (response.status === 401) {
|
|
50
|
+
throw new Error('Unauthorized');
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
if (response.status === 404) {
|
|
54
|
+
const body: NotFoundProblemResponse = await response.json();
|
|
55
|
+
throw new Error(body.code);
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
return response;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
async authStatus(
|
|
62
|
+
accessToken: string,
|
|
63
|
+
region: Region,
|
|
64
|
+
verbose = false
|
|
65
|
+
): Promise<{ viewerId: string; organizations: string[] }> {
|
|
66
|
+
try {
|
|
67
|
+
const response = await this.request('', { headers: { authorization: accessToken } }, region);
|
|
68
|
+
|
|
69
|
+
return await response.json();
|
|
70
|
+
} catch (error) {
|
|
71
|
+
if (verbose) {
|
|
72
|
+
console.log(error);
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
throw error;
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
async prepareFileUpload({
|
|
80
|
+
organizationId,
|
|
81
|
+
name,
|
|
82
|
+
version,
|
|
83
|
+
filesHash,
|
|
84
|
+
filename,
|
|
85
|
+
isUpsert,
|
|
86
|
+
}: PrepareFileuploadParams): Promise<PrepareFileuploadOKResponse> {
|
|
87
|
+
const response = await this.request(
|
|
88
|
+
`/${organizationId}/${name}/${version}/prepare-file-upload`,
|
|
89
|
+
{
|
|
90
|
+
method: 'POST',
|
|
91
|
+
headers: {
|
|
92
|
+
'content-type': 'application/json',
|
|
93
|
+
authorization: this.accessToken,
|
|
94
|
+
} as HeadersInit,
|
|
95
|
+
body: JSON.stringify({
|
|
96
|
+
filesHash,
|
|
97
|
+
filename,
|
|
98
|
+
isUpsert,
|
|
99
|
+
}),
|
|
100
|
+
},
|
|
101
|
+
this.region
|
|
102
|
+
);
|
|
103
|
+
|
|
104
|
+
if (response.ok) {
|
|
105
|
+
return response.json();
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
throw new Error('Could not prepare file upload');
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
async pushApi({
|
|
112
|
+
organizationId,
|
|
113
|
+
name,
|
|
114
|
+
version,
|
|
115
|
+
rootFilePath,
|
|
116
|
+
filePaths,
|
|
117
|
+
branch,
|
|
118
|
+
isUpsert,
|
|
119
|
+
isPublic,
|
|
120
|
+
batchId,
|
|
121
|
+
batchSize,
|
|
122
|
+
}: PushApiParams) {
|
|
123
|
+
const response = await this.request(
|
|
124
|
+
`/${organizationId}/${name}/${version}`,
|
|
125
|
+
{
|
|
126
|
+
method: 'PUT',
|
|
127
|
+
headers: {
|
|
128
|
+
'content-type': 'application/json',
|
|
129
|
+
authorization: this.accessToken,
|
|
130
|
+
} as HeadersInit,
|
|
131
|
+
body: JSON.stringify({
|
|
132
|
+
rootFilePath,
|
|
133
|
+
filePaths,
|
|
134
|
+
branch,
|
|
135
|
+
isUpsert,
|
|
136
|
+
isPublic,
|
|
137
|
+
batchId,
|
|
138
|
+
batchSize,
|
|
139
|
+
}),
|
|
140
|
+
},
|
|
141
|
+
this.region
|
|
142
|
+
);
|
|
143
|
+
|
|
144
|
+
if (response.ok) {
|
|
145
|
+
return;
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
throw new Error('Could not push api');
|
|
149
|
+
}
|
|
150
|
+
}
|
package/src/ref-utils.ts
ADDED
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
import { Source } from './resolve';
|
|
2
|
+
import { OasRef } from './typings/openapi';
|
|
3
|
+
import { isTruthy } from './utils';
|
|
4
|
+
|
|
5
|
+
export function joinPointer(base: string, key: string | number) {
|
|
6
|
+
if (base === '') base = '#/';
|
|
7
|
+
return base[base.length - 1] === '/' ? base + key : base + '/' + key;
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
export function isRef(node: any): node is OasRef {
|
|
11
|
+
return node && typeof node.$ref === 'string';
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
export class Location {
|
|
15
|
+
constructor(public source: Source, public pointer: string) {}
|
|
16
|
+
|
|
17
|
+
child(components: (string | number)[] | string | number) {
|
|
18
|
+
return new Location(
|
|
19
|
+
this.source,
|
|
20
|
+
joinPointer(
|
|
21
|
+
this.pointer,
|
|
22
|
+
(Array.isArray(components) ? components : [components]).map(escapePointer).join('/')
|
|
23
|
+
)
|
|
24
|
+
);
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
key() {
|
|
28
|
+
return { ...this, reportOnKey: true };
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
get absolutePointer() {
|
|
32
|
+
return this.source.absoluteRef + (this.pointer === '#/' ? '' : this.pointer);
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
export function unescapePointer(fragment: string): string {
|
|
37
|
+
return decodeURIComponent(fragment.replace(/~1/g, '/').replace(/~0/g, '~'));
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
export function escapePointer<T extends string | number>(fragment: T): T {
|
|
41
|
+
if (typeof fragment === 'number') return fragment;
|
|
42
|
+
return (fragment as string).replace(/~/g, '~0').replace(/\//g, '~1') as T;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
export function parseRef(ref: string): { uri: string | null; pointer: string[] } {
|
|
46
|
+
const [uri, pointer] = ref.split('#/');
|
|
47
|
+
return {
|
|
48
|
+
uri: uri || null,
|
|
49
|
+
pointer: pointer ? pointer.split('/').map(unescapePointer).filter(isTruthy) : [],
|
|
50
|
+
};
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
export function parsePointer(pointer: string) {
|
|
54
|
+
return pointer.substr(2).split('/').map(unescapePointer);
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
export function pointerBaseName(pointer: string) {
|
|
58
|
+
const parts = pointer.split('/');
|
|
59
|
+
return parts[parts.length - 1];
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
export function refBaseName(ref: string) {
|
|
63
|
+
const parts = ref.split(/[\/\\]/); // split by '\' and '/'
|
|
64
|
+
return parts[parts.length - 1].replace(/\.[^.]+$/, ''); // replace extension with empty string
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
export function isAbsoluteUrl(ref: string) {
|
|
68
|
+
return ref.startsWith('http://') || ref.startsWith('https://');
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
export function isMappingRef(mapping: string) {
|
|
72
|
+
// TODO: proper detection of mapping refs
|
|
73
|
+
return (
|
|
74
|
+
mapping.startsWith('#') ||
|
|
75
|
+
mapping.startsWith('https://') ||
|
|
76
|
+
mapping.startsWith('http://') ||
|
|
77
|
+
mapping.startsWith('./') ||
|
|
78
|
+
mapping.startsWith('../') ||
|
|
79
|
+
mapping.indexOf('/') > -1
|
|
80
|
+
);
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
export function isAnchor(ref: string) {
|
|
84
|
+
return /^#[A-Za-z][A-Za-z0-9\-_:.]*$/.test(ref);
|
|
85
|
+
}
|
package/src/resolve.ts
ADDED
|
@@ -0,0 +1,417 @@
|
|
|
1
|
+
import * as fs from 'fs';
|
|
2
|
+
import * as path from 'path';
|
|
3
|
+
import { OasRef } from './typings/openapi';
|
|
4
|
+
import { isRef, joinPointer, escapePointer, parseRef, isAbsoluteUrl, isAnchor } from './ref-utils';
|
|
5
|
+
import type { YAMLNode, LoadOptions } from 'yaml-ast-parser';
|
|
6
|
+
import { NormalizedNodeType, isNamedType, SpecExtension } from './types';
|
|
7
|
+
import { readFileFromUrl, parseYaml, nextTick } from './utils';
|
|
8
|
+
import { ResolveConfig } from './config/types';
|
|
9
|
+
|
|
10
|
+
export type CollectedRefs = Map<string /* absoluteFilePath */, Document>;
|
|
11
|
+
|
|
12
|
+
export class Source {
|
|
13
|
+
constructor(public absoluteRef: string, public body: string, public mimeType?: string) {}
|
|
14
|
+
|
|
15
|
+
private _ast: YAMLNode | undefined;
|
|
16
|
+
private _lines: string[] | undefined;
|
|
17
|
+
|
|
18
|
+
// pass safeLoad as argument to separate it from browser bundle
|
|
19
|
+
getAst(safeLoad: (input: string, options?: LoadOptions | undefined) => YAMLNode) {
|
|
20
|
+
if (this._ast === undefined) {
|
|
21
|
+
this._ast = safeLoad(this.body, { filename: this.absoluteRef }) ?? undefined;
|
|
22
|
+
|
|
23
|
+
// fix ast representation of file with newlines only
|
|
24
|
+
if (
|
|
25
|
+
this._ast &&
|
|
26
|
+
this._ast.kind === 0 && // KIND.scalar = 0
|
|
27
|
+
this._ast.value === '' &&
|
|
28
|
+
this._ast.startPosition !== 1
|
|
29
|
+
) {
|
|
30
|
+
this._ast.startPosition = 1;
|
|
31
|
+
this._ast.endPosition = 1;
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
return this._ast;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
getLines() {
|
|
38
|
+
if (this._lines === undefined) {
|
|
39
|
+
this._lines = this.body.split(/\r\n|[\n\r]/g);
|
|
40
|
+
}
|
|
41
|
+
return this._lines;
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
export class ResolveError extends Error {
|
|
46
|
+
constructor(public originalError: Error) {
|
|
47
|
+
super(originalError.message);
|
|
48
|
+
// Set the prototype explicitly.
|
|
49
|
+
Object.setPrototypeOf(this, ResolveError.prototype);
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
const jsYamlErrorLineColRegexp = /\((\d+):(\d+)\)$/;
|
|
54
|
+
|
|
55
|
+
export class YamlParseError extends Error {
|
|
56
|
+
col: number;
|
|
57
|
+
line: number;
|
|
58
|
+
|
|
59
|
+
constructor(public originalError: Error, public source: Source) {
|
|
60
|
+
super(originalError.message.split('\n')[0]);
|
|
61
|
+
// Set the prototype explicitly.
|
|
62
|
+
Object.setPrototypeOf(this, YamlParseError.prototype);
|
|
63
|
+
|
|
64
|
+
const [, line, col] = this.message.match(jsYamlErrorLineColRegexp) || [];
|
|
65
|
+
this.line = parseInt(line, 10);
|
|
66
|
+
this.col = parseInt(col, 10);
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
export type Document = {
|
|
71
|
+
source: Source;
|
|
72
|
+
parsed: any;
|
|
73
|
+
};
|
|
74
|
+
|
|
75
|
+
export function makeRefId(absoluteRef: string, pointer: string) {
|
|
76
|
+
return absoluteRef + '::' + pointer;
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
export function makeDocumentFromString(sourceString: string, absoluteRef: string) {
|
|
80
|
+
const source = new Source(absoluteRef, sourceString);
|
|
81
|
+
try {
|
|
82
|
+
return {
|
|
83
|
+
source,
|
|
84
|
+
parsed: parseYaml(sourceString, { filename: absoluteRef }),
|
|
85
|
+
};
|
|
86
|
+
} catch (e) {
|
|
87
|
+
throw new YamlParseError(e, source);
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
export class BaseResolver {
|
|
92
|
+
cache: Map<string, Promise<Document | ResolveError>> = new Map();
|
|
93
|
+
|
|
94
|
+
constructor(protected config: ResolveConfig = { http: { headers: [] } }) {}
|
|
95
|
+
|
|
96
|
+
getFiles() {
|
|
97
|
+
return new Set(Array.from(this.cache.keys()));
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
resolveExternalRef(base: string | null, ref: string): string {
|
|
101
|
+
if (isAbsoluteUrl(ref)) {
|
|
102
|
+
return ref;
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
if (base && isAbsoluteUrl(base)) {
|
|
106
|
+
return new URL(ref, base).href;
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
return path.resolve(base ? path.dirname(base) : process.cwd(), ref);
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
async loadExternalRef(absoluteRef: string): Promise<Source> {
|
|
113
|
+
try {
|
|
114
|
+
if (isAbsoluteUrl(absoluteRef)) {
|
|
115
|
+
const { body, mimeType } = await readFileFromUrl(absoluteRef, this.config.http);
|
|
116
|
+
return new Source(absoluteRef, body, mimeType);
|
|
117
|
+
} else {
|
|
118
|
+
if (fs.lstatSync(absoluteRef).isDirectory()) {
|
|
119
|
+
throw new Error(`Expected a file but received a folder at ${absoluteRef}`);
|
|
120
|
+
}
|
|
121
|
+
const content = await fs.promises.readFile(absoluteRef, 'utf-8');
|
|
122
|
+
// In some cases file have \r\n line delimeters like on windows, we should skip it.
|
|
123
|
+
return new Source(absoluteRef, content.replace(/\r\n/g, '\n'));
|
|
124
|
+
}
|
|
125
|
+
} catch (error) {
|
|
126
|
+
error.message = error.message.replace(', lstat', '');
|
|
127
|
+
throw new ResolveError(error);
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
parseDocument(source: Source, isRoot: boolean = false): Document {
|
|
132
|
+
const ext = source.absoluteRef.substr(source.absoluteRef.lastIndexOf('.'));
|
|
133
|
+
if (
|
|
134
|
+
!['.json', '.json', '.yml', '.yaml'].includes(ext) &&
|
|
135
|
+
!source.mimeType?.match(/(json|yaml|openapi)/) &&
|
|
136
|
+
!isRoot // always parse root
|
|
137
|
+
) {
|
|
138
|
+
return { source, parsed: source.body };
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
try {
|
|
142
|
+
return {
|
|
143
|
+
source,
|
|
144
|
+
parsed: parseYaml(source.body, { filename: source.absoluteRef }),
|
|
145
|
+
};
|
|
146
|
+
} catch (e) {
|
|
147
|
+
throw new YamlParseError(e, source);
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
async resolveDocument(
|
|
152
|
+
base: string | null,
|
|
153
|
+
ref: string,
|
|
154
|
+
isRoot: boolean = false
|
|
155
|
+
): Promise<Document | ResolveError | YamlParseError> {
|
|
156
|
+
const absoluteRef = this.resolveExternalRef(base, ref);
|
|
157
|
+
|
|
158
|
+
const cachedDocument = this.cache.get(absoluteRef);
|
|
159
|
+
if (cachedDocument) {
|
|
160
|
+
return cachedDocument;
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
const doc = this.loadExternalRef(absoluteRef).then((source) => {
|
|
164
|
+
return this.parseDocument(source, isRoot);
|
|
165
|
+
});
|
|
166
|
+
|
|
167
|
+
this.cache.set(absoluteRef, doc);
|
|
168
|
+
|
|
169
|
+
return doc;
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
export type ResolvedRef =
|
|
174
|
+
| {
|
|
175
|
+
resolved: false;
|
|
176
|
+
isRemote: boolean;
|
|
177
|
+
nodePointer?: string;
|
|
178
|
+
document?: Document;
|
|
179
|
+
source?: Source;
|
|
180
|
+
error?: ResolveError | YamlParseError;
|
|
181
|
+
node?: any;
|
|
182
|
+
}
|
|
183
|
+
| {
|
|
184
|
+
resolved: true;
|
|
185
|
+
node: any;
|
|
186
|
+
document: Document;
|
|
187
|
+
nodePointer: string;
|
|
188
|
+
isRemote: boolean;
|
|
189
|
+
error?: undefined;
|
|
190
|
+
};
|
|
191
|
+
|
|
192
|
+
export type ResolvedRefMap = Map<string, ResolvedRef>;
|
|
193
|
+
|
|
194
|
+
type RefFrame = {
|
|
195
|
+
prev: RefFrame | null;
|
|
196
|
+
node: any;
|
|
197
|
+
};
|
|
198
|
+
|
|
199
|
+
function pushRef(head: RefFrame, node: any): RefFrame {
|
|
200
|
+
return {
|
|
201
|
+
prev: head,
|
|
202
|
+
node,
|
|
203
|
+
};
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
function hasRef(head: RefFrame | null, node: any): boolean {
|
|
207
|
+
while (head) {
|
|
208
|
+
if (head.node === node) {
|
|
209
|
+
return true;
|
|
210
|
+
}
|
|
211
|
+
head = head.prev;
|
|
212
|
+
}
|
|
213
|
+
return false;
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
const unknownType = { name: 'unknown', properties: {} };
|
|
217
|
+
const resolvableScalarType = { name: 'scalar', properties: {} };
|
|
218
|
+
|
|
219
|
+
export async function resolveDocument(opts: {
|
|
220
|
+
rootDocument: Document;
|
|
221
|
+
externalRefResolver: BaseResolver;
|
|
222
|
+
rootType: NormalizedNodeType;
|
|
223
|
+
}): Promise<ResolvedRefMap> {
|
|
224
|
+
const { rootDocument, externalRefResolver, rootType } = opts;
|
|
225
|
+
const resolvedRefMap: ResolvedRefMap = new Map();
|
|
226
|
+
const seedNodes = new Set<string>(); // format "${type}::${absoluteRef}${pointer}"
|
|
227
|
+
|
|
228
|
+
const resolvePromises: Array<Promise<void>> = [];
|
|
229
|
+
resolveRefsInParallel(rootDocument.parsed, rootDocument, '#/', rootType);
|
|
230
|
+
|
|
231
|
+
let resolved;
|
|
232
|
+
do {
|
|
233
|
+
resolved = await Promise.all(resolvePromises);
|
|
234
|
+
} while (resolvePromises.length !== resolved.length);
|
|
235
|
+
|
|
236
|
+
return resolvedRefMap;
|
|
237
|
+
|
|
238
|
+
function resolveRefsInParallel(
|
|
239
|
+
rootNode: any,
|
|
240
|
+
rootNodeDocument: Document,
|
|
241
|
+
rootNodePointer: string,
|
|
242
|
+
type: any
|
|
243
|
+
) {
|
|
244
|
+
const rootNodeDocAbsoluteRef = rootNodeDocument.source.absoluteRef;
|
|
245
|
+
const anchorRefsMap: Map<string, any> = new Map();
|
|
246
|
+
|
|
247
|
+
walk(rootNode, type, rootNodeDocAbsoluteRef + rootNodePointer);
|
|
248
|
+
|
|
249
|
+
function walk(node: any, type: NormalizedNodeType, nodeAbsoluteRef: string) {
|
|
250
|
+
if (typeof node !== 'object' || node === null) {
|
|
251
|
+
return;
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
const nodeId = `${type.name}::${nodeAbsoluteRef}`;
|
|
255
|
+
if (seedNodes.has(nodeId)) {
|
|
256
|
+
return;
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
seedNodes.add(nodeId);
|
|
260
|
+
|
|
261
|
+
const [_, anchor] = Object.entries(node).find(([key]) => key === '$anchor') || [];
|
|
262
|
+
if (anchor) {
|
|
263
|
+
anchorRefsMap.set(`#${anchor}`, node);
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
if (Array.isArray(node)) {
|
|
267
|
+
const itemsType = type.items;
|
|
268
|
+
// we continue resolving unknown types, but stop early on known scalars
|
|
269
|
+
if (itemsType === undefined && type !== unknownType && type !== SpecExtension) {
|
|
270
|
+
return;
|
|
271
|
+
}
|
|
272
|
+
for (let i = 0; i < node.length; i++) {
|
|
273
|
+
walk(node[i], itemsType || unknownType, joinPointer(nodeAbsoluteRef, i));
|
|
274
|
+
}
|
|
275
|
+
return;
|
|
276
|
+
}
|
|
277
|
+
|
|
278
|
+
for (const propName of Object.keys(node)) {
|
|
279
|
+
let propValue = node[propName];
|
|
280
|
+
let propType = type.properties[propName];
|
|
281
|
+
if (propType === undefined) propType = type.additionalProperties;
|
|
282
|
+
if (typeof propType === 'function') propType = propType(propValue, propName);
|
|
283
|
+
if (propType === undefined) propType = unknownType;
|
|
284
|
+
if (
|
|
285
|
+
type.extensionsPrefix &&
|
|
286
|
+
propName.startsWith(type.extensionsPrefix) &&
|
|
287
|
+
propType === unknownType
|
|
288
|
+
) {
|
|
289
|
+
propType = SpecExtension;
|
|
290
|
+
}
|
|
291
|
+
|
|
292
|
+
if (!isNamedType(propType) && propType?.directResolveAs) {
|
|
293
|
+
propType = propType.directResolveAs;
|
|
294
|
+
propValue = { $ref: propValue };
|
|
295
|
+
}
|
|
296
|
+
|
|
297
|
+
if (propType && propType.name === undefined && propType.resolvable !== false) {
|
|
298
|
+
propType = resolvableScalarType;
|
|
299
|
+
}
|
|
300
|
+
|
|
301
|
+
if (!isNamedType(propType) || typeof propValue !== 'object') {
|
|
302
|
+
continue;
|
|
303
|
+
}
|
|
304
|
+
|
|
305
|
+
walk(propValue, propType, joinPointer(nodeAbsoluteRef, escapePointer(propName)));
|
|
306
|
+
}
|
|
307
|
+
|
|
308
|
+
if (isRef(node)) {
|
|
309
|
+
const promise = followRef(rootNodeDocument, node, {
|
|
310
|
+
prev: null,
|
|
311
|
+
node,
|
|
312
|
+
}).then((resolvedRef) => {
|
|
313
|
+
if (resolvedRef.resolved) {
|
|
314
|
+
resolveRefsInParallel(
|
|
315
|
+
resolvedRef.node,
|
|
316
|
+
resolvedRef.document,
|
|
317
|
+
resolvedRef.nodePointer!,
|
|
318
|
+
type
|
|
319
|
+
);
|
|
320
|
+
}
|
|
321
|
+
});
|
|
322
|
+
resolvePromises.push(promise);
|
|
323
|
+
}
|
|
324
|
+
}
|
|
325
|
+
|
|
326
|
+
async function followRef(
|
|
327
|
+
document: Document,
|
|
328
|
+
ref: OasRef,
|
|
329
|
+
refStack: RefFrame
|
|
330
|
+
): Promise<ResolvedRef> {
|
|
331
|
+
if (hasRef(refStack.prev, ref)) {
|
|
332
|
+
throw new Error('Self-referencing circular pointer');
|
|
333
|
+
}
|
|
334
|
+
|
|
335
|
+
if (isAnchor(ref.$ref)) {
|
|
336
|
+
// Wait for all anchors in the document to be collected firstly.
|
|
337
|
+
await nextTick();
|
|
338
|
+
const resolvedRef: ResolvedRef = {
|
|
339
|
+
resolved: true,
|
|
340
|
+
isRemote: false,
|
|
341
|
+
node: anchorRefsMap.get(ref.$ref),
|
|
342
|
+
document,
|
|
343
|
+
nodePointer: ref.$ref,
|
|
344
|
+
};
|
|
345
|
+
const refId = makeRefId(document.source.absoluteRef, ref.$ref);
|
|
346
|
+
resolvedRefMap.set(refId, resolvedRef);
|
|
347
|
+
return resolvedRef;
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
const { uri, pointer } = parseRef(ref.$ref);
|
|
351
|
+
const isRemote = uri !== null;
|
|
352
|
+
let targetDoc: Document;
|
|
353
|
+
try {
|
|
354
|
+
targetDoc = isRemote
|
|
355
|
+
? ((await externalRefResolver.resolveDocument(
|
|
356
|
+
document.source.absoluteRef,
|
|
357
|
+
uri!
|
|
358
|
+
)) as Document)
|
|
359
|
+
: document;
|
|
360
|
+
} catch (error) {
|
|
361
|
+
const resolvedRef = {
|
|
362
|
+
resolved: false as const,
|
|
363
|
+
isRemote,
|
|
364
|
+
document: undefined,
|
|
365
|
+
error: error,
|
|
366
|
+
};
|
|
367
|
+
const refId = makeRefId(document.source.absoluteRef, ref.$ref);
|
|
368
|
+
resolvedRefMap.set(refId, resolvedRef);
|
|
369
|
+
return resolvedRef;
|
|
370
|
+
}
|
|
371
|
+
|
|
372
|
+
let resolvedRef: ResolvedRef = {
|
|
373
|
+
resolved: true,
|
|
374
|
+
document: targetDoc,
|
|
375
|
+
isRemote,
|
|
376
|
+
node: document.parsed,
|
|
377
|
+
nodePointer: '#/',
|
|
378
|
+
};
|
|
379
|
+
|
|
380
|
+
let target = targetDoc.parsed as any;
|
|
381
|
+
|
|
382
|
+
const segments = pointer;
|
|
383
|
+
for (const segment of segments) {
|
|
384
|
+
if (typeof target !== 'object') {
|
|
385
|
+
target = undefined;
|
|
386
|
+
break;
|
|
387
|
+
} else if (target[segment] !== undefined) {
|
|
388
|
+
target = target[segment];
|
|
389
|
+
resolvedRef.nodePointer = joinPointer(resolvedRef.nodePointer!, escapePointer(segment));
|
|
390
|
+
} else if (isRef(target)) {
|
|
391
|
+
resolvedRef = await followRef(targetDoc, target, pushRef(refStack, target));
|
|
392
|
+
targetDoc = resolvedRef.document || targetDoc;
|
|
393
|
+
|
|
394
|
+
if (typeof resolvedRef.node !== 'object') {
|
|
395
|
+
target = undefined;
|
|
396
|
+
break;
|
|
397
|
+
}
|
|
398
|
+
|
|
399
|
+
target = resolvedRef.node[segment];
|
|
400
|
+
resolvedRef.nodePointer = joinPointer(resolvedRef.nodePointer!, escapePointer(segment));
|
|
401
|
+
} else {
|
|
402
|
+
target = undefined;
|
|
403
|
+
break;
|
|
404
|
+
}
|
|
405
|
+
}
|
|
406
|
+
|
|
407
|
+
resolvedRef.node = target;
|
|
408
|
+
resolvedRef.document = targetDoc;
|
|
409
|
+
const refId = makeRefId(document.source.absoluteRef, ref.$ref);
|
|
410
|
+
if (resolvedRef.document && isRef(target)) {
|
|
411
|
+
resolvedRef = await followRef(resolvedRef.document, target, pushRef(refStack, target));
|
|
412
|
+
}
|
|
413
|
+
resolvedRefMap.set(refId, resolvedRef);
|
|
414
|
+
return { ...resolvedRef };
|
|
415
|
+
}
|
|
416
|
+
}
|
|
417
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
test: '
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
test: 1
|