cdk-assets 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 (56) hide show
  1. package/LICENSE +202 -0
  2. package/NOTICE +2 -0
  3. package/README.md +195 -0
  4. package/bin/cdk-assets +2 -0
  5. package/bin/cdk-assets.d.ts +1 -0
  6. package/bin/cdk-assets.js +70 -0
  7. package/bin/docker-credential-cdk-assets +2 -0
  8. package/bin/docker-credential-cdk-assets.d.ts +13 -0
  9. package/bin/docker-credential-cdk-assets.js +39 -0
  10. package/bin/list.d.ts +3 -0
  11. package/bin/list.js +10 -0
  12. package/bin/logging.d.ts +4 -0
  13. package/bin/logging.js +28 -0
  14. package/bin/publish.d.ts +5 -0
  15. package/bin/publish.js +49 -0
  16. package/lib/asset-manifest.d.ts +162 -0
  17. package/lib/asset-manifest.js +245 -0
  18. package/lib/aws-types.d.ts +1660 -0
  19. package/lib/aws-types.js +138 -0
  20. package/lib/aws.d.ts +70 -0
  21. package/lib/aws.js +125 -0
  22. package/lib/index.d.ts +4 -0
  23. package/lib/index.js +21 -0
  24. package/lib/private/archive.d.ts +3 -0
  25. package/lib/private/archive.js +87 -0
  26. package/lib/private/asset-handler.d.ts +54 -0
  27. package/lib/private/asset-handler.js +3 -0
  28. package/lib/private/docker-credentials.d.ts +35 -0
  29. package/lib/private/docker-credentials.js +89 -0
  30. package/lib/private/docker.d.ts +98 -0
  31. package/lib/private/docker.js +232 -0
  32. package/lib/private/fs-extra.d.ts +3 -0
  33. package/lib/private/fs-extra.js +37 -0
  34. package/lib/private/handlers/client-options.d.ts +3 -0
  35. package/lib/private/handlers/client-options.js +12 -0
  36. package/lib/private/handlers/container-images.d.ts +22 -0
  37. package/lib/private/handlers/container-images.js +224 -0
  38. package/lib/private/handlers/files.d.ts +14 -0
  39. package/lib/private/handlers/files.js +288 -0
  40. package/lib/private/handlers/index.d.ts +3 -0
  41. package/lib/private/handlers/index.js +16 -0
  42. package/lib/private/p-limit.d.ts +10 -0
  43. package/lib/private/p-limit.js +51 -0
  44. package/lib/private/placeholders.d.ts +10 -0
  45. package/lib/private/placeholders.js +34 -0
  46. package/lib/private/shell.d.ts +24 -0
  47. package/lib/private/shell.js +131 -0
  48. package/lib/private/util.d.ts +5 -0
  49. package/lib/private/util.js +16 -0
  50. package/lib/progress.d.ts +114 -0
  51. package/lib/progress.js +104 -0
  52. package/lib/publishing.d.ts +118 -0
  53. package/lib/publishing.js +193 -0
  54. package/package.json +103 -0
  55. package/scripts/manual-test-manifest.json +12 -0
  56. package/scripts/manual-test.sh +22 -0
