Skip to content

Commit 94ba772

Browse files
authoredJan 21, 2025··
fix(cx-api): cannot detect CloudAssembly across different libraries (#32998)
### Reason for this change We are publishing the `cx-api` package twice: Once as a standalone package `@aws-cdk/cx-api` and once as part of the construct library under `aws-cdk-lib/cx-api`. The code is copied during the release and the same versions of the packages will have the same code. However this makes it difficult for other packages to take a type dependency on types from this package. The most common class that's used from `cx-api` is `CloudAssembly` - the result of `app.synth()`. Previously a package had to take a dependency on the very large `aws-cdk-lib` just to use a single type. It would be better if other packages could instead depend on the smaller, much more focused `@aws-cdk/cx-api` package. ### Description of changes This adds the same mechanism to `CloudAssembly` to detect cross-library compatibility, that we already use for constructs like `Stack` or `App`. In TypeScript, it's now possible for a consuming package to receive an object from either package and check at runtime if it satisfies the requirements. We cannot get around type checking with this. Instead we introduce a new type `ICloudAssembly` into the Cloud Assembly Schema package (cdklabs/cloud-assembly-schema#133). This interface only declares a single property: `directory`. Consumers can use this type to indicate where they would like to receive a `CloudAssembly`. They can then use runtime code to either confirm a provided object already satisfies the requirements or fallback to creating a new `CloudAssembly` from the directory. Because the `CloudAssembly` in `cxapi` implements the new interface, this approach will work in all jsii languages. In TypeScript it's even compatible with older version of `aws-cdk-lib`. Jsii language will only support this going forward. #### Allowed breaking changes ``` weakened:aws-cdk-lib.cloud_assembly_schema.MetadataEntry weakened:aws-cdk-lib.cx_api.MetadataEntryResult ``` This PR updates the version of `@aws-cdk/cloud-assembly-schema` to make new of the new interface. However the update also includes a change to `MetadataEntry` which was introduced in cdklabs/cloud-assembly-schema#121. That change is weakening a type, because in #31041, the CDK started emitting booleans and numbers as metadata values. But since these types weren't officially declared in the schema, jsii runtime type checking failed to load them. The fix was to officially extend the type union to include `boolean` and `number` primitive values. This is considered breaking, because when used as an output any consuming code will now need to account for the possibility of the value being a `boolean` or `number`. In static languages, the type would already have been treated as a generic Object with required runtime checks. ### Describe any new or updated permissions being added n/a ### Description of how you validated changes Unit tests. ### Checklist - [x] My code adheres to the [CONTRIBUTING GUIDE](https://github.com/aws/aws-cdk/blob/main/CONTRIBUTING.md) and [DESIGN GUIDELINES](https://github.com/aws/aws-cdk/blob/main/docs/DESIGN_GUIDELINES.md) ---- *By submitting this pull request, I confirm that my contribution is made under the terms of the Apache-2.0 license*
1 parent 6d834c0 commit 94ba772

File tree

17 files changed

+314
-10762
lines changed

17 files changed

+314
-10762
lines changed
 

‎allowed-breaking-changes.txt

+9-1
Original file line numberDiff line numberDiff line change
@@ -942,4 +942,12 @@ change-return-type:aws-cdk-lib.aws_lambda.FilterRule.null
942942
# output property was mistakenly marked as required even though it should have allowed
943943
# for undefined, i.e optional
944944
changed-type:@aws-cdk/cx-api.CloudFormationStackArtifact.notificationArns
945-
changed-type:aws-cdk-lib.cx_api.CloudFormationStackArtifact.notificationArns
945+
changed-type:aws-cdk-lib.cx_api.CloudFormationStackArtifact.notificationArns
946+
947+
# In aws/aws-cdk#31041, the CDK started emitting booleans and numbers as metadata values.
948+
# Since these types weren't officially declared in the schema, jsii runtime type checking failed to load them.
949+
# The type has now weakened, and when it's used as an output any consuming code will need to account for the possibility of the value being a boolean or number.
950+
# In static languages, the type would already have been treated as a generic Object with required runtime checks.
951+
# See: https://github.com/cdklabs/cloud-assembly-schema/pull/121
952+
weakened:aws-cdk-lib.cloud_assembly_schema.MetadataEntry
953+
weakened:aws-cdk-lib.cx_api.MetadataEntryResult

‎packages/@aws-cdk/cli-lib-alpha/THIRD_PARTY_LICENSES

+252-10,738
Large diffs are not rendered by default.

‎packages/@aws-cdk/cx-api/package.json

+4-4
Original file line numberDiff line numberDiff line change
@@ -62,7 +62,7 @@
6262
"gen": "cdk-copy cx-api",
6363
"watch": "cdk-watch",
6464
"lint": "cdk-lint && madge --circular --extensions js lib",
65-
"test": "cdk-test",
65+
"test": "jest",
6666
"pkglint": "pkglint -f",
6767
"package": "cdk-package",
6868
"awslint": "cdk-awslint",
@@ -82,12 +82,12 @@
8282
"semver": "^7.6.3"
8383
},
8484
"peerDependencies": {
85-
"@aws-cdk/cloud-assembly-schema": "^39.0.0"
85+
"@aws-cdk/cloud-assembly-schema": "^39.2.0"
8686
},
8787
"license": "Apache-2.0",
8888
"devDependencies": {
8989
"@aws-cdk/cdk-build-tools": "0.0.0",
90-
"@aws-cdk/cloud-assembly-schema": "^39.0.1",
90+
"@aws-cdk/cloud-assembly-schema": "^39.2.0",
9191
"@aws-cdk/pkglint": "0.0.0",
9292
"@types/jest": "^29.5.14",
9393
"@types/mock-fs": "^4.13.4",
@@ -120,4 +120,4 @@
120120
"publishConfig": {
121121
"tag": "latest"
122122
}
123-
}
123+
}

