@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.
Files changed (257) hide show
  1. package/.eslintrc.js +9 -0
  2. package/LICENSE +202 -0
  3. package/NOTICE +16 -0
  4. package/README.md +205 -0
  5. package/bin/apply-patches +22 -0
  6. package/bin/download-and-run-old-tests +52 -0
  7. package/bin/query-github +2 -0
  8. package/bin/query-github.d.ts +1 -0
  9. package/bin/query-github.js +54 -0
  10. package/bin/query-github.ts +56 -0
  11. package/bin/run-suite +2 -0
  12. package/bin/run-suite.d.ts +1 -0
  13. package/bin/run-suite.js +131 -0
  14. package/bin/run-suite.ts +140 -0
  15. package/bin/stage-distribution +2 -0
  16. package/bin/stage-distribution.d.ts +1 -0
  17. package/bin/stage-distribution.js +217 -0
  18. package/bin/stage-distribution.ts +267 -0
  19. package/bin/test-root +2 -0
  20. package/bin/test-root.d.ts +1 -0
  21. package/bin/test-root.js +6 -0
  22. package/bin/test-root.ts +3 -0
  23. package/entrypoints/test-cli-regression-against-current-code.sh +11 -0
  24. package/entrypoints/test-cli-regression-against-latest-release.sh +11 -0
  25. package/entrypoints/test-cli-regression.bash +83 -0
  26. package/entrypoints/test.sh +12 -0
  27. package/lib/aws.d.ts +51 -0
  28. package/lib/aws.js +206 -0
  29. package/lib/aws.ts +263 -0
  30. package/lib/corking.d.ts +12 -0
  31. package/lib/corking.js +35 -0
  32. package/lib/corking.ts +33 -0
  33. package/lib/eventually.d.ts +20 -0
  34. package/lib/eventually.js +34 -0
  35. package/lib/eventually.ts +42 -0
  36. package/lib/files.d.ts +15 -0
  37. package/lib/files.js +80 -0
  38. package/lib/files.ts +80 -0
  39. package/lib/github.d.ts +4 -0
  40. package/lib/github.js +43 -0
  41. package/lib/github.ts +43 -0
  42. package/lib/index.d.ts +13 -0
  43. package/lib/index.js +30 -0
  44. package/lib/index.ts +13 -0
  45. package/lib/integ-test.d.ts +10 -0
  46. package/lib/integ-test.js +70 -0
  47. package/lib/integ-test.ts +81 -0
  48. package/lib/lists.d.ts +1 -0
  49. package/lib/lists.js +11 -0
  50. package/lib/lists.ts +9 -0
  51. package/lib/memoize.d.ts +6 -0
  52. package/lib/memoize.js +18 -0
  53. package/lib/memoize.ts +14 -0
  54. package/lib/npm.d.ts +8 -0
  55. package/lib/npm.js +38 -0
  56. package/lib/npm.ts +41 -0
  57. package/lib/package-sources/release-source.d.ts +23 -0
  58. package/lib/package-sources/release-source.js +71 -0
  59. package/lib/package-sources/release-source.ts +81 -0
  60. package/lib/package-sources/repo-source.d.ts +30 -0
  61. package/lib/package-sources/repo-source.js +97 -0
  62. package/lib/package-sources/repo-source.ts +111 -0
  63. package/lib/package-sources/repo-tools/npm +2 -0
  64. package/lib/package-sources/repo-tools/npm.d.ts +1 -0
  65. package/lib/package-sources/repo-tools/npm.js +43 -0
  66. package/lib/package-sources/repo-tools/npm.ts +48 -0
  67. package/lib/package-sources/source.d.ts +28 -0
  68. package/lib/package-sources/source.js +3 -0
  69. package/lib/package-sources/source.ts +35 -0
  70. package/lib/package-sources/subprocess.d.ts +3 -0
  71. package/lib/package-sources/subprocess.js +17 -0
  72. package/lib/package-sources/subprocess.ts +15 -0
  73. package/lib/resource-pool.d.ts +50 -0
  74. package/lib/resource-pool.js +117 -0
  75. package/lib/resource-pool.ts +140 -0
  76. package/lib/resources.d.ts +1 -0
  77. package/lib/resources.js +6 -0
  78. package/lib/resources.ts +4 -0
  79. package/lib/shell.d.ts +56 -0
  80. package/lib/shell.js +123 -0
  81. package/lib/shell.ts +168 -0
  82. package/lib/staging/codeartifact.d.ts +44 -0
  83. package/lib/staging/codeartifact.js +281 -0
  84. package/lib/staging/codeartifact.ts +387 -0
  85. package/lib/staging/maven.d.ts +5 -0
  86. package/lib/staging/maven.js +91 -0
  87. package/lib/staging/maven.ts +95 -0
  88. package/lib/staging/npm.d.ts +4 -0
  89. package/lib/staging/npm.js +55 -0
  90. package/lib/staging/npm.ts +62 -0
  91. package/lib/staging/nuget.d.ts +4 -0
  92. package/lib/staging/nuget.js +69 -0
  93. package/lib/staging/nuget.ts +75 -0
  94. package/lib/staging/parallel-shell.d.ts +5 -0
  95. package/lib/staging/parallel-shell.js +45 -0
  96. package/lib/staging/parallel-shell.ts +51 -0
  97. package/lib/staging/pypi.d.ts +4 -0
  98. package/lib/staging/pypi.js +48 -0
  99. package/lib/staging/pypi.ts +50 -0
  100. package/lib/staging/usage-dir.d.ts +31 -0
  101. package/lib/staging/usage-dir.js +87 -0
  102. package/lib/staging/usage-dir.ts +99 -0
  103. package/lib/with-aws.d.ts +14 -0
  104. package/lib/with-aws.js +60 -0
  105. package/lib/with-aws.ts +67 -0
  106. package/lib/with-cdk-app.d.ts +210 -0
  107. package/lib/with-cdk-app.js +539 -0
  108. package/lib/with-cdk-app.ts +742 -0
  109. package/lib/with-cli-lib.d.ts +17 -0
  110. package/lib/with-cli-lib.js +123 -0
  111. package/lib/with-cli-lib.ts +134 -0
  112. package/lib/with-packages.d.ts +5 -0
  113. package/lib/with-packages.js +13 -0
  114. package/lib/with-packages.ts +15 -0
  115. package/lib/with-sam.d.ts +33 -0
  116. package/lib/with-sam.js +258 -0
  117. package/lib/with-sam.ts +288 -0
  118. package/lib/with-temporary-directory.d.ts +5 -0
  119. package/lib/with-temporary-directory.js +31 -0
  120. package/lib/with-temporary-directory.ts +35 -0
  121. package/lib/with-timeout.d.ts +19 -0
  122. package/lib/with-timeout.js +34 -0
  123. package/lib/with-timeout.ts +33 -0
  124. package/lib/xpmutex.d.ts +43 -0
  125. package/lib/xpmutex.js +207 -0
  126. package/lib/xpmutex.ts +218 -0
  127. package/package.json +111 -0
  128. package/resources/bootstrap-templates/session-tags.all-roles-deny-all.yaml +703 -0
  129. package/resources/bootstrap-templates/session-tags.deploy-role-deny-sqs.yaml +700 -0
  130. package/resources/cdk-apps/app/app.js +926 -0
  131. package/resources/cdk-apps/app/appsync.hotswap.graphql +3 -0
  132. package/resources/cdk-apps/app/cdk.json +7 -0
  133. package/resources/cdk-apps/app/docker/Dockerfile +2 -0
  134. package/resources/cdk-apps/app/docker/Dockerfile.Custom +2 -0
  135. package/resources/cdk-apps/app/lambda/index.js +4 -0
  136. package/resources/cdk-apps/app/lambda/response.json +3 -0
  137. package/resources/cdk-apps/app/nested-stack.js +65 -0
  138. package/resources/cdk-apps/cfn-include-app/cdk.json +4 -0
  139. package/resources/cdk-apps/cfn-include-app/cfn-include-app.js +21 -0
  140. package/resources/cdk-apps/cfn-include-app/example-template.json +13 -0
  141. package/resources/cdk-apps/rollback-test-app/app.js +110 -0
  142. package/resources/cdk-apps/rollback-test-app/cdk.json +7 -0
  143. package/resources/cdk-apps/sam_cdk_integ_app/bin/test-app.js +11 -0
  144. package/resources/cdk-apps/sam_cdk_integ_app/cdk.json +6 -0
  145. package/resources/cdk-apps/sam_cdk_integ_app/lib/nested-stack.js +19 -0
  146. package/resources/cdk-apps/sam_cdk_integ_app/lib/test-stack.js +134 -0
  147. package/resources/cdk-apps/sam_cdk_integ_app/src/docker/DockerImageFunctionConstruct/.no-packagejson-validator +0 -0
  148. package/resources/cdk-apps/sam_cdk_integ_app/src/docker/DockerImageFunctionConstruct/Dockerfile +9 -0
  149. package/resources/cdk-apps/sam_cdk_integ_app/src/docker/DockerImageFunctionConstruct/app.js +22 -0
  150. package/resources/cdk-apps/sam_cdk_integ_app/src/docker/DockerImageFunctionConstruct/package.json +18 -0
  151. package/resources/cdk-apps/sam_cdk_integ_app/src/go/GoFunctionConstruct/go.mod +5 -0
  152. package/resources/cdk-apps/sam_cdk_integ_app/src/go/GoFunctionConstruct/go.sum +17 -0
  153. package/resources/cdk-apps/sam_cdk_integ_app/src/go/GoFunctionConstruct/main.go +17 -0
  154. package/resources/cdk-apps/sam_cdk_integ_app/src/nodejs/NodeJsFunctionConstruct/.no-packagejson-validator +0 -0
  155. package/resources/cdk-apps/sam_cdk_integ_app/src/nodejs/NodeJsFunctionConstruct/app.ts +16 -0
  156. package/resources/cdk-apps/sam_cdk_integ_app/src/nodejs/NodeJsFunctionConstruct/package-lock.json +12 -0
  157. package/resources/cdk-apps/sam_cdk_integ_app/src/nodejs/NodeJsFunctionConstruct/package.json +5 -0
  158. package/resources/cdk-apps/sam_cdk_integ_app/src/python/Function/app.py +15 -0
  159. package/resources/cdk-apps/sam_cdk_integ_app/src/python/Function/requirements.txt +1 -0
  160. package/resources/cdk-apps/sam_cdk_integ_app/src/python/Layer/layer_version_dependency.py +5 -0
  161. package/resources/cdk-apps/sam_cdk_integ_app/src/python/Layer/requirements.txt +1 -0
  162. package/resources/cdk-apps/sam_cdk_integ_app/src/rest-api-definition.yaml +12 -0
  163. package/resources/cdk-apps/simple-app/app.js +26 -0
  164. package/resources/cdk-apps/simple-app/cdk.json +7 -0
  165. package/resources/cli-regression-patches/v1.119.0/NOTES.md +5 -0
  166. package/resources/cli-regression-patches/v1.119.0/cli.integtest.js +659 -0
  167. package/resources/cli-regression-patches/v1.130.0/NOTES.md +12 -0
  168. package/resources/cli-regression-patches/v1.130.0/app/app.js +378 -0
  169. package/resources/cli-regression-patches/v1.130.0/bootstrapping.integtest.js +220 -0
  170. package/resources/cli-regression-patches/v1.44.0/NOTES.md +18 -0
  171. package/resources/cli-regression-patches/v1.44.0/bootstrapping.integtest.js +126 -0
  172. package/resources/cli-regression-patches/v1.44.0/test.sh +26 -0
  173. package/resources/cli-regression-patches/v1.61.1/NOTES.md +2 -0
  174. package/resources/cli-regression-patches/v1.61.1/skip-tests.txt +16 -0
  175. package/resources/cli-regression-patches/v1.62.0/NOTES.md +2 -0
  176. package/resources/cli-regression-patches/v1.62.0/aws-helpers.js +245 -0
  177. package/resources/cli-regression-patches/v1.63.0/NOTES.md +1 -0
  178. package/resources/cli-regression-patches/v1.63.0/skip-tests.txt +7 -0
  179. package/resources/cli-regression-patches/v1.64.0/NOTES.md +3 -0
  180. package/resources/cli-regression-patches/v1.64.0/cdk-helpers.js +325 -0
  181. package/resources/cli-regression-patches/v1.64.0/cli.integtest.js +599 -0
  182. package/resources/cli-regression-patches/v1.64.1/NOTES.md +3 -0
  183. package/resources/cli-regression-patches/v1.64.1/cdk-helpers.js +324 -0
  184. package/resources/cli-regression-patches/v1.64.1/cli.integtest.js +599 -0
  185. package/resources/cli-regression-patches/v1.67.0/NOTES.md +2 -0
  186. package/resources/cli-regression-patches/v1.67.0/cdk-helpers.js +331 -0
  187. package/resources/cli-regression-patches/v2.130.0/NOTES.md +1 -0
  188. 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
  189. 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
  190. package/resources/cli-regression-patches/v2.130.0/skip-tests.txt +5 -0
  191. package/resources/cli-regression-patches/v2.132.0/NOTES.md +1 -0
  192. package/resources/cli-regression-patches/v2.132.0/skip-tests.txt +4 -0
  193. package/resources/cli-regression-patches/v2.142.0/NOTES.md +1 -0
  194. package/resources/cli-regression-patches/v2.142.0/skip-tests.txt +4 -0
  195. package/resources/cli-regression-patches/v2.160.0/skip-tests.txt +2 -0
  196. package/resources/cli-regression-patches/v2.161.0/NOTES.md +1 -0
  197. package/resources/cli-regression-patches/v2.161.0/skip-tests.txt +5 -0
  198. package/resources/cli-regression-patches/v2.166.0/NOTES.md +1 -0
  199. package/resources/cli-regression-patches/v2.166.0/skip-tests.txt +2 -0
  200. package/resources/cloud-assemblies/0.36.0/InitStack.template.json +1 -0
  201. package/resources/cloud-assemblies/0.36.0/cdk.out +1 -0
  202. package/resources/cloud-assemblies/0.36.0/manifest.json +19 -0
  203. package/resources/cloud-assemblies/1.10.0-lookup-default-vpc/InitStack.template.json +2 -0
  204. package/resources/cloud-assemblies/1.10.0-lookup-default-vpc/cdk.out +1 -0
  205. package/resources/cloud-assemblies/1.10.0-lookup-default-vpc/manifest.json.js +37 -0
  206. package/resources/cloud-assemblies/1.10.0-request-azs/InitStack.template.json +2 -0
  207. package/resources/cloud-assemblies/1.10.0-request-azs/cdk.out +1 -0
  208. package/resources/cloud-assemblies/1.10.0-request-azs/manifest.json.js +34 -0
  209. package/resources/integ.jest.config.js +25 -0
  210. package/resources/templates/sqs-template.json +36 -0
  211. package/skip-tests.txt +8 -0
  212. package/tests/cli-integ-tests/README.md +47 -0
  213. package/tests/cli-integ-tests/bootstrapping.integtest.d.ts +1 -0
  214. package/tests/cli-integ-tests/bootstrapping.integtest.js +412 -0
  215. package/tests/cli-integ-tests/bootstrapping.integtest.ts +493 -0
  216. package/tests/cli-integ-tests/cli-lib.integtest.d.ts +1 -0
  217. package/tests/cli-integ-tests/cli-lib.integtest.js +62 -0
  218. package/tests/cli-integ-tests/cli-lib.integtest.ts +90 -0
  219. package/tests/cli-integ-tests/cli.integtest.d.ts +1 -0
  220. package/tests/cli-integ-tests/cli.integtest.js +2104 -0
  221. package/tests/cli-integ-tests/cli.integtest.ts +2874 -0
  222. package/tests/cli-integ-tests/garbage-collection.integtest.d.ts +1 -0
  223. package/tests/cli-integ-tests/garbage-collection.integtest.js +314 -0
  224. package/tests/cli-integ-tests/garbage-collection.integtest.ts +392 -0
  225. package/tests/init-csharp/init-csharp.integtest.d.ts +1 -0
  226. package/tests/init-csharp/init-csharp.integtest.js +14 -0
  227. package/tests/init-csharp/init-csharp.integtest.ts +15 -0
  228. package/tests/init-fsharp/init-fsharp.integtest.d.ts +1 -0
  229. package/tests/init-fsharp/init-fsharp.integtest.js +14 -0
  230. package/tests/init-fsharp/init-fsharp.integtest.ts +15 -0
  231. package/tests/init-go/init-go.integtest.d.ts +1 -0
  232. package/tests/init-go/init-go.integtest.js +21 -0
  233. package/tests/init-go/init-go.integtest.ts +23 -0
  234. package/tests/init-java/init-java.integtest.d.ts +1 -0
  235. package/tests/init-java/init-java.integtest.js +14 -0
  236. package/tests/init-java/init-java.integtest.ts +14 -0
  237. package/tests/init-javascript/init-javascript.integtest.d.ts +1 -0
  238. package/tests/init-javascript/init-javascript.integtest.js +53 -0
  239. package/tests/init-javascript/init-javascript.integtest.ts +59 -0
  240. package/tests/init-python/init-python.integtest.d.ts +1 -0
  241. package/tests/init-python/init-python.integtest.js +19 -0
  242. package/tests/init-python/init-python.integtest.ts +20 -0
  243. package/tests/init-typescript-app/init-typescript-app.integtest.d.ts +1 -0
  244. package/tests/init-typescript-app/init-typescript-app.integtest.js +54 -0
  245. package/tests/init-typescript-app/init-typescript-app.integtest.ts +66 -0
  246. package/tests/init-typescript-lib/init-typescript-lib.integtest.d.ts +1 -0
  247. package/tests/init-typescript-lib/init-typescript-lib.integtest.js +13 -0
  248. package/tests/init-typescript-lib/init-typescript-lib.integtest.ts +13 -0
  249. package/tests/tool-integrations/amplify.integtest.d.ts +1 -0
  250. package/tests/tool-integrations/amplify.integtest.js +39 -0
  251. package/tests/tool-integrations/amplify.integtest.ts +43 -0
  252. package/tests/tool-integrations/with-tool-context.d.ts +9 -0
  253. package/tests/tool-integrations/with-tool-context.js +13 -0
  254. package/tests/tool-integrations/with-tool-context.ts +14 -0
  255. package/tests/uberpackage/uberpackage.integtest.d.ts +1 -0
  256. package/tests/uberpackage/uberpackage.integtest.js +11 -0
  257. package/tests/uberpackage/uberpackage.integtest.ts +11 -0