@@ -0,0 +1,193 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.AssetPublishing = void 0;
4
+ const docker_1 = require("./private/docker");
5
+ const handlers_1 = require("./private/handlers");
6
+ const p_limit_1 = require("./private/p-limit");
7
+ const progress_1 = require("./progress");
8
+ class AssetPublishing {
9
+ constructor(manifest, options) {
10
+ this.manifest = manifest;
11
+ this.options = options;
12
+ /**
13
+ * The message for the IPublishProgress interface
14
+ */
15
+ this.message = 'Starting';
16
+ this.failures = new Array();
17
+ this.completedOperations = 0;
18
+ this.aborted = false;
19
+ this.handlerCache = new Map();
20
+ this.assets = manifest.entries;
21
+ this.totalOperations = this.assets.length;
22
+ this.publishInParallel = options.publishInParallel ?? false;
23
+ this.buildAssets = options.buildAssets ?? true;
24
+ this.publishAssets = options.publishAssets ?? true;
25
+ const self = this;
26
+ this.handlerHost = {
27
+ aws: this.options.aws,
28
+ get aborted() {
29
+ return self.aborted;
30
+ },
31
+ emitMessage(t, m) {
32
+ self.progressEvent(t, m);
33
+ },
34
+ dockerFactory: new docker_1.DockerFactory(),
35
+ };
36
+ }
37
+ /**
38
+ * Publish all assets from the manifest
39
+ */
40
+ async publish(options = {}) {
41
+ if (this.publishInParallel) {
42
+ const limit = (0, p_limit_1.pLimit)(20);
43
+ // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism
44
+ await Promise.all(this.assets.map((asset) => limit(async () => this.publishAsset(asset, options))));
45
+ }
46
+ else {
47
+ for (const asset of this.assets) {
48
+ if (!(await this.publishAsset(asset, options))) {
49
+ break;
50
+ }
51
+ }
52
+ }
53
+ if ((this.options.throwOnError ?? true) && this.failures.length > 0) {
54
+ throw new Error(`Error publishing: ${this.failures.map((e) => e.error.message)}`);
55
+ }
56
+ }
57
+ /**
58
+ * Build a single asset from the manifest
59
+ */
60
+ async buildEntry(asset) {
61
+ try {
62
+ if (this.progressEvent(progress_1.EventType.START, `Building ${asset.id}`)) {
63
+ return false;
64
+ }
65
+ const handler = this.assetHandler(asset);
66
+ await handler.build();
67
+ if (this.aborted) {
68
+ throw new Error('Aborted');
69
+ }
70
+ this.completedOperations++;
71
+ if (this.progressEvent(progress_1.EventType.SUCCESS, `Built ${asset.id}`)) {
72
+ return false;
73
+ }
74
+ }
75
+ catch (e) {
76
+ this.failures.push({ asset, error: e });
77
+ this.completedOperations++;
78
+ if (this.progressEvent(progress_1.EventType.FAIL, e.message)) {
79
+ return false;
80
+ }
81
+ }
82
+ return true;
83
+ }
84
+ /**
85
+ * Publish a single asset from the manifest
86
+ */
87
+ async publishEntry(asset, options = {}) {
88
+ try {
89
+ if (this.progressEvent(progress_1.EventType.START, `Publishing ${asset.id}`)) {
90
+ return false;
91
+ }
92
+ const handler = this.assetHandler(asset);
93
+ await handler.publish(options);
94
+ if (this.aborted) {
95
+ throw new Error('Aborted');
96
+ }
97
+ this.completedOperations++;
98
+ if (this.progressEvent(progress_1.EventType.SUCCESS, `Published ${asset.id}`)) {
99
+ return false;
100
+ }
101
+ }
102
+ catch (e) {
103
+ this.failures.push({ asset, error: e });
104
+ this.completedOperations++;
105
+ if (this.progressEvent(progress_1.EventType.FAIL, e.message)) {
106
+ return false;
107
+ }
108
+ }
109
+ return true;
110
+ }
111
+ /**
112
+ * Return whether a single asset is published
113
+ */
114
+ isEntryPublished(asset) {
115
+ const handler = this.assetHandler(asset);
116
+ return handler.isPublished();
117
+ }
118
+ /**
119
+ * publish an asset (used by 'publish()')
120
+ * @param asset The asset to publish
121
+ * @returns false when publishing should stop
122
+ */
123
+ async publishAsset(asset, options = {}) {
124
+ try {
125
+ if (this.progressEvent(progress_1.EventType.START, `Publishing ${asset.id}`)) {
126
+ return false;
127
+ }
128
+ const handler = this.assetHandler(asset);
129
+ if (this.buildAssets) {
130
+ await handler.build();
131
+ }
132
+ if (this.publishAssets) {
133
+ await handler.publish(options);
134
+ }
135
+ if (this.aborted) {
136
+ throw new Error('Aborted');
137
+ }
138
+ this.completedOperations++;
139
+ if (this.progressEvent(progress_1.EventType.SUCCESS, `Published ${asset.id}`)) {
140
+ return false;
141
+ }
142
+ }
143
+ catch (e) {
144
+ this.failures.push({ asset, error: e });
145
+ this.completedOperations++;
146
+ if (this.progressEvent(progress_1.EventType.FAIL, e.message)) {
147
+ return false;
148
+ }
149
+ }
150
+ return true;
151
+ }
152
+ get percentComplete() {
153
+ if (this.totalOperations === 0) {
154
+ return 100;
155
+ }
156
+ return Math.floor((this.completedOperations / this.totalOperations) * 100);
157
+ }
158
+ abort() {
159
+ this.aborted = true;
160
+ }
161
+ get hasFailures() {
162
+ return this.failures.length > 0;
163
+ }
164
+ /**
165
+ * Publish a progress event to the listener, if present.
166
+ *
167
+ * Returns whether an abort is requested. Helper to get rid of repetitive code in publish().
168
+ */
169
+ progressEvent(event, message) {
170
+ this.message = message;
171
+ if (this.options.progressListener) {
172
+ this.options.progressListener.onPublishEvent(event, this);
173
+ }
174
+ return this.aborted;
175
+ }
176
+ assetHandler(asset) {
177
+ const existing = this.handlerCache.get(asset);
178
+ if (existing) {
179
+ return existing;
180
+ }
181
+ if (this.options.quiet !== undefined && this.options.subprocessOutputDestination) {
182
+ throw new Error('Cannot set both quiet and subprocessOutputDestination. Please use only subprocessOutputDestination');
183
+ }
184
+ const subprocessOutputDestination = this.options.subprocessOutputDestination ?? (this.options.quiet ? 'ignore' : 'stdio');
185
+ const ret = (0, handlers_1.makeAssetHandler)(this.manifest, asset, this.handlerHost, {
186
+ subprocessOutputDestination,
187
+ });
188
+ this.handlerCache.set(asset, ret);
189
+ return ret;
190
+ }
191
+ }
192
+ exports.AssetPublishing = AssetPublishing;
193
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"publishing.js","sourceRoot":"","sources":["publishing.ts"],"names":[],"mappings":"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","sourcesContent":["import { AssetManifest, IManifestEntry } from './asset-manifest';\nimport { IAws } from './aws';\nimport {\n  IAssetHandler,\n  IHandlerHost,\n  type PublishOptions,\n  SubprocessOutputDestination,\n} from './private/asset-handler';\nimport { DockerFactory } from './private/docker';\nimport { makeAssetHandler } from './private/handlers';\nimport { pLimit } from './private/p-limit';\nimport { EventType, IPublishProgress, IPublishProgressListener } from './progress';\n\nexport interface AssetPublishingOptions {\n  /**\n   * Entry point for AWS client\n   */\n  readonly aws: IAws;\n\n  /**\n   * Listener for progress events\n   *\n   * @default No listener\n   */\n  readonly progressListener?: IPublishProgressListener;\n\n  /**\n   * Whether to throw at the end if there were errors\n   *\n   * @default true\n   */\n  readonly throwOnError?: boolean;\n\n  /**\n   * Whether to publish in parallel, when 'publish()' is called\n   *\n   * @default false\n   */\n  readonly publishInParallel?: boolean;\n\n  /**\n   * Whether to build assets, when 'publish()' is called\n   *\n   * @default true\n   */\n  readonly buildAssets?: boolean;\n\n  /**\n   * Whether to publish assets, when 'publish()' is called\n   *\n   * @default true\n   */\n  readonly publishAssets?: boolean;\n\n  /**\n   * @deprecated use {@link #subprocessOutputDestination} instead\n   */\n  readonly quiet?: boolean;\n\n  /**\n   * Where to send output of a subprocesses\n   *\n   * @default 'stdio'\n   */\n  subprocessOutputDestination?: SubprocessOutputDestination;\n}\n\n/**\n * A failure to publish an asset\n */\nexport interface FailedAsset {\n  /**\n   * The asset that failed to publish\n   */\n  readonly asset: IManifestEntry;\n\n  /**\n   * The failure that occurred\n   */\n  readonly error: Error;\n}\n\nexport class AssetPublishing implements IPublishProgress {\n  /**\n   * The message for the IPublishProgress interface\n   */\n  public message: string = 'Starting';\n\n  /**\n   * The current asset for the IPublishProgress interface\n   */\n  public currentAsset?: IManifestEntry;\n  public readonly failures = new Array<FailedAsset>();\n  private readonly assets: IManifestEntry[];\n\n  private readonly totalOperations: number;\n  private completedOperations: number = 0;\n  private aborted = false;\n  private readonly handlerHost: IHandlerHost;\n  private readonly publishInParallel: boolean;\n  private readonly buildAssets: boolean;\n  private readonly publishAssets: boolean;\n  private readonly handlerCache = new Map<IManifestEntry, IAssetHandler>();\n\n  constructor(\n    private readonly manifest: AssetManifest,\n    private readonly options: AssetPublishingOptions,\n  ) {\n    this.assets = manifest.entries;\n    this.totalOperations = this.assets.length;\n    this.publishInParallel = options.publishInParallel ?? false;\n    this.buildAssets = options.buildAssets ?? true;\n    this.publishAssets = options.publishAssets ?? true;\n\n    const self = this;\n    this.handlerHost = {\n      aws: this.options.aws,\n      get aborted() {\n        return self.aborted;\n      },\n      emitMessage(t, m) {\n        self.progressEvent(t, m);\n      },\n      dockerFactory: new DockerFactory(),\n    };\n  }\n\n  /**\n   * Publish all assets from the manifest\n   */\n  public async publish(options: PublishOptions = {}): Promise<void> {\n    if (this.publishInParallel) {\n      const limit = pLimit(20);\n      // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n      await Promise.all(\n        this.assets.map((asset) => limit(async () => this.publishAsset(asset, options))),\n      );\n    } else {\n      for (const asset of this.assets) {\n        if (!(await this.publishAsset(asset, options))) {\n          break;\n        }\n      }\n    }\n\n    if ((this.options.throwOnError ?? true) && this.failures.length > 0) {\n      throw new Error(`Error publishing: ${this.failures.map((e) => e.error.message)}`);\n    }\n  }\n\n  /**\n   * Build a single asset from the manifest\n   */\n  public async buildEntry(asset: IManifestEntry) {\n    try {\n      if (this.progressEvent(EventType.START, `Building ${asset.id}`)) {\n        return false;\n      }\n\n      const handler = this.assetHandler(asset);\n      await handler.build();\n\n      if (this.aborted) {\n        throw new Error('Aborted');\n      }\n\n      this.completedOperations++;\n      if (this.progressEvent(EventType.SUCCESS, `Built ${asset.id}`)) {\n        return false;\n      }\n    } catch (e: any) {\n      this.failures.push({ asset, error: e });\n      this.completedOperations++;\n      if (this.progressEvent(EventType.FAIL, e.message)) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  /**\n   * Publish a single asset from the manifest\n   */\n  public async publishEntry(asset: IManifestEntry, options: PublishOptions = {}) {\n    try {\n      if (this.progressEvent(EventType.START, `Publishing ${asset.id}`)) {\n        return false;\n      }\n\n      const handler = this.assetHandler(asset);\n      await handler.publish(options);\n\n      if (this.aborted) {\n        throw new Error('Aborted');\n      }\n\n      this.completedOperations++;\n      if (this.progressEvent(EventType.SUCCESS, `Published ${asset.id}`)) {\n        return false;\n      }\n    } catch (e: any) {\n      this.failures.push({ asset, error: e });\n      this.completedOperations++;\n      if (this.progressEvent(EventType.FAIL, e.message)) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  /**\n   * Return whether a single asset is published\n   */\n  public isEntryPublished(asset: IManifestEntry) {\n    const handler = this.assetHandler(asset);\n    return handler.isPublished();\n  }\n\n  /**\n   * publish an asset (used by 'publish()')\n   * @param asset The asset to publish\n   * @returns false when publishing should stop\n   */\n  private async publishAsset(asset: IManifestEntry, options: PublishOptions = {}) {\n    try {\n      if (this.progressEvent(EventType.START, `Publishing ${asset.id}`)) {\n        return false;\n      }\n\n      const handler = this.assetHandler(asset);\n\n      if (this.buildAssets) {\n        await handler.build();\n      }\n\n      if (this.publishAssets) {\n        await handler.publish(options);\n      }\n\n      if (this.aborted) {\n        throw new Error('Aborted');\n      }\n\n      this.completedOperations++;\n      if (this.progressEvent(EventType.SUCCESS, `Published ${asset.id}`)) {\n        return false;\n      }\n    } catch (e: any) {\n      this.failures.push({ asset, error: e });\n      this.completedOperations++;\n      if (this.progressEvent(EventType.FAIL, e.message)) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  public get percentComplete() {\n    if (this.totalOperations === 0) {\n      return 100;\n    }\n    return Math.floor((this.completedOperations / this.totalOperations) * 100);\n  }\n\n  public abort(): void {\n    this.aborted = true;\n  }\n\n  public get hasFailures() {\n    return this.failures.length > 0;\n  }\n\n  /**\n   * Publish a progress event to the listener, if present.\n   *\n   * Returns whether an abort is requested. Helper to get rid of repetitive code in publish().\n   */\n  private progressEvent(event: EventType, message: string): boolean {\n    this.message = message;\n    if (this.options.progressListener) {\n      this.options.progressListener.onPublishEvent(event, this);\n    }\n    return this.aborted;\n  }\n\n  private assetHandler(asset: IManifestEntry) {\n    const existing = this.handlerCache.get(asset);\n    if (existing) {\n      return existing;\n    }\n    if (this.options.quiet !== undefined && this.options.subprocessOutputDestination) {\n      throw new Error(\n        'Cannot set both quiet and subprocessOutputDestination. Please use only subprocessOutputDestination',\n      );\n    }\n    const subprocessOutputDestination =\n      this.options.subprocessOutputDestination ?? (this.options.quiet ? 'ignore' : 'stdio');\n    const ret = makeAssetHandler(this.manifest, asset, this.handlerHost, {\n      subprocessOutputDestination,\n    });\n    this.handlerCache.set(asset, ret);\n    return ret;\n  }\n}\n"]}
package/package.json ADDED
@@ -0,0 +1,103 @@
1
+ {
2
+ "name": "cdk-assets",
3
+ "description": "CDK Asset Publishing Tool",
4
+ "repository": {
5
+ "type": "git",
6
+ "url": "https://github.com/aws/aws-cdk-cli",
7
+ "directory": "packages/cdk-assets"
8
+ },
9
+ "bin": {
10
+ "cdk-assets": "bin/cdk-assets",
11
+ "docker-credential-cdk-assets": "bin/docker-credential-cdk-assets"
12
+ },
13
+ "scripts": {
14
+ "build": "npx projen build",
15
+ "bump": "npx projen bump",
16
+ "check-for-updates": "npx projen check-for-updates",
17
+ "compile": "npx projen compile",
18
+ "default": "npx projen default",
19
+ "eslint": "npx projen eslint",
20
+ "gather-versions": "npx projen gather-versions",
21
+ "package": "npx projen package",
22
+ "post-compile": "npx projen post-compile",
23
+ "pre-compile": "npx projen pre-compile",
24
+ "shrinkwrap": "npx projen shrinkwrap",
25
+ "test": "npx projen test",
26
+ "test:watch": "npx projen test:watch",
27
+ "unbump": "npx projen unbump",
28
+ "watch": "npx projen watch",
29
+ "projen": "npx projen"
30
+ },
31
+ "author": {
32
+ "name": "Amazon Web Services",
33
+ "url": "https://aws.amazon.com",
34
+ "organization": true
35
+ },
36
+ "devDependencies": {
37
+ "@cdklabs/eslint-plugin": "^1.3.2",
38
+ "@smithy/types": "^4.1.0",
39
+ "@smithy/util-stream": "^4.0.2",
40
+ "@stylistic/eslint-plugin": "^3.1.0",
41
+ "@types/archiver": "^6.0.3",
42
+ "@types/glob": "^8.1.0",
43
+ "@types/jest": "^29.5.14",
44
+ "@types/mime": "^2",
45
+ "@types/mock-fs": "^4",
46
+ "@types/node": "^16",
47
+ "@types/yargs": "^17.0.33",
48
+ "@typescript-eslint/eslint-plugin": "^8",
49
+ "@typescript-eslint/parser": "^8",
50
+ "aws-sdk-client-mock": "^4.1.0",
51
+ "aws-sdk-client-mock-jest": "^4.1.0",
52
+ "commit-and-tag-version": "^12",
53
+ "constructs": "^10.0.0",
54
+ "eslint": "^9",
55
+ "eslint-config-prettier": "^10.0.1",
56
+ "eslint-import-resolver-typescript": "^3.7.0",
57
+ "eslint-plugin-import": "^2.31.0",
58
+ "eslint-plugin-jest": "^28.11.0",
59
+ "eslint-plugin-prettier": "^5.2.3",
60
+ "fs-extra": "^11.3.0",
61
+ "graceful-fs": "^4.2.11",
62
+ "jest": "^29.7.0",
63
+ "jest-junit": "^15",
64
+ "jszip": "^3.10.1",
65
+ "mock-fs": "^5",
66
+ "prettier": "^2.8",
67
+ "projen": "^0.91.9",
68
+ "ts-jest": "^29.2.5",
69
+ "typescript": "5.6"
70
+ },
71
+ "dependencies": {
72
+ "@aws-cdk/cloud-assembly-schema": "^39.0.0",
73
+ "@aws-cdk/cx-api": "^2.178.2",
74
+ "@aws-sdk/client-ecr": "3.741",
75
+ "@aws-sdk/client-s3": "3.741",
76
+ "@aws-sdk/client-secrets-manager": "3.741",
77
+ "@aws-sdk/client-sts": "3.741",
78
+ "@aws-sdk/credential-providers": "^3.744.0",
79
+ "@aws-sdk/lib-storage": "^3.744.0",
80
+ "@smithy/config-resolver": "^4.0.1",
81
+ "@smithy/node-config-provider": "^4.0.1",
82
+ "archiver": "^7.0.1",
83
+ "glob": "^11.0.1",
84
+ "mime": "^2",
85
+ "yargs": "^17.7.2"
86
+ },
87
+ "keywords": [
88
+ "aws",
89
+ "cdk"
90
+ ],
91
+ "engines": {
92
+ "node": ">= 16.0.0"
93
+ },
94
+ "main": "lib/index.js",
95
+ "license": "Apache-2.0",
96
+ "homepage": "https://github.com/aws/aws-cdk",
97
+ "publishConfig": {
98
+ "access": "public"
99
+ },
100
+ "version": "0.0.0",
101
+ "types": "lib/index.d.ts",
102
+ "//": "~~ Generated by projen. To modify, edit .projenrc.js and run \"npx projen\"."
103
+ }
@@ -0,0 +1,12 @@
1
+ {
2
+ "version": "38.0.1",
3
+ "files": {
4
+ "asset1": {
5
+ "type": "file",
6
+ "source": { "path": "/Users/huijbers/Downloads/HEY-arm64.dmg" },
7
+ "destinations": {
8
+ "dest1": { "bucketName": "huijbers-test-objectlock", "objectKey": "hey.dmg" }
9
+ }
10
+ }
11
+ }
12
+ }
@@ -0,0 +1,22 @@
1
+ #!/bin/bash
2
+ set -eu
3
+ scriptdir=$(cd $(dirname $0) && pwd)
4
+
5
+ path=$(realpath $1)
6
+
7
+ cat <<EOF > $scriptdir/manual-test-manifest.json
8
+ {
9
+ "version": "38.0.1",
10
+ "files": {
11
+ "asset1": {
12
+ "type": "file",
13
+ "source": { "path": "$path" },
14
+ "destinations": {
15
+ "dest1": { "bucketName": "$2", "objectKey": "$3" }
16
+ }
17
+ }
18
+ }
19
+ }
20
+ EOF
21
+
22
+ npx ts-node $scriptdir/../bin/cdk-assets.ts -v -p $scriptdir/manual-test-manifest.json publish