@aws-cdk-testing/cli-integ 0.0.0
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/.eslintrc.js +9 -0
- package/LICENSE +202 -0
- package/NOTICE +16 -0
- package/README.md +205 -0
- package/bin/apply-patches +22 -0
- package/bin/download-and-run-old-tests +52 -0
- package/bin/query-github +2 -0
- package/bin/query-github.d.ts +1 -0
- package/bin/query-github.js +54 -0
- package/bin/query-github.ts +56 -0
- package/bin/run-suite +2 -0
- package/bin/run-suite.d.ts +1 -0
- package/bin/run-suite.js +131 -0
- package/bin/run-suite.ts +140 -0
- package/bin/stage-distribution +2 -0
- package/bin/stage-distribution.d.ts +1 -0
- package/bin/stage-distribution.js +217 -0
- package/bin/stage-distribution.ts +267 -0
- package/bin/test-root +2 -0
- package/bin/test-root.d.ts +1 -0
- package/bin/test-root.js +6 -0
- package/bin/test-root.ts +3 -0
- package/entrypoints/test-cli-regression-against-current-code.sh +11 -0
- package/entrypoints/test-cli-regression-against-latest-release.sh +11 -0
- package/entrypoints/test-cli-regression.bash +83 -0
- package/entrypoints/test.sh +12 -0
- package/lib/aws.d.ts +51 -0
- package/lib/aws.js +206 -0
- package/lib/aws.ts +263 -0
- package/lib/corking.d.ts +12 -0
- package/lib/corking.js +35 -0
- package/lib/corking.ts +33 -0
- package/lib/eventually.d.ts +20 -0
- package/lib/eventually.js +34 -0
- package/lib/eventually.ts +42 -0
- package/lib/files.d.ts +15 -0
- package/lib/files.js +80 -0
- package/lib/files.ts +80 -0
- package/lib/github.d.ts +4 -0
- package/lib/github.js +43 -0
- package/lib/github.ts +43 -0
- package/lib/index.d.ts +13 -0
- package/lib/index.js +30 -0
- package/lib/index.ts +13 -0
- package/lib/integ-test.d.ts +10 -0
- package/lib/integ-test.js +70 -0
- package/lib/integ-test.ts +81 -0
- package/lib/lists.d.ts +1 -0
- package/lib/lists.js +11 -0
- package/lib/lists.ts +9 -0
- package/lib/memoize.d.ts +6 -0
- package/lib/memoize.js +18 -0
- package/lib/memoize.ts +14 -0
- package/lib/npm.d.ts +8 -0
- package/lib/npm.js +38 -0
- package/lib/npm.ts +41 -0
- package/lib/package-sources/release-source.d.ts +23 -0
- package/lib/package-sources/release-source.js +71 -0
- package/lib/package-sources/release-source.ts +81 -0
- package/lib/package-sources/repo-source.d.ts +30 -0
- package/lib/package-sources/repo-source.js +97 -0
- package/lib/package-sources/repo-source.ts +111 -0
- package/lib/package-sources/repo-tools/npm +2 -0
- package/lib/package-sources/repo-tools/npm.d.ts +1 -0
- package/lib/package-sources/repo-tools/npm.js +43 -0
- package/lib/package-sources/repo-tools/npm.ts +48 -0
- package/lib/package-sources/source.d.ts +28 -0
- package/lib/package-sources/source.js +3 -0
- package/lib/package-sources/source.ts +35 -0
- package/lib/package-sources/subprocess.d.ts +3 -0
- package/lib/package-sources/subprocess.js +17 -0
- package/lib/package-sources/subprocess.ts +15 -0
- package/lib/resource-pool.d.ts +50 -0
- package/lib/resource-pool.js +117 -0
- package/lib/resource-pool.ts +140 -0
- package/lib/resources.d.ts +1 -0
- package/lib/resources.js +6 -0
- package/lib/resources.ts +4 -0
- package/lib/shell.d.ts +56 -0
- package/lib/shell.js +123 -0
- package/lib/shell.ts +168 -0
- package/lib/staging/codeartifact.d.ts +44 -0
- package/lib/staging/codeartifact.js +281 -0
- package/lib/staging/codeartifact.ts +387 -0
- package/lib/staging/maven.d.ts +5 -0
- package/lib/staging/maven.js +91 -0
- package/lib/staging/maven.ts +95 -0
- package/lib/staging/npm.d.ts +4 -0
- package/lib/staging/npm.js +55 -0
- package/lib/staging/npm.ts +62 -0
- package/lib/staging/nuget.d.ts +4 -0
- package/lib/staging/nuget.js +69 -0
- package/lib/staging/nuget.ts +75 -0
- package/lib/staging/parallel-shell.d.ts +5 -0
- package/lib/staging/parallel-shell.js +45 -0
- package/lib/staging/parallel-shell.ts +51 -0
- package/lib/staging/pypi.d.ts +4 -0
- package/lib/staging/pypi.js +48 -0
- package/lib/staging/pypi.ts +50 -0
- package/lib/staging/usage-dir.d.ts +31 -0
- package/lib/staging/usage-dir.js +87 -0
- package/lib/staging/usage-dir.ts +99 -0
- package/lib/with-aws.d.ts +14 -0
- package/lib/with-aws.js +60 -0
- package/lib/with-aws.ts +67 -0
- package/lib/with-cdk-app.d.ts +210 -0
- package/lib/with-cdk-app.js +539 -0
- package/lib/with-cdk-app.ts +742 -0
- package/lib/with-cli-lib.d.ts +17 -0
- package/lib/with-cli-lib.js +123 -0
- package/lib/with-cli-lib.ts +134 -0
- package/lib/with-packages.d.ts +5 -0
- package/lib/with-packages.js +13 -0
- package/lib/with-packages.ts +15 -0
- package/lib/with-sam.d.ts +33 -0
- package/lib/with-sam.js +258 -0
- package/lib/with-sam.ts +288 -0
- package/lib/with-temporary-directory.d.ts +5 -0
- package/lib/with-temporary-directory.js +31 -0
- package/lib/with-temporary-directory.ts +35 -0
- package/lib/with-timeout.d.ts +19 -0
- package/lib/with-timeout.js +34 -0
- package/lib/with-timeout.ts +33 -0
- package/lib/xpmutex.d.ts +43 -0
- package/lib/xpmutex.js +207 -0
- package/lib/xpmutex.ts +218 -0
- package/package.json +111 -0
- package/resources/bootstrap-templates/session-tags.all-roles-deny-all.yaml +703 -0
- package/resources/bootstrap-templates/session-tags.deploy-role-deny-sqs.yaml +700 -0
- package/resources/cdk-apps/app/app.js +926 -0
- package/resources/cdk-apps/app/appsync.hotswap.graphql +3 -0
- package/resources/cdk-apps/app/cdk.json +7 -0
- package/resources/cdk-apps/app/docker/Dockerfile +2 -0
- package/resources/cdk-apps/app/docker/Dockerfile.Custom +2 -0
- package/resources/cdk-apps/app/lambda/index.js +4 -0
- package/resources/cdk-apps/app/lambda/response.json +3 -0
- package/resources/cdk-apps/app/nested-stack.js +65 -0
- package/resources/cdk-apps/cfn-include-app/cdk.json +4 -0
- package/resources/cdk-apps/cfn-include-app/cfn-include-app.js +21 -0
- package/resources/cdk-apps/cfn-include-app/example-template.json +13 -0
- package/resources/cdk-apps/rollback-test-app/app.js +110 -0
- package/resources/cdk-apps/rollback-test-app/cdk.json +7 -0
- package/resources/cdk-apps/sam_cdk_integ_app/bin/test-app.js +11 -0
- package/resources/cdk-apps/sam_cdk_integ_app/cdk.json +6 -0
- package/resources/cdk-apps/sam_cdk_integ_app/lib/nested-stack.js +19 -0
- package/resources/cdk-apps/sam_cdk_integ_app/lib/test-stack.js +134 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/docker/DockerImageFunctionConstruct/.no-packagejson-validator +0 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/docker/DockerImageFunctionConstruct/Dockerfile +9 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/docker/DockerImageFunctionConstruct/app.js +22 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/docker/DockerImageFunctionConstruct/package.json +18 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/go/GoFunctionConstruct/go.mod +5 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/go/GoFunctionConstruct/go.sum +17 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/go/GoFunctionConstruct/main.go +17 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/nodejs/NodeJsFunctionConstruct/.no-packagejson-validator +0 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/nodejs/NodeJsFunctionConstruct/app.ts +16 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/nodejs/NodeJsFunctionConstruct/package-lock.json +12 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/nodejs/NodeJsFunctionConstruct/package.json +5 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/python/Function/app.py +15 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/python/Function/requirements.txt +1 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/python/Layer/layer_version_dependency.py +5 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/python/Layer/requirements.txt +1 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/rest-api-definition.yaml +12 -0
- package/resources/cdk-apps/simple-app/app.js +26 -0
- package/resources/cdk-apps/simple-app/cdk.json +7 -0
- package/resources/cli-regression-patches/v1.119.0/NOTES.md +5 -0
- package/resources/cli-regression-patches/v1.119.0/cli.integtest.js +659 -0
- package/resources/cli-regression-patches/v1.130.0/NOTES.md +12 -0
- package/resources/cli-regression-patches/v1.130.0/app/app.js +378 -0
- package/resources/cli-regression-patches/v1.130.0/bootstrapping.integtest.js +220 -0
- package/resources/cli-regression-patches/v1.44.0/NOTES.md +18 -0
- package/resources/cli-regression-patches/v1.44.0/bootstrapping.integtest.js +126 -0
- package/resources/cli-regression-patches/v1.44.0/test.sh +26 -0
- package/resources/cli-regression-patches/v1.61.1/NOTES.md +2 -0
- package/resources/cli-regression-patches/v1.61.1/skip-tests.txt +16 -0
- package/resources/cli-regression-patches/v1.62.0/NOTES.md +2 -0
- package/resources/cli-regression-patches/v1.62.0/aws-helpers.js +245 -0
- package/resources/cli-regression-patches/v1.63.0/NOTES.md +1 -0
- package/resources/cli-regression-patches/v1.63.0/skip-tests.txt +7 -0
- package/resources/cli-regression-patches/v1.64.0/NOTES.md +3 -0
- package/resources/cli-regression-patches/v1.64.0/cdk-helpers.js +325 -0
- package/resources/cli-regression-patches/v1.64.0/cli.integtest.js +599 -0
- package/resources/cli-regression-patches/v1.64.1/NOTES.md +3 -0
- package/resources/cli-regression-patches/v1.64.1/cdk-helpers.js +324 -0
- package/resources/cli-regression-patches/v1.64.1/cli.integtest.js +599 -0
- package/resources/cli-regression-patches/v1.67.0/NOTES.md +2 -0
- package/resources/cli-regression-patches/v1.67.0/cdk-helpers.js +331 -0
- package/resources/cli-regression-patches/v2.130.0/NOTES.md +1 -0
- package/resources/cli-regression-patches/v2.130.0/node_modules/@aws-cdk-testing/cli-integ/resources/cdk-apps/sam_cdk_integ_app/lib/nested-stack.js +19 -0
- package/resources/cli-regression-patches/v2.130.0/node_modules/@aws-cdk-testing/cli-integ/resources/cdk-apps/sam_cdk_integ_app/lib/test-stack.js +134 -0
- package/resources/cli-regression-patches/v2.130.0/skip-tests.txt +5 -0
- package/resources/cli-regression-patches/v2.132.0/NOTES.md +1 -0
- package/resources/cli-regression-patches/v2.132.0/skip-tests.txt +4 -0
- package/resources/cli-regression-patches/v2.142.0/NOTES.md +1 -0
- package/resources/cli-regression-patches/v2.142.0/skip-tests.txt +4 -0
- package/resources/cli-regression-patches/v2.160.0/skip-tests.txt +2 -0
- package/resources/cli-regression-patches/v2.161.0/NOTES.md +1 -0
- package/resources/cli-regression-patches/v2.161.0/skip-tests.txt +5 -0
- package/resources/cli-regression-patches/v2.166.0/NOTES.md +1 -0
- package/resources/cli-regression-patches/v2.166.0/skip-tests.txt +2 -0
- package/resources/cloud-assemblies/0.36.0/InitStack.template.json +1 -0
- package/resources/cloud-assemblies/0.36.0/cdk.out +1 -0
- package/resources/cloud-assemblies/0.36.0/manifest.json +19 -0
- package/resources/cloud-assemblies/1.10.0-lookup-default-vpc/InitStack.template.json +2 -0
- package/resources/cloud-assemblies/1.10.0-lookup-default-vpc/cdk.out +1 -0
- package/resources/cloud-assemblies/1.10.0-lookup-default-vpc/manifest.json.js +37 -0
- package/resources/cloud-assemblies/1.10.0-request-azs/InitStack.template.json +2 -0
- package/resources/cloud-assemblies/1.10.0-request-azs/cdk.out +1 -0
- package/resources/cloud-assemblies/1.10.0-request-azs/manifest.json.js +34 -0
- package/resources/integ.jest.config.js +25 -0
- package/resources/templates/sqs-template.json +36 -0
- package/skip-tests.txt +8 -0
- package/tests/cli-integ-tests/README.md +47 -0
- package/tests/cli-integ-tests/bootstrapping.integtest.d.ts +1 -0
- package/tests/cli-integ-tests/bootstrapping.integtest.js +412 -0
- package/tests/cli-integ-tests/bootstrapping.integtest.ts +493 -0
- package/tests/cli-integ-tests/cli-lib.integtest.d.ts +1 -0
- package/tests/cli-integ-tests/cli-lib.integtest.js +62 -0
- package/tests/cli-integ-tests/cli-lib.integtest.ts +90 -0
- package/tests/cli-integ-tests/cli.integtest.d.ts +1 -0
- package/tests/cli-integ-tests/cli.integtest.js +2104 -0
- package/tests/cli-integ-tests/cli.integtest.ts +2874 -0
- package/tests/cli-integ-tests/garbage-collection.integtest.d.ts +1 -0
- package/tests/cli-integ-tests/garbage-collection.integtest.js +314 -0
- package/tests/cli-integ-tests/garbage-collection.integtest.ts +392 -0
- package/tests/init-csharp/init-csharp.integtest.d.ts +1 -0
- package/tests/init-csharp/init-csharp.integtest.js +14 -0
- package/tests/init-csharp/init-csharp.integtest.ts +15 -0
- package/tests/init-fsharp/init-fsharp.integtest.d.ts +1 -0
- package/tests/init-fsharp/init-fsharp.integtest.js +14 -0
- package/tests/init-fsharp/init-fsharp.integtest.ts +15 -0
- package/tests/init-go/init-go.integtest.d.ts +1 -0
- package/tests/init-go/init-go.integtest.js +21 -0
- package/tests/init-go/init-go.integtest.ts +23 -0
- package/tests/init-java/init-java.integtest.d.ts +1 -0
- package/tests/init-java/init-java.integtest.js +14 -0
- package/tests/init-java/init-java.integtest.ts +14 -0
- package/tests/init-javascript/init-javascript.integtest.d.ts +1 -0
- package/tests/init-javascript/init-javascript.integtest.js +53 -0
- package/tests/init-javascript/init-javascript.integtest.ts +59 -0
- package/tests/init-python/init-python.integtest.d.ts +1 -0
- package/tests/init-python/init-python.integtest.js +19 -0
- package/tests/init-python/init-python.integtest.ts +20 -0
- package/tests/init-typescript-app/init-typescript-app.integtest.d.ts +1 -0
- package/tests/init-typescript-app/init-typescript-app.integtest.js +54 -0
- package/tests/init-typescript-app/init-typescript-app.integtest.ts +66 -0
- package/tests/init-typescript-lib/init-typescript-lib.integtest.d.ts +1 -0
- package/tests/init-typescript-lib/init-typescript-lib.integtest.js +13 -0
- package/tests/init-typescript-lib/init-typescript-lib.integtest.ts +13 -0
- package/tests/tool-integrations/amplify.integtest.d.ts +1 -0
- package/tests/tool-integrations/amplify.integtest.js +39 -0
- package/tests/tool-integrations/amplify.integtest.ts +43 -0
- package/tests/tool-integrations/with-tool-context.d.ts +9 -0
- package/tests/tool-integrations/with-tool-context.js +13 -0
- package/tests/tool-integrations/with-tool-context.ts +14 -0
- package/tests/uberpackage/uberpackage.integtest.d.ts +1 -0
- package/tests/uberpackage/uberpackage.integtest.js +11 -0
- package/tests/uberpackage/uberpackage.integtest.ts +11 -0
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
export interface IPackageSourceSetup {
|
|
2
|
+
readonly name: string;
|
|
3
|
+
readonly description: string;
|
|
4
|
+
prepare(): Promise<void>;
|
|
5
|
+
cleanup(): Promise<void>;
|
|
6
|
+
}
|
|
7
|
+
export interface IPackageSource {
|
|
8
|
+
makeCliAvailable(): Promise<void>;
|
|
9
|
+
assertJsiiPackagesAvailable(): void;
|
|
10
|
+
majorVersion(): string;
|
|
11
|
+
initializeDotnetPackages(targetDir: string): Promise<void>;
|
|
12
|
+
/**
|
|
13
|
+
* CLI version
|
|
14
|
+
*/
|
|
15
|
+
requestedCliVersion(): string;
|
|
16
|
+
/**
|
|
17
|
+
* Framework version if it's different than the CLI version
|
|
18
|
+
*
|
|
19
|
+
* Not all tests will respect this.
|
|
20
|
+
*/
|
|
21
|
+
requestedFrameworkVersion(): string;
|
|
22
|
+
/**
|
|
23
|
+
* Versions of alpha packages if different than the CLI version
|
|
24
|
+
*
|
|
25
|
+
* Not all tests will respect this.
|
|
26
|
+
*/
|
|
27
|
+
requestedAlphaVersion(): string;
|
|
28
|
+
}
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic291cmNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsic291cmNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgaW50ZXJmYWNlIElQYWNrYWdlU291cmNlU2V0dXAge1xuICByZWFkb25seSBuYW1lOiBzdHJpbmc7XG4gIHJlYWRvbmx5IGRlc2NyaXB0aW9uOiBzdHJpbmc7XG5cbiAgcHJlcGFyZSgpOiBQcm9taXNlPHZvaWQ+O1xuICBjbGVhbnVwKCk6IFByb21pc2U8dm9pZD47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgSVBhY2thZ2VTb3VyY2Uge1xuICBtYWtlQ2xpQXZhaWxhYmxlKCk6IFByb21pc2U8dm9pZD47XG5cbiAgYXNzZXJ0SnNpaVBhY2thZ2VzQXZhaWxhYmxlKCk6IHZvaWQ7XG4gIG1ham9yVmVyc2lvbigpOiBzdHJpbmc7XG5cbiAgaW5pdGlhbGl6ZURvdG5ldFBhY2thZ2VzKHRhcmdldERpcjogc3RyaW5nKTogUHJvbWlzZTx2b2lkPjtcblxuICAvKipcbiAgICogQ0xJIHZlcnNpb25cbiAgICovXG4gIHJlcXVlc3RlZENsaVZlcnNpb24oKTogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBGcmFtZXdvcmsgdmVyc2lvbiBpZiBpdCdzIGRpZmZlcmVudCB0aGFuIHRoZSBDTEkgdmVyc2lvblxuICAgKlxuICAgKiBOb3QgYWxsIHRlc3RzIHdpbGwgcmVzcGVjdCB0aGlzLlxuICAgKi9cbiAgcmVxdWVzdGVkRnJhbWV3b3JrVmVyc2lvbigpOiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIFZlcnNpb25zIG9mIGFscGhhIHBhY2thZ2VzIGlmIGRpZmZlcmVudCB0aGFuIHRoZSBDTEkgdmVyc2lvblxuICAgKlxuICAgKiBOb3QgYWxsIHRlc3RzIHdpbGwgcmVzcGVjdCB0aGlzLlxuICAgKi9cbiAgcmVxdWVzdGVkQWxwaGFWZXJzaW9uKCk6IHN0cmluZztcbn1cbiJdfQ==
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
export interface IPackageSourceSetup {
|
|
2
|
+
readonly name: string;
|
|
3
|
+
readonly description: string;
|
|
4
|
+
|
|
5
|
+
prepare(): Promise<void>;
|
|
6
|
+
cleanup(): Promise<void>;
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
export interface IPackageSource {
|
|
10
|
+
makeCliAvailable(): Promise<void>;
|
|
11
|
+
|
|
12
|
+
assertJsiiPackagesAvailable(): void;
|
|
13
|
+
majorVersion(): string;
|
|
14
|
+
|
|
15
|
+
initializeDotnetPackages(targetDir: string): Promise<void>;
|
|
16
|
+
|
|
17
|
+
/**
|
|
18
|
+
* CLI version
|
|
19
|
+
*/
|
|
20
|
+
requestedCliVersion(): string;
|
|
21
|
+
|
|
22
|
+
/**
|
|
23
|
+
* Framework version if it's different than the CLI version
|
|
24
|
+
*
|
|
25
|
+
* Not all tests will respect this.
|
|
26
|
+
*/
|
|
27
|
+
requestedFrameworkVersion(): string;
|
|
28
|
+
|
|
29
|
+
/**
|
|
30
|
+
* Versions of alpha packages if different than the CLI version
|
|
31
|
+
*
|
|
32
|
+
* Not all tests will respect this.
|
|
33
|
+
*/
|
|
34
|
+
requestedAlphaVersion(): string;
|
|
35
|
+
}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.serializeForSubprocess = serializeForSubprocess;
|
|
4
|
+
exports.packageSourceInSubprocess = packageSourceInSubprocess;
|
|
5
|
+
const release_source_1 = require("./release-source");
|
|
6
|
+
const repo_source_1 = require("./repo-source");
|
|
7
|
+
function serializeForSubprocess(s) {
|
|
8
|
+
process.env.PACKAGE_SOURCE = s.name;
|
|
9
|
+
}
|
|
10
|
+
function packageSourceInSubprocess() {
|
|
11
|
+
switch (process.env.PACKAGE_SOURCE) {
|
|
12
|
+
case 'repo': return new repo_source_1.RepoPackageSource();
|
|
13
|
+
case 'release': return new release_source_1.ReleasePackageSource();
|
|
14
|
+
default: throw new Error(`Unrecognized package source: ${process.env.PACKAGE_SOURCE}`);
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3VicHJvY2Vzcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInN1YnByb2Nlc3MudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFJQSx3REFFQztBQUVELDhEQU1DO0FBZEQscURBQXdEO0FBQ3hELCtDQUFrRDtBQUdsRCxTQUFnQixzQkFBc0IsQ0FBQyxDQUFzQjtJQUMzRCxPQUFPLENBQUMsR0FBRyxDQUFDLGNBQWMsR0FBRyxDQUFDLENBQUMsSUFBSSxDQUFDO0FBQ3RDLENBQUM7QUFFRCxTQUFnQix5QkFBeUI7SUFDdkMsUUFBUSxPQUFPLENBQUMsR0FBRyxDQUFDLGNBQWMsRUFBRSxDQUFDO1FBQ25DLEtBQUssTUFBTSxDQUFDLENBQUMsT0FBTyxJQUFJLCtCQUFpQixFQUFFLENBQUM7UUFDNUMsS0FBSyxTQUFTLENBQUMsQ0FBQyxPQUFPLElBQUkscUNBQW9CLEVBQUUsQ0FBQztRQUNsRCxPQUFPLENBQUMsQ0FBQyxNQUFNLElBQUksS0FBSyxDQUFDLGdDQUFnQyxPQUFPLENBQUMsR0FBRyxDQUFDLGNBQWMsRUFBRSxDQUFDLENBQUM7SUFDekYsQ0FBQztBQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBSZWxlYXNlUGFja2FnZVNvdXJjZSB9IGZyb20gJy4vcmVsZWFzZS1zb3VyY2UnO1xuaW1wb3J0IHsgUmVwb1BhY2thZ2VTb3VyY2UgfSBmcm9tICcuL3JlcG8tc291cmNlJztcbmltcG9ydCB7IElQYWNrYWdlU291cmNlU2V0dXAsIElQYWNrYWdlU291cmNlIH0gZnJvbSAnLi9zb3VyY2UnO1xuXG5leHBvcnQgZnVuY3Rpb24gc2VyaWFsaXplRm9yU3VicHJvY2VzcyhzOiBJUGFja2FnZVNvdXJjZVNldHVwKSB7XG4gIHByb2Nlc3MuZW52LlBBQ0tBR0VfU09VUkNFID0gcy5uYW1lO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gcGFja2FnZVNvdXJjZUluU3VicHJvY2VzcygpOiBJUGFja2FnZVNvdXJjZSB7XG4gIHN3aXRjaCAocHJvY2Vzcy5lbnYuUEFDS0FHRV9TT1VSQ0UpIHtcbiAgICBjYXNlICdyZXBvJzogcmV0dXJuIG5ldyBSZXBvUGFja2FnZVNvdXJjZSgpO1xuICAgIGNhc2UgJ3JlbGVhc2UnOiByZXR1cm4gbmV3IFJlbGVhc2VQYWNrYWdlU291cmNlKCk7XG4gICAgZGVmYXVsdDogdGhyb3cgbmV3IEVycm9yKGBVbnJlY29nbml6ZWQgcGFja2FnZSBzb3VyY2U6ICR7cHJvY2Vzcy5lbnYuUEFDS0FHRV9TT1VSQ0V9YCk7XG4gIH1cbn1cbiJdfQ==
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import { ReleasePackageSource } from './release-source';
|
|
2
|
+
import { RepoPackageSource } from './repo-source';
|
|
3
|
+
import { IPackageSourceSetup, IPackageSource } from './source';
|
|
4
|
+
|
|
5
|
+
export function serializeForSubprocess(s: IPackageSourceSetup) {
|
|
6
|
+
process.env.PACKAGE_SOURCE = s.name;
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
export function packageSourceInSubprocess(): IPackageSource {
|
|
10
|
+
switch (process.env.PACKAGE_SOURCE) {
|
|
11
|
+
case 'repo': return new RepoPackageSource();
|
|
12
|
+
case 'release': return new ReleasePackageSource();
|
|
13
|
+
default: throw new Error(`Unrecognized package source: ${process.env.PACKAGE_SOURCE}`);
|
|
14
|
+
}
|
|
15
|
+
}
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* A class that holds a pool of resources and gives them out and returns them on-demand
|
|
3
|
+
*
|
|
4
|
+
* The resources will be given out front to back, when they are returned
|
|
5
|
+
* the most recently returned version will be given out again (for best
|
|
6
|
+
* cache coherency).
|
|
7
|
+
*
|
|
8
|
+
* If there are multiple consumers waiting for a resource, consumers are serviced
|
|
9
|
+
* in FIFO order for most fairness.
|
|
10
|
+
*/
|
|
11
|
+
export declare class ResourcePool<A extends string = string> {
|
|
12
|
+
private readonly pool;
|
|
13
|
+
static withResources<A extends string>(name: string, resources: A[]): ResourcePool<A>;
|
|
14
|
+
private readonly resources;
|
|
15
|
+
private readonly mutexes;
|
|
16
|
+
private readonly locks;
|
|
17
|
+
private constructor();
|
|
18
|
+
/**
|
|
19
|
+
* Take one value from the resource pool
|
|
20
|
+
*
|
|
21
|
+
* If no such value is currently available, wait until it is.
|
|
22
|
+
*/
|
|
23
|
+
take(): Promise<ILease<A>>;
|
|
24
|
+
/**
|
|
25
|
+
* Execute a block using a single resource from the pool
|
|
26
|
+
*/
|
|
27
|
+
using<B>(block: (x: A) => B | Promise<B>): Promise<B>;
|
|
28
|
+
private tryObtainLease;
|
|
29
|
+
private makeLease;
|
|
30
|
+
/**
|
|
31
|
+
* When a value is returned:
|
|
32
|
+
*
|
|
33
|
+
* - If someone's waiting for it, give it to them
|
|
34
|
+
* - Otherwise put it back into the pool
|
|
35
|
+
*/
|
|
36
|
+
private returnValue;
|
|
37
|
+
}
|
|
38
|
+
/**
|
|
39
|
+
* A single value taken from the pool
|
|
40
|
+
*/
|
|
41
|
+
export interface ILease<A> {
|
|
42
|
+
/**
|
|
43
|
+
* The value obtained by the lease
|
|
44
|
+
*/
|
|
45
|
+
readonly value: A;
|
|
46
|
+
/**
|
|
47
|
+
* Return the leased value to the pool
|
|
48
|
+
*/
|
|
49
|
+
dispose(): Promise<void>;
|
|
50
|
+
}
|
|
@@ -0,0 +1,117 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.ResourcePool = void 0;
|
|
4
|
+
const xpmutex_1 = require("./xpmutex");
|
|
5
|
+
/**
|
|
6
|
+
* A class that holds a pool of resources and gives them out and returns them on-demand
|
|
7
|
+
*
|
|
8
|
+
* The resources will be given out front to back, when they are returned
|
|
9
|
+
* the most recently returned version will be given out again (for best
|
|
10
|
+
* cache coherency).
|
|
11
|
+
*
|
|
12
|
+
* If there are multiple consumers waiting for a resource, consumers are serviced
|
|
13
|
+
* in FIFO order for most fairness.
|
|
14
|
+
*/
|
|
15
|
+
class ResourcePool {
|
|
16
|
+
static withResources(name, resources) {
|
|
17
|
+
const pool = xpmutex_1.XpMutexPool.fromName(name);
|
|
18
|
+
return new ResourcePool(pool, resources);
|
|
19
|
+
}
|
|
20
|
+
constructor(pool, resources) {
|
|
21
|
+
this.pool = pool;
|
|
22
|
+
this.mutexes = {};
|
|
23
|
+
this.locks = {};
|
|
24
|
+
if (resources.length === 0) {
|
|
25
|
+
throw new Error('Must have at least one resource in the pool');
|
|
26
|
+
}
|
|
27
|
+
// Shuffle to reduce contention
|
|
28
|
+
resources = [...resources];
|
|
29
|
+
fisherYatesShuffle(resources);
|
|
30
|
+
this.resources = resources;
|
|
31
|
+
for (const res of resources) {
|
|
32
|
+
this.mutexes[res] = this.pool.mutex(res);
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Take one value from the resource pool
|
|
37
|
+
*
|
|
38
|
+
* If no such value is currently available, wait until it is.
|
|
39
|
+
*/
|
|
40
|
+
async take() {
|
|
41
|
+
while (true) {
|
|
42
|
+
// Start a wait on the unlock now -- if the unlock signal comes after
|
|
43
|
+
// we try to acquire but before we start the wait, we might miss it.
|
|
44
|
+
//
|
|
45
|
+
// (The timeout is in case the unlock signal doesn't come for whatever reason).
|
|
46
|
+
const wait = this.pool.awaitUnlock(10000);
|
|
47
|
+
// Try all mutexes, we might need to reacquire an expired lock
|
|
48
|
+
for (const res of this.resources) {
|
|
49
|
+
const lease = await this.tryObtainLease(res);
|
|
50
|
+
if (lease) {
|
|
51
|
+
// Ignore the wait (count as handled)
|
|
52
|
+
wait.then(() => { }, () => { });
|
|
53
|
+
return lease;
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
// None available, wait until one gets unlocked then try again
|
|
57
|
+
await wait;
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* Execute a block using a single resource from the pool
|
|
62
|
+
*/
|
|
63
|
+
async using(block) {
|
|
64
|
+
const lease = await this.take();
|
|
65
|
+
try {
|
|
66
|
+
return await block(lease.value);
|
|
67
|
+
}
|
|
68
|
+
finally {
|
|
69
|
+
await lease.dispose();
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
async tryObtainLease(value) {
|
|
73
|
+
const lock = await this.mutexes[value].tryAcquire();
|
|
74
|
+
if (!lock) {
|
|
75
|
+
return undefined;
|
|
76
|
+
}
|
|
77
|
+
this.locks[value] = lock;
|
|
78
|
+
return this.makeLease(value);
|
|
79
|
+
}
|
|
80
|
+
makeLease(value) {
|
|
81
|
+
let disposed = false;
|
|
82
|
+
return {
|
|
83
|
+
value,
|
|
84
|
+
dispose: async () => {
|
|
85
|
+
if (disposed) {
|
|
86
|
+
throw new Error('Calling dispose() on an already-disposed lease.');
|
|
87
|
+
}
|
|
88
|
+
disposed = true;
|
|
89
|
+
return this.returnValue(value);
|
|
90
|
+
},
|
|
91
|
+
};
|
|
92
|
+
}
|
|
93
|
+
/**
|
|
94
|
+
* When a value is returned:
|
|
95
|
+
*
|
|
96
|
+
* - If someone's waiting for it, give it to them
|
|
97
|
+
* - Otherwise put it back into the pool
|
|
98
|
+
*/
|
|
99
|
+
async returnValue(value) {
|
|
100
|
+
const lock = this.locks[value];
|
|
101
|
+
delete this.locks[value];
|
|
102
|
+
await (lock === null || lock === void 0 ? void 0 : lock.release());
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
exports.ResourcePool = ResourcePool;
|
|
106
|
+
/**
|
|
107
|
+
* Shuffle an array in-place
|
|
108
|
+
*/
|
|
109
|
+
function fisherYatesShuffle(xs) {
|
|
110
|
+
for (let i = xs.length - 1; i >= 1; i--) {
|
|
111
|
+
const j = Math.floor(Math.random() * i);
|
|
112
|
+
const h = xs[j];
|
|
113
|
+
xs[j] = xs[i];
|
|
114
|
+
xs[i] = h;
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resource-pool.js","sourceRoot":"","sources":["resource-pool.ts"],"names":[],"mappings":";;;AAAA,uCAAwD;AAExD;;;;;;;;;GASG;AACH,MAAa,YAAY;IAChB,MAAM,CAAC,aAAa,CAAmB,IAAY,EAAE,SAAc;QACxE,MAAM,IAAI,GAAG,qBAAW,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;QACxC,OAAO,IAAI,YAAY,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;IAC3C,CAAC;IAMD,YAAqC,IAAiB,EAAE,SAAc;QAAjC,SAAI,GAAJ,IAAI,CAAa;QAHrC,YAAO,GAA4B,EAAE,CAAC;QACtC,UAAK,GAAsC,EAAE,CAAC;QAG7D,IAAI,SAAS,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAC3B,MAAM,IAAI,KAAK,CAAC,6CAA6C,CAAC,CAAC;QACjE,CAAC;QAED,+BAA+B;QAC/B,SAAS,GAAG,CAAC,GAAG,SAAS,CAAC,CAAC;QAC3B,kBAAkB,CAAC,SAAS,CAAC,CAAC;QAC9B,IAAI,CAAC,SAAS,GAAG,SAAS,CAAC;QAE3B,KAAK,MAAM,GAAG,IAAI,SAAS,EAAE,CAAC;YAC5B,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAC3C,CAAC;IACH,CAAC;IAED;;;;OAIG;IACI,KAAK,CAAC,IAAI;QACf,OAAO,IAAI,EAAE,CAAC;YACZ,qEAAqE;YACrE,oEAAoE;YACpE,EAAE;YACF,+EAA+E;YAC/E,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,CAAC;YAE3C,8DAA8D;YAC9D,KAAK,MAAM,GAAG,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;gBACjC,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC;gBAC7C,IAAI,KAAK,EAAE,CAAC;oBACV,qCAAqC;oBACrC,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,GAAE,CAAC,EAAE,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;oBAC9B,OAAO,KAAK,CAAC;gBACf,CAAC;YACH,CAAC;YAED,8DAA8D;YAC9D,MAAM,IAAI,CAAC;QACb,CAAC;IACH,CAAC;IAED;;OAEG;IACI,KAAK,CAAC,KAAK,CAAI,KAA+B;QACnD,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,IAAI,EAAE,CAAC;QAChC,IAAI,CAAC;YACH,OAAO,MAAM,KAAK,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QAClC,CAAC;gBAAS,CAAC;YACT,MAAM,KAAK,CAAC,OAAO,EAAE,CAAC;QACxB,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,cAAc,CAAC,KAAQ;QACnC,MAAM,IAAI,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,UAAU,EAAE,CAAC;QACpD,IAAI,CAAC,IAAI,EAAE,CAAC;YACV,OAAO,SAAS,CAAC;QACnB,CAAC;QAED,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,GAAG,IAAI,CAAC;QACzB,OAAO,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;IAC/B,CAAC;IAEO,SAAS,CAAC,KAAQ;QACxB,IAAI,QAAQ,GAAG,KAAK,CAAC;QACrB,OAAO;YACL,KAAK;YACL,OAAO,EAAE,KAAK,IAAI,EAAE;gBAClB,IAAI,QAAQ,EAAE,CAAC;oBACb,MAAM,IAAI,KAAK,CAAC,iDAAiD,CAAC,CAAC;gBACrE,CAAC;gBACD,QAAQ,GAAG,IAAI,CAAC;gBAChB,OAAO,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;YACjC,CAAC;SACF,CAAC;IACJ,CAAC;IAED;;;;;OAKG;IACK,KAAK,CAAC,WAAW,CAAC,KAAa;QACrC,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QAC/B,OAAO,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QACzB,MAAM,CAAA,IAAI,aAAJ,IAAI,uBAAJ,IAAI,CAAE,OAAO,EAAE,CAAA,CAAC;IACxB,CAAC;CACF;AApGD,oCAoGC;AAiBD;;GAEG;AACH,SAAS,kBAAkB,CAAI,EAAO;IACpC,KAAK,IAAI,CAAC,GAAG,EAAE,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;QACxC,MAAM,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC;QACxC,MAAM,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC,CAAC;QAChB,EAAE,CAAC,CAAC,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC,CAAC;QACd,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;IACZ,CAAC;AACH,CAAC","sourcesContent":["import { ILock, XpMutex, XpMutexPool } from './xpmutex';\n\n/**\n * A class that holds a pool of resources and gives them out and returns them on-demand\n *\n * The resources will be given out front to back, when they are returned\n * the most recently returned version will be given out again (for best\n * cache coherency).\n *\n * If there are multiple consumers waiting for a resource, consumers are serviced\n * in FIFO order for most fairness.\n */\nexport class ResourcePool<A extends string=string> {\n  public static withResources<A extends string>(name: string, resources: A[]) {\n    const pool = XpMutexPool.fromName(name);\n    return new ResourcePool(pool, resources);\n  }\n\n  private readonly resources: ReadonlyArray<A>;\n  private readonly mutexes: Record<string, XpMutex> = {};\n  private readonly locks: Record<string, ILock | undefined> = {};\n\n  private constructor(private readonly pool: XpMutexPool, resources: A[]) {\n    if (resources.length === 0) {\n      throw new Error('Must have at least one resource in the pool');\n    }\n\n    // Shuffle to reduce contention\n    resources = [...resources];\n    fisherYatesShuffle(resources);\n    this.resources = resources;\n\n    for (const res of resources) {\n      this.mutexes[res] = this.pool.mutex(res);\n    }\n  }\n\n  /**\n   * Take one value from the resource pool\n   *\n   * If no such value is currently available, wait until it is.\n   */\n  public async take(): Promise<ILease<A>> {\n    while (true) {\n      // Start a wait on the unlock now -- if the unlock signal comes after\n      // we try to acquire but before we start the wait, we might miss it.\n      //\n      // (The timeout is in case the unlock signal doesn't come for whatever reason).\n      const wait = this.pool.awaitUnlock(10_000);\n\n      // Try all mutexes, we might need to reacquire an expired lock\n      for (const res of this.resources) {\n        const lease = await this.tryObtainLease(res);\n        if (lease) {\n          // Ignore the wait (count as handled)\n          wait.then(() => {}, () => {});\n          return lease;\n        }\n      }\n\n      // None available, wait until one gets unlocked then try again\n      await wait;\n    }\n  }\n\n  /**\n   * Execute a block using a single resource from the pool\n   */\n  public async using<B>(block: (x: A) => B | Promise<B>): Promise<B> {\n    const lease = await this.take();\n    try {\n      return await block(lease.value);\n    } finally {\n      await lease.dispose();\n    }\n  }\n\n  private async tryObtainLease(value: A) {\n    const lock = await this.mutexes[value].tryAcquire();\n    if (!lock) {\n      return undefined;\n    }\n\n    this.locks[value] = lock;\n    return this.makeLease(value);\n  }\n\n  private makeLease(value: A): ILease<A> {\n    let disposed = false;\n    return {\n      value,\n      dispose: async () => {\n        if (disposed) {\n          throw new Error('Calling dispose() on an already-disposed lease.');\n        }\n        disposed = true;\n        return this.returnValue(value);\n      },\n    };\n  }\n\n  /**\n   * When a value is returned:\n   *\n   * - If someone's waiting for it, give it to them\n   * - Otherwise put it back into the pool\n   */\n  private async returnValue(value: string) {\n    const lock = this.locks[value];\n    delete this.locks[value];\n    await lock?.release();\n  }\n}\n\n/**\n * A single value taken from the pool\n */\nexport interface ILease<A> {\n  /**\n   * The value obtained by the lease\n   */\n  readonly value: A;\n\n  /**\n   * Return the leased value to the pool\n   */\n  dispose(): Promise<void>;\n}\n\n/**\n * Shuffle an array in-place\n */\nfunction fisherYatesShuffle<A>(xs: A[]) {\n  for (let i = xs.length - 1; i >= 1; i--) {\n    const j = Math.floor(Math.random() * i);\n    const h = xs[j];\n    xs[j] = xs[i];\n    xs[i] = h;\n  }\n}\n"]}
|
|
@@ -0,0 +1,140 @@
|
|
|
1
|
+
import { ILock, XpMutex, XpMutexPool } from './xpmutex';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* A class that holds a pool of resources and gives them out and returns them on-demand
|
|
5
|
+
*
|
|
6
|
+
* The resources will be given out front to back, when they are returned
|
|
7
|
+
* the most recently returned version will be given out again (for best
|
|
8
|
+
* cache coherency).
|
|
9
|
+
*
|
|
10
|
+
* If there are multiple consumers waiting for a resource, consumers are serviced
|
|
11
|
+
* in FIFO order for most fairness.
|
|
12
|
+
*/
|
|
13
|
+
export class ResourcePool<A extends string=string> {
|
|
14
|
+
public static withResources<A extends string>(name: string, resources: A[]) {
|
|
15
|
+
const pool = XpMutexPool.fromName(name);
|
|
16
|
+
return new ResourcePool(pool, resources);
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
private readonly resources: ReadonlyArray<A>;
|
|
20
|
+
private readonly mutexes: Record<string, XpMutex> = {};
|
|
21
|
+
private readonly locks: Record<string, ILock | undefined> = {};
|
|
22
|
+
|
|
23
|
+
private constructor(private readonly pool: XpMutexPool, resources: A[]) {
|
|
24
|
+
if (resources.length === 0) {
|
|
25
|
+
throw new Error('Must have at least one resource in the pool');
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
// Shuffle to reduce contention
|
|
29
|
+
resources = [...resources];
|
|
30
|
+
fisherYatesShuffle(resources);
|
|
31
|
+
this.resources = resources;
|
|
32
|
+
|
|
33
|
+
for (const res of resources) {
|
|
34
|
+
this.mutexes[res] = this.pool.mutex(res);
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* Take one value from the resource pool
|
|
40
|
+
*
|
|
41
|
+
* If no such value is currently available, wait until it is.
|
|
42
|
+
*/
|
|
43
|
+
public async take(): Promise<ILease<A>> {
|
|
44
|
+
while (true) {
|
|
45
|
+
// Start a wait on the unlock now -- if the unlock signal comes after
|
|
46
|
+
// we try to acquire but before we start the wait, we might miss it.
|
|
47
|
+
//
|
|
48
|
+
// (The timeout is in case the unlock signal doesn't come for whatever reason).
|
|
49
|
+
const wait = this.pool.awaitUnlock(10_000);
|
|
50
|
+
|
|
51
|
+
// Try all mutexes, we might need to reacquire an expired lock
|
|
52
|
+
for (const res of this.resources) {
|
|
53
|
+
const lease = await this.tryObtainLease(res);
|
|
54
|
+
if (lease) {
|
|
55
|
+
// Ignore the wait (count as handled)
|
|
56
|
+
wait.then(() => {}, () => {});
|
|
57
|
+
return lease;
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
// None available, wait until one gets unlocked then try again
|
|
62
|
+
await wait;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
/**
|
|
67
|
+
* Execute a block using a single resource from the pool
|
|
68
|
+
*/
|
|
69
|
+
public async using<B>(block: (x: A) => B | Promise<B>): Promise<B> {
|
|
70
|
+
const lease = await this.take();
|
|
71
|
+
try {
|
|
72
|
+
return await block(lease.value);
|
|
73
|
+
} finally {
|
|
74
|
+
await lease.dispose();
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
private async tryObtainLease(value: A) {
|
|
79
|
+
const lock = await this.mutexes[value].tryAcquire();
|
|
80
|
+
if (!lock) {
|
|
81
|
+
return undefined;
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
this.locks[value] = lock;
|
|
85
|
+
return this.makeLease(value);
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
private makeLease(value: A): ILease<A> {
|
|
89
|
+
let disposed = false;
|
|
90
|
+
return {
|
|
91
|
+
value,
|
|
92
|
+
dispose: async () => {
|
|
93
|
+
if (disposed) {
|
|
94
|
+
throw new Error('Calling dispose() on an already-disposed lease.');
|
|
95
|
+
}
|
|
96
|
+
disposed = true;
|
|
97
|
+
return this.returnValue(value);
|
|
98
|
+
},
|
|
99
|
+
};
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
/**
|
|
103
|
+
* When a value is returned:
|
|
104
|
+
*
|
|
105
|
+
* - If someone's waiting for it, give it to them
|
|
106
|
+
* - Otherwise put it back into the pool
|
|
107
|
+
*/
|
|
108
|
+
private async returnValue(value: string) {
|
|
109
|
+
const lock = this.locks[value];
|
|
110
|
+
delete this.locks[value];
|
|
111
|
+
await lock?.release();
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
/**
|
|
116
|
+
* A single value taken from the pool
|
|
117
|
+
*/
|
|
118
|
+
export interface ILease<A> {
|
|
119
|
+
/**
|
|
120
|
+
* The value obtained by the lease
|
|
121
|
+
*/
|
|
122
|
+
readonly value: A;
|
|
123
|
+
|
|
124
|
+
/**
|
|
125
|
+
* Return the leased value to the pool
|
|
126
|
+
*/
|
|
127
|
+
dispose(): Promise<void>;
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
/**
|
|
131
|
+
* Shuffle an array in-place
|
|
132
|
+
*/
|
|
133
|
+
function fisherYatesShuffle<A>(xs: A[]) {
|
|
134
|
+
for (let i = xs.length - 1; i >= 1; i--) {
|
|
135
|
+
const j = Math.floor(Math.random() * i);
|
|
136
|
+
const h = xs[j];
|
|
137
|
+
xs[j] = xs[i];
|
|
138
|
+
xs[i] = h;
|
|
139
|
+
}
|
|
140
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export declare const RESOURCES_DIR: string;
|
package/lib/resources.js
ADDED
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.RESOURCES_DIR = void 0;
|
|
4
|
+
const path = require("path");
|
|
5
|
+
exports.RESOURCES_DIR = path.resolve(__dirname, '..', 'resources');
|
|
6
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVzb3VyY2VzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsicmVzb3VyY2VzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLDZCQUE2QjtBQUVoQixRQUFBLGFBQWEsR0FBRyxJQUFJLENBQUMsT0FBTyxDQUFDLFNBQVMsRUFBRSxJQUFJLEVBQUUsV0FBVyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBwYXRoIGZyb20gJ3BhdGgnO1xuXG5leHBvcnQgY29uc3QgUkVTT1VSQ0VTX0RJUiA9IHBhdGgucmVzb2x2ZShfX2Rpcm5hbWUsICcuLicsICdyZXNvdXJjZXMnKTtcblxuIl19
|
package/lib/resources.ts
ADDED
package/lib/shell.d.ts
ADDED
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
import * as child_process from 'child_process';
|
|
2
|
+
import { TestContext } from './integ-test';
|
|
3
|
+
import { TemporaryDirectoryContext } from './with-temporary-directory';
|
|
4
|
+
/**
|
|
5
|
+
* A shell command that does what you want
|
|
6
|
+
*
|
|
7
|
+
* Is platform-aware, handles errors nicely.
|
|
8
|
+
*/
|
|
9
|
+
export declare function shell(command: string[], options?: ShellOptions): Promise<string>;
|
|
10
|
+
export interface ShellOptions extends child_process.SpawnOptions {
|
|
11
|
+
/**
|
|
12
|
+
* Properties to add to 'env'
|
|
13
|
+
*/
|
|
14
|
+
readonly modEnv?: Record<string, string>;
|
|
15
|
+
/**
|
|
16
|
+
* Don't fail when exiting with an error
|
|
17
|
+
*
|
|
18
|
+
* @default false
|
|
19
|
+
*/
|
|
20
|
+
readonly allowErrExit?: boolean;
|
|
21
|
+
/**
|
|
22
|
+
* Whether to capture stderr
|
|
23
|
+
*
|
|
24
|
+
* @default true
|
|
25
|
+
*/
|
|
26
|
+
readonly captureStderr?: boolean;
|
|
27
|
+
/**
|
|
28
|
+
* Pass output here
|
|
29
|
+
*/
|
|
30
|
+
readonly outputs?: NodeJS.WritableStream[];
|
|
31
|
+
/**
|
|
32
|
+
* Only return stderr. For example, this is used to validate
|
|
33
|
+
* that when CI=true, all logs are sent to stdout.
|
|
34
|
+
*
|
|
35
|
+
* @default false
|
|
36
|
+
*/
|
|
37
|
+
readonly onlyStderr?: boolean;
|
|
38
|
+
/**
|
|
39
|
+
* Don't log to stdout
|
|
40
|
+
*
|
|
41
|
+
* @default always
|
|
42
|
+
*/
|
|
43
|
+
readonly show?: 'always' | 'never' | 'error';
|
|
44
|
+
}
|
|
45
|
+
export declare class ShellHelper {
|
|
46
|
+
private readonly _cwd;
|
|
47
|
+
private readonly _output;
|
|
48
|
+
static fromContext(context: TestContext & TemporaryDirectoryContext): ShellHelper;
|
|
49
|
+
constructor(_cwd: string, _output: NodeJS.WritableStream);
|
|
50
|
+
shell(command: string[], options?: Omit<ShellOptions, 'cwd' | 'outputs'>): Promise<string>;
|
|
51
|
+
}
|
|
52
|
+
/**
|
|
53
|
+
* rm -rf reimplementation, don't want to depend on an NPM package for this
|
|
54
|
+
*/
|
|
55
|
+
export declare function rimraf(fsPath: string): void;
|
|
56
|
+
export declare function addToShellPath(x: string): void;
|
package/lib/shell.js
ADDED
|
@@ -0,0 +1,123 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.ShellHelper = void 0;
|
|
4
|
+
exports.shell = shell;
|
|
5
|
+
exports.rimraf = rimraf;
|
|
6
|
+
exports.addToShellPath = addToShellPath;
|
|
7
|
+
const child_process = require("child_process");
|
|
8
|
+
const fs = require("fs");
|
|
9
|
+
const path = require("path");
|
|
10
|
+
/**
|
|
11
|
+
* A shell command that does what you want
|
|
12
|
+
*
|
|
13
|
+
* Is platform-aware, handles errors nicely.
|
|
14
|
+
*/
|
|
15
|
+
async function shell(command, options = {}) {
|
|
16
|
+
var _a, _b;
|
|
17
|
+
if (options.modEnv && options.env) {
|
|
18
|
+
throw new Error('Use either env or modEnv but not both');
|
|
19
|
+
}
|
|
20
|
+
const outputs = new Set(options.outputs);
|
|
21
|
+
const writeToOutputs = (x) => {
|
|
22
|
+
for (const outputStream of outputs) {
|
|
23
|
+
outputStream.write(x);
|
|
24
|
+
}
|
|
25
|
+
};
|
|
26
|
+
// Always output the command
|
|
27
|
+
writeToOutputs(`💻 ${command.join(' ')}\n`);
|
|
28
|
+
const show = (_a = options.show) !== null && _a !== void 0 ? _a : 'always';
|
|
29
|
+
if (process.env.VERBOSE) {
|
|
30
|
+
outputs.add(process.stdout);
|
|
31
|
+
}
|
|
32
|
+
const env = (_b = options.env) !== null && _b !== void 0 ? _b : (options.modEnv ? { ...process.env, ...options.modEnv } : process.env);
|
|
33
|
+
const child = child_process.spawn(command[0], command.slice(1), {
|
|
34
|
+
...options,
|
|
35
|
+
env,
|
|
36
|
+
// Need this for Windows where we want .cmd and .bat to be found as well.
|
|
37
|
+
shell: true,
|
|
38
|
+
stdio: ['ignore', 'pipe', 'pipe'],
|
|
39
|
+
});
|
|
40
|
+
return new Promise((resolve, reject) => {
|
|
41
|
+
const stdout = new Array();
|
|
42
|
+
const stderr = new Array();
|
|
43
|
+
child.stdout.on('data', chunk => {
|
|
44
|
+
if (show === 'always') {
|
|
45
|
+
writeToOutputs(chunk);
|
|
46
|
+
}
|
|
47
|
+
stdout.push(chunk);
|
|
48
|
+
});
|
|
49
|
+
child.stderr.on('data', chunk => {
|
|
50
|
+
var _a;
|
|
51
|
+
if (show === 'always') {
|
|
52
|
+
writeToOutputs(chunk);
|
|
53
|
+
}
|
|
54
|
+
if ((_a = options.captureStderr) !== null && _a !== void 0 ? _a : true) {
|
|
55
|
+
stderr.push(chunk);
|
|
56
|
+
}
|
|
57
|
+
});
|
|
58
|
+
child.once('error', reject);
|
|
59
|
+
child.once('close', code => {
|
|
60
|
+
const stderrOutput = Buffer.concat(stderr).toString('utf-8');
|
|
61
|
+
const stdoutOutput = Buffer.concat(stdout).toString('utf-8');
|
|
62
|
+
const out = (options.onlyStderr ? stderrOutput : stdoutOutput + stderrOutput).trim();
|
|
63
|
+
if (code === 0 || options.allowErrExit) {
|
|
64
|
+
resolve(out);
|
|
65
|
+
}
|
|
66
|
+
else {
|
|
67
|
+
if (show === 'error') {
|
|
68
|
+
writeToOutputs(`${out}\n`);
|
|
69
|
+
}
|
|
70
|
+
reject(new Error(`'${command.join(' ')}' exited with error code ${code}.`));
|
|
71
|
+
}
|
|
72
|
+
});
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
class ShellHelper {
|
|
76
|
+
static fromContext(context) {
|
|
77
|
+
return new ShellHelper(context.integTestDir, context.output);
|
|
78
|
+
}
|
|
79
|
+
constructor(_cwd, _output) {
|
|
80
|
+
this._cwd = _cwd;
|
|
81
|
+
this._output = _output;
|
|
82
|
+
}
|
|
83
|
+
async shell(command, options = {}) {
|
|
84
|
+
return shell(command, {
|
|
85
|
+
outputs: [this._output],
|
|
86
|
+
cwd: this._cwd,
|
|
87
|
+
...options,
|
|
88
|
+
});
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
exports.ShellHelper = ShellHelper;
|
|
92
|
+
/**
|
|
93
|
+
* rm -rf reimplementation, don't want to depend on an NPM package for this
|
|
94
|
+
*/
|
|
95
|
+
function rimraf(fsPath) {
|
|
96
|
+
try {
|
|
97
|
+
const isDir = fs.lstatSync(fsPath).isDirectory();
|
|
98
|
+
if (isDir) {
|
|
99
|
+
for (const file of fs.readdirSync(fsPath)) {
|
|
100
|
+
rimraf(path.join(fsPath, file));
|
|
101
|
+
}
|
|
102
|
+
fs.rmdirSync(fsPath);
|
|
103
|
+
}
|
|
104
|
+
else {
|
|
105
|
+
fs.unlinkSync(fsPath);
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
catch (e) {
|
|
109
|
+
// We will survive ENOENT
|
|
110
|
+
if (e.code !== 'ENOENT') {
|
|
111
|
+
throw e;
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
function addToShellPath(x) {
|
|
116
|
+
var _a, _b;
|
|
117
|
+
const parts = (_b = (_a = process.env.PATH) === null || _a === void 0 ? void 0 : _a.split(':')) !== null && _b !== void 0 ? _b : [];
|
|
118
|
+
if (!parts.includes(x)) {
|
|
119
|
+
parts.unshift(x);
|
|
120
|
+
}
|
|
121
|
+
process.env.PATH = parts.join(':');
|
|
122
|
+
}
|
|
123
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"shell.js","sourceRoot":"","sources":["shell.ts"],"names":[],"mappings":"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","sourcesContent":["import * as child_process from 'child_process';\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport { TestContext } from './integ-test';\nimport { TemporaryDirectoryContext } from './with-temporary-directory';\n\n/**\n * A shell command that does what you want\n *\n * Is platform-aware, handles errors nicely.\n */\nexport async function shell(command: string[], options: ShellOptions = {}): Promise<string> {\n  if (options.modEnv && options.env) {\n    throw new Error('Use either env or modEnv but not both');\n  }\n\n  const outputs = new Set(options.outputs);\n  const writeToOutputs = (x: string) => {\n    for (const outputStream of outputs) {\n      outputStream.write(x);\n    }\n  };\n\n  // Always output the command\n  writeToOutputs(`💻 ${command.join(' ')}\\n`);\n  const show = options.show ?? 'always';\n\n  if (process.env.VERBOSE) {\n    outputs.add(process.stdout);\n  }\n\n  const env = options.env ?? (options.modEnv ? { ...process.env, ...options.modEnv } : process.env);\n\n  const child = child_process.spawn(command[0], command.slice(1), {\n    ...options,\n    env,\n    // Need this for Windows where we want .cmd and .bat to be found as well.\n    shell: true,\n    stdio: ['ignore', 'pipe', 'pipe'],\n  });\n\n  return new Promise<string>((resolve, reject) => {\n    const stdout = new Array<Buffer>();\n    const stderr = new Array<Buffer>();\n\n    child.stdout!.on('data', chunk => {\n      if (show === 'always') {\n        writeToOutputs(chunk);\n      }\n      stdout.push(chunk);\n    });\n\n    child.stderr!.on('data', chunk => {\n      if (show === 'always') {\n        writeToOutputs(chunk);\n      }\n      if (options.captureStderr ?? true) {\n        stderr.push(chunk);\n      }\n    });\n\n    child.once('error', reject);\n\n    child.once('close', code => {\n      const stderrOutput = Buffer.concat(stderr).toString('utf-8');\n      const stdoutOutput = Buffer.concat(stdout).toString('utf-8');\n      const out = (options.onlyStderr ? stderrOutput : stdoutOutput + stderrOutput).trim();\n      if (code === 0 || options.allowErrExit) {\n        resolve(out);\n      } else {\n        if (show === 'error') {\n          writeToOutputs(`${out}\\n`);\n        }\n        reject(new Error(`'${command.join(' ')}' exited with error code ${code}.`));\n      }\n    });\n  });\n}\n\nexport interface ShellOptions extends child_process.SpawnOptions {\n  /**\n   * Properties to add to 'env'\n   */\n  readonly modEnv?: Record<string, string>;\n\n  /**\n   * Don't fail when exiting with an error\n   *\n   * @default false\n   */\n  readonly allowErrExit?: boolean;\n\n  /**\n   * Whether to capture stderr\n   *\n   * @default true\n   */\n  readonly captureStderr?: boolean;\n\n  /**\n   * Pass output here\n   */\n  readonly outputs?: NodeJS.WritableStream[];\n\n  /**\n   * Only return stderr. For example, this is used to validate\n   * that when CI=true, all logs are sent to stdout.\n   *\n   * @default false\n   */\n  readonly onlyStderr?: boolean;\n\n  /**\n   * Don't log to stdout\n   *\n   * @default always\n   */\n  readonly show?: 'always' | 'never' | 'error';\n}\n\nexport class ShellHelper {\n  public static fromContext(context: TestContext & TemporaryDirectoryContext) {\n    return new ShellHelper(context.integTestDir, context.output);\n  }\n\n  constructor(\n    private readonly _cwd: string,\n    private readonly _output: NodeJS.WritableStream) { }\n\n  public async shell(command: string[], options: Omit<ShellOptions, 'cwd' | 'outputs'> = {}): Promise<string> {\n    return shell(command, {\n      outputs: [this._output],\n      cwd: this._cwd,\n      ...options,\n    });\n  }\n}\n\n/**\n * rm -rf reimplementation, don't want to depend on an NPM package for this\n */\nexport function rimraf(fsPath: string) {\n  try {\n    const isDir = fs.lstatSync(fsPath).isDirectory();\n\n    if (isDir) {\n      for (const file of fs.readdirSync(fsPath)) {\n        rimraf(path.join(fsPath, file));\n      }\n      fs.rmdirSync(fsPath);\n    } else {\n      fs.unlinkSync(fsPath);\n    }\n  } catch (e: any) {\n    // We will survive ENOENT\n    if (e.code !== 'ENOENT') { throw e; }\n  }\n}\n\nexport function addToShellPath(x: string) {\n  const parts = process.env.PATH?.split(':') ?? [];\n\n  if (!parts.includes(x)) {\n    parts.unshift(x);\n  }\n\n  process.env.PATH = parts.join(':');\n}\n"]}
|