‎packages/@aws-cdk/integ-runner/package.json

+2-2
Original file line numberDiff line numberDiff line change
@@ -72,7 +72,7 @@
7272
},
7373
"dependencies": {
7474
"chokidar": "^3.6.0",
75-
"@aws-cdk/cloud-assembly-schema": "^39.0.0",
75+
"@aws-cdk/cloud-assembly-schema": "^39.2.0",
7676
"@aws-cdk/cloudformation-diff": "0.0.0",
7777
"@aws-cdk/cx-api": "0.0.0",
7878
"@aws-cdk/aws-service-spec": "^0.1.49",
@@ -109,4 +109,4 @@
109109
"publishConfig": {
110110
"tag": "latest"
111111
}
112-
}
112+
}

‎packages/@aws-cdk/toolkit/lib/api/cloud-assembly/private/source-builder.ts

+8-2
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
import * as cxapi from '@aws-cdk/cx-api';
12
import * as fs from 'fs-extra';
23
import type { ICloudAssemblySource } from '../';
34
import { ContextAwareCloudAssembly, ContextAwareCloudAssemblyProps } from './context-aware-source';
@@ -10,7 +11,6 @@ import { debug } from '../../io/private';
1011
import { AssemblyBuilder, CdkAppSourceProps } from '../source-builder';
1112

