@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
@@ -0,0 +1,62 @@
1
+ /* eslint-disable no-console */
2
+ import * as path from 'path';
3
+ import { LoginInformation } from './codeartifact';
4
+ import { parallelShell } from './parallel-shell';
5
+ import { UsageDir } from './usage-dir';
6
+ import { updateIniKey, loadLines, writeLines } from '../files';
7
+ import { shell } from '../shell';
8
+
9
+ export async function npmLogin(login: LoginInformation, usageDir: UsageDir) {
10
+ // Creating an ~/.npmrc that references an envvar is what you're supposed to do. (https://docs.npmjs.com/private-modules/ci-server-config)
11
+ await writeNpmLoginToken(usageDir, login.npmEndpoint, '${NPM_TOKEN}');
12
+
13
+ // Add variables to env file
14
+ await usageDir.addToEnv(npmEnv(usageDir, login));
15
+ }
16
+
17
+ function npmEnv(usageDir: UsageDir, login: LoginInformation) {
18
+ return {
19
+ npm_config_userconfig: path.join(usageDir.directory, '.npmrc'),
20
+ npm_config_registry: login.npmEndpoint,
21
+ npm_config_always_auth: 'true', // Necessary for NPM 6, otherwise it will sometimes not pass the token
22
+ NPM_TOKEN: login.authToken,
23
+ };
24
+ }
25
+
26
+ export async function uploadNpmPackages(packages: string[], login: LoginInformation, usageDir: UsageDir) {
27
+ await parallelShell(packages, async (pkg, output) => {
28
+ console.log(`⏳ ${pkg}`);
29
+
30
+ // path.resolve() is required -- if the filename ends up looking like `js/bla.tgz` then NPM thinks it's a short form GitHub name.
31
+ await shell(['node', require.resolve('npm'), 'publish', path.resolve(pkg)], {
32
+ modEnv: npmEnv(usageDir, login),
33
+ show: 'error',
34
+ outputs: [output],
35
+ });
36
+
37
+ console.log(`✅ ${pkg}`);
38
+ }, (pkg, output) => {
39
+ if (output.toString().includes('code EPUBLISHCONFLICT')) {
40
+ console.log(`❌ ${pkg}: already exists. Skipped.`);
41
+ return 'skip';
42
+ }
43
+ if (output.toString().includes('code EPRIVATE')) {
44
+ console.log(`❌ ${pkg}: is private. Skipped.`);
45
+ return 'skip';
46
+ }
47
+ return 'fail';
48
+ });
49
+ }
50
+
51
+ async function writeNpmLoginToken(usageDir: UsageDir, endpoint: string, token: string) {
52
+ const rcFile = path.join(usageDir.directory, '.npmrc');
53
+ const lines = await loadLines(rcFile);
54
+
55
+ const key = `${endpoint.replace(/^https:/, '')}:_authToken`;
56
+ updateIniKey(lines, key, token);
57
+
58
+ await writeLines(rcFile, lines);
59
+ return rcFile;
60
+ }
61
+
62
+ // Environment variable, .npmrc in same directory as package.json or in home dir
@@ -0,0 +1,4 @@
1
+ import { LoginInformation } from './codeartifact';
2
+ import { UsageDir } from './usage-dir';
3
+ export declare function nugetLogin(login: LoginInformation, usageDir: UsageDir): Promise<void>;
4
+ export declare function uploadDotnetPackages(packages: string[], usageDir: UsageDir): Promise<void>;
@@ -0,0 +1,69 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.nugetLogin = nugetLogin;
4
+ exports.uploadDotnetPackages = uploadDotnetPackages;
5
+ const parallel_shell_1 = require("./parallel-shell");
6
+ const files_1 = require("../files");
7
+ const shell_1 = require("../shell");
8
+ async function nugetLogin(login, usageDir) {
9
+ // NuGet.Config MUST live in the current directory or in the home directory, and there is no environment
10
+ // variable to configure its location.
11
+ await writeNuGetConfigFile(usageDir.cwdFile('NuGet.Config'), login);
12
+ }
13
+ async function uploadDotnetPackages(packages, usageDir) {
14
+ await usageDir.copyCwdFileHere('NuGet.Config');
15
+ await (0, parallel_shell_1.parallelShell)(packages, async (pkg, output) => {
16
+ console.log(`⏳ ${pkg}`);
17
+ await (0, shell_1.shell)(['dotnet', 'nuget', 'push',
18
+ pkg,
19
+ '--source', 'CodeArtifact',
20
+ '--no-symbols',
21
+ '--force-english-output',
22
+ '--disable-buffering',
23
+ '--timeout', '600',
24
+ '--skip-duplicate'], {
25
+ outputs: [output],
26
+ });
27
+ console.log(`✅ ${pkg}`);
28
+ }, (pkg, output) => {
29
+ if (output.toString().includes('Conflict')) {
30
+ console.log(`❌ ${pkg}: already exists. Skipped.`);
31
+ return 'skip';
32
+ }
33
+ if (output.includes('System.Threading.AbandonedMutexException')) {
34
+ console.log(`♻️ ${pkg}: AbandonedMutexException. Probably a sign of throttling, retrying.`);
35
+ return 'retry';
36
+ }
37
+ if (output.includes('Too Many Requests')) {
38
+ console.log(`♻️ ${pkg}: Too many requests. Retrying.`);
39
+ return 'retry';
40
+ }
41
+ if (output.includes('System.IO.IOException: The system cannot open the device or file specified.')) {
42
+ console.log(`♻️ ${pkg}: Some error that we've seen before as a result of throttling. Retrying.`);
43
+ return 'retry';
44
+ }
45
+ return 'fail';
46
+ });
47
+ }
48
+ async function writeNuGetConfigFile(filename, login) {
49
+ // `dotnet nuget push` has an `--api-key` parameter, but CodeArtifact
50
+ // does not support that. We must authenticate with Basic auth.
51
+ await (0, files_1.writeFile)(filename, `<?xml version="1.0" encoding="utf-8"?>
52
+ <configuration>
53
+ <packageSources>
54
+ <add key="nuget.org" value="https://api.nuget.org/v3/index.json" protocolVersion="3" />
55
+ <add key="CodeArtifact" value="${login.nugetEndpoint}v3/index.json" />
56
+ </packageSources>
57
+ <activePackageSource>
58
+ <add key="CodeArtifact" value="${login.nugetEndpoint}v3/index.json" />
59
+ </activePackageSource>
60
+ <packageSourceCredentials>
61
+ <CodeArtifact>
62
+ <add key="Username" value="aws" />
63
+ <add key="ClearTextPassword" value="${login.authToken}" />
64
+ </CodeArtifact>
65
+ </packageSourceCredentials>
66
+ </configuration>`);
67
+ }
68
+ // NuGet.Config in current directory
69
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibnVnZXQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJudWdldC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQU9BLGdDQUlDO0FBRUQsb0RBc0NDO0FBakRELHFEQUFpRDtBQUVqRCxvQ0FBcUM7QUFDckMsb0NBQWlDO0FBRTFCLEtBQUssVUFBVSxVQUFVLENBQUMsS0FBdUIsRUFBRSxRQUFrQjtJQUMxRSx3R0FBd0c7SUFDeEcsc0NBQXNDO0lBQ3RDLE1BQU0sb0JBQW9CLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxjQUFjLENBQUMsRUFBRSxLQUFLLENBQUMsQ0FBQztBQUN0RSxDQUFDO0FBRU0sS0FBSyxVQUFVLG9CQUFvQixDQUFDLFFBQWtCLEVBQUUsUUFBa0I7SUFDL0UsTUFBTSxRQUFRLENBQUMsZUFBZSxDQUFDLGNBQWMsQ0FBQyxDQUFDO0lBRS9DLE1BQU0sSUFBQSw4QkFBYSxFQUFDLFFBQVEsRUFBRSxLQUFLLEVBQUUsR0FBRyxFQUFFLE1BQU0sRUFBRSxFQUFFO1FBQ2xELE9BQU8sQ0FBQyxHQUFHLENBQUMsS0FBSyxHQUFHLEVBQUUsQ0FBQyxDQUFDO1FBRXhCLE1BQU0sSUFBQSxhQUFLLEVBQUMsQ0FBQyxRQUFRLEVBQUUsT0FBTyxFQUFFLE1BQU07WUFDcEMsR0FBRztZQUNILFVBQVUsRUFBRSxjQUFjO1lBQzFCLGNBQWM7WUFDZCx3QkFBd0I7WUFDeEIscUJBQXFCO1lBQ3JCLFdBQVcsRUFBRSxLQUFLO1lBQ2xCLGtCQUFrQixDQUFDLEVBQUU7WUFDckIsT0FBTyxFQUFFLENBQUMsTUFBTSxDQUFDO1NBQ2xCLENBQUMsQ0FBQztRQUVILE9BQU8sQ0FBQyxHQUFHLENBQUMsS0FBSyxHQUFHLEVBQUUsQ0FBQyxDQUFDO0lBQzFCLENBQUMsRUFDRCxDQUFDLEdBQUcsRUFBRSxNQUFNLEVBQUUsRUFBRTtRQUNkLElBQUksTUFBTSxDQUFDLFFBQVEsRUFBRSxDQUFDLFFBQVEsQ0FBQyxVQUFVLENBQUMsRUFBRSxDQUFDO1lBQzNDLE9BQU8sQ0FBQyxHQUFHLENBQUMsS0FBSyxHQUFHLDRCQUE0QixDQUFDLENBQUM7WUFDbEQsT0FBTyxNQUFNLENBQUM7UUFDaEIsQ0FBQztRQUNELElBQUksTUFBTSxDQUFDLFFBQVEsQ0FBQywwQ0FBMEMsQ0FBQyxFQUFFLENBQUM7WUFDaEUsT0FBTyxDQUFDLEdBQUcsQ0FBQyxNQUFNLEdBQUcscUVBQXFFLENBQUMsQ0FBQztZQUM1RixPQUFPLE9BQU8sQ0FBQztRQUNqQixDQUFDO1FBQ0QsSUFBSSxNQUFNLENBQUMsUUFBUSxDQUFDLG1CQUFtQixDQUFDLEVBQUUsQ0FBQztZQUN6QyxPQUFPLENBQUMsR0FBRyxDQUFDLE1BQU0sR0FBRyxnQ0FBZ0MsQ0FBQyxDQUFDO1lBQ3ZELE9BQU8sT0FBTyxDQUFDO1FBQ2pCLENBQUM7UUFDRCxJQUFJLE1BQU0sQ0FBQyxRQUFRLENBQUMsNkVBQTZFLENBQUMsRUFBRSxDQUFDO1lBQ25HLE9BQU8sQ0FBQyxHQUFHLENBQUMsTUFBTSxHQUFHLDBFQUEwRSxDQUFDLENBQUM7WUFDakcsT0FBTyxPQUFPLENBQUM7UUFDakIsQ0FBQztRQUNELE9BQU8sTUFBTSxDQUFDO0lBQ2hCLENBQUMsQ0FBQyxDQUFDO0FBQ0wsQ0FBQztBQUVELEtBQUssVUFBVSxvQkFBb0IsQ0FBQyxRQUFnQixFQUFFLEtBQXVCO0lBQzNFLHFFQUFxRTtJQUNyRSwrREFBK0Q7SUFDL0QsTUFBTSxJQUFBLGlCQUFTLEVBQUMsUUFBUSxFQUFFOzs7O3FDQUlTLEtBQUssQ0FBQyxhQUFhOzs7cUNBR25CLEtBQUssQ0FBQyxhQUFhOzs7Ozs4Q0FLVixLQUFLLENBQUMsU0FBUzs7O2lCQUc1QyxDQUFDLENBQUM7QUFDbkIsQ0FBQztBQUVELG9DQUFvQyIsInNvdXJjZXNDb250ZW50IjpbIi8qIGVzbGludC1kaXNhYmxlIG5vLWNvbnNvbGUgKi9cbmltcG9ydCB7IExvZ2luSW5mb3JtYXRpb24gfSBmcm9tICcuL2NvZGVhcnRpZmFjdCc7XG5pbXBvcnQgeyBwYXJhbGxlbFNoZWxsIH0gZnJvbSAnLi9wYXJhbGxlbC1zaGVsbCc7XG5pbXBvcnQgeyBVc2FnZURpciB9IGZyb20gJy4vdXNhZ2UtZGlyJztcbmltcG9ydCB7IHdyaXRlRmlsZSB9IGZyb20gJy4uL2ZpbGVzJztcbmltcG9ydCB7IHNoZWxsIH0gZnJvbSAnLi4vc2hlbGwnO1xuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gbnVnZXRMb2dpbihsb2dpbjogTG9naW5JbmZvcm1hdGlvbiwgdXNhZ2VEaXI6IFVzYWdlRGlyKSB7XG4gIC8vIE51R2V0LkNvbmZpZyBNVVNUIGxpdmUgaW4gdGhlIGN1cnJlbnQgZGlyZWN0b3J5IG9yIGluIHRoZSBob21lIGRpcmVjdG9yeSwgYW5kIHRoZXJlIGlzIG5vIGVudmlyb25tZW50XG4gIC8vIHZhcmlhYmxlIHRvIGNvbmZpZ3VyZSBpdHMgbG9jYXRpb24uXG4gIGF3YWl0IHdyaXRlTnVHZXRDb25maWdGaWxlKHVzYWdlRGlyLmN3ZEZpbGUoJ051R2V0LkNvbmZpZycpLCBsb2dpbik7XG59XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiB1cGxvYWREb3RuZXRQYWNrYWdlcyhwYWNrYWdlczogc3RyaW5nW10sIHVzYWdlRGlyOiBVc2FnZURpcikge1xuICBhd2FpdCB1c2FnZURpci5jb3B5Q3dkRmlsZUhlcmUoJ051R2V0LkNvbmZpZycpO1xuXG4gIGF3YWl0IHBhcmFsbGVsU2hlbGwocGFja2FnZXMsIGFzeW5jIChwa2csIG91dHB1dCkgPT4ge1xuICAgIGNvbnNvbGUubG9nKGDij7MgJHtwa2d9YCk7XG5cbiAgICBhd2FpdCBzaGVsbChbJ2RvdG5ldCcsICdudWdldCcsICdwdXNoJyxcbiAgICAgIHBrZyxcbiAgICAgICctLXNvdXJjZScsICdDb2RlQXJ0aWZhY3QnLFxuICAgICAgJy0tbm8tc3ltYm9scycsXG4gICAgICAnLS1mb3JjZS1lbmdsaXNoLW91dHB1dCcsXG4gICAgICAnLS1kaXNhYmxlLWJ1ZmZlcmluZycsXG4gICAgICAnLS10aW1lb3V0JywgJzYwMCcsXG4gICAgICAnLS1za2lwLWR1cGxpY2F0ZSddLCB7XG4gICAgICBvdXRwdXRzOiBbb3V0cHV0XSxcbiAgICB9KTtcblxuICAgIGNvbnNvbGUubG9nKGDinIUgJHtwa2d9YCk7XG4gIH0sXG4gIChwa2csIG91dHB1dCkgPT4ge1xuICAgIGlmIChvdXRwdXQudG9TdHJpbmcoKS5pbmNsdWRlcygnQ29uZmxpY3QnKSkge1xuICAgICAgY29uc29sZS5sb2coYOKdjCAke3BrZ306IGFscmVhZHkgZXhpc3RzLiBTa2lwcGVkLmApO1xuICAgICAgcmV0dXJuICdza2lwJztcbiAgICB9XG4gICAgaWYgKG91dHB1dC5pbmNsdWRlcygnU3lzdGVtLlRocmVhZGluZy5BYmFuZG9uZWRNdXRleEV4Y2VwdGlvbicpKSB7XG4gICAgICBjb25zb2xlLmxvZyhg4pm777iPICR7cGtnfTogQWJhbmRvbmVkTXV0ZXhFeGNlcHRpb24uIFByb2JhYmx5IGEgc2lnbiBvZiB0aHJvdHRsaW5nLCByZXRyeWluZy5gKTtcbiAgICAgIHJldHVybiAncmV0cnknO1xuICAgIH1cbiAgICBpZiAob3V0cHV0LmluY2x1ZGVzKCdUb28gTWFueSBSZXF1ZXN0cycpKSB7XG4gICAgICBjb25zb2xlLmxvZyhg4pm777iPICR7cGtnfTogVG9vIG1hbnkgcmVxdWVzdHMuIFJldHJ5aW5nLmApO1xuICAgICAgcmV0dXJuICdyZXRyeSc7XG4gICAgfVxuICAgIGlmIChvdXRwdXQuaW5jbHVkZXMoJ1N5c3RlbS5JTy5JT0V4Y2VwdGlvbjogVGhlIHN5c3RlbSBjYW5ub3Qgb3BlbiB0aGUgZGV2aWNlIG9yIGZpbGUgc3BlY2lmaWVkLicpKSB7XG4gICAgICBjb25zb2xlLmxvZyhg4pm777iPICR7cGtnfTogU29tZSBlcnJvciB0aGF0IHdlJ3ZlIHNlZW4gYmVmb3JlIGFzIGEgcmVzdWx0IG9mIHRocm90dGxpbmcuIFJldHJ5aW5nLmApO1xuICAgICAgcmV0dXJuICdyZXRyeSc7XG4gICAgfVxuICAgIHJldHVybiAnZmFpbCc7XG4gIH0pO1xufVxuXG5hc3luYyBmdW5jdGlvbiB3cml0ZU51R2V0Q29uZmlnRmlsZShmaWxlbmFtZTogc3RyaW5nLCBsb2dpbjogTG9naW5JbmZvcm1hdGlvbikge1xuICAvLyBgZG90bmV0IG51Z2V0IHB1c2hgIGhhcyBhbiBgLS1hcGkta2V5YCBwYXJhbWV0ZXIsIGJ1dCBDb2RlQXJ0aWZhY3RcbiAgLy8gZG9lcyBub3Qgc3VwcG9ydCB0aGF0LiBXZSBtdXN0IGF1dGhlbnRpY2F0ZSB3aXRoIEJhc2ljIGF1dGguXG4gIGF3YWl0IHdyaXRlRmlsZShmaWxlbmFtZSwgYDw/eG1sIHZlcnNpb249XCIxLjBcIiBlbmNvZGluZz1cInV0Zi04XCI/PlxuPGNvbmZpZ3VyYXRpb24+XG4gIDxwYWNrYWdlU291cmNlcz5cbiAgICA8YWRkIGtleT1cIm51Z2V0Lm9yZ1wiIHZhbHVlPVwiaHR0cHM6Ly9hcGkubnVnZXQub3JnL3YzL2luZGV4Lmpzb25cIiBwcm90b2NvbFZlcnNpb249XCIzXCIgLz5cbiAgICA8YWRkIGtleT1cIkNvZGVBcnRpZmFjdFwiIHZhbHVlPVwiJHtsb2dpbi5udWdldEVuZHBvaW50fXYzL2luZGV4Lmpzb25cIiAvPlxuICA8L3BhY2thZ2VTb3VyY2VzPlxuICA8YWN0aXZlUGFja2FnZVNvdXJjZT5cbiAgICA8YWRkIGtleT1cIkNvZGVBcnRpZmFjdFwiIHZhbHVlPVwiJHtsb2dpbi5udWdldEVuZHBvaW50fXYzL2luZGV4Lmpzb25cIiAvPlxuICA8L2FjdGl2ZVBhY2thZ2VTb3VyY2U+XG4gIDxwYWNrYWdlU291cmNlQ3JlZGVudGlhbHM+XG4gICAgPENvZGVBcnRpZmFjdD5cbiAgICAgICAgPGFkZCBrZXk9XCJVc2VybmFtZVwiIHZhbHVlPVwiYXdzXCIgLz5cbiAgICAgICAgPGFkZCBrZXk9XCJDbGVhclRleHRQYXNzd29yZFwiIHZhbHVlPVwiJHtsb2dpbi5hdXRoVG9rZW59XCIgLz5cbiAgICAgIDwvQ29kZUFydGlmYWN0PlxuICA8L3BhY2thZ2VTb3VyY2VDcmVkZW50aWFscz5cbjwvY29uZmlndXJhdGlvbj5gKTtcbn1cblxuLy8gTnVHZXQuQ29uZmlnIGluIGN1cnJlbnQgZGlyZWN0b3J5XG4iXX0=
@@ -0,0 +1,75 @@
1
+ /* eslint-disable no-console */
2
+ import { LoginInformation } from './codeartifact';
3
+ import { parallelShell } from './parallel-shell';
4
+ import { UsageDir } from './usage-dir';
5
+ import { writeFile } from '../files';
6
+ import { shell } from '../shell';
7
+
8
+ export async function nugetLogin(login: LoginInformation, usageDir: UsageDir) {
9
+ // NuGet.Config MUST live in the current directory or in the home directory, and there is no environment
10
+ // variable to configure its location.
11
+ await writeNuGetConfigFile(usageDir.cwdFile('NuGet.Config'), login);
12
+ }
13
+
14
+ export async function uploadDotnetPackages(packages: string[], usageDir: UsageDir) {
15
+ await usageDir.copyCwdFileHere('NuGet.Config');
16
+
17
+ await parallelShell(packages, async (pkg, output) => {
18
+ console.log(`⏳ ${pkg}`);
19
+
20
+ await shell(['dotnet', 'nuget', 'push',
21
+ pkg,
22
+ '--source', 'CodeArtifact',
23
+ '--no-symbols',
24
+ '--force-english-output',
25
+ '--disable-buffering',
26
+ '--timeout', '600',
27
+ '--skip-duplicate'], {
28
+ outputs: [output],
29
+ });
30
+
31
+ console.log(`✅ ${pkg}`);
32
+ },
33
+ (pkg, output) => {
34
+ if (output.toString().includes('Conflict')) {
35
+ console.log(`❌ ${pkg}: already exists. Skipped.`);
36
+ return 'skip';
37
+ }
38
+ if (output.includes('System.Threading.AbandonedMutexException')) {
39
+ console.log(`♻️ ${pkg}: AbandonedMutexException. Probably a sign of throttling, retrying.`);
40
+ return 'retry';
41
+ }
42
+ if (output.includes('Too Many Requests')) {
43
+ console.log(`♻️ ${pkg}: Too many requests. Retrying.`);
44
+ return 'retry';
45
+ }
46
+ if (output.includes('System.IO.IOException: The system cannot open the device or file specified.')) {
47
+ console.log(`♻️ ${pkg}: Some error that we've seen before as a result of throttling. Retrying.`);
48
+ return 'retry';
49
+ }
50
+ return 'fail';
51
+ });
52
+ }
53
+
54
+ async function writeNuGetConfigFile(filename: string, login: LoginInformation) {
55
+ // `dotnet nuget push` has an `--api-key` parameter, but CodeArtifact
56
+ // does not support that. We must authenticate with Basic auth.
57
+ await writeFile(filename, `<?xml version="1.0" encoding="utf-8"?>
58
+ <configuration>
59
+ <packageSources>
60
+ <add key="nuget.org" value="https://api.nuget.org/v3/index.json" protocolVersion="3" />
61
+ <add key="CodeArtifact" value="${login.nugetEndpoint}v3/index.json" />
62
+ </packageSources>
63
+ <activePackageSource>
64
+ <add key="CodeArtifact" value="${login.nugetEndpoint}v3/index.json" />
65
+ </activePackageSource>
66
+ <packageSourceCredentials>
67
+ <CodeArtifact>
68
+ <add key="Username" value="aws" />
69
+ <add key="ClearTextPassword" value="${login.authToken}" />
70
+ </CodeArtifact>
71
+ </packageSourceCredentials>
72
+ </configuration>`);
73
+ }
74
+
75
+ // NuGet.Config in current directory
@@ -0,0 +1,5 @@
1
+ export type ErrorResponse = 'fail' | 'skip' | 'retry';
2
+ /**
3
+ * Run a function in parallel with cached output
4
+ */
5
+ export declare function parallelShell<A>(inputs: A[], block: (x: A, output: NodeJS.WritableStream) => Promise<void>, swallowError?: (x: A, output: string) => ErrorResponse): Promise<void>;
@@ -0,0 +1,45 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.parallelShell = parallelShell;
4
+ const p_queue_1 = require("p-queue");
5
+ const aws_1 = require("../aws");
6
+ const corking_1 = require("../corking");
7
+ /**
8
+ * Run a function in parallel with cached output
9
+ */
10
+ async function parallelShell(inputs, block, swallowError) {
11
+ // Limit to 10 for now, too many instances of Maven exhaust the CodeBuild instance memory
12
+ const q = new p_queue_1.default({ concurrency: Number(process.env.CONCURRENCY) || 10 });
13
+ await q.addAll(inputs.map(input => async () => {
14
+ let attempts = 10;
15
+ let sleepMs = 500;
16
+ while (true) {
17
+ const output = new corking_1.MemoryStream();
18
+ try {
19
+ await block(input, output);
20
+ return;
21
+ }
22
+ catch (e) {
23
+ switch (swallowError === null || swallowError === void 0 ? void 0 : swallowError(input, output.toString())) {
24
+ case 'skip':
25
+ return;
26
+ case 'retry':
27
+ if (--attempts > 0) {
28
+ await (0, aws_1.sleep)(Math.floor(Math.random() * sleepMs));
29
+ sleepMs *= 2;
30
+ continue;
31
+ }
32
+ break;
33
+ case 'fail':
34
+ case undefined:
35
+ break;
36
+ }
37
+ // eslint-disable-next-line no-console
38
+ console.error(output.toString());
39
+ throw e;
40
+ }
41
+ }
42
+ }));
43
+ await q.onEmpty();
44
+ }
45
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGFyYWxsZWwtc2hlbGwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJwYXJhbGxlbC1zaGVsbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQVNBLHNDQXlDQztBQWxERCxxQ0FBNkI7QUFDN0IsZ0NBQStCO0FBQy9CLHdDQUEwQztBQUkxQzs7R0FFRztBQUNJLEtBQUssVUFBVSxhQUFhLENBQ2pDLE1BQVcsRUFDWCxLQUE2RCxFQUM3RCxZQUFzRDtJQUV0RCx5RkFBeUY7SUFDekYsTUFBTSxDQUFDLEdBQUcsSUFBSSxpQkFBTSxDQUFDLEVBQUUsV0FBVyxFQUFFLE1BQU0sQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLFdBQVcsQ0FBQyxJQUFJLEVBQUUsRUFBRSxDQUFDLENBQUM7SUFDN0UsTUFBTSxDQUFDLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQyxLQUFLLElBQUksRUFBRTtRQUM1QyxJQUFJLFFBQVEsR0FBRyxFQUFFLENBQUM7UUFDbEIsSUFBSSxPQUFPLEdBQUcsR0FBRyxDQUFDO1FBQ2xCLE9BQU8sSUFBSSxFQUFFLENBQUM7WUFDWixNQUFNLE1BQU0sR0FBRyxJQUFJLHNCQUFZLEVBQUUsQ0FBQztZQUNsQyxJQUFJLENBQUM7Z0JBQ0gsTUFBTSxLQUFLLENBQUMsS0FBSyxFQUFFLE1BQU0sQ0FBQyxDQUFDO2dCQUMzQixPQUFPO1lBQ1QsQ0FBQztZQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUM7Z0JBQ1gsUUFBUSxZQUFZLGFBQVosWUFBWSx1QkFBWixZQUFZLENBQUcsS0FBSyxFQUFFLE1BQU0sQ0FBQyxRQUFRLEVBQUUsQ0FBQyxFQUFFLENBQUM7b0JBQ2pELEtBQUssTUFBTTt3QkFDVCxPQUFPO29CQUVULEtBQUssT0FBTzt3QkFDVixJQUFJLEVBQUUsUUFBUSxHQUFHLENBQUMsRUFBRSxDQUFDOzRCQUNuQixNQUFNLElBQUEsV0FBSyxFQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLE1BQU0sRUFBRSxHQUFHLE9BQU8sQ0FBQyxDQUFDLENBQUM7NEJBQ2pELE9BQU8sSUFBSSxDQUFDLENBQUM7NEJBQ2IsU0FBUzt3QkFDWCxDQUFDO3dCQUNELE1BQU07b0JBRVIsS0FBSyxNQUFNLENBQUM7b0JBQ1osS0FBSyxTQUFTO3dCQUNaLE1BQU07Z0JBQ1YsQ0FBQztnQkFFRCxzQ0FBc0M7Z0JBQ3RDLE9BQU8sQ0FBQyxLQUFLLENBQUMsTUFBTSxDQUFDLFFBQVEsRUFBRSxDQUFDLENBQUM7Z0JBQ2pDLE1BQU0sQ0FBQyxDQUFDO1lBQ1YsQ0FBQztRQUNILENBQUM7SUFDSCxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBRUosTUFBTSxDQUFDLENBQUMsT0FBTyxFQUFFLENBQUM7QUFDcEIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBQUXVldWUgZnJvbSAncC1xdWV1ZSc7XG5pbXBvcnQgeyBzbGVlcCB9IGZyb20gJy4uL2F3cyc7XG5pbXBvcnQgeyBNZW1vcnlTdHJlYW0gfSBmcm9tICcuLi9jb3JraW5nJztcblxuZXhwb3J0IHR5cGUgRXJyb3JSZXNwb25zZSA9ICdmYWlsJyB8ICdza2lwJyB8ICdyZXRyeSc7XG5cbi8qKlxuICogUnVuIGEgZnVuY3Rpb24gaW4gcGFyYWxsZWwgd2l0aCBjYWNoZWQgb3V0cHV0XG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBwYXJhbGxlbFNoZWxsPEE+KFxuICBpbnB1dHM6IEFbXSxcbiAgYmxvY2s6ICh4OiBBLCBvdXRwdXQ6IE5vZGVKUy5Xcml0YWJsZVN0cmVhbSkgPT4gUHJvbWlzZTx2b2lkPixcbiAgc3dhbGxvd0Vycm9yPzogKHg6IEEsIG91dHB1dDogc3RyaW5nKSA9PiBFcnJvclJlc3BvbnNlLFxuKSB7XG4gIC8vIExpbWl0IHRvIDEwIGZvciBub3csIHRvbyBtYW55IGluc3RhbmNlcyBvZiBNYXZlbiBleGhhdXN0IHRoZSBDb2RlQnVpbGQgaW5zdGFuY2UgbWVtb3J5XG4gIGNvbnN0IHEgPSBuZXcgUFF1ZXVlKHsgY29uY3VycmVuY3k6IE51bWJlcihwcm9jZXNzLmVudi5DT05DVVJSRU5DWSkgfHwgMTAgfSk7XG4gIGF3YWl0IHEuYWRkQWxsKGlucHV0cy5tYXAoaW5wdXQgPT4gYXN5bmMgKCkgPT4ge1xuICAgIGxldCBhdHRlbXB0cyA9IDEwO1xuICAgIGxldCBzbGVlcE1zID0gNTAwO1xuICAgIHdoaWxlICh0cnVlKSB7XG4gICAgICBjb25zdCBvdXRwdXQgPSBuZXcgTWVtb3J5U3RyZWFtKCk7XG4gICAgICB0cnkge1xuICAgICAgICBhd2FpdCBibG9jayhpbnB1dCwgb3V0cHV0KTtcbiAgICAgICAgcmV0dXJuO1xuICAgICAgfSBjYXRjaCAoZSkge1xuICAgICAgICBzd2l0Y2ggKHN3YWxsb3dFcnJvcj8uKGlucHV0LCBvdXRwdXQudG9TdHJpbmcoKSkpIHtcbiAgICAgICAgICBjYXNlICdza2lwJzpcbiAgICAgICAgICAgIHJldHVybjtcblxuICAgICAgICAgIGNhc2UgJ3JldHJ5JzpcbiAgICAgICAgICAgIGlmICgtLWF0dGVtcHRzID4gMCkge1xuICAgICAgICAgICAgICBhd2FpdCBzbGVlcChNYXRoLmZsb29yKE1hdGgucmFuZG9tKCkgKiBzbGVlcE1zKSk7XG4gICAgICAgICAgICAgIHNsZWVwTXMgKj0gMjtcbiAgICAgICAgICAgICAgY29udGludWU7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBicmVhaztcblxuICAgICAgICAgIGNhc2UgJ2ZhaWwnOlxuICAgICAgICAgIGNhc2UgdW5kZWZpbmVkOlxuICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgIH1cblxuICAgICAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgbm8tY29uc29sZVxuICAgICAgICBjb25zb2xlLmVycm9yKG91dHB1dC50b1N0cmluZygpKTtcbiAgICAgICAgdGhyb3cgZTtcbiAgICAgIH1cbiAgICB9XG4gIH0pKTtcblxuICBhd2FpdCBxLm9uRW1wdHkoKTtcbn1cbiJdfQ==
@@ -0,0 +1,51 @@
1
+ import PQueue from 'p-queue';
2
+ import { sleep } from '../aws';
3
+ import { MemoryStream } from '../corking';
4
+
5
+ export type ErrorResponse = 'fail' | 'skip' | 'retry';
6
+
7
+ /**
8
+ * Run a function in parallel with cached output
9
+ */
10
+ export async function parallelShell<A>(
11
+ inputs: A[],
12
+ block: (x: A, output: NodeJS.WritableStream) => Promise<void>,
13
+ swallowError?: (x: A, output: string) => ErrorResponse,
14
+ ) {
15
+ // Limit to 10 for now, too many instances of Maven exhaust the CodeBuild instance memory
16
+ const q = new PQueue({ concurrency: Number(process.env.CONCURRENCY) || 10 });
17
+ await q.addAll(inputs.map(input => async () => {
18
+ let attempts = 10;
19
+ let sleepMs = 500;
20
+ while (true) {
21
+ const output = new MemoryStream();
22
+ try {
23
+ await block(input, output);
24
+ return;
25
+ } catch (e) {
26
+ switch (swallowError?.(input, output.toString())) {
27
+ case 'skip':
28
+ return;
29
+
30
+ case 'retry':
31
+ if (--attempts > 0) {
32
+ await sleep(Math.floor(Math.random() * sleepMs));
33
+ sleepMs *= 2;
34
+ continue;
35
+ }
36
+ break;
37
+
38
+ case 'fail':
39
+ case undefined:
40
+ break;
41
+ }
42
+
43
+ // eslint-disable-next-line no-console
44
+ console.error(output.toString());
45
+ throw e;
46
+ }
47
+ }
48
+ }));
49
+
50
+ await q.onEmpty();
51
+ }
@@ -0,0 +1,4 @@
1
+ import { LoginInformation } from './codeartifact';
2
+ import { UsageDir } from './usage-dir';
3
+ export declare function pypiLogin(login: LoginInformation, usageDir: UsageDir): Promise<void>;
4
+ export declare function uploadPythonPackages(packages: string[], login: LoginInformation): Promise<void>;
@@ -0,0 +1,48 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.pypiLogin = pypiLogin;
4
+ exports.uploadPythonPackages = uploadPythonPackages;
5
+ /* eslint-disable no-console */
6
+ const path = require("path");
7
+ const parallel_shell_1 = require("./parallel-shell");
8
+ const files_1 = require("../files");
9
+ const shell_1 = require("../shell");
10
+ async function pypiLogin(login, usageDir) {
11
+ // Write pip config file and set environment var
12
+ await (0, files_1.writeFile)(path.join(usageDir.directory, 'pip.conf'), [
13
+ '[global]',
14
+ `index-url = https://aws:${login.authToken}@${login.pypiEndpoint.replace(/^https:\/\//, '')}simple/`,
15
+ ].join('\n'));
16
+ await usageDir.addToEnv({
17
+ PIP_CONFIG_FILE: `${usageDir.directory}/pip.conf`,
18
+ });
19
+ }
20
+ async function uploadPythonPackages(packages, login) {
21
+ await (0, shell_1.shell)(['pip', 'install', 'twine'], { show: 'error' });
22
+ // Even though twine supports uploading all packages in one go, we have to upload them
23
+ // individually since CodeArtifact does not support Twine's `--skip-existing`. Fun beans.
24
+ await (0, parallel_shell_1.parallelShell)(packages, async (pkg, output) => {
25
+ console.log(`⏳ ${pkg}`);
26
+ await (0, shell_1.shell)(['twine', 'upload', '--verbose', pkg], {
27
+ modEnv: {
28
+ TWINE_USERNAME: 'aws',
29
+ TWINE_PASSWORD: login.authToken,
30
+ TWINE_REPOSITORY_URL: login.pypiEndpoint,
31
+ },
32
+ show: 'error',
33
+ outputs: [output],
34
+ });
35
+ console.log(`✅ ${pkg}`);
36
+ }, (pkg, output) => {
37
+ if (output.toString().includes('This package is configured to block new versions') || output.toString().includes('409 Conflict')) {
38
+ console.log(`❌ ${pkg}: already exists. Skipped.`);
39
+ return 'skip';
40
+ }
41
+ if (output.includes('429 Too Many Requests ')) {
42
+ console.log(`♻️ ${pkg}: 429 Too Many Requests. Retrying.`);
43
+ return 'retry';
44
+ }
45
+ return 'fail';
46
+ });
47
+ }
48
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHlwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInB5cGkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFRQSw4QkFTQztBQUVELG9EQThCQztBQWpERCwrQkFBK0I7QUFDL0IsNkJBQTZCO0FBRTdCLHFEQUFpRDtBQUVqRCxvQ0FBcUM7QUFDckMsb0NBQWlDO0FBRTFCLEtBQUssVUFBVSxTQUFTLENBQUMsS0FBdUIsRUFBRSxRQUFrQjtJQUN6RSxnREFBZ0Q7SUFDaEQsTUFBTSxJQUFBLGlCQUFTLEVBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsU0FBUyxFQUFFLFVBQVUsQ0FBQyxFQUFFO1FBQ3pELFVBQVU7UUFDViwyQkFBMkIsS0FBSyxDQUFDLFNBQVMsSUFBSSxLQUFLLENBQUMsWUFBWSxDQUFDLE9BQU8sQ0FBQyxhQUFhLEVBQUUsRUFBRSxDQUFDLFNBQVM7S0FDckcsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQztJQUNkLE1BQU0sUUFBUSxDQUFDLFFBQVEsQ0FBQztRQUN0QixlQUFlLEVBQUUsR0FBRyxRQUFRLENBQUMsU0FBUyxXQUFXO0tBQ2xELENBQUMsQ0FBQztBQUNMLENBQUM7QUFFTSxLQUFLLFVBQVUsb0JBQW9CLENBQUMsUUFBa0IsRUFBRSxLQUF1QjtJQUNwRixNQUFNLElBQUEsYUFBSyxFQUFDLENBQUMsS0FBSyxFQUFFLFNBQVMsRUFBRSxPQUFPLENBQUMsRUFBRSxFQUFFLElBQUksRUFBRSxPQUFPLEVBQUUsQ0FBQyxDQUFDO0lBRTVELHNGQUFzRjtJQUN0Rix5RkFBeUY7SUFDekYsTUFBTSxJQUFBLDhCQUFhLEVBQUMsUUFBUSxFQUFFLEtBQUssRUFBRSxHQUFHLEVBQUUsTUFBTSxFQUFFLEVBQUU7UUFDbEQsT0FBTyxDQUFDLEdBQUcsQ0FBQyxLQUFLLEdBQUcsRUFBRSxDQUFDLENBQUM7UUFFeEIsTUFBTSxJQUFBLGFBQUssRUFBQyxDQUFDLE9BQU8sRUFBRSxRQUFRLEVBQUUsV0FBVyxFQUFFLEdBQUcsQ0FBQyxFQUFFO1lBQ2pELE1BQU0sRUFBRTtnQkFDTixjQUFjLEVBQUUsS0FBSztnQkFDckIsY0FBYyxFQUFFLEtBQUssQ0FBQyxTQUFTO2dCQUMvQixvQkFBb0IsRUFBRSxLQUFLLENBQUMsWUFBWTthQUN6QztZQUNELElBQUksRUFBRSxPQUFPO1lBQ2IsT0FBTyxFQUFFLENBQUMsTUFBTSxDQUFDO1NBQ2xCLENBQUMsQ0FBQztRQUVILE9BQU8sQ0FBQyxHQUFHLENBQUMsS0FBSyxHQUFHLEVBQUUsQ0FBQyxDQUFDO0lBQzFCLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxNQUFNLEVBQUUsRUFBRTtRQUNqQixJQUFJLE1BQU0sQ0FBQyxRQUFRLEVBQUUsQ0FBQyxRQUFRLENBQUMsa0RBQWtELENBQUMsSUFBSSxNQUFNLENBQUMsUUFBUSxFQUFFLENBQUMsUUFBUSxDQUFDLGNBQWMsQ0FBQyxFQUFFLENBQUM7WUFDakksT0FBTyxDQUFDLEdBQUcsQ0FBQyxLQUFLLEdBQUcsNEJBQTRCLENBQUMsQ0FBQztZQUNsRCxPQUFPLE1BQU0sQ0FBQztRQUNoQixDQUFDO1FBQ0QsSUFBSSxNQUFNLENBQUMsUUFBUSxDQUFDLHdCQUF3QixDQUFDLEVBQUUsQ0FBQztZQUM5QyxPQUFPLENBQUMsR0FBRyxDQUFDLE1BQU0sR0FBRyxvQ0FBb0MsQ0FBQyxDQUFDO1lBQzNELE9BQU8sT0FBTyxDQUFDO1FBQ2pCLENBQUM7UUFDRCxPQUFPLE1BQU0sQ0FBQztJQUNoQixDQUFDLENBQUMsQ0FBQztBQUNMLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBlc2xpbnQtZGlzYWJsZSBuby1jb25zb2xlICovXG5pbXBvcnQgKiBhcyBwYXRoIGZyb20gJ3BhdGgnO1xuaW1wb3J0IHsgTG9naW5JbmZvcm1hdGlvbiB9IGZyb20gJy4vY29kZWFydGlmYWN0JztcbmltcG9ydCB7IHBhcmFsbGVsU2hlbGwgfSBmcm9tICcuL3BhcmFsbGVsLXNoZWxsJztcbmltcG9ydCB7IFVzYWdlRGlyIH0gZnJvbSAnLi91c2FnZS1kaXInO1xuaW1wb3J0IHsgd3JpdGVGaWxlIH0gZnJvbSAnLi4vZmlsZXMnO1xuaW1wb3J0IHsgc2hlbGwgfSBmcm9tICcuLi9zaGVsbCc7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBweXBpTG9naW4obG9naW46IExvZ2luSW5mb3JtYXRpb24sIHVzYWdlRGlyOiBVc2FnZURpcikge1xuICAvLyBXcml0ZSBwaXAgY29uZmlnIGZpbGUgYW5kIHNldCBlbnZpcm9ubWVudCB2YXJcbiAgYXdhaXQgd3JpdGVGaWxlKHBhdGguam9pbih1c2FnZURpci5kaXJlY3RvcnksICdwaXAuY29uZicpLCBbXG4gICAgJ1tnbG9iYWxdJyxcbiAgICBgaW5kZXgtdXJsID0gaHR0cHM6Ly9hd3M6JHtsb2dpbi5hdXRoVG9rZW59QCR7bG9naW4ucHlwaUVuZHBvaW50LnJlcGxhY2UoL15odHRwczpcXC9cXC8vLCAnJyl9c2ltcGxlL2AsXG4gIF0uam9pbignXFxuJykpO1xuICBhd2FpdCB1c2FnZURpci5hZGRUb0Vudih7XG4gICAgUElQX0NPTkZJR19GSUxFOiBgJHt1c2FnZURpci5kaXJlY3Rvcnl9L3BpcC5jb25mYCxcbiAgfSk7XG59XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiB1cGxvYWRQeXRob25QYWNrYWdlcyhwYWNrYWdlczogc3RyaW5nW10sIGxvZ2luOiBMb2dpbkluZm9ybWF0aW9uKSB7XG4gIGF3YWl0IHNoZWxsKFsncGlwJywgJ2luc3RhbGwnLCAndHdpbmUnXSwgeyBzaG93OiAnZXJyb3InIH0pO1xuXG4gIC8vIEV2ZW4gdGhvdWdoIHR3aW5lIHN1cHBvcnRzIHVwbG9hZGluZyBhbGwgcGFja2FnZXMgaW4gb25lIGdvLCB3ZSBoYXZlIHRvIHVwbG9hZCB0aGVtXG4gIC8vIGluZGl2aWR1YWxseSBzaW5jZSBDb2RlQXJ0aWZhY3QgZG9lcyBub3Qgc3VwcG9ydCBUd2luZSdzIGAtLXNraXAtZXhpc3RpbmdgLiBGdW4gYmVhbnMuXG4gIGF3YWl0IHBhcmFsbGVsU2hlbGwocGFja2FnZXMsIGFzeW5jIChwa2csIG91dHB1dCkgPT4ge1xuICAgIGNvbnNvbGUubG9nKGDij7MgJHtwa2d9YCk7XG5cbiAgICBhd2FpdCBzaGVsbChbJ3R3aW5lJywgJ3VwbG9hZCcsICctLXZlcmJvc2UnLCBwa2ddLCB7XG4gICAgICBtb2RFbnY6IHtcbiAgICAgICAgVFdJTkVfVVNFUk5BTUU6ICdhd3MnLFxuICAgICAgICBUV0lORV9QQVNTV09SRDogbG9naW4uYXV0aFRva2VuLFxuICAgICAgICBUV0lORV9SRVBPU0lUT1JZX1VSTDogbG9naW4ucHlwaUVuZHBvaW50LFxuICAgICAgfSxcbiAgICAgIHNob3c6ICdlcnJvcicsXG4gICAgICBvdXRwdXRzOiBbb3V0cHV0XSxcbiAgICB9KTtcblxuICAgIGNvbnNvbGUubG9nKGDinIUgJHtwa2d9YCk7XG4gIH0sIChwa2csIG91dHB1dCkgPT4ge1xuICAgIGlmIChvdXRwdXQudG9TdHJpbmcoKS5pbmNsdWRlcygnVGhpcyBwYWNrYWdlIGlzIGNvbmZpZ3VyZWQgdG8gYmxvY2sgbmV3IHZlcnNpb25zJykgfHwgb3V0cHV0LnRvU3RyaW5nKCkuaW5jbHVkZXMoJzQwOSBDb25mbGljdCcpKSB7XG4gICAgICBjb25zb2xlLmxvZyhg4p2MICR7cGtnfTogYWxyZWFkeSBleGlzdHMuIFNraXBwZWQuYCk7XG4gICAgICByZXR1cm4gJ3NraXAnO1xuICAgIH1cbiAgICBpZiAob3V0cHV0LmluY2x1ZGVzKCc0MjkgVG9vIE1hbnkgUmVxdWVzdHMgJykpIHtcbiAgICAgIGNvbnNvbGUubG9nKGDimbvvuI8gJHtwa2d9OiA0MjkgVG9vIE1hbnkgUmVxdWVzdHMuIFJldHJ5aW5nLmApO1xuICAgICAgcmV0dXJuICdyZXRyeSc7XG4gICAgfVxuICAgIHJldHVybiAnZmFpbCc7XG4gIH0pO1xufVxuIl19
@@ -0,0 +1,50 @@
1
+ /* eslint-disable no-console */
2
+ import * as path from 'path';
3
+ import { LoginInformation } from './codeartifact';
4
+ import { parallelShell } from './parallel-shell';
5
+ import { UsageDir } from './usage-dir';
6
+ import { writeFile } from '../files';
7
+ import { shell } from '../shell';
8
+
9
+ export async function pypiLogin(login: LoginInformation, usageDir: UsageDir) {
10
+ // Write pip config file and set environment var
11
+ await writeFile(path.join(usageDir.directory, 'pip.conf'), [
12
+ '[global]',
13
+ `index-url = https://aws:${login.authToken}@${login.pypiEndpoint.replace(/^https:\/\//, '')}simple/`,
14
+ ].join('\n'));
15
+ await usageDir.addToEnv({
16
+ PIP_CONFIG_FILE: `${usageDir.directory}/pip.conf`,
17
+ });
18
+ }
19
+
20
+ export async function uploadPythonPackages(packages: string[], login: LoginInformation) {
21
+ await shell(['pip', 'install', 'twine'], { show: 'error' });
22
+
23
+ // Even though twine supports uploading all packages in one go, we have to upload them
24
+ // individually since CodeArtifact does not support Twine's `--skip-existing`. Fun beans.
25
+ await parallelShell(packages, async (pkg, output) => {
26
+ console.log(`⏳ ${pkg}`);
27
+
28
+ await shell(['twine', 'upload', '--verbose', pkg], {
29
+ modEnv: {
30
+ TWINE_USERNAME: 'aws',
31
+ TWINE_PASSWORD: login.authToken,
32
+ TWINE_REPOSITORY_URL: login.pypiEndpoint,
33
+ },
34
+ show: 'error',
35
+ outputs: [output],
36
+ });
37
+
38
+ console.log(`✅ ${pkg}`);
39
+ }, (pkg, output) => {
40
+ if (output.toString().includes('This package is configured to block new versions') || output.toString().includes('409 Conflict')) {
41
+ console.log(`❌ ${pkg}: already exists. Skipped.`);
42
+ return 'skip';
43
+ }
44
+ if (output.includes('429 Too Many Requests ')) {
45
+ console.log(`♻️ ${pkg}: 429 Too Many Requests. Retrying.`);
46
+ return 'retry';
47
+ }
48
+ return 'fail';
49
+ });
50
+ }
@@ -0,0 +1,31 @@
1
+ export declare const DEFAULT_USAGE_DIR: string;
2
+ /**
3
+ * The usage directory is where we write per-session config files to access the CodeArtifact repository.
4
+ *
5
+ * Some config files may be written in a system-global location, but they will not be active unless the
6
+ * contents of this directory have been sourced/copied into the current terminal.
7
+ *
8
+ * CONTRACT
9
+ *
10
+ * There are two special entries:
11
+ *
12
+ * - `env`, a file with `key=value` entries for environment variables to set.
13
+ * - `cwd/`, a directory with files that need to be copied into the current directory before each command.
14
+ *
15
+ * Other than these, code may write tempfiles to this directory if it wants, but there is no meaning
16
+ * implied for other files.
17
+ */
18
+ export declare class UsageDir {
19
+ readonly directory: string;
20
+ static default(): UsageDir;
21
+ readonly envFile: string;
22
+ readonly cwdDir: string;
23
+ private constructor();
24
+ clean(): Promise<void>;
25
+ addToEnv(settings: Record<string, string>): Promise<void>;
26
+ currentEnv(): Promise<Record<string, string>>;
27
+ cwdFile(filename: string): string;
28
+ activateInCurrentProcess(): Promise<void>;
29
+ copyCwdFileHere(...filenames: string[]): Promise<void>;
30
+ advertise(): void;
31
+ }
@@ -0,0 +1,87 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.UsageDir = exports.DEFAULT_USAGE_DIR = void 0;
4
+ const path = require("path");
5
+ const fs = require("fs-extra");
6
+ const files_1 = require("../files");
7
+ exports.DEFAULT_USAGE_DIR = path.join((0, files_1.homeDir)(), '.codeartifact/usage');
8
+ /**
9
+ * The usage directory is where we write per-session config files to access the CodeArtifact repository.
10
+ *
11
+ * Some config files may be written in a system-global location, but they will not be active unless the
12
+ * contents of this directory have been sourced/copied into the current terminal.
13
+ *
14
+ * CONTRACT
15
+ *
16
+ * There are two special entries:
17
+ *
18
+ * - `env`, a file with `key=value` entries for environment variables to set.
19
+ * - `cwd/`, a directory with files that need to be copied into the current directory before each command.
20
+ *
21
+ * Other than these, code may write tempfiles to this directory if it wants, but there is no meaning
22
+ * implied for other files.
23
+ */
24
+ class UsageDir {
25
+ static default() {
26
+ return new UsageDir(exports.DEFAULT_USAGE_DIR);
27
+ }
28
+ constructor(directory) {
29
+ this.directory = directory;
30
+ this.envFile = path.join(this.directory, 'env');
31
+ this.cwdDir = path.join(this.directory, 'cwd');
32
+ }
33
+ async clean() {
34
+ await fs.rm(this.directory, { recursive: true, force: true });
35
+ await fs.mkdirp(path.join(this.directory, 'cwd'));
36
+ await fs.writeFile(path.join(this.directory, 'env'), '', { encoding: 'utf-8' });
37
+ await this.addToEnv({
38
+ CWD_FILES_DIR: path.join(this.directory, 'cwd'),
39
+ });
40
+ // Write a bash helper to load these settings
41
+ await fs.writeFile(path.join(this.directory, 'activate.bash'), [
42
+ `while read -u10 line; do [[ -z $line ]] || export "$line"; done 10<${this.directory}/env`,
43
+ 'cp -R $CWD_FILES_DIR/ .', // Copy files from directory even if it is empty
44
+ ].join('\n'), { encoding: 'utf-8' });
45
+ }
46
+ async addToEnv(settings) {
47
+ const lines = await (0, files_1.loadLines)(this.envFile);
48
+ for (const [k, v] of Object.entries(settings)) {
49
+ (0, files_1.updateIniKey)(lines, k, v);
50
+ }
51
+ await (0, files_1.writeLines)(this.envFile, lines);
52
+ }
53
+ async currentEnv() {
54
+ const lines = await (0, files_1.loadLines)(this.envFile);
55
+ const splitter = /^([a-zA-Z0-9_-]+)\s*=\s*(.*)$/;
56
+ const ret = {};
57
+ for (const line of lines) {
58
+ const m = line.match(splitter);
59
+ if (m) {
60
+ ret[m[1]] = m[2];
61
+ }
62
+ }
63
+ return ret;
64
+ }
65
+ cwdFile(filename) {
66
+ return path.join(this.cwdDir, filename);
67
+ }
68
+ async activateInCurrentProcess() {
69
+ for (const [k, v] of Object.entries(await this.currentEnv())) {
70
+ process.env[k] = v;
71
+ }
72
+ await (0, files_1.copyDirectoryContents)(this.cwdDir, '.');
73
+ }
74
+ async copyCwdFileHere(...filenames) {
75
+ for (const file of filenames) {
76
+ await fs.copyFile(path.join(this.cwdDir, file), file);
77
+ }
78
+ }
79
+ advertise() {
80
+ // eslint-disable-next-line no-console
81
+ console.log('To activate these settings in the current terminal:');
82
+ // eslint-disable-next-line no-console
83
+ console.log(` source ${this.directory}/activate.bash`);
84
+ }
85
+ }
86
+ exports.UsageDir = UsageDir;
87
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"usage-dir.js","sourceRoot":"","sources":["usage-dir.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport * as fs from 'fs-extra';\nimport { copyDirectoryContents, homeDir, loadLines, updateIniKey, writeLines } from '../files';\n\nexport const DEFAULT_USAGE_DIR = path.join(homeDir(), '.codeartifact/usage');\n\n/**\n * The usage directory is where we write per-session config files to access the CodeArtifact repository.\n *\n * Some config files may be written in a system-global location, but they will not be active unless the\n * contents of this directory have been sourced/copied into the current terminal.\n *\n * CONTRACT\n *\n * There are two special entries:\n *\n * - `env`, a file with `key=value` entries for environment variables to  set.\n * - `cwd/`, a directory with files that need to be copied into the current directory before each command.\n *\n * Other than these, code may write tempfiles to this directory if it wants, but there is no meaning\n * implied for other files.\n */\nexport class UsageDir {\n  public static default() {\n    return new UsageDir(DEFAULT_USAGE_DIR);\n  }\n\n  public readonly envFile: string;\n  public readonly cwdDir: string;\n\n  private constructor(public readonly directory: string) {\n    this.envFile = path.join(this.directory, 'env');\n    this.cwdDir = path.join(this.directory, 'cwd');\n  }\n\n  public async clean() {\n    await fs.rm(this.directory, { recursive: true, force: true });\n    await fs.mkdirp(path.join(this.directory, 'cwd'));\n    await fs.writeFile(path.join(this.directory, 'env'), '', { encoding: 'utf-8' });\n\n    await this.addToEnv({\n      CWD_FILES_DIR: path.join(this.directory, 'cwd'),\n    });\n\n    // Write a bash helper to load these settings\n    await fs.writeFile(path.join(this.directory, 'activate.bash'), [\n      `while read -u10 line; do [[ -z $line ]] || export \"$line\"; done 10<${this.directory}/env`,\n      'cp -R $CWD_FILES_DIR/ .', // Copy files from directory even if it is empty\n    ].join('\\n'), { encoding: 'utf-8' });\n  }\n\n  public async addToEnv(settings: Record<string, string>) {\n    const lines = await loadLines(this.envFile);\n    for (const [k, v] of Object.entries(settings)) {\n      updateIniKey(lines, k, v);\n    }\n    await writeLines(this.envFile, lines);\n  }\n\n  public async currentEnv(): Promise<Record<string, string>> {\n    const lines = await loadLines(this.envFile);\n\n    const splitter = /^([a-zA-Z0-9_-]+)\\s*=\\s*(.*)$/;\n\n    const ret: Record<string, string> = {};\n    for (const line of lines) {\n      const m = line.match(splitter);\n      if (m) {\n        ret[m[1]] = m[2];\n      }\n    }\n    return ret;\n  }\n\n  public cwdFile(filename: string) {\n    return path.join(this.cwdDir, filename);\n  }\n\n  public async activateInCurrentProcess() {\n    for (const [k, v] of Object.entries(await this.currentEnv())) {\n      process.env[k] = v;\n    }\n\n    await copyDirectoryContents(this.cwdDir, '.');\n  }\n\n  public async copyCwdFileHere(...filenames: string[]) {\n    for (const file of filenames) {\n      await fs.copyFile(path.join(this.cwdDir, file), file);\n    }\n  }\n\n  public advertise() {\n    // eslint-disable-next-line no-console\n    console.log('To activate these settings in the current terminal:');\n    // eslint-disable-next-line no-console\n    console.log(`    source ${this.directory}/activate.bash`);\n  }\n}\n"]}