@aws-cdk-testing/cli-integ 2.61.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +201 -0
- package/NOTICE +16 -0
- package/README.md +151 -0
- package/bin/apply-patches +19 -0
- package/bin/download-and-run-old-tests +52 -0
- package/bin/query-github +2 -0
- package/bin/query-github.d.ts +1 -0
- package/bin/query-github.js +55 -0
- package/bin/run-suite +2 -0
- package/bin/run-suite.d.ts +1 -0
- package/bin/run-suite.js +126 -0
- package/bin/stage-distribution +2 -0
- package/bin/stage-distribution.d.ts +1 -0
- package/bin/stage-distribution.js +209 -0
- package/bin/test-root +2 -0
- package/bin/test-root.d.ts +1 -0
- package/bin/test-root.js +6 -0
- package/entrypoints/test-cli-regression-against-current-code.sh +11 -0
- package/entrypoints/test-cli-regression-against-latest-release.sh +11 -0
- package/entrypoints/test-cli-regression.bash +83 -0
- package/entrypoints/test.sh +12 -0
- package/lib/aws.d.ts +55 -0
- package/lib/aws.js +243 -0
- package/lib/corking.d.ts +13 -0
- package/lib/corking.js +34 -0
- package/lib/files.d.ts +15 -0
- package/lib/files.js +80 -0
- package/lib/github.d.ts +4 -0
- package/lib/github.js +43 -0
- package/lib/index.d.ts +12 -0
- package/lib/index.js +25 -0
- package/lib/integ-test.d.ts +11 -0
- package/lib/integ-test.js +55 -0
- package/lib/lists.d.ts +1 -0
- package/lib/lists.js +12 -0
- package/lib/memoize.d.ts +6 -0
- package/lib/memoize.js +19 -0
- package/lib/npm.d.ts +4 -0
- package/lib/npm.js +15 -0
- package/lib/package-sources/release-source.d.ts +22 -0
- package/lib/package-sources/release-source.js +67 -0
- package/lib/package-sources/repo-source.d.ts +23 -0
- package/lib/package-sources/repo-source.js +92 -0
- package/lib/package-sources/repo-tools/npm +2 -0
- package/lib/package-sources/repo-tools/npm.d.ts +1 -0
- package/lib/package-sources/repo-tools/npm.js +42 -0
- package/lib/package-sources/source.d.ts +24 -0
- package/lib/package-sources/source.js +3 -0
- package/lib/package-sources/subprocess.d.ts +3 -0
- package/lib/package-sources/subprocess.js +18 -0
- package/lib/resource-pool.d.ts +54 -0
- package/lib/resource-pool.js +120 -0
- package/lib/resources.d.ts +1 -0
- package/lib/resources.js +6 -0
- package/lib/shell.d.ts +59 -0
- package/lib/shell.js +118 -0
- package/lib/staging/codeartifact.d.ts +44 -0
- package/lib/staging/codeartifact.js +258 -0
- package/lib/staging/maven.d.ts +5 -0
- package/lib/staging/maven.js +83 -0
- package/lib/staging/npm.d.ts +4 -0
- package/lib/staging/npm.js +56 -0
- package/lib/staging/nuget.d.ts +4 -0
- package/lib/staging/nuget.js +71 -0
- package/lib/staging/parallel-shell.d.ts +6 -0
- package/lib/staging/parallel-shell.js +46 -0
- package/lib/staging/pypi.d.ts +4 -0
- package/lib/staging/pypi.js +49 -0
- package/lib/staging/usage-dir.d.ts +31 -0
- package/lib/staging/usage-dir.js +87 -0
- package/lib/with-aws.d.ts +13 -0
- package/lib/with-aws.js +60 -0
- package/lib/with-cdk-app.d.ts +146 -0
- package/lib/with-cdk-app.js +398 -0
- package/lib/with-packages.d.ts +5 -0
- package/lib/with-packages.js +14 -0
- package/lib/with-sam.d.ts +33 -0
- package/lib/with-sam.js +240 -0
- package/lib/with-temporary-directory.d.ts +5 -0
- package/lib/with-temporary-directory.js +32 -0
- package/lib/xpmutex.d.ts +43 -0
- package/lib/xpmutex.js +207 -0
- package/package.json +73 -0
- package/resources/cdk-apps/app/app.js +463 -0
- package/resources/cdk-apps/app/cdk.json +7 -0
- package/resources/cdk-apps/app/docker/Dockerfile +2 -0
- package/resources/cdk-apps/app/docker/Dockerfile.Custom +2 -0
- package/resources/cdk-apps/app/lambda/index.js +4 -0
- package/resources/cdk-apps/app/lambda/response.json +3 -0
- package/resources/cdk-apps/app/nested-stack.js +49 -0
- package/resources/cdk-apps/cfn-include-app/.gitignore +1 -0
- package/resources/cdk-apps/cfn-include-app/cdk.json +4 -0
- package/resources/cdk-apps/cfn-include-app/cfn-include-app.js +21 -0
- package/resources/cdk-apps/cfn-include-app/example-template.json +13 -0
- package/resources/cdk-apps/sam_cdk_integ_app/bin/test-app.js +11 -0
- package/resources/cdk-apps/sam_cdk_integ_app/cdk.json +6 -0
- package/resources/cdk-apps/sam_cdk_integ_app/lib/nested-stack.js +19 -0
- package/resources/cdk-apps/sam_cdk_integ_app/lib/test-stack.js +134 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/docker/DockerImageFunctionConstruct/.no-packagejson-validator +0 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/docker/DockerImageFunctionConstruct/Dockerfile +9 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/docker/DockerImageFunctionConstruct/app.js +22 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/docker/DockerImageFunctionConstruct/package.json +18 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/go/GoFunctionConstruct/go.mod +5 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/go/GoFunctionConstruct/go.sum +17 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/go/GoFunctionConstruct/main.go +17 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/nodejs/NodeJsFunctionConstruct/.no-packagejson-validator +0 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/nodejs/NodeJsFunctionConstruct/app.ts +16 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/nodejs/NodeJsFunctionConstruct/package-lock.json +12 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/nodejs/NodeJsFunctionConstruct/package.json +5 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/python/Function/app.py +15 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/python/Function/requirements.txt +1 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/python/Layer/layer_version_dependency.py +5 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/python/Layer/requirements.txt +1 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/rest-api-definition.yaml +12 -0
- package/resources/cli-regression-patches/v1.119.0/NOTES.md +5 -0
- package/resources/cli-regression-patches/v1.119.0/cli.integtest.js +659 -0
- package/resources/cli-regression-patches/v1.130.0/NOTES.md +12 -0
- package/resources/cli-regression-patches/v1.130.0/app/app.js +378 -0
- package/resources/cli-regression-patches/v1.130.0/bootstrapping.integtest.js +220 -0
- package/resources/cli-regression-patches/v1.44.0/NOTES.md +18 -0
- package/resources/cli-regression-patches/v1.44.0/bootstrapping.integtest.js +126 -0
- package/resources/cli-regression-patches/v1.44.0/test.sh +26 -0
- package/resources/cli-regression-patches/v1.61.1/NOTES.md +2 -0
- package/resources/cli-regression-patches/v1.61.1/skip-tests.txt +16 -0
- package/resources/cli-regression-patches/v1.62.0/NOTES.md +2 -0
- package/resources/cli-regression-patches/v1.62.0/aws-helpers.js +245 -0
- package/resources/cli-regression-patches/v1.63.0/NOTES.md +1 -0
- package/resources/cli-regression-patches/v1.63.0/skip-tests.txt +7 -0
- package/resources/cli-regression-patches/v1.64.0/NOTES.md +3 -0
- package/resources/cli-regression-patches/v1.64.0/cdk-helpers.js +325 -0
- package/resources/cli-regression-patches/v1.64.0/cli.integtest.js +599 -0
- package/resources/cli-regression-patches/v1.64.1/NOTES.md +3 -0
- package/resources/cli-regression-patches/v1.64.1/cdk-helpers.js +324 -0
- package/resources/cli-regression-patches/v1.64.1/cli.integtest.js +599 -0
- package/resources/cli-regression-patches/v1.67.0/NOTES.md +2 -0
- package/resources/cli-regression-patches/v1.67.0/cdk-helpers.js +331 -0
- package/resources/cloud-assemblies/0.36.0/InitStack.template.json +1 -0
- package/resources/cloud-assemblies/0.36.0/manifest.json +19 -0
- package/resources/cloud-assemblies/1.10.0-lookup-default-vpc/InitStack.template.json +2 -0
- package/resources/cloud-assemblies/1.10.0-lookup-default-vpc/manifest.json.js +37 -0
- package/resources/cloud-assemblies/1.10.0-request-azs/InitStack.template.json +2 -0
- package/resources/cloud-assemblies/1.10.0-request-azs/manifest.json.js +34 -0
- package/resources/integ.jest.config.js +25 -0
- package/skip-tests.txt +8 -0
- package/tests/cli-integ-tests/README.md +47 -0
- package/tests/cli-integ-tests/bootstrapping.integtest.d.ts +1 -0
- package/tests/cli-integ-tests/bootstrapping.integtest.js +271 -0
- package/tests/cli-integ-tests/cli.integtest.d.ts +1 -0
- package/tests/cli-integ-tests/cli.integtest.js +1029 -0
- package/tests/init-csharp/init-csharp.integtest.d.ts +1 -0
- package/tests/init-csharp/init-csharp.integtest.js +14 -0
- package/tests/init-fsharp/init-fsharp.integtest.d.ts +1 -0
- package/tests/init-fsharp/init-fsharp.integtest.js +14 -0
- package/tests/init-java/init-java.integtest.d.ts +1 -0
- package/tests/init-java/init-java.integtest.js +14 -0
- package/tests/init-javascript/init-javascript.integtest.d.ts +1 -0
- package/tests/init-javascript/init-javascript.integtest.js +15 -0
- package/tests/init-python/init-python.integtest.d.ts +1 -0
- package/tests/init-python/init-python.integtest.js +19 -0
- package/tests/init-typescript-app/init-typescript-app.integtest.d.ts +1 -0
- package/tests/init-typescript-app/init-typescript-app.integtest.js +49 -0
- package/tests/init-typescript-lib/init-typescript-lib.integtest.d.ts +1 -0
- package/tests/init-typescript-lib/init-typescript-lib.integtest.js +13 -0
- package/tests/uberpackage/uberpackage.integtest.d.ts +1 -0
- package/tests/uberpackage/uberpackage.integtest.js +11 -0
package/lib/shell.d.ts
ADDED
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
/// <reference types="node" />
|
|
2
|
+
import * as child_process from 'child_process';
|
|
3
|
+
import { TestContext } from './integ-test';
|
|
4
|
+
import { TemporaryDirectoryContext } from './with-temporary-directory';
|
|
5
|
+
/**
|
|
6
|
+
* A shell command that does what you want
|
|
7
|
+
*
|
|
8
|
+
* Is platform-aware, handles errors nicely.
|
|
9
|
+
*/
|
|
10
|
+
export declare function shell(command: string[], options?: ShellOptions): Promise<string>;
|
|
11
|
+
export interface ShellOptions extends child_process.SpawnOptions {
|
|
12
|
+
/**
|
|
13
|
+
* Properties to add to 'env'
|
|
14
|
+
*/
|
|
15
|
+
readonly modEnv?: Record<string, string>;
|
|
16
|
+
/**
|
|
17
|
+
* Don't fail when exiting with an error
|
|
18
|
+
*
|
|
19
|
+
* @default false
|
|
20
|
+
*/
|
|
21
|
+
readonly allowErrExit?: boolean;
|
|
22
|
+
/**
|
|
23
|
+
* Whether to capture stderr
|
|
24
|
+
*
|
|
25
|
+
* @default true
|
|
26
|
+
*/
|
|
27
|
+
readonly captureStderr?: boolean;
|
|
28
|
+
/**
|
|
29
|
+
* Pass output here
|
|
30
|
+
*
|
|
31
|
+
* @default stdout unless quiet=true
|
|
32
|
+
*/
|
|
33
|
+
readonly output?: NodeJS.WritableStream;
|
|
34
|
+
/**
|
|
35
|
+
* Only return stderr. For example, this is used to validate
|
|
36
|
+
* that when CI=true, all logs are sent to stdout.
|
|
37
|
+
*
|
|
38
|
+
* @default false
|
|
39
|
+
*/
|
|
40
|
+
readonly onlyStderr?: boolean;
|
|
41
|
+
/**
|
|
42
|
+
* Don't log to stdout
|
|
43
|
+
*
|
|
44
|
+
* @default always
|
|
45
|
+
*/
|
|
46
|
+
readonly show?: 'always' | 'never' | 'error';
|
|
47
|
+
}
|
|
48
|
+
export declare class ShellHelper {
|
|
49
|
+
private readonly _cwd;
|
|
50
|
+
private readonly _output;
|
|
51
|
+
static fromContext(context: TestContext & TemporaryDirectoryContext): ShellHelper;
|
|
52
|
+
constructor(_cwd: string, _output: NodeJS.WritableStream);
|
|
53
|
+
shell(command: string[], options?: Omit<ShellOptions, 'cwd' | 'output'>): Promise<string>;
|
|
54
|
+
}
|
|
55
|
+
/**
|
|
56
|
+
* rm -rf reimplementation, don't want to depend on an NPM package for this
|
|
57
|
+
*/
|
|
58
|
+
export declare function rimraf(fsPath: string): void;
|
|
59
|
+
export declare function addToShellPath(x: string): void;
|
package/lib/shell.js
ADDED
|
@@ -0,0 +1,118 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.addToShellPath = exports.rimraf = exports.ShellHelper = exports.shell = void 0;
|
|
4
|
+
const child_process = require("child_process");
|
|
5
|
+
const fs = require("fs");
|
|
6
|
+
const path = require("path");
|
|
7
|
+
/**
|
|
8
|
+
* A shell command that does what you want
|
|
9
|
+
*
|
|
10
|
+
* Is platform-aware, handles errors nicely.
|
|
11
|
+
*/
|
|
12
|
+
async function shell(command, options = {}) {
|
|
13
|
+
if (options.modEnv && options.env) {
|
|
14
|
+
throw new Error('Use either env or modEnv but not both');
|
|
15
|
+
}
|
|
16
|
+
// Always output the command
|
|
17
|
+
(options.output ?? process.stdout).write(`💻 ${command.join(' ')}\n`);
|
|
18
|
+
let output = options.output ?? process.stdout;
|
|
19
|
+
switch (options.show ?? 'always') {
|
|
20
|
+
case 'always':
|
|
21
|
+
break;
|
|
22
|
+
case 'never':
|
|
23
|
+
case 'error':
|
|
24
|
+
output = undefined;
|
|
25
|
+
break;
|
|
26
|
+
}
|
|
27
|
+
if (process.env.VERBOSE) {
|
|
28
|
+
output = process.stdout;
|
|
29
|
+
}
|
|
30
|
+
const env = options.env ?? (options.modEnv ? { ...process.env, ...options.modEnv } : process.env);
|
|
31
|
+
const child = child_process.spawn(command[0], command.slice(1), {
|
|
32
|
+
...options,
|
|
33
|
+
env,
|
|
34
|
+
// Need this for Windows where we want .cmd and .bat to be found as well.
|
|
35
|
+
shell: true,
|
|
36
|
+
stdio: ['ignore', 'pipe', 'pipe'],
|
|
37
|
+
});
|
|
38
|
+
return new Promise((resolve, reject) => {
|
|
39
|
+
const stdout = new Array();
|
|
40
|
+
const stderr = new Array();
|
|
41
|
+
child.stdout.on('data', chunk => {
|
|
42
|
+
output?.write(chunk);
|
|
43
|
+
stdout.push(chunk);
|
|
44
|
+
});
|
|
45
|
+
child.stderr.on('data', chunk => {
|
|
46
|
+
output?.write(chunk);
|
|
47
|
+
if (options.captureStderr ?? true) {
|
|
48
|
+
stderr.push(chunk);
|
|
49
|
+
}
|
|
50
|
+
});
|
|
51
|
+
child.once('error', reject);
|
|
52
|
+
child.once('close', code => {
|
|
53
|
+
const stderrOutput = Buffer.concat(stderr).toString('utf-8');
|
|
54
|
+
const stdoutOutput = Buffer.concat(stdout).toString('utf-8');
|
|
55
|
+
const out = (options.onlyStderr ? stderrOutput : stdoutOutput + stderrOutput).trim();
|
|
56
|
+
if (code === 0 || options.allowErrExit) {
|
|
57
|
+
resolve(out);
|
|
58
|
+
}
|
|
59
|
+
else {
|
|
60
|
+
if (options.show === 'error') {
|
|
61
|
+
(options.output ?? process.stdout).write(out + '\n');
|
|
62
|
+
}
|
|
63
|
+
reject(new Error(`'${command.join(' ')}' exited with error code ${code}.`));
|
|
64
|
+
}
|
|
65
|
+
});
|
|
66
|
+
});
|
|
67
|
+
}
|
|
68
|
+
exports.shell = shell;
|
|
69
|
+
class ShellHelper {
|
|
70
|
+
constructor(_cwd, _output) {
|
|
71
|
+
this._cwd = _cwd;
|
|
72
|
+
this._output = _output;
|
|
73
|
+
}
|
|
74
|
+
static fromContext(context) {
|
|
75
|
+
return new ShellHelper(context.integTestDir, context.output);
|
|
76
|
+
}
|
|
77
|
+
async shell(command, options = {}) {
|
|
78
|
+
return shell(command, {
|
|
79
|
+
output: this._output,
|
|
80
|
+
cwd: this._cwd,
|
|
81
|
+
...options,
|
|
82
|
+
});
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
exports.ShellHelper = ShellHelper;
|
|
86
|
+
/**
|
|
87
|
+
* rm -rf reimplementation, don't want to depend on an NPM package for this
|
|
88
|
+
*/
|
|
89
|
+
function rimraf(fsPath) {
|
|
90
|
+
try {
|
|
91
|
+
const isDir = fs.lstatSync(fsPath).isDirectory();
|
|
92
|
+
if (isDir) {
|
|
93
|
+
for (const file of fs.readdirSync(fsPath)) {
|
|
94
|
+
rimraf(path.join(fsPath, file));
|
|
95
|
+
}
|
|
96
|
+
fs.rmdirSync(fsPath);
|
|
97
|
+
}
|
|
98
|
+
else {
|
|
99
|
+
fs.unlinkSync(fsPath);
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
catch (e) {
|
|
103
|
+
// We will survive ENOENT
|
|
104
|
+
if (e.code !== 'ENOENT') {
|
|
105
|
+
throw e;
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
exports.rimraf = rimraf;
|
|
110
|
+
function addToShellPath(x) {
|
|
111
|
+
const parts = process.env.PATH?.split(':') ?? [];
|
|
112
|
+
if (!parts.includes(x)) {
|
|
113
|
+
parts.unshift(x);
|
|
114
|
+
}
|
|
115
|
+
process.env.PATH = parts.join(':');
|
|
116
|
+
}
|
|
117
|
+
exports.addToShellPath = addToShellPath;
|
|
118
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"shell.js","sourceRoot":"","sources":["shell.ts"],"names":[],"mappings":"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","sourcesContent":["import * as child_process from 'child_process';\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport { TestContext } from './integ-test';\nimport { TemporaryDirectoryContext } from './with-temporary-directory';\n\n/**\n * A shell command that does what you want\n *\n * Is platform-aware, handles errors nicely.\n */\nexport async function shell(command: string[], options: ShellOptions = {}): Promise<string> {\n  if (options.modEnv && options.env) {\n    throw new Error('Use either env or modEnv but not both');\n  }\n\n  // Always output the command\n  (options.output ?? process.stdout).write(`💻 ${command.join(' ')}\\n`);\n\n  let output: NodeJS.WritableStream | undefined = options.output ?? process.stdout;\n  switch (options.show ?? 'always') {\n    case 'always':\n      break;\n    case 'never':\n    case 'error':\n      output = undefined;\n      break;\n  }\n\n  if (process.env.VERBOSE) {\n    output = process.stdout;\n  }\n\n  const env = options.env ?? (options.modEnv ? { ...process.env, ...options.modEnv } : process.env);\n\n  const child = child_process.spawn(command[0], command.slice(1), {\n    ...options,\n    env,\n    // Need this for Windows where we want .cmd and .bat to be found as well.\n    shell: true,\n    stdio: ['ignore', 'pipe', 'pipe'],\n  });\n\n  return new Promise<string>((resolve, reject) => {\n    const stdout = new Array<Buffer>();\n    const stderr = new Array<Buffer>();\n\n    child.stdout!.on('data', chunk => {\n      output?.write(chunk);\n      stdout.push(chunk);\n    });\n\n    child.stderr!.on('data', chunk => {\n      output?.write(chunk);\n      if (options.captureStderr ?? true) {\n        stderr.push(chunk);\n      }\n    });\n\n    child.once('error', reject);\n\n    child.once('close', code => {\n      const stderrOutput = Buffer.concat(stderr).toString('utf-8');\n      const stdoutOutput = Buffer.concat(stdout).toString('utf-8');\n      const out = (options.onlyStderr ? stderrOutput : stdoutOutput + stderrOutput).trim();\n      if (code === 0 || options.allowErrExit) {\n        resolve(out);\n      } else {\n        if (options.show === 'error') {\n          (options.output ?? process.stdout).write(out + '\\n');\n        }\n        reject(new Error(`'${command.join(' ')}' exited with error code ${code}.`));\n      }\n    });\n  });\n}\n\nexport interface ShellOptions extends child_process.SpawnOptions {\n  /**\n   * Properties to add to 'env'\n   */\n  readonly modEnv?: Record<string, string>;\n\n  /**\n   * Don't fail when exiting with an error\n   *\n   * @default false\n   */\n  readonly allowErrExit?: boolean;\n\n  /**\n   * Whether to capture stderr\n   *\n   * @default true\n   */\n  readonly captureStderr?: boolean;\n\n  /**\n   * Pass output here\n   *\n   * @default stdout unless quiet=true\n   */\n  readonly output?: NodeJS.WritableStream;\n\n  /**\n   * Only return stderr. For example, this is used to validate\n   * that when CI=true, all logs are sent to stdout.\n   *\n   * @default false\n   */\n  readonly onlyStderr?: boolean;\n\n  /**\n   * Don't log to stdout\n   *\n   * @default always\n   */\n  readonly show?: 'always' | 'never' | 'error';\n}\n\nexport class ShellHelper {\n  public static fromContext(context: TestContext & TemporaryDirectoryContext) {\n    return new ShellHelper(context.integTestDir, context.output);\n  }\n\n  constructor(\n    private readonly _cwd: string,\n    private readonly _output: NodeJS.WritableStream) { }\n\n  public async shell(command: string[], options: Omit<ShellOptions, 'cwd' | 'output'> = {}): Promise<string> {\n    return shell(command, {\n      output: this._output,\n      cwd: this._cwd,\n      ...options,\n    });\n  }\n}\n\n/**\n * rm -rf reimplementation, don't want to depend on an NPM package for this\n */\nexport function rimraf(fsPath: string) {\n  try {\n    const isDir = fs.lstatSync(fsPath).isDirectory();\n\n    if (isDir) {\n      for (const file of fs.readdirSync(fsPath)) {\n        rimraf(path.join(fsPath, file));\n      }\n      fs.rmdirSync(fsPath);\n    } else {\n      fs.unlinkSync(fsPath);\n    }\n  } catch (e) {\n    // We will survive ENOENT\n    if (e.code !== 'ENOENT') { throw e; }\n  }\n}\n\nexport function addToShellPath(x: string) {\n  const parts = process.env.PATH?.split(':') ?? [];\n\n  if (!parts.includes(x)) {\n    parts.unshift(x);\n  }\n\n  process.env.PATH = parts.join(':');\n}"]}
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
export declare class TestRepository {
|
|
2
|
+
readonly repositoryName: string;
|
|
3
|
+
static readonly DEFAULT_DOMAIN = "test-cdk";
|
|
4
|
+
static newRandom(): Promise<TestRepository>;
|
|
5
|
+
static newWithName(name: string): Promise<TestRepository>;
|
|
6
|
+
static existing(repositoryName: string): TestRepository;
|
|
7
|
+
/**
|
|
8
|
+
* Garbage collect repositories
|
|
9
|
+
*/
|
|
10
|
+
static gc(): Promise<void>;
|
|
11
|
+
readonly npmUpstream = "npm-upstream";
|
|
12
|
+
readonly pypiUpstream = "pypi-upstream";
|
|
13
|
+
readonly nugetUpstream = "nuget-upstream";
|
|
14
|
+
readonly mavenUpstream = "maven-upstream";
|
|
15
|
+
readonly domain = "test-cdk";
|
|
16
|
+
private readonly codeArtifact;
|
|
17
|
+
private _loginInformation;
|
|
18
|
+
private constructor();
|
|
19
|
+
prepare(): Promise<void>;
|
|
20
|
+
loginInformation(): Promise<LoginInformation>;
|
|
21
|
+
delete(): Promise<void>;
|
|
22
|
+
/**
|
|
23
|
+
* List all packages and mark them as "allow upstream versions".
|
|
24
|
+
*
|
|
25
|
+
* If we don't do this and we publish `foo@2.3.4-rc.0`, then we can't
|
|
26
|
+
* download `foo@2.3.0` anymore because by default CodeArtifact will
|
|
27
|
+
* block different versions from the same package.
|
|
28
|
+
*/
|
|
29
|
+
markAllUpstreamAllow(): Promise<void>;
|
|
30
|
+
private ensureDomain;
|
|
31
|
+
private ensureUpstreams;
|
|
32
|
+
private ensureRepository;
|
|
33
|
+
private domainExists;
|
|
34
|
+
private repositoryExists;
|
|
35
|
+
private listPackages;
|
|
36
|
+
}
|
|
37
|
+
export interface LoginInformation {
|
|
38
|
+
readonly authToken: string;
|
|
39
|
+
readonly repositoryName: string;
|
|
40
|
+
readonly npmEndpoint: string;
|
|
41
|
+
readonly mavenEndpoint: string;
|
|
42
|
+
readonly nugetEndpoint: string;
|
|
43
|
+
readonly pypiEndpoint: string;
|
|
44
|
+
}
|
|
@@ -0,0 +1,258 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.TestRepository = void 0;
|
|
4
|
+
const AWS = require("aws-sdk");
|
|
5
|
+
const aws_1 = require("../aws");
|
|
6
|
+
const COLLECT_BY_TAG = 'collect-by';
|
|
7
|
+
const REPO_LIFETIME_MS = 24 * 3600 * 1000; // One day
|
|
8
|
+
class TestRepository {
|
|
9
|
+
constructor(repositoryName) {
|
|
10
|
+
this.repositoryName = repositoryName;
|
|
11
|
+
this.npmUpstream = 'npm-upstream';
|
|
12
|
+
this.pypiUpstream = 'pypi-upstream';
|
|
13
|
+
this.nugetUpstream = 'nuget-upstream';
|
|
14
|
+
this.mavenUpstream = 'maven-upstream';
|
|
15
|
+
this.domain = TestRepository.DEFAULT_DOMAIN;
|
|
16
|
+
this.codeArtifact = new AWS.CodeArtifact();
|
|
17
|
+
}
|
|
18
|
+
static async newRandom() {
|
|
19
|
+
const qualifier = Math.random().toString(36).replace(/[^a-z0-9]+/g, '');
|
|
20
|
+
const repo = new TestRepository(`test-${qualifier}`);
|
|
21
|
+
await repo.prepare();
|
|
22
|
+
return repo;
|
|
23
|
+
}
|
|
24
|
+
static async newWithName(name) {
|
|
25
|
+
const repo = new TestRepository(name);
|
|
26
|
+
await repo.prepare();
|
|
27
|
+
return repo;
|
|
28
|
+
}
|
|
29
|
+
static existing(repositoryName) {
|
|
30
|
+
return new TestRepository(repositoryName);
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Garbage collect repositories
|
|
34
|
+
*/
|
|
35
|
+
static async gc() {
|
|
36
|
+
if (!await TestRepository.existing('*dummy*').domainExists()) {
|
|
37
|
+
return;
|
|
38
|
+
}
|
|
39
|
+
const codeArtifact = new AWS.CodeArtifact();
|
|
40
|
+
let nextToken;
|
|
41
|
+
do {
|
|
42
|
+
const page = await codeArtifact.listRepositories({ nextToken }).promise();
|
|
43
|
+
for (const repo of page.repositories ?? []) {
|
|
44
|
+
const tags = await codeArtifact.listTagsForResource({ resourceArn: repo.arn }).promise();
|
|
45
|
+
const collectable = tags?.tags?.find(t => t.key === COLLECT_BY_TAG && Number(t.value) < Date.now());
|
|
46
|
+
if (collectable) {
|
|
47
|
+
// eslint-disable-next-line no-console
|
|
48
|
+
console.log('Deleting', repo.name);
|
|
49
|
+
await codeArtifact.deleteRepository({
|
|
50
|
+
domain: repo.domainName,
|
|
51
|
+
repository: repo.name,
|
|
52
|
+
}).promise();
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
nextToken = page.nextToken;
|
|
56
|
+
} while (nextToken);
|
|
57
|
+
}
|
|
58
|
+
async prepare() {
|
|
59
|
+
await this.ensureDomain();
|
|
60
|
+
await this.ensureUpstreams();
|
|
61
|
+
await this.ensureRepository(this.repositoryName, {
|
|
62
|
+
description: 'Testing repository',
|
|
63
|
+
upstreams: [
|
|
64
|
+
this.npmUpstream,
|
|
65
|
+
this.pypiUpstream,
|
|
66
|
+
this.nugetUpstream,
|
|
67
|
+
this.mavenUpstream,
|
|
68
|
+
],
|
|
69
|
+
tags: {
|
|
70
|
+
[COLLECT_BY_TAG]: `${Date.now() + REPO_LIFETIME_MS}`,
|
|
71
|
+
},
|
|
72
|
+
});
|
|
73
|
+
}
|
|
74
|
+
async loginInformation() {
|
|
75
|
+
if (this._loginInformation) {
|
|
76
|
+
return this._loginInformation;
|
|
77
|
+
}
|
|
78
|
+
this._loginInformation = {
|
|
79
|
+
authToken: (await this.codeArtifact.getAuthorizationToken({ domain: this.domain, durationSeconds: 12 * 3600 }).promise()).authorizationToken,
|
|
80
|
+
repositoryName: this.repositoryName,
|
|
81
|
+
npmEndpoint: (await this.codeArtifact.getRepositoryEndpoint({ domain: this.domain, repository: this.repositoryName, format: 'npm' }).promise()).repositoryEndpoint,
|
|
82
|
+
mavenEndpoint: (await this.codeArtifact.getRepositoryEndpoint({ domain: this.domain, repository: this.repositoryName, format: 'maven' }).promise()).repositoryEndpoint,
|
|
83
|
+
nugetEndpoint: (await this.codeArtifact.getRepositoryEndpoint({ domain: this.domain, repository: this.repositoryName, format: 'nuget' }).promise()).repositoryEndpoint,
|
|
84
|
+
pypiEndpoint: (await this.codeArtifact.getRepositoryEndpoint({ domain: this.domain, repository: this.repositoryName, format: 'pypi' }).promise()).repositoryEndpoint,
|
|
85
|
+
};
|
|
86
|
+
return this._loginInformation;
|
|
87
|
+
}
|
|
88
|
+
async delete() {
|
|
89
|
+
try {
|
|
90
|
+
await this.codeArtifact.deleteRepository({
|
|
91
|
+
domain: this.domain,
|
|
92
|
+
repository: this.repositoryName,
|
|
93
|
+
}).promise();
|
|
94
|
+
// eslint-disable-next-line no-console
|
|
95
|
+
console.log('Deleted', this.repositoryName);
|
|
96
|
+
}
|
|
97
|
+
catch (e) {
|
|
98
|
+
if (e.code !== 'ResourceNotFoundException') {
|
|
99
|
+
throw e;
|
|
100
|
+
}
|
|
101
|
+
// Okay
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
/**
|
|
105
|
+
* List all packages and mark them as "allow upstream versions".
|
|
106
|
+
*
|
|
107
|
+
* If we don't do this and we publish `foo@2.3.4-rc.0`, then we can't
|
|
108
|
+
* download `foo@2.3.0` anymore because by default CodeArtifact will
|
|
109
|
+
* block different versions from the same package.
|
|
110
|
+
*/
|
|
111
|
+
async markAllUpstreamAllow() {
|
|
112
|
+
for await (const pkg of this.listPackages({ upstream: 'BLOCK' })) {
|
|
113
|
+
await retryThrottled(() => this.codeArtifact.putPackageOriginConfiguration({
|
|
114
|
+
domain: this.domain,
|
|
115
|
+
repository: this.repositoryName,
|
|
116
|
+
format: pkg.format,
|
|
117
|
+
package: pkg.package,
|
|
118
|
+
namespace: pkg.namespace,
|
|
119
|
+
restrictions: {
|
|
120
|
+
publish: 'ALLOW',
|
|
121
|
+
upstream: 'ALLOW',
|
|
122
|
+
},
|
|
123
|
+
}).promise());
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
async ensureDomain() {
|
|
127
|
+
if (await this.domainExists()) {
|
|
128
|
+
return;
|
|
129
|
+
}
|
|
130
|
+
await this.codeArtifact.createDomain({
|
|
131
|
+
domain: this.domain,
|
|
132
|
+
tags: [{ key: 'testing', value: 'true' }],
|
|
133
|
+
}).promise();
|
|
134
|
+
}
|
|
135
|
+
async ensureUpstreams() {
|
|
136
|
+
await this.ensureRepository(this.npmUpstream, {
|
|
137
|
+
description: 'The upstream repository for NPM',
|
|
138
|
+
external: 'public:npmjs',
|
|
139
|
+
});
|
|
140
|
+
await this.ensureRepository(this.mavenUpstream, {
|
|
141
|
+
description: 'The upstream repository for Maven',
|
|
142
|
+
external: 'public:maven-central',
|
|
143
|
+
});
|
|
144
|
+
await this.ensureRepository(this.nugetUpstream, {
|
|
145
|
+
description: 'The upstream repository for NuGet',
|
|
146
|
+
external: 'public:nuget-org',
|
|
147
|
+
});
|
|
148
|
+
await this.ensureRepository(this.pypiUpstream, {
|
|
149
|
+
description: 'The upstream repository for PyPI',
|
|
150
|
+
external: 'public:pypi',
|
|
151
|
+
});
|
|
152
|
+
}
|
|
153
|
+
async ensureRepository(name, options) {
|
|
154
|
+
if (await this.repositoryExists(name)) {
|
|
155
|
+
return;
|
|
156
|
+
}
|
|
157
|
+
await this.codeArtifact.createRepository({
|
|
158
|
+
domain: this.domain,
|
|
159
|
+
repository: name,
|
|
160
|
+
description: options?.description,
|
|
161
|
+
upstreams: options?.upstreams?.map(repositoryName => ({ repositoryName })),
|
|
162
|
+
tags: options?.tags ? Object.entries(options.tags).map(([key, value]) => ({ key, value })) : undefined,
|
|
163
|
+
}).promise();
|
|
164
|
+
if (options?.external) {
|
|
165
|
+
const externalConnection = options.external;
|
|
166
|
+
await retry(() => this.codeArtifact.associateExternalConnection({
|
|
167
|
+
domain: this.domain,
|
|
168
|
+
repository: name,
|
|
169
|
+
externalConnection,
|
|
170
|
+
}).promise());
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
async domainExists() {
|
|
174
|
+
try {
|
|
175
|
+
await this.codeArtifact.describeDomain({ domain: this.domain }).promise();
|
|
176
|
+
return true;
|
|
177
|
+
}
|
|
178
|
+
catch (e) {
|
|
179
|
+
if (e.code !== 'ResourceNotFoundException') {
|
|
180
|
+
throw e;
|
|
181
|
+
}
|
|
182
|
+
return false;
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
async repositoryExists(name) {
|
|
186
|
+
try {
|
|
187
|
+
await this.codeArtifact.describeRepository({ domain: this.domain, repository: name }).promise();
|
|
188
|
+
return true;
|
|
189
|
+
}
|
|
190
|
+
catch (e) {
|
|
191
|
+
if (e.code !== 'ResourceNotFoundException') {
|
|
192
|
+
throw e;
|
|
193
|
+
}
|
|
194
|
+
return false;
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
async *listPackages(filter = {}) {
|
|
198
|
+
let response = await retryThrottled(() => this.codeArtifact.listPackages({
|
|
199
|
+
domain: this.domain,
|
|
200
|
+
repository: this.repositoryName,
|
|
201
|
+
...filter,
|
|
202
|
+
}).promise());
|
|
203
|
+
while (true) {
|
|
204
|
+
for (const p of response.packages ?? []) {
|
|
205
|
+
yield p;
|
|
206
|
+
}
|
|
207
|
+
if (!response.nextToken) {
|
|
208
|
+
break;
|
|
209
|
+
}
|
|
210
|
+
response = await retryThrottled(() => this.codeArtifact.listPackages({
|
|
211
|
+
domain: this.domain,
|
|
212
|
+
repository: this.repositoryName,
|
|
213
|
+
...filter,
|
|
214
|
+
nextToken: response.nextToken,
|
|
215
|
+
}).promise());
|
|
216
|
+
}
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
exports.TestRepository = TestRepository;
|
|
220
|
+
TestRepository.DEFAULT_DOMAIN = 'test-cdk';
|
|
221
|
+
async function retry(block) {
|
|
222
|
+
let attempts = 3;
|
|
223
|
+
while (true) {
|
|
224
|
+
try {
|
|
225
|
+
return await block();
|
|
226
|
+
}
|
|
227
|
+
catch (e) {
|
|
228
|
+
if (attempts-- === 0) {
|
|
229
|
+
throw e;
|
|
230
|
+
}
|
|
231
|
+
// eslint-disable-next-line no-console
|
|
232
|
+
console.debug(e.message);
|
|
233
|
+
await aws_1.sleep(500);
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
async function retryThrottled(block) {
|
|
238
|
+
let time = 100;
|
|
239
|
+
let attempts = 15;
|
|
240
|
+
while (true) {
|
|
241
|
+
try {
|
|
242
|
+
return await block();
|
|
243
|
+
}
|
|
244
|
+
catch (e) {
|
|
245
|
+
// eslint-disable-next-line no-console
|
|
246
|
+
console.debug(e.message);
|
|
247
|
+
if (e.code !== 'ThrottlingException') {
|
|
248
|
+
throw e;
|
|
249
|
+
}
|
|
250
|
+
if (attempts-- === 0) {
|
|
251
|
+
throw e;
|
|
252
|
+
}
|
|
253
|
+
await aws_1.sleep(Math.floor(Math.random() * time));
|
|
254
|
+
time *= 2;
|
|
255
|
+
}
|
|
256
|
+
}
|
|
257
|
+
}
|
|
258
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"codeartifact.js","sourceRoot":"","sources":["codeartifact.ts"],"names":[],"mappings":"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","sourcesContent":["import * as AWS from 'aws-sdk';\nimport { sleep } from '../aws';\n\nconst COLLECT_BY_TAG = 'collect-by';\nconst REPO_LIFETIME_MS = 24 * 3600 * 1000; // One day\n\nexport class TestRepository {\n  public static readonly DEFAULT_DOMAIN = 'test-cdk';\n\n  public static async newRandom() {\n    const qualifier = Math.random().toString(36).replace(/[^a-z0-9]+/g, '');\n\n    const repo = new TestRepository(`test-${qualifier}`);\n    await repo.prepare();\n    return repo;\n  }\n\n  public static async newWithName(name: string) {\n    const repo = new TestRepository(name);\n    await repo.prepare();\n    return repo;\n  }\n\n  public static existing(repositoryName: string) {\n    return new TestRepository(repositoryName);\n  }\n\n  /**\n   * Garbage collect repositories\n   */\n  public static async gc() {\n    if (!await TestRepository.existing('*dummy*').domainExists()) {\n      return;\n    }\n\n    const codeArtifact = new AWS.CodeArtifact();\n\n    let nextToken: string | undefined;\n    do {\n      const page = await codeArtifact.listRepositories({ nextToken }).promise();\n\n      for (const repo of page.repositories ?? []) {\n        const tags = await codeArtifact.listTagsForResource({ resourceArn: repo.arn! }).promise();\n        const collectable = tags?.tags?.find(t => t.key === COLLECT_BY_TAG && Number(t.value) < Date.now());\n        if (collectable) {\n          // eslint-disable-next-line no-console\n          console.log('Deleting', repo.name);\n          await codeArtifact.deleteRepository({\n            domain: repo.domainName!,\n            repository: repo.name!,\n          }).promise();\n        }\n      }\n\n      nextToken = page.nextToken;\n    } while (nextToken);\n  }\n\n  public readonly npmUpstream = 'npm-upstream';\n  public readonly pypiUpstream = 'pypi-upstream';\n  public readonly nugetUpstream = 'nuget-upstream';\n  public readonly mavenUpstream = 'maven-upstream';\n  public readonly domain = TestRepository.DEFAULT_DOMAIN;\n\n  private readonly codeArtifact = new AWS.CodeArtifact();\n\n  private _loginInformation: LoginInformation | undefined;\n\n  private constructor(public readonly repositoryName: string) {\n  }\n\n  public async prepare() {\n    await this.ensureDomain();\n    await this.ensureUpstreams();\n\n    await this.ensureRepository(this.repositoryName, {\n      description: 'Testing repository',\n      upstreams: [\n        this.npmUpstream,\n        this.pypiUpstream,\n        this.nugetUpstream,\n        this.mavenUpstream,\n      ],\n      tags: {\n        [COLLECT_BY_TAG]: `${Date.now() + REPO_LIFETIME_MS}`,\n      },\n    });\n  }\n\n  public async loginInformation(): Promise<LoginInformation> {\n    if (this._loginInformation) {\n      return this._loginInformation;\n    }\n\n    this._loginInformation = {\n      authToken: (await this.codeArtifact.getAuthorizationToken({ domain: this.domain, durationSeconds: 12 * 3600 }).promise()).authorizationToken!,\n      repositoryName: this.repositoryName,\n      npmEndpoint: (await this.codeArtifact.getRepositoryEndpoint({ domain: this.domain, repository: this.repositoryName, format: 'npm' }).promise()).repositoryEndpoint!,\n      mavenEndpoint: (await this.codeArtifact.getRepositoryEndpoint({ domain: this.domain, repository: this.repositoryName, format: 'maven' }).promise()).repositoryEndpoint!,\n      nugetEndpoint: (await this.codeArtifact.getRepositoryEndpoint({ domain: this.domain, repository: this.repositoryName, format: 'nuget' }).promise()).repositoryEndpoint!,\n      pypiEndpoint: (await this.codeArtifact.getRepositoryEndpoint({ domain: this.domain, repository: this.repositoryName, format: 'pypi' }).promise()).repositoryEndpoint!,\n    };\n    return this._loginInformation;\n  }\n\n  public async delete() {\n    try {\n      await this.codeArtifact.deleteRepository({\n        domain: this.domain,\n        repository: this.repositoryName,\n      }).promise();\n\n      // eslint-disable-next-line no-console\n      console.log('Deleted', this.repositoryName);\n    } catch (e) {\n      if (e.code !== 'ResourceNotFoundException') { throw e; }\n      // Okay\n    }\n  }\n\n  /**\n   * List all packages and mark them as \"allow upstream versions\".\n   *\n   * If we don't do this and we publish `foo@2.3.4-rc.0`, then we can't\n   * download `foo@2.3.0` anymore because by default CodeArtifact will\n   * block different versions from the same package.\n   */\n  public async markAllUpstreamAllow() {\n    for await (const pkg of this.listPackages({ upstream: 'BLOCK' })) {\n      await retryThrottled(() => this.codeArtifact.putPackageOriginConfiguration({\n        domain: this.domain,\n        repository: this.repositoryName,\n\n        format: pkg.format!,\n        package: pkg.package!,\n        namespace: pkg.namespace!,\n        restrictions: {\n          publish: 'ALLOW',\n          upstream: 'ALLOW',\n        },\n      }).promise());\n    }\n  }\n\n  private async ensureDomain() {\n    if (await this.domainExists()) { return; }\n    await this.codeArtifact.createDomain({\n      domain: this.domain,\n      tags: [{ key: 'testing', value: 'true' }],\n    }).promise();\n  }\n\n  private async ensureUpstreams() {\n    await this.ensureRepository(this.npmUpstream, {\n      description: 'The upstream repository for NPM',\n      external: 'public:npmjs',\n    });\n    await this.ensureRepository(this.mavenUpstream, {\n      description: 'The upstream repository for Maven',\n      external: 'public:maven-central',\n    });\n    await this.ensureRepository(this.nugetUpstream, {\n      description: 'The upstream repository for NuGet',\n      external: 'public:nuget-org',\n    });\n    await this.ensureRepository(this.pypiUpstream, {\n      description: 'The upstream repository for PyPI',\n      external: 'public:pypi',\n    });\n  }\n\n  private async ensureRepository(name: string, options?: {\n    readonly description?: string,\n    readonly external?: string,\n    readonly upstreams?: string[],\n    readonly tags?: Record<string, string>,\n  }) {\n    if (await this.repositoryExists(name)) { return; }\n\n    await this.codeArtifact.createRepository({\n      domain: this.domain,\n      repository: name,\n      description: options?.description,\n      upstreams: options?.upstreams?.map(repositoryName => ({ repositoryName })),\n      tags: options?.tags ? Object.entries(options.tags).map(([key, value]) => ({ key, value })) : undefined,\n    }).promise();\n\n    if (options?.external) {\n      const externalConnection = options.external;\n      await retry(() => this.codeArtifact.associateExternalConnection({\n        domain: this.domain,\n        repository: name,\n        externalConnection,\n      }).promise());\n    }\n  }\n\n  private async domainExists() {\n    try {\n      await this.codeArtifact.describeDomain({ domain: this.domain }).promise();\n      return true;\n    } catch (e) {\n      if (e.code !== 'ResourceNotFoundException') { throw e; }\n      return false;\n    }\n  }\n\n  private async repositoryExists(name: string) {\n    try {\n      await this.codeArtifact.describeRepository({ domain: this.domain, repository: name }).promise();\n      return true;\n    } catch (e) {\n      if (e.code !== 'ResourceNotFoundException') { throw e; }\n      return false;\n    }\n  }\n\n  private async* listPackages(filter: Pick<AWS.CodeArtifact.ListPackagesRequest, 'upstream'|'publish'|'format'> = {}) {\n    let response = await retryThrottled(() => this.codeArtifact.listPackages({\n      domain: this.domain,\n      repository: this.repositoryName,\n      ...filter,\n    }).promise());\n\n    while (true) {\n      for (const p of response.packages ?? []) {\n        yield p;\n      }\n\n      if (!response.nextToken) {\n        break;\n      }\n\n      response = await retryThrottled(() => this.codeArtifact.listPackages({\n        domain: this.domain,\n        repository: this.repositoryName,\n        ...filter,\n        nextToken: response.nextToken,\n      }).promise());\n    }\n  }\n}\n\nasync function retry<A>(block: () => Promise<A>) {\n  let attempts = 3;\n  while (true) {\n    try {\n      return await block();\n    } catch (e) {\n      if (attempts-- === 0) { throw e; }\n      // eslint-disable-next-line no-console\n      console.debug(e.message);\n      await sleep(500);\n    }\n  }\n}\n\nasync function retryThrottled<A>(block: () => Promise<A>) {\n  let time = 100;\n  let attempts = 15;\n  while (true) {\n    try {\n      return await block();\n    } catch (e) {\n      // eslint-disable-next-line no-console\n      console.debug(e.message);\n      if (e.code !== 'ThrottlingException') { throw e; }\n      if (attempts-- === 0) { throw e; }\n      await sleep(Math.floor(Math.random() * time));\n      time *= 2;\n    }\n  }\n}\n\nexport interface LoginInformation {\n  readonly authToken: string;\n  readonly repositoryName: string;\n  readonly npmEndpoint: string;\n  readonly mavenEndpoint: string;\n  readonly nugetEndpoint: string;\n  readonly pypiEndpoint: string;\n}"]}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
import { LoginInformation } from './codeartifact';
|
|
2
|
+
import { UsageDir } from './usage-dir';
|
|
3
|
+
export declare function mavenLogin(login: LoginInformation, usageDir: UsageDir): Promise<void>;
|
|
4
|
+
export declare function uploadJavaPackages(packages: string[], login: LoginInformation, usageDir: UsageDir): Promise<void>;
|
|
5
|
+
export declare function writeMavenSettingsFile(filename: string, login: LoginInformation): Promise<void>;
|
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.writeMavenSettingsFile = exports.uploadJavaPackages = exports.mavenLogin = void 0;
|
|
4
|
+
/* eslint-disable no-console */
|
|
5
|
+
const path = require("path");
|
|
6
|
+
const files_1 = require("../files");
|
|
7
|
+
const shell_1 = require("../shell");
|
|
8
|
+
const parallel_shell_1 = require("./parallel-shell");
|
|
9
|
+
// Do not try to JIT the Maven binary
|
|
10
|
+
const NO_JIT = '-XX:+TieredCompilation -XX:TieredStopAtLevel=1';
|
|
11
|
+
async function mavenLogin(login, usageDir) {
|
|
12
|
+
await writeMavenSettingsFile(settingsFile(usageDir), login);
|
|
13
|
+
// Write env var
|
|
14
|
+
// Twiddle JVM settings a bit to make Maven survive running on a CodeBuild box.
|
|
15
|
+
await usageDir.addToEnv({
|
|
16
|
+
MAVEN_OPTS: `-Duser.home=${usageDir.directory} ${NO_JIT} ${process.env.MAVEN_OPTS ?? ''}`.trim(),
|
|
17
|
+
});
|
|
18
|
+
}
|
|
19
|
+
exports.mavenLogin = mavenLogin;
|
|
20
|
+
function settingsFile(usageDir) {
|
|
21
|
+
// If we configure usageDir as a fake home directory Maven will find this file.
|
|
22
|
+
// (No other way to configure the settings file as part of the environment).
|
|
23
|
+
return path.join(usageDir.directory, '.m2', 'settings.xml');
|
|
24
|
+
}
|
|
25
|
+
async function uploadJavaPackages(packages, login, usageDir) {
|
|
26
|
+
await parallel_shell_1.parallelShell(packages, async (pkg, output) => {
|
|
27
|
+
console.log(`⏳ ${pkg}`);
|
|
28
|
+
await shell_1.shell(['mvn',
|
|
29
|
+
`--settings=${settingsFile(usageDir)}`,
|
|
30
|
+
'org.apache.maven.plugins:maven-deploy-plugin:3.0.0:deploy-file',
|
|
31
|
+
`-Durl=${login.mavenEndpoint}`,
|
|
32
|
+
'-DrepositoryId=codeartifact',
|
|
33
|
+
`-DpomFile=${pkg}`,
|
|
34
|
+
`-Dfile=${pkg.replace(/.pom$/, '.jar')}`,
|
|
35
|
+
`-Dsources=${pkg.replace(/.pom$/, '-sources.jar')}`,
|
|
36
|
+
`-Djavadoc=${pkg.replace(/.pom$/, '-javadoc.jar')}`], {
|
|
37
|
+
output,
|
|
38
|
+
modEnv: {
|
|
39
|
+
// Do not try to JIT the Maven binary
|
|
40
|
+
MAVEN_OPTS: `${NO_JIT} ${process.env.MAVEN_OPTS ?? ''}`.trim(),
|
|
41
|
+
},
|
|
42
|
+
});
|
|
43
|
+
console.log(`✅ ${pkg}`);
|
|
44
|
+
}, (pkg, output) => {
|
|
45
|
+
if (output.toString().includes('409 Conflict')) {
|
|
46
|
+
console.log(`❌ ${pkg}: already exists. Skipped.`);
|
|
47
|
+
return 'skip';
|
|
48
|
+
}
|
|
49
|
+
return 'fail';
|
|
50
|
+
});
|
|
51
|
+
}
|
|
52
|
+
exports.uploadJavaPackages = uploadJavaPackages;
|
|
53
|
+
async function writeMavenSettingsFile(filename, login) {
|
|
54
|
+
await files_1.writeFile(filename, `<?xml version="1.0" encoding="UTF-8" ?>
|
|
55
|
+
<settings xmlns="http://maven.apache.org/SETTINGS/1.0.0"
|
|
56
|
+
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
|
57
|
+
xsi:schemaLocation="http://maven.apache.org/SETTINGS/1.0.0
|
|
58
|
+
http://maven.apache.org/xsd/settings-1.0.0.xsd">
|
|
59
|
+
<servers>
|
|
60
|
+
<server>
|
|
61
|
+
<id>codeartifact</id>
|
|
62
|
+
<username>aws</username>
|
|
63
|
+
<password>${login.authToken}</password>
|
|
64
|
+
</server>
|
|
65
|
+
</servers>
|
|
66
|
+
<profiles>
|
|
67
|
+
<profile>
|
|
68
|
+
<id>default</id>
|
|
69
|
+
<repositories>
|
|
70
|
+
<repository>
|
|
71
|
+
<id>codeartifact</id>
|
|
72
|
+
<url>${login.mavenEndpoint}</url>
|
|
73
|
+
</repository>
|
|
74
|
+
</repositories>
|
|
75
|
+
</profile>
|
|
76
|
+
</profiles>
|
|
77
|
+
<activeProfiles>
|
|
78
|
+
<activeProfile>default</activeProfile>
|
|
79
|
+
</activeProfiles>
|
|
80
|
+
</settings>`);
|
|
81
|
+
}
|
|
82
|
+
exports.writeMavenSettingsFile = writeMavenSettingsFile;
|
|
83
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWF2ZW4uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJtYXZlbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSwrQkFBK0I7QUFDL0IsNkJBQTZCO0FBQzdCLG9DQUFxQztBQUNyQyxvQ0FBaUM7QUFFakMscURBQWlEO0FBR2pELHFDQUFxQztBQUNyQyxNQUFNLE1BQU0sR0FBRyxnREFBZ0QsQ0FBQztBQUV6RCxLQUFLLFVBQVUsVUFBVSxDQUFDLEtBQXVCLEVBQUUsUUFBa0I7SUFDMUUsTUFBTSxzQkFBc0IsQ0FBQyxZQUFZLENBQUMsUUFBUSxDQUFDLEVBQUUsS0FBSyxDQUFDLENBQUM7SUFFNUQsZ0JBQWdCO0lBQ2hCLCtFQUErRTtJQUMvRSxNQUFNLFFBQVEsQ0FBQyxRQUFRLENBQUM7UUFDdEIsVUFBVSxFQUFFLGVBQWUsUUFBUSxDQUFDLFNBQVMsSUFBSSxNQUFNLElBQUksT0FBTyxDQUFDLEdBQUcsQ0FBQyxVQUFVLElBQUksRUFBRSxFQUFFLENBQUMsSUFBSSxFQUFFO0tBQ2pHLENBQUMsQ0FBQztBQUNMLENBQUM7QUFSRCxnQ0FRQztBQUVELFNBQVMsWUFBWSxDQUFDLFFBQWtCO0lBQ3RDLCtFQUErRTtJQUMvRSw0RUFBNEU7SUFDNUUsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxTQUFTLEVBQUUsS0FBSyxFQUFFLGNBQWMsQ0FBQyxDQUFDO0FBQzlELENBQUM7QUFFTSxLQUFLLFVBQVUsa0JBQWtCLENBQUMsUUFBa0IsRUFBRSxLQUF1QixFQUFFLFFBQWtCO0lBQ3RHLE1BQU0sOEJBQWEsQ0FBQyxRQUFRLEVBQUUsS0FBSyxFQUFFLEdBQUcsRUFBRSxNQUFNLEVBQUUsRUFBRTtRQUNsRCxPQUFPLENBQUMsR0FBRyxDQUFDLEtBQUssR0FBRyxFQUFFLENBQUMsQ0FBQztRQUV4QixNQUFNLGFBQUssQ0FBQyxDQUFDLEtBQUs7WUFDaEIsY0FBYyxZQUFZLENBQUMsUUFBUSxDQUFDLEVBQUU7WUFDdEMsZ0VBQWdFO1lBQ2hFLFNBQVMsS0FBSyxDQUFDLGFBQWEsRUFBRTtZQUM5Qiw2QkFBNkI7WUFDN0IsYUFBYSxHQUFHLEVBQUU7WUFDbEIsVUFBVSxHQUFHLENBQUMsT0FBTyxDQUFDLE9BQU8sRUFBRSxNQUFNLENBQUMsRUFBRTtZQUN4QyxhQUFhLEdBQUcsQ0FBQyxPQUFPLENBQUMsT0FBTyxFQUFFLGNBQWMsQ0FBQyxFQUFFO1lBQ25ELGFBQWEsR0FBRyxDQUFDLE9BQU8sQ0FBQyxPQUFPLEVBQUUsY0FBYyxDQUFDLEVBQUUsQ0FBQyxFQUFFO1lBQ3RELE1BQU07WUFDTixNQUFNLEVBQUU7Z0JBQ04scUNBQXFDO2dCQUNyQyxVQUFVLEVBQUUsR0FBRyxNQUFNLElBQUksT0FBTyxDQUFDLEdBQUcsQ0FBQyxVQUFVLElBQUksRUFBRSxFQUFFLENBQUMsSUFBSSxFQUFFO2FBQy9EO1NBQ0YsQ0FBQyxDQUFDO1FBRUgsT0FBTyxDQUFDLEdBQUcsQ0FBQyxLQUFLLEdBQUcsRUFBRSxDQUFDLENBQUM7SUFDMUIsQ0FBQyxFQUNELENBQUMsR0FBRyxFQUFFLE1BQU0sRUFBRSxFQUFFO1FBQ2QsSUFBSSxNQUFNLENBQUMsUUFBUSxFQUFFLENBQUMsUUFBUSxDQUFDLGNBQWMsQ0FBQyxFQUFFO1lBQzlDLE9BQU8sQ0FBQyxHQUFHLENBQUMsS0FBSyxHQUFHLDRCQUE0QixDQUFDLENBQUM7WUFDbEQsT0FBTyxNQUFNLENBQUM7U0FDZjtRQUNELE9BQU8sTUFBTSxDQUFDO0lBQ2hCLENBQUMsQ0FBQyxDQUFDO0FBQ0wsQ0FBQztBQTdCRCxnREE2QkM7QUFFTSxLQUFLLFVBQVUsc0JBQXNCLENBQUMsUUFBZ0IsRUFBRSxLQUF1QjtJQUNwRixNQUFNLGlCQUFTLENBQUMsUUFBUSxFQUFFOzs7Ozs7Ozs7b0JBU1IsS0FBSyxDQUFDLFNBQVM7Ozs7Ozs7OzttQkFTaEIsS0FBSyxDQUFDLGFBQWE7Ozs7Ozs7O2NBUXhCLENBQUMsQ0FBQztBQUNoQixDQUFDO0FBNUJELHdEQTRCQyIsInNvdXJjZXNDb250ZW50IjpbIi8qIGVzbGludC1kaXNhYmxlIG5vLWNvbnNvbGUgKi9cbmltcG9ydCAqIGFzIHBhdGggZnJvbSAncGF0aCc7XG5pbXBvcnQgeyB3cml0ZUZpbGUgfSBmcm9tICcuLi9maWxlcyc7XG5pbXBvcnQgeyBzaGVsbCB9IGZyb20gJy4uL3NoZWxsJztcbmltcG9ydCB7IExvZ2luSW5mb3JtYXRpb24gfSBmcm9tICcuL2NvZGVhcnRpZmFjdCc7XG5pbXBvcnQgeyBwYXJhbGxlbFNoZWxsIH0gZnJvbSAnLi9wYXJhbGxlbC1zaGVsbCc7XG5pbXBvcnQgeyBVc2FnZURpciB9IGZyb20gJy4vdXNhZ2UtZGlyJztcblxuLy8gRG8gbm90IHRyeSB0byBKSVQgdGhlIE1hdmVuIGJpbmFyeVxuY29uc3QgTk9fSklUID0gJy1YWDorVGllcmVkQ29tcGlsYXRpb24gLVhYOlRpZXJlZFN0b3BBdExldmVsPTEnO1xuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gbWF2ZW5Mb2dpbihsb2dpbjogTG9naW5JbmZvcm1hdGlvbiwgdXNhZ2VEaXI6IFVzYWdlRGlyKSB7XG4gIGF3YWl0IHdyaXRlTWF2ZW5TZXR0aW5nc0ZpbGUoc2V0dGluZ3NGaWxlKHVzYWdlRGlyKSwgbG9naW4pO1xuXG4gIC8vIFdyaXRlIGVudiB2YXJcbiAgLy8gVHdpZGRsZSBKVk0gc2V0dGluZ3MgYSBiaXQgdG8gbWFrZSBNYXZlbiBzdXJ2aXZlIHJ1bm5pbmcgb24gYSBDb2RlQnVpbGQgYm94LlxuICBhd2FpdCB1c2FnZURpci5hZGRUb0Vudih7XG4gICAgTUFWRU5fT1BUUzogYC1EdXNlci5ob21lPSR7dXNhZ2VEaXIuZGlyZWN0b3J5fSAke05PX0pJVH0gJHtwcm9jZXNzLmVudi5NQVZFTl9PUFRTID8/ICcnfWAudHJpbSgpLFxuICB9KTtcbn1cblxuZnVuY3Rpb24gc2V0dGluZ3NGaWxlKHVzYWdlRGlyOiBVc2FnZURpcikge1xuICAvLyBJZiB3ZSBjb25maWd1cmUgdXNhZ2VEaXIgYXMgYSBmYWtlIGhvbWUgZGlyZWN0b3J5IE1hdmVuIHdpbGwgZmluZCB0aGlzIGZpbGUuXG4gIC8vIChObyBvdGhlciB3YXkgdG8gY29uZmlndXJlIHRoZSBzZXR0aW5ncyBmaWxlIGFzIHBhcnQgb2YgdGhlIGVudmlyb25tZW50KS5cbiAgcmV0dXJuIHBhdGguam9pbih1c2FnZURpci5kaXJlY3RvcnksICcubTInLCAnc2V0dGluZ3MueG1sJyk7XG59XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiB1cGxvYWRKYXZhUGFja2FnZXMocGFja2FnZXM6IHN0cmluZ1tdLCBsb2dpbjogTG9naW5JbmZvcm1hdGlvbiwgdXNhZ2VEaXI6IFVzYWdlRGlyKSB7XG4gIGF3YWl0IHBhcmFsbGVsU2hlbGwocGFja2FnZXMsIGFzeW5jIChwa2csIG91dHB1dCkgPT4ge1xuICAgIGNvbnNvbGUubG9nKGDij7MgJHtwa2d9YCk7XG5cbiAgICBhd2FpdCBzaGVsbChbJ212bicsXG4gICAgICBgLS1zZXR0aW5ncz0ke3NldHRpbmdzRmlsZSh1c2FnZURpcil9YCxcbiAgICAgICdvcmcuYXBhY2hlLm1hdmVuLnBsdWdpbnM6bWF2ZW4tZGVwbG95LXBsdWdpbjozLjAuMDpkZXBsb3ktZmlsZScsXG4gICAgICBgLUR1cmw9JHtsb2dpbi5tYXZlbkVuZHBvaW50fWAsXG4gICAgICAnLURyZXBvc2l0b3J5SWQ9Y29kZWFydGlmYWN0JyxcbiAgICAgIGAtRHBvbUZpbGU9JHtwa2d9YCxcbiAgICAgIGAtRGZpbGU9JHtwa2cucmVwbGFjZSgvLnBvbSQvLCAnLmphcicpfWAsXG4gICAgICBgLURzb3VyY2VzPSR7cGtnLnJlcGxhY2UoLy5wb20kLywgJy1zb3VyY2VzLmphcicpfWAsXG4gICAgICBgLURqYXZhZG9jPSR7cGtnLnJlcGxhY2UoLy5wb20kLywgJy1qYXZhZG9jLmphcicpfWBdLCB7XG4gICAgICBvdXRwdXQsXG4gICAgICBtb2RFbnY6IHtcbiAgICAgICAgLy8gRG8gbm90IHRyeSB0byBKSVQgdGhlIE1hdmVuIGJpbmFyeVxuICAgICAgICBNQVZFTl9PUFRTOiBgJHtOT19KSVR9ICR7cHJvY2Vzcy5lbnYuTUFWRU5fT1BUUyA/PyAnJ31gLnRyaW0oKSxcbiAgICAgIH0sXG4gICAgfSk7XG5cbiAgICBjb25zb2xlLmxvZyhg4pyFICR7cGtnfWApO1xuICB9LFxuICAocGtnLCBvdXRwdXQpID0+IHtcbiAgICBpZiAob3V0cHV0LnRvU3RyaW5nKCkuaW5jbHVkZXMoJzQwOSBDb25mbGljdCcpKSB7XG4gICAgICBjb25zb2xlLmxvZyhg4p2MICR7cGtnfTogYWxyZWFkeSBleGlzdHMuIFNraXBwZWQuYCk7XG4gICAgICByZXR1cm4gJ3NraXAnO1xuICAgIH1cbiAgICByZXR1cm4gJ2ZhaWwnO1xuICB9KTtcbn1cblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIHdyaXRlTWF2ZW5TZXR0aW5nc0ZpbGUoZmlsZW5hbWU6IHN0cmluZywgbG9naW46IExvZ2luSW5mb3JtYXRpb24pIHtcbiAgYXdhaXQgd3JpdGVGaWxlKGZpbGVuYW1lLCBgPD94bWwgdmVyc2lvbj1cIjEuMFwiIGVuY29kaW5nPVwiVVRGLThcIiA/PlxuICA8c2V0dGluZ3MgeG1sbnM9XCJodHRwOi8vbWF2ZW4uYXBhY2hlLm9yZy9TRVRUSU5HUy8xLjAuMFwiXG4gICAgICAgICAgICB4bWxuczp4c2k9XCJodHRwOi8vd3d3LnczLm9yZy8yMDAxL1hNTFNjaGVtYS1pbnN0YW5jZVwiXG4gICAgICAgICAgICB4c2k6c2NoZW1hTG9jYXRpb249XCJodHRwOi8vbWF2ZW4uYXBhY2hlLm9yZy9TRVRUSU5HUy8xLjAuMFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBodHRwOi8vbWF2ZW4uYXBhY2hlLm9yZy94c2Qvc2V0dGluZ3MtMS4wLjAueHNkXCI+XG4gICAgPHNlcnZlcnM+XG4gICAgICA8c2VydmVyPlxuICAgICAgICA8aWQ+Y29kZWFydGlmYWN0PC9pZD5cbiAgICAgICAgPHVzZXJuYW1lPmF3czwvdXNlcm5hbWU+XG4gICAgICAgIDxwYXNzd29yZD4ke2xvZ2luLmF1dGhUb2tlbn08L3Bhc3N3b3JkPlxuICAgICAgPC9zZXJ2ZXI+XG4gICAgPC9zZXJ2ZXJzPlxuICAgIDxwcm9maWxlcz5cbiAgICAgIDxwcm9maWxlPlxuICAgICAgICA8aWQ+ZGVmYXVsdDwvaWQ+XG4gICAgICAgIDxyZXBvc2l0b3JpZXM+XG4gICAgICAgICAgPHJlcG9zaXRvcnk+XG4gICAgICAgICAgICA8aWQ+Y29kZWFydGlmYWN0PC9pZD5cbiAgICAgICAgICAgIDx1cmw+JHtsb2dpbi5tYXZlbkVuZHBvaW50fTwvdXJsPlxuICAgICAgICAgIDwvcmVwb3NpdG9yeT5cbiAgICAgICAgPC9yZXBvc2l0b3JpZXM+XG4gICAgICA8L3Byb2ZpbGU+XG4gICAgPC9wcm9maWxlcz5cbiAgICA8YWN0aXZlUHJvZmlsZXM+XG4gICAgICA8YWN0aXZlUHJvZmlsZT5kZWZhdWx0PC9hY3RpdmVQcm9maWxlPlxuICAgIDwvYWN0aXZlUHJvZmlsZXM+XG4gIDwvc2V0dGluZ3M+YCk7XG59XG4iXX0=
|
|
@@ -0,0 +1,4 @@
|
|
|
1
|
+
import { LoginInformation } from './codeartifact';
|
|
2
|
+
import { UsageDir } from './usage-dir';
|
|
3
|
+
export declare function npmLogin(login: LoginInformation, usageDir: UsageDir): Promise<void>;
|
|
4
|
+
export declare function uploadNpmPackages(packages: string[], login: LoginInformation, usageDir: UsageDir): Promise<void>;
|