1213
export abstract class CloudAssemblySourceBuilder {
13-
1414
/**
1515
* Helper to provide the CloudAssemblySourceBuilder with required toolkit services
1616
* @deprecated this should move to the toolkit really.
@@ -40,13 +40,19 @@ export abstract class CloudAssemblySourceBuilder {
4040
produce: async () => {
4141
const outdir = determineOutputDirectory(props.outdir);
4242
const env = await prepareDefaultEnvironment(services, { outdir });
43-
return changeDir(async () =>
43+
const assembly = await changeDir(async () =>
4444
withContext(context.all, env, props.synthOptions ?? {}, async (envWithContext, ctx) =>
4545
withEnv(envWithContext, () => builder({
4646
outdir,
4747
context: ctx,
4848
})),
4949
), props.workingDirectory);
50+
51+
if (cxapi.CloudAssembly.isCloudAssembly(assembly)) {
52+
return assembly;
53+
}
54+
55+
return new cxapi.CloudAssembly(assembly.directory);
5056
},
5157
},
5258
contextAssemblyProps,

‎packages/@aws-cdk/toolkit/lib/api/cloud-assembly/source-builder.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import type * as cxapi from '@aws-cdk/cx-api';
1+
import type * as cxschema from '@aws-cdk/cloud-assembly-schema';
22

33
export interface AppProps {
44
/**
@@ -12,7 +12,7 @@ export interface AppProps {
1212
readonly context?: { [key: string]: any };
1313
}
1414

15-
export type AssemblyBuilder = (props: AppProps) => Promise<cxapi.CloudAssembly>;
15+
export type AssemblyBuilder = (props: AppProps) => Promise<cxschema.ICloudAssembly>;
1616

1717
/**
1818
* Configuration for creating a CLI from an AWS CDK App directory

‎packages/@aws-cdk/toolkit/package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -57,7 +57,7 @@
5757
"typescript": "~5.6.3"
5858
},
5959
"dependencies": {
60-
"@aws-cdk/cloud-assembly-schema": "^39.0.1",
60+
"@aws-cdk/cloud-assembly-schema": "^39.2.0",
6161
"@aws-cdk/cloudformation-diff": "0.0.0",
6262
"@aws-cdk/cx-api": "0.0.0",
6363
"@aws-cdk/region-info": "0.0.0",

‎packages/@aws-cdk/toolkit/test/_fixtures/external-context/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,5 +7,5 @@ export default async () => {
77
new s3.Bucket(stack, 'MyBucket', {
88
bucketName: app.node.tryGetContext('externally-provided-bucket-name'),
99
});
10-
return app.synth() as any;
10+
return app.synth();
1111
};

‎packages/@aws-cdk/toolkit/test/_fixtures/stack-with-bucket/index.js

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

‎packages/@aws-cdk/toolkit/test/_fixtures/stack-with-bucket/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -5,5 +5,5 @@ export default async () => {
55
const app = new core.App();
66
const stack = new core.Stack(app, 'Stack1');
77
new s3.Bucket(stack, 'MyBucket');
8-
return app.synth() as any;
8+
return app.synth();
99
};

‎packages/@aws-cdk/toolkit/test/_fixtures/two-empty-stacks/index.js

+1-2
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

‎packages/@aws-cdk/toolkit/test/_fixtures/two-empty-stacks/index.ts

+1-2
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,5 @@ export default async () => {
55
new core.Stack(app, 'Stack1');
66
new core.Stack(app, 'Stack2');
77

8-
// @todo fix api
9-
return app.synth() as any;
8+
return app.synth();
109
};

‎packages/aws-cdk-lib/cx-api/lib/cloud-assembly.ts

+18-1
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,19 @@
11
import * as fs from 'fs';
22
import * as os from 'os';
33
import * as path from 'path';
4+
// This is deliberately importing the interface from the external package.
5+
// We want this, so that jsii language packages can depend on @aws-cdk/cloud-assembly-schema
6+
// instead of being forced to take a dependency on the much larger aws-cdk-lib.
7+
import type { ICloudAssembly } from '@aws-cdk/cloud-assembly-schema';
48
import { CloudFormationStackArtifact } from './artifacts/cloudformation-artifact';
59
import { NestedCloudAssemblyArtifact } from './artifacts/nested-cloud-assembly-artifact';
610
import { TreeCloudArtifact } from './artifacts/tree-cloud-artifact';
711
import { CloudArtifact } from './cloud-artifact';
812
import { topologicalSort } from './toposort';
913
import * as cxschema from '../../cloud-assembly-schema';
1014

15+
const CLOUD_ASSEMBLY_SYMBOL = Symbol.for('@aws-cdk/cx-api.CloudAssembly');
16+
1117
/**
1218
* The name of the root manifest file of the assembly.
1319
*/
@@ -16,7 +22,16 @@ const MANIFEST_FILE = 'manifest.json';
1622
/**
1723
* Represents a deployable cloud application.
1824
*/
19-
export class CloudAssembly {
25+
export class CloudAssembly implements ICloudAssembly {
26+
/**
27+
* Return whether the given object is a CloudAssembly.
28+
*
29+
* We do attribute detection since we can't reliably use 'instanceof'.
30+
*/
31+
public static isCloudAssembly(x: any): x is CloudAssembly {
32+
return x !== null && typeof(x) === 'object' && CLOUD_ASSEMBLY_SYMBOL in x;
33+
}
34+
2035
/**
2136
* The root directory of the cloud assembly.
2237
*/
@@ -54,6 +69,8 @@ export class CloudAssembly {
5469
this.artifacts = this.renderArtifacts(loadOptions?.topoSort ?? true);
5570
this.runtime = this.manifest.runtime || { libraries: { } };
5671

72+
Object.defineProperty(this, CLOUD_ASSEMBLY_SYMBOL, { value: true });
73+
5774
// force validation of deps by accessing 'depends' on all artifacts
5875
this.validateDeps();
5976
}

‎packages/aws-cdk-lib/cx-api/test/cloud-assembly.test.ts

+9
Original file line numberDiff line numberDiff line change
@@ -185,3 +185,12 @@ test('getStackArtifact retrieves a stack by artifact id from a nested assembly',
185185
expect(assembly.getStackArtifact('stack1').id).toEqual('stack1');
186186
expect(assembly.getStackArtifact('stack2').id).toEqual('stack2');
187187
});
188+
189+
test('isCloudAssembly correctly detects Cloud Assemblies', () => {
190+
const assembly = new CloudAssembly(path.join(FIXTURES, 'nested-assemblies'));
191+
const inheritedAssembly = new (class extends CloudAssembly {})(path.join(FIXTURES, 'nested-assemblies'));
192+
193+
expect(CloudAssembly.isCloudAssembly(assembly)).toBe(true);
194+
expect(CloudAssembly.isCloudAssembly(inheritedAssembly)).toBe(true);
195+
expect(CloudAssembly.isCloudAssembly({})).toBe(false);
196+
});

‎packages/aws-cdk-lib/package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -122,7 +122,7 @@
122122
"@aws-cdk/asset-awscli-v1": "^2.2.208",
123123
"@aws-cdk/asset-kubectl-v20": "^2.1.3",
124124
"@aws-cdk/asset-node-proxy-agent-v6": "^2.1.0",
125-
"@aws-cdk/cloud-assembly-schema": "^39.0.1",
125+
"@aws-cdk/cloud-assembly-schema": "^39.2.0",
126126
"@balena/dockerignore": "^1.0.2",
127127
"case": "1.6.3",
128128
"fs-extra": "^11.2.0",

‎packages/aws-cdk/package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -140,7 +140,7 @@
140140
"xml-js": "^1.6.11"
141141
},
142142
"dependencies": {
143-
"@aws-cdk/cloud-assembly-schema": "^39.0.0",
143+
"@aws-cdk/cloud-assembly-schema": "^39.2.0",
144144
"@aws-cdk/cloudformation-diff": "0.0.0",
145145
"@aws-cdk/cx-api": "0.0.0",
146146
"@aws-cdk/region-info": "0.0.0",

‎yarn.lock

+2-2
Original file line numberDiff line numberDiff line change
@@ -71,15 +71,15 @@
7171
"@aws-cdk/service-spec-types" "^0.0.115"
7272
"@cdklabs/tskb" "^0.0.3"
7373

74-
"@aws-cdk/cloud-assembly-schema@^39.0.0", "@aws-cdk/cloud-assembly-schema@^39.0.1", "@aws-cdk/cloud-assembly-schema@^39.1.34":
74+
"@aws-cdk/cloud-assembly-schema@^39.1.34":
7575
version "39.1.34"
7676
resolved "https://registry.npmjs.org/@aws-cdk/cloud-assembly-schema/-/cloud-assembly-schema-39.1.34.tgz#06bfc07cd892bf431f97c8ff784c8b001480c134"
7777
integrity sha512-/uDvlrOD67PAaMxEX/gbWA8tce9SdVkGBtxEXV/R+DN7z6T4BztX3aZhkjMP/mzWu0UbkgOkwbOzqD4terCGwg==
7878
dependencies:
7979
jsonschema "^1.4.1"
8080
semver "^7.6.3"
8181

82-
"@aws-cdk/cloud-assembly-schema@^39.1.49":
82+
"@aws-cdk/cloud-assembly-schema@^39.1.49", "@aws-cdk/cloud-assembly-schema@^39.2.0":
8383
version "39.2.0"
8484
resolved "https://registry.npmjs.org/@aws-cdk/cloud-assembly-schema/-/cloud-assembly-schema-39.2.0.tgz#2abf4afb91c873a0206b5b6467fe53e505f7a9b7"
8585
integrity sha512-ymGG+ab4xN40iPx9O0zuuvu6qZi4RY+hr3YScSg5Ye0dkcchQ49RBINHrqqy7fZvcMbV7bkxf/Cxj9yxSF3BnA==

0 commit comments

Comments
 (0)
Please sign in to comment.