package/lib/aws.js ADDED
@@ -0,0 +1,206 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.AwsClients = void 0;
4
+ exports.isStackMissingError = isStackMissingError;
5
+ exports.isBucketMissingError = isBucketMissingError;
6
+ exports.retry = retry;
7
+ exports.outputFromStack = outputFromStack;
8
+ exports.sleep = sleep;
9
+ const client_cloudformation_1 = require("@aws-sdk/client-cloudformation");
10
+ const client_ecr_1 = require("@aws-sdk/client-ecr");
11
+ const client_ecs_1 = require("@aws-sdk/client-ecs");
12
+ const client_iam_1 = require("@aws-sdk/client-iam");
13
+ const client_lambda_1 = require("@aws-sdk/client-lambda");
14
+ const client_s3_1 = require("@aws-sdk/client-s3");
15
+ const client_sns_1 = require("@aws-sdk/client-sns");
16
+ const client_sso_1 = require("@aws-sdk/client-sso");
17
+ const client_sts_1 = require("@aws-sdk/client-sts");
18
+ const credential_providers_1 = require("@aws-sdk/credential-providers");
19
+ const util_retry_1 = require("@smithy/util-retry");
20
+ class AwsClients {
21
+ static async default(output) {
22
+ var _a, _b;
23
+ const region = (_b = (_a = process.env.AWS_REGION) !== null && _a !== void 0 ? _a : process.env.AWS_DEFAULT_REGION) !== null && _b !== void 0 ? _b : 'us-east-1';
24
+ return AwsClients.forRegion(region, output);
25
+ }
26
+ static async forRegion(region, output) {
27
+ return new AwsClients(region, output);
28
+ }
29
+ constructor(region, output) {
30
+ this.region = region;
31
+ this.output = output;
32
+ this.config = {
33
+ credentials: chainableCredentials(this.region),
34
+ region: this.region,
35
+ retryStrategy: new util_retry_1.ConfiguredRetryStrategy(9, (attempt) => attempt ** 500),
36
+ };
37
+ this.cloudFormation = new client_cloudformation_1.CloudFormationClient(this.config);
38
+ this.s3 = new client_s3_1.S3Client(this.config);
39
+ this.ecr = new client_ecr_1.ECRClient(this.config);
40
+ this.ecs = new client_ecs_1.ECSClient(this.config);
41
+ this.sso = new client_sso_1.SSOClient(this.config);
42
+ this.sns = new client_sns_1.SNSClient(this.config);
43
+ this.iam = new client_iam_1.IAMClient(this.config);
44
+ this.lambda = new client_lambda_1.LambdaClient(this.config);
45
+ this.sts = new client_sts_1.STSClient(this.config);
46
+ }
47
+ async account() {
48
+ // Reduce # of retries, we use this as a circuit breaker for detecting no-config
49
+ const stsClient = new client_sts_1.STSClient({
50
+ credentials: this.config.credentials,
51
+ region: this.config.region,
52
+ maxAttempts: 2,
53
+ });
54
+ return (await stsClient.send(new client_sts_1.GetCallerIdentityCommand({}))).Account;
55
+ }
56
+ async deleteStacks(...stackNames) {
57
+ if (stackNames.length === 0) {
58
+ return;
59
+ }
60
+ // We purposely do all stacks serially, because they've been ordered
61
+ // to do the bootstrap stack last.
62
+ for (const stackName of stackNames) {
63
+ await this.cloudFormation.send(new client_cloudformation_1.UpdateTerminationProtectionCommand({
64
+ EnableTerminationProtection: false,
65
+ StackName: stackName,
66
+ }));
67
+ await this.cloudFormation.send(new client_cloudformation_1.DeleteStackCommand({
68
+ StackName: stackName,
69
+ }));
70
+ await retry(this.output, `Deleting ${stackName}`, retry.forSeconds(600), async () => {
71
+ const status = await this.stackStatus(stackName);
72
+ if (status !== undefined && status.endsWith('_FAILED')) {
73
+ throw retry.abort(new Error(`'${stackName}' is in state '${status}'`));
74
+ }
75
+ if (status !== undefined) {
76
+ throw new Error(`Delete of '${stackName}' not complete yet, status: '${status}'`);
77
+ }
78
+ });
79
+ }
80
+ }
81
+ async stackStatus(stackName) {
82
+ var _a;
83
+ try {
84
+ return (_a = (await this.cloudFormation.send(new client_cloudformation_1.DescribeStacksCommand({
85
+ StackName: stackName,
86
+ }))).Stacks) === null || _a === void 0 ? void 0 : _a[0].StackStatus;
87
+ }
88
+ catch (e) {
89
+ if (isStackMissingError(e)) {
90
+ return undefined;
91
+ }
92
+ throw e;
93
+ }
94
+ }
95
+ async emptyBucket(bucketName, options) {
96
+ const objects = await this.s3.send(new client_s3_1.ListObjectVersionsCommand({
97
+ Bucket: bucketName,
98
+ }));
99
+ const deletes = [...(objects.Versions || []), ...(objects.DeleteMarkers || [])].reduce((acc, obj) => {
100
+ if (typeof obj.VersionId !== 'undefined' && typeof obj.Key !== 'undefined') {
101
+ acc.push({ Key: obj.Key, VersionId: obj.VersionId });
102
+ }
103
+ else if (typeof obj.Key !== 'undefined') {
104
+ acc.push({ Key: obj.Key });
105
+ }
106
+ return acc;
107
+ }, []);
108
+ if (deletes.length === 0) {
109
+ return Promise.resolve();
110
+ }
111
+ return this.s3.send(new client_s3_1.DeleteObjectsCommand({
112
+ Bucket: bucketName,
113
+ Delete: {
114
+ Objects: deletes,
115
+ Quiet: false,
116
+ },
117
+ BypassGovernanceRetention: (options === null || options === void 0 ? void 0 : options.bypassGovernance) ? true : undefined,
118
+ }));
119
+ }
120
+ async deleteImageRepository(repositoryName) {
121
+ await this.ecr.send(new client_ecr_1.DeleteRepositoryCommand({
122
+ repositoryName: repositoryName,
123
+ force: true,
124
+ }));
125
+ }
126
+ async deleteBucket(bucketName) {
127
+ try {
128
+ await this.emptyBucket(bucketName);
129
+ await this.s3.send(new client_s3_1.DeleteBucketCommand({
130
+ Bucket: bucketName,
131
+ }));
132
+ }
133
+ catch (e) {
134
+ if (isBucketMissingError(e)) {
135
+ return;
136
+ }
137
+ throw e;
138
+ }
139
+ }
140
+ }
141
+ exports.AwsClients = AwsClients;
142
+ function isStackMissingError(e) {
143
+ return e.message.indexOf('does not exist') > -1;
144
+ }
145
+ function isBucketMissingError(e) {
146
+ return e.message.indexOf('does not exist') > -1;
147
+ }
148
+ /**
149
+ * Retry an async operation until a deadline is hit.
150
+ *
151
+ * Use `retry.forSeconds()` to construct a deadline relative to right now.
152
+ *
153
+ * Exceptions will cause the operation to retry. Use `retry.abort` to annotate an exception
154
+ * to stop the retry and end in a failure.
155
+ */
156
+ async function retry(output, operation, deadline, block) {
157
+ let i = 0;
158
+ output.write(`💈 ${operation}\n`);
159
+ while (true) {
160
+ try {
161
+ i++;
162
+ const ret = await block();
163
+ output.write(`💈 ${operation}: succeeded after ${i} attempts\n`);
164
+ return ret;
165
+ }
166
+ catch (e) {
167
+ if (e.abort || Date.now() > deadline.getTime()) {
168
+ throw new Error(`${operation}: did not succeed after ${i} attempts: ${e}`);
169
+ }
170
+ output.write(`⏳ ${operation} (${e.message})\n`);
171
+ await sleep(5000);
172
+ }
173
+ }
174
+ }
175
+ /**
176
+ * Make a deadline for the `retry` function relative to the current time.
177
+ */
178
+ retry.forSeconds = (seconds) => {
179
+ return new Date(Date.now() + seconds * 1000);
180
+ };
181
+ /**
182
+ * Annotate an error to stop the retrying
183
+ */
184
+ retry.abort = (e) => {
185
+ e.abort = true;
186
+ return e;
187
+ };
188
+ function outputFromStack(key, stack) {
189
+ var _a, _b;
190
+ return (_b = ((_a = stack.Outputs) !== null && _a !== void 0 ? _a : []).find((o) => o.OutputKey === key)) === null || _b === void 0 ? void 0 : _b.OutputValue;
191
+ }
192
+ async function sleep(ms) {
193
+ return new Promise((ok) => setTimeout(ok, ms));
194
+ }
195
+ function chainableCredentials(region) {
196
+ if (process.env.CODEBUILD_BUILD_ARN && process.env.AWS_PROFILE) {
197
+ // in codebuild we must assume the role that the cdk uses
198
+ // otherwise credentials will just be picked up by the normal sdk
199
+ // heuristics and expire after an hour.
200
+ return (0, credential_providers_1.fromIni)({
201
+ clientConfig: { region },
202
+ });
203
+ }
204
+ return undefined;
205
+ }
206
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aws.js","sourceRoot":"","sources":["aws.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  CloudFormationClient,\n  DeleteStackCommand,\n  DescribeStacksCommand,\n  UpdateTerminationProtectionCommand,\n  type Stack,\n} from '@aws-sdk/client-cloudformation';\nimport { DeleteRepositoryCommand, ECRClient } from '@aws-sdk/client-ecr';\nimport { ECSClient } from '@aws-sdk/client-ecs';\nimport { IAMClient } from '@aws-sdk/client-iam';\nimport { LambdaClient } from '@aws-sdk/client-lambda';\nimport {\n  S3Client,\n  DeleteObjectsCommand,\n  ListObjectVersionsCommand,\n  type ObjectIdentifier,\n  DeleteBucketCommand,\n} from '@aws-sdk/client-s3';\nimport { SNSClient } from '@aws-sdk/client-sns';\nimport { SSOClient } from '@aws-sdk/client-sso';\nimport { STSClient, GetCallerIdentityCommand } from '@aws-sdk/client-sts';\nimport { fromIni } from '@aws-sdk/credential-providers';\nimport type { AwsCredentialIdentityProvider } from '@smithy/types';\nimport { ConfiguredRetryStrategy } from '@smithy/util-retry';\ninterface ClientConfig {\n  readonly credentials?: AwsCredentialIdentityProvider;\n  readonly region: string;\n  readonly retryStrategy: ConfiguredRetryStrategy;\n}\n\nexport class AwsClients {\n  public static async default(output: NodeJS.WritableStream) {\n    const region = process.env.AWS_REGION ?? process.env.AWS_DEFAULT_REGION ?? 'us-east-1';\n    return AwsClients.forRegion(region, output);\n  }\n\n  public static async forRegion(region: string, output: NodeJS.WritableStream) {\n    return new AwsClients(region, output);\n  }\n\n  private readonly config: ClientConfig;\n\n  public readonly cloudFormation: CloudFormationClient;\n  public readonly s3: S3Client;\n  public readonly ecr: ECRClient;\n  public readonly ecs: ECSClient;\n  public readonly sso: SSOClient;\n  public readonly sns: SNSClient;\n  public readonly iam: IAMClient;\n  public readonly lambda: LambdaClient;\n  public readonly sts: STSClient;\n\n  constructor(public readonly region: string, private readonly output: NodeJS.WritableStream) {\n    this.config = {\n      credentials: chainableCredentials(this.region),\n      region: this.region,\n      retryStrategy: new ConfiguredRetryStrategy(9, (attempt: number) => attempt ** 500),\n    };\n    this.cloudFormation = new CloudFormationClient(this.config);\n    this.s3 = new S3Client(this.config);\n    this.ecr = new ECRClient(this.config);\n    this.ecs = new ECSClient(this.config);\n    this.sso = new SSOClient(this.config);\n    this.sns = new SNSClient(this.config);\n    this.iam = new IAMClient(this.config);\n    this.lambda = new LambdaClient(this.config);\n    this.sts = new STSClient(this.config);\n  }\n\n  public async account(): Promise<string> {\n    // Reduce # of retries, we use this as a circuit breaker for detecting no-config\n    const stsClient = new STSClient({\n      credentials: this.config.credentials,\n      region: this.config.region,\n      maxAttempts: 2,\n    });\n\n    return (await stsClient.send(new GetCallerIdentityCommand({}))).Account!;\n  }\n\n  public async deleteStacks(...stackNames: string[]) {\n    if (stackNames.length === 0) {\n      return;\n    }\n\n    // We purposely do all stacks serially, because they've been ordered\n    // to do the bootstrap stack last.\n    for (const stackName of stackNames) {\n      await this.cloudFormation.send(\n        new UpdateTerminationProtectionCommand({\n          EnableTerminationProtection: false,\n          StackName: stackName,\n        }),\n      );\n      await this.cloudFormation.send(\n        new DeleteStackCommand({\n          StackName: stackName,\n        }),\n      );\n\n      await retry(this.output, `Deleting ${stackName}`, retry.forSeconds(600), async () => {\n        const status = await this.stackStatus(stackName);\n        if (status !== undefined && status.endsWith('_FAILED')) {\n          throw retry.abort(new Error(`'${stackName}' is in state '${status}'`));\n        }\n        if (status !== undefined) {\n          throw new Error(`Delete of '${stackName}' not complete yet, status: '${status}'`);\n        }\n      });\n    }\n  }\n\n  public async stackStatus(stackName: string): Promise<string | undefined> {\n    try {\n      return (\n        await this.cloudFormation.send(\n          new DescribeStacksCommand({\n            StackName: stackName,\n          }),\n        )\n      ).Stacks?.[0].StackStatus;\n    } catch (e: any) {\n      if (isStackMissingError(e)) {\n        return undefined;\n      }\n      throw e;\n    }\n  }\n\n  public async emptyBucket(bucketName: string, options?: { bypassGovernance?: boolean }) {\n    const objects = await this.s3.send(\n      new ListObjectVersionsCommand({\n        Bucket: bucketName,\n      }),\n    );\n\n    const deletes = [...(objects.Versions || []), ...(objects.DeleteMarkers || [])].reduce((acc, obj) => {\n      if (typeof obj.VersionId !== 'undefined' && typeof obj.Key !== 'undefined') {\n        acc.push({ Key: obj.Key, VersionId: obj.VersionId });\n      } else if (typeof obj.Key !== 'undefined') {\n        acc.push({ Key: obj.Key });\n      }\n      return acc;\n    }, [] as ObjectIdentifier[]);\n\n    if (deletes.length === 0) {\n      return Promise.resolve();\n    }\n\n    return this.s3.send(\n      new DeleteObjectsCommand({\n        Bucket: bucketName,\n        Delete: {\n          Objects: deletes,\n          Quiet: false,\n        },\n        BypassGovernanceRetention: options?.bypassGovernance ? true : undefined,\n      }),\n    );\n  }\n\n  public async deleteImageRepository(repositoryName: string) {\n    await this.ecr.send(\n      new DeleteRepositoryCommand({\n        repositoryName: repositoryName,\n        force: true,\n      }),\n    );\n  }\n\n  public async deleteBucket(bucketName: string) {\n    try {\n      await this.emptyBucket(bucketName);\n\n      await this.s3.send(\n        new DeleteBucketCommand({\n          Bucket: bucketName,\n        }),\n      );\n    } catch (e: any) {\n      if (isBucketMissingError(e)) {\n        return;\n      }\n      throw e;\n    }\n  }\n}\n\nexport function isStackMissingError(e: Error) {\n  return e.message.indexOf('does not exist') > -1;\n}\n\nexport function isBucketMissingError(e: Error) {\n  return e.message.indexOf('does not exist') > -1;\n}\n\n/**\n * Retry an async operation until a deadline is hit.\n *\n * Use `retry.forSeconds()` to construct a deadline relative to right now.\n *\n * Exceptions will cause the operation to retry. Use `retry.abort` to annotate an exception\n * to stop the retry and end in a failure.\n */\nexport async function retry<A>(\n  output: NodeJS.WritableStream,\n  operation: string,\n  deadline: Date,\n  block: () => Promise<A>,\n): Promise<A> {\n  let i = 0;\n  output.write(`💈 ${operation}\\n`);\n  while (true) {\n    try {\n      i++;\n      const ret = await block();\n      output.write(`💈 ${operation}: succeeded after ${i} attempts\\n`);\n      return ret;\n    } catch (e: any) {\n      if (e.abort || Date.now() > deadline.getTime()) {\n        throw new Error(`${operation}: did not succeed after ${i} attempts: ${e}`);\n      }\n      output.write(`⏳ ${operation} (${e.message})\\n`);\n      await sleep(5000);\n    }\n  }\n}\n\n/**\n * Make a deadline for the `retry` function relative to the current time.\n */\nretry.forSeconds = (seconds: number): Date => {\n  return new Date(Date.now() + seconds * 1000);\n};\n\n/**\n * Annotate an error to stop the retrying\n */\nretry.abort = (e: Error): Error => {\n  (e as any).abort = true;\n  return e;\n};\n\nexport function outputFromStack(key: string, stack: Stack): string | undefined {\n  return (stack.Outputs ?? []).find((o) => o.OutputKey === key)?.OutputValue;\n}\n\nexport async function sleep(ms: number) {\n  return new Promise((ok) => setTimeout(ok, ms));\n}\n\nfunction chainableCredentials(region: string): AwsCredentialIdentityProvider | undefined {\n  if (process.env.CODEBUILD_BUILD_ARN && process.env.AWS_PROFILE) {\n    // in codebuild we must assume the role that the cdk uses\n    // otherwise credentials will just be picked up by the normal sdk\n    // heuristics and expire after an hour.\n    return fromIni({\n      clientConfig: { region },\n    });\n  }\n\n  return undefined;\n}\n"]}
package/lib/aws.ts ADDED
@@ -0,0 +1,263 @@
1
+ import {
2
+ CloudFormationClient,
3
+ DeleteStackCommand,
4
+ DescribeStacksCommand,
5
+ UpdateTerminationProtectionCommand,
6
+ type Stack,
7
+ } from '@aws-sdk/client-cloudformation';
8
+ import { DeleteRepositoryCommand, ECRClient } from '@aws-sdk/client-ecr';
9
+ import { ECSClient } from '@aws-sdk/client-ecs';
10
+ import { IAMClient } from '@aws-sdk/client-iam';
11
+ import { LambdaClient } from '@aws-sdk/client-lambda';
12
+ import {
13
+ S3Client,
14
+ DeleteObjectsCommand,
15
+ ListObjectVersionsCommand,
16
+ type ObjectIdentifier,
17
+ DeleteBucketCommand,
18
+ } from '@aws-sdk/client-s3';
19
+ import { SNSClient } from '@aws-sdk/client-sns';
20
+ import { SSOClient } from '@aws-sdk/client-sso';
21
+ import { STSClient, GetCallerIdentityCommand } from '@aws-sdk/client-sts';
22
+ import { fromIni } from '@aws-sdk/credential-providers';
23
+ import type { AwsCredentialIdentityProvider } from '@smithy/types';
24
+ import { ConfiguredRetryStrategy } from '@smithy/util-retry';
25
+ interface ClientConfig {
26
+ readonly credentials?: AwsCredentialIdentityProvider;
27
+ readonly region: string;
28
+ readonly retryStrategy: ConfiguredRetryStrategy;
29
+ }
30
+
31
+ export class AwsClients {
32
+ public static async default(output: NodeJS.WritableStream) {
33
+ const region = process.env.AWS_REGION ?? process.env.AWS_DEFAULT_REGION ?? 'us-east-1';
34
+ return AwsClients.forRegion(region, output);
35
+ }
36
+
37
+ public static async forRegion(region: string, output: NodeJS.WritableStream) {
38
+ return new AwsClients(region, output);
39
+ }
40
+
41
+ private readonly config: ClientConfig;
42
+
43
+ public readonly cloudFormation: CloudFormationClient;
44
+ public readonly s3: S3Client;
45
+ public readonly ecr: ECRClient;
46
+ public readonly ecs: ECSClient;
47
+ public readonly sso: SSOClient;
48
+ public readonly sns: SNSClient;
49
+ public readonly iam: IAMClient;
50
+ public readonly lambda: LambdaClient;
51
+ public readonly sts: STSClient;
52
+
53
+ constructor(public readonly region: string, private readonly output: NodeJS.WritableStream) {
54
+ this.config = {
55
+ credentials: chainableCredentials(this.region),
56
+ region: this.region,
57
+ retryStrategy: new ConfiguredRetryStrategy(9, (attempt: number) => attempt ** 500),
58
+ };
59
+ this.cloudFormation = new CloudFormationClient(this.config);
60
+ this.s3 = new S3Client(this.config);
61
+ this.ecr = new ECRClient(this.config);
62
+ this.ecs = new ECSClient(this.config);
63
+ this.sso = new SSOClient(this.config);
64
+ this.sns = new SNSClient(this.config);
65
+ this.iam = new IAMClient(this.config);
66
+ this.lambda = new LambdaClient(this.config);
67
+ this.sts = new STSClient(this.config);
68
+ }
69
+
70
+ public async account(): Promise<string> {
71
+ // Reduce # of retries, we use this as a circuit breaker for detecting no-config
72
+ const stsClient = new STSClient({
73
+ credentials: this.config.credentials,
74
+ region: this.config.region,
75
+ maxAttempts: 2,
76
+ });
77
+
78
+ return (await stsClient.send(new GetCallerIdentityCommand({}))).Account!;
79
+ }
80
+
81
+ public async deleteStacks(...stackNames: string[]) {
82
+ if (stackNames.length === 0) {
83
+ return;
84
+ }
85
+
86
+ // We purposely do all stacks serially, because they've been ordered
87
+ // to do the bootstrap stack last.
88
+ for (const stackName of stackNames) {
89
+ await this.cloudFormation.send(
90
+ new UpdateTerminationProtectionCommand({
91
+ EnableTerminationProtection: false,
92
+ StackName: stackName,
93
+ }),
94
+ );
95
+ await this.cloudFormation.send(
96
+ new DeleteStackCommand({
97
+ StackName: stackName,
98
+ }),
99
+ );
100
+
101
+ await retry(this.output, `Deleting ${stackName}`, retry.forSeconds(600), async () => {
102
+ const status = await this.stackStatus(stackName);
103
+ if (status !== undefined && status.endsWith('_FAILED')) {
104
+ throw retry.abort(new Error(`'${stackName}' is in state '${status}'`));
105
+ }
106
+ if (status !== undefined) {
107
+ throw new Error(`Delete of '${stackName}' not complete yet, status: '${status}'`);
108
+ }
109
+ });
110
+ }
111
+ }
112
+
113
+ public async stackStatus(stackName: string): Promise<string | undefined> {
114
+ try {
115
+ return (
116
+ await this.cloudFormation.send(
117
+ new DescribeStacksCommand({
118
+ StackName: stackName,
119
+ }),
120
+ )
121
+ ).Stacks?.[0].StackStatus;
122
+ } catch (e: any) {
123
+ if (isStackMissingError(e)) {
124
+ return undefined;
125
+ }
126
+ throw e;
127
+ }
128
+ }
129
+
130
+ public async emptyBucket(bucketName: string, options?: { bypassGovernance?: boolean }) {
131
+ const objects = await this.s3.send(
132
+ new ListObjectVersionsCommand({
133
+ Bucket: bucketName,
134
+ }),
135
+ );
136
+
137
+ const deletes = [...(objects.Versions || []), ...(objects.DeleteMarkers || [])].reduce((acc, obj) => {
138
+ if (typeof obj.VersionId !== 'undefined' && typeof obj.Key !== 'undefined') {
139
+ acc.push({ Key: obj.Key, VersionId: obj.VersionId });
140
+ } else if (typeof obj.Key !== 'undefined') {
141
+ acc.push({ Key: obj.Key });
142
+ }
143
+ return acc;
144
+ }, [] as ObjectIdentifier[]);
145
+
146
+ if (deletes.length === 0) {
147
+ return Promise.resolve();
148
+ }
149
+
150
+ return this.s3.send(
151
+ new DeleteObjectsCommand({
152
+ Bucket: bucketName,
153
+ Delete: {
154
+ Objects: deletes,
155
+ Quiet: false,
156
+ },
157
+ BypassGovernanceRetention: options?.bypassGovernance ? true : undefined,
158
+ }),
159
+ );
160
+ }
161
+
162
+ public async deleteImageRepository(repositoryName: string) {
163
+ await this.ecr.send(
164
+ new DeleteRepositoryCommand({
165
+ repositoryName: repositoryName,
166
+ force: true,
167
+ }),
168
+ );
169
+ }
170
+
171
+ public async deleteBucket(bucketName: string) {
172
+ try {
173
+ await this.emptyBucket(bucketName);
174
+
175
+ await this.s3.send(
176
+ new DeleteBucketCommand({
177
+ Bucket: bucketName,
178
+ }),
179
+ );
180
+ } catch (e: any) {
181
+ if (isBucketMissingError(e)) {
182
+ return;
183
+ }
184
+ throw e;
185
+ }
186
+ }
187
+ }
188
+
189
+ export function isStackMissingError(e: Error) {
190
+ return e.message.indexOf('does not exist') > -1;
191
+ }
192
+
193
+ export function isBucketMissingError(e: Error) {
194
+ return e.message.indexOf('does not exist') > -1;
195
+ }
196
+
197
+ /**
198
+ * Retry an async operation until a deadline is hit.
199
+ *
200
+ * Use `retry.forSeconds()` to construct a deadline relative to right now.
201
+ *
202
+ * Exceptions will cause the operation to retry. Use `retry.abort` to annotate an exception
203
+ * to stop the retry and end in a failure.
204
+ */
205
+ export async function retry<A>(
206
+ output: NodeJS.WritableStream,
207
+ operation: string,
208
+ deadline: Date,
209
+ block: () => Promise<A>,
210
+ ): Promise<A> {
211
+ let i = 0;
212
+ output.write(`💈 ${operation}\n`);
213
+ while (true) {
214
+ try {
215
+ i++;
216
+ const ret = await block();
217
+ output.write(`💈 ${operation}: succeeded after ${i} attempts\n`);
218
+ return ret;
219
+ } catch (e: any) {
220
+ if (e.abort || Date.now() > deadline.getTime()) {
221
+ throw new Error(`${operation}: did not succeed after ${i} attempts: ${e}`);
222
+ }
223
+ output.write(`⏳ ${operation} (${e.message})\n`);
224
+ await sleep(5000);
225
+ }
226
+ }
227
+ }
228
+
229
+ /**
230
+ * Make a deadline for the `retry` function relative to the current time.
231
+ */
232
+ retry.forSeconds = (seconds: number): Date => {
233
+ return new Date(Date.now() + seconds * 1000);
234
+ };
235
+
236
+ /**
237
+ * Annotate an error to stop the retrying
238
+ */
239
+ retry.abort = (e: Error): Error => {
240
+ (e as any).abort = true;
241
+ return e;
242
+ };
243
+
244
+ export function outputFromStack(key: string, stack: Stack): string | undefined {
245
+ return (stack.Outputs ?? []).find((o) => o.OutputKey === key)?.OutputValue;
246
+ }
247
+
248
+ export async function sleep(ms: number) {
249
+ return new Promise((ok) => setTimeout(ok, ms));
250
+ }
251
+
252
+ function chainableCredentials(region: string): AwsCredentialIdentityProvider | undefined {
253
+ if (process.env.CODEBUILD_BUILD_ARN && process.env.AWS_PROFILE) {
254
+ // in codebuild we must assume the role that the cdk uses
255
+ // otherwise credentials will just be picked up by the normal sdk
256
+ // heuristics and expire after an hour.
257
+ return fromIni({
258
+ clientConfig: { region },
259
+ });
260
+ }
261
+
262
+ return undefined;
263
+ }
@@ -0,0 +1,12 @@
1
+ /**
2
+ * Routines for corking stdout and stderr
3
+ */
4
+ import * as stream from 'stream';
5
+ export declare class MemoryStream extends stream.Writable {
6
+ private parts;
7
+ _write(chunk: Buffer, _encoding: string, callback: (error?: Error | null) => void): void;
8
+ buffer(): Buffer;
9
+ clear(): void;
10
+ flushTo(strm: NodeJS.WritableStream): Promise<void>;
11
+ toString(): string;
12
+ }
package/lib/corking.js ADDED
@@ -0,0 +1,35 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.MemoryStream = void 0;
4
+ /**
5
+ * Routines for corking stdout and stderr
6
+ */
7
+ const stream = require("stream");
8
+ class MemoryStream extends stream.Writable {
9
+ constructor() {
10
+ super(...arguments);
11
+ this.parts = new Array();
12
+ }
13
+ _write(chunk, _encoding, callback) {
14
+ this.parts.push(chunk);
15
+ callback();
16
+ }
17
+ buffer() {
18
+ return Buffer.concat(this.parts);
19
+ }
20
+ clear() {
21
+ this.parts.splice(0, this.parts.length);
22
+ }
23
+ async flushTo(strm) {
24
+ const flushed = strm.write(this.buffer());
25
+ if (!flushed) {
26
+ return new Promise(ok => strm.once('drain', ok));
27
+ }
28
+ return;
29
+ }
30
+ toString() {
31
+ return this.buffer().toString();
32
+ }
33
+ }
34
+ exports.MemoryStream = MemoryStream;
35
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29ya2luZy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImNvcmtpbmcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUE7O0dBRUc7QUFDSCxpQ0FBaUM7QUFFakMsTUFBYSxZQUFhLFNBQVEsTUFBTSxDQUFDLFFBQVE7SUFBakQ7O1FBQ1UsVUFBSyxHQUFHLElBQUksS0FBSyxFQUFVLENBQUM7SUEwQnRDLENBQUM7SUF4QlEsTUFBTSxDQUFDLEtBQWEsRUFBRSxTQUFpQixFQUFFLFFBQXdDO1FBQ3RGLElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3ZCLFFBQVEsRUFBRSxDQUFDO0lBQ2IsQ0FBQztJQUVNLE1BQU07UUFDWCxPQUFPLE1BQU0sQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQ25DLENBQUM7SUFFTSxLQUFLO1FBQ1YsSUFBSSxDQUFDLEtBQUssQ0FBQyxNQUFNLENBQUMsQ0FBQyxFQUFFLElBQUksQ0FBQyxLQUFLLENBQUMsTUFBTSxDQUFDLENBQUM7SUFDMUMsQ0FBQztJQUVNLEtBQUssQ0FBQyxPQUFPLENBQUMsSUFBMkI7UUFDOUMsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLENBQUMsQ0FBQztRQUMxQyxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7WUFDYixPQUFPLElBQUksT0FBTyxDQUFDLEVBQUUsQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxPQUFPLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQztRQUNuRCxDQUFDO1FBQ0QsT0FBTztJQUNULENBQUM7SUFFTSxRQUFRO1FBQ2IsT0FBTyxJQUFJLENBQUMsTUFBTSxFQUFFLENBQUMsUUFBUSxFQUFFLENBQUM7SUFDbEMsQ0FBQztDQUNGO0FBM0JELG9DQTJCQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogUm91dGluZXMgZm9yIGNvcmtpbmcgc3Rkb3V0IGFuZCBzdGRlcnJcbiAqL1xuaW1wb3J0ICogYXMgc3RyZWFtIGZyb20gJ3N0cmVhbSc7XG5cbmV4cG9ydCBjbGFzcyBNZW1vcnlTdHJlYW0gZXh0ZW5kcyBzdHJlYW0uV3JpdGFibGUge1xuICBwcml2YXRlIHBhcnRzID0gbmV3IEFycmF5PEJ1ZmZlcj4oKTtcblxuICBwdWJsaWMgX3dyaXRlKGNodW5rOiBCdWZmZXIsIF9lbmNvZGluZzogc3RyaW5nLCBjYWxsYmFjazogKGVycm9yPzogRXJyb3IgfCBudWxsKSA9PiB2b2lkKTogdm9pZCB7XG4gICAgdGhpcy5wYXJ0cy5wdXNoKGNodW5rKTtcbiAgICBjYWxsYmFjaygpO1xuICB9XG5cbiAgcHVibGljIGJ1ZmZlcigpIHtcbiAgICByZXR1cm4gQnVmZmVyLmNvbmNhdCh0aGlzLnBhcnRzKTtcbiAgfVxuXG4gIHB1YmxpYyBjbGVhcigpIHtcbiAgICB0aGlzLnBhcnRzLnNwbGljZSgwLCB0aGlzLnBhcnRzLmxlbmd0aCk7XG4gIH1cblxuICBwdWJsaWMgYXN5bmMgZmx1c2hUbyhzdHJtOiBOb2RlSlMuV3JpdGFibGVTdHJlYW0pOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBjb25zdCBmbHVzaGVkID0gc3RybS53cml0ZSh0aGlzLmJ1ZmZlcigpKTtcbiAgICBpZiAoIWZsdXNoZWQpIHtcbiAgICAgIHJldHVybiBuZXcgUHJvbWlzZShvayA9PiBzdHJtLm9uY2UoJ2RyYWluJywgb2spKTtcbiAgICB9XG4gICAgcmV0dXJuO1xuICB9XG5cbiAgcHVibGljIHRvU3RyaW5nKCkge1xuICAgIHJldHVybiB0aGlzLmJ1ZmZlcigpLnRvU3RyaW5nKCk7XG4gIH1cbn1cbiJdfQ==
package/lib/corking.ts ADDED
@@ -0,0 +1,33 @@
1
+ /**
2
+ * Routines for corking stdout and stderr
3
+ */
4
+ import * as stream from 'stream';
5
+
6
+ export class MemoryStream extends stream.Writable {
7
+ private parts = new Array<Buffer>();
8
+
9
+ public _write(chunk: Buffer, _encoding: string, callback: (error?: Error | null) => void): void {
10
+ this.parts.push(chunk);
11
+ callback();
12
+ }
13
+
14
+ public buffer() {
15
+ return Buffer.concat(this.parts);
16
+ }
17
+
18
+ public clear() {
19
+ this.parts.splice(0, this.parts.length);
20
+ }
21
+
22
+ public async flushTo(strm: NodeJS.WritableStream): Promise<void> {
23
+ const flushed = strm.write(this.buffer());
24
+ if (!flushed) {
25
+ return new Promise(ok => strm.once('drain', ok));
26
+ }
27
+ return;
28
+ }
29
+
30
+ public toString() {
31
+ return this.buffer().toString();
32
+ }
33
+ }
@@ -0,0 +1,20 @@
1
+ /**
2
+ * @param maxAttempts the maximum number of attempts
3
+ * @param interval interval in milliseconds to observe between attempts
4
+ */
5
+ export type EventuallyOptions = {
6
+ maxAttempts?: number;
7
+ interval?: number;
8
+ };
9
+ /**
10
+ * Runs a function on an interval until the maximum number of attempts has
11
+ * been reached.
12
+ *
13
+ * Default interval = 1000 milliseconds
14
+ * Default maxAttempts = 10
15
+ *
16
+ * @param fn function to run
17
+ * @param options EventuallyOptions
18
+ */
19
+ declare const eventually: <T>(call: () => Promise<T>, options?: EventuallyOptions) => Promise<T>;
20
+ export default eventually;
@@ -0,0 +1,34 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const wait = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
4
+ const DEFAULT_INTERVAL = 1000;
5
+ const DEFAULT_MAX_ATTEMPTS = 10;
6
+ /**
7
+ * Runs a function on an interval until the maximum number of attempts has
8
+ * been reached.
9
+ *
10
+ * Default interval = 1000 milliseconds
11
+ * Default maxAttempts = 10
12
+ *
13
+ * @param fn function to run
14
+ * @param options EventuallyOptions
15
+ */
16
+ const eventually = async (call, options) => {
17
+ const opts = {
18
+ interval: (options === null || options === void 0 ? void 0 : options.interval) ? options.interval : DEFAULT_INTERVAL,
19
+ maxAttempts: (options === null || options === void 0 ? void 0 : options.maxAttempts) ? options.maxAttempts : DEFAULT_MAX_ATTEMPTS,
20
+ };
21
+ while (opts.maxAttempts-- >= 0) {
22
+ try {
23
+ return await call();
24
+ }
25
+ catch (err) {
26
+ if (opts.maxAttempts <= 0)
27
+ throw err;
28
+ }
29
+ await wait(opts.interval);
30
+ }
31
+ throw new Error('An unexpected error has occurred.');
32
+ };
33
+ exports.default = eventually;
34
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXZlbnR1YWxseS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImV2ZW50dWFsbHkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFTQSxNQUFNLElBQUksR0FBRyxDQUFDLEVBQVUsRUFBaUIsRUFBRSxDQUFDLElBQUksT0FBTyxDQUFDLENBQUMsT0FBTyxFQUFFLEVBQUUsQ0FBQyxVQUFVLENBQUMsT0FBTyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUM7QUFDOUYsTUFBTSxnQkFBZ0IsR0FBRyxJQUFJLENBQUM7QUFDOUIsTUFBTSxvQkFBb0IsR0FBRyxFQUFFLENBQUM7QUFFaEM7Ozs7Ozs7OztHQVNHO0FBQ0gsTUFBTSxVQUFVLEdBQUcsS0FBSyxFQUFLLElBQXNCLEVBQUUsT0FBMkIsRUFBYyxFQUFFO0lBQzlGLE1BQU0sSUFBSSxHQUFHO1FBQ1gsUUFBUSxFQUFFLENBQUEsT0FBTyxhQUFQLE9BQU8sdUJBQVAsT0FBTyxDQUFFLFFBQVEsRUFBQyxDQUFDLENBQUMsT0FBTyxDQUFDLFFBQVEsQ0FBQyxDQUFDLENBQUMsZ0JBQWdCO1FBQ2pFLFdBQVcsRUFBRSxDQUFBLE9BQU8sYUFBUCxPQUFPLHVCQUFQLE9BQU8sQ0FBRSxXQUFXLEVBQUMsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxXQUFXLENBQUMsQ0FBQyxDQUFDLG9CQUFvQjtLQUMvRSxDQUFDO0lBRUYsT0FBTyxJQUFJLENBQUMsV0FBVyxFQUFFLElBQUksQ0FBQyxFQUFFLENBQUM7UUFDL0IsSUFBSSxDQUFDO1lBQ0gsT0FBTyxNQUFNLElBQUksRUFBRSxDQUFDO1FBQ3RCLENBQUM7UUFBQyxPQUFPLEdBQUcsRUFBRSxDQUFDO1lBQ2IsSUFBSSxJQUFJLENBQUMsV0FBVyxJQUFJLENBQUM7Z0JBQUUsTUFBTSxHQUFHLENBQUM7UUFDdkMsQ0FBQztRQUNELE1BQU0sSUFBSSxDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsQ0FBQztJQUM1QixDQUFDO0lBRUQsTUFBTSxJQUFJLEtBQUssQ0FBQyxtQ0FBbUMsQ0FBQyxDQUFDO0FBQ3ZELENBQUMsQ0FBQztBQUVGLGtCQUFlLFVBQVUsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQHBhcmFtIG1heEF0dGVtcHRzIHRoZSBtYXhpbXVtIG51bWJlciBvZiBhdHRlbXB0c1xuICogQHBhcmFtIGludGVydmFsIGludGVydmFsIGluIG1pbGxpc2Vjb25kcyB0byBvYnNlcnZlIGJldHdlZW4gYXR0ZW1wdHNcbiAqL1xuZXhwb3J0IHR5cGUgRXZlbnR1YWxseU9wdGlvbnMgPSB7XG4gIG1heEF0dGVtcHRzPzogbnVtYmVyO1xuICBpbnRlcnZhbD86IG51bWJlcjtcbn07XG5cbmNvbnN0IHdhaXQgPSAobXM6IG51bWJlcik6IFByb21pc2U8dm9pZD4gPT4gbmV3IFByb21pc2UoKHJlc29sdmUpID0+IHNldFRpbWVvdXQocmVzb2x2ZSwgbXMpKTtcbmNvbnN0IERFRkFVTFRfSU5URVJWQUwgPSAxMDAwO1xuY29uc3QgREVGQVVMVF9NQVhfQVRURU1QVFMgPSAxMDtcblxuLyoqXG4gKiBSdW5zIGEgZnVuY3Rpb24gb24gYW4gaW50ZXJ2YWwgdW50aWwgdGhlIG1heGltdW0gbnVtYmVyIG9mIGF0dGVtcHRzIGhhc1xuICogYmVlbiByZWFjaGVkLlxuICpcbiAqIERlZmF1bHQgaW50ZXJ2YWwgPSAxMDAwIG1pbGxpc2Vjb25kc1xuICogRGVmYXVsdCBtYXhBdHRlbXB0cyA9IDEwXG4gKlxuICogQHBhcmFtIGZuIGZ1bmN0aW9uIHRvIHJ1blxuICogQHBhcmFtIG9wdGlvbnMgRXZlbnR1YWxseU9wdGlvbnNcbiAqL1xuY29uc3QgZXZlbnR1YWxseSA9IGFzeW5jIDxUPihjYWxsOiAoKSA9PiBQcm9taXNlPFQ+LCBvcHRpb25zPzogRXZlbnR1YWxseU9wdGlvbnMpOiBQcm9taXNlPFQ+ID0+IHtcbiAgY29uc3Qgb3B0cyA9IHtcbiAgICBpbnRlcnZhbDogb3B0aW9ucz8uaW50ZXJ2YWwgPyBvcHRpb25zLmludGVydmFsIDogREVGQVVMVF9JTlRFUlZBTCxcbiAgICBtYXhBdHRlbXB0czogb3B0aW9ucz8ubWF4QXR0ZW1wdHMgPyBvcHRpb25zLm1heEF0dGVtcHRzIDogREVGQVVMVF9NQVhfQVRURU1QVFMsXG4gIH07XG5cbiAgd2hpbGUgKG9wdHMubWF4QXR0ZW1wdHMtLSA+PSAwKSB7XG4gICAgdHJ5IHtcbiAgICAgIHJldHVybiBhd2FpdCBjYWxsKCk7XG4gICAgfSBjYXRjaCAoZXJyKSB7XG4gICAgICBpZiAob3B0cy5tYXhBdHRlbXB0cyA8PSAwKSB0aHJvdyBlcnI7XG4gICAgfVxuICAgIGF3YWl0IHdhaXQob3B0cy5pbnRlcnZhbCk7XG4gIH1cblxuICB0aHJvdyBuZXcgRXJyb3IoJ0FuIHVuZXhwZWN0ZWQgZXJyb3IgaGFzIG9jY3VycmVkLicpO1xufTtcblxuZXhwb3J0IGRlZmF1bHQgZXZlbnR1YWxseTtcbiJdfQ==