diff --git a/packages/@aws-cdk/aws-ecs/test/test.ecs-cluster.ts b/packages/@aws-cdk/aws-ecs/test/test.ecs-cluster.ts index 9c683245b8bea..961d93c38c2d8 100644 --- a/packages/@aws-cdk/aws-ecs/test/test.ecs-cluster.ts +++ b/packages/@aws-cdk/aws-ecs/test/test.ecs-cluster.ts @@ -155,6 +155,22 @@ export = { test.done(); }, + "multiple clusters with default capacity"(test: Test) { + // GIVEN + const stack = new cdk.Stack(); + const vpc = new ec2.Vpc(stack, 'MyVpc', {}); + + // WHEN + for (let i = 0; i < 2; i++) { + const cluster = new ecs.Cluster(stack, `EcsCluster${i}`, { vpc, }); + cluster.addCapacity('MyCapacity', { + instanceType: new ec2.InstanceType('m3.medium'), + }); + } + + test.done(); + }, + 'lifecycle hook is automatically added'(test: Test) { // GIVEN const stack = new cdk.Stack(); diff --git a/packages/@aws-cdk/aws-ssm/lib/parameter.ts b/packages/@aws-cdk/aws-ssm/lib/parameter.ts index a5605c4af01b8..fcb05a68709d0 100644 --- a/packages/@aws-cdk/aws-ssm/lib/parameter.ts +++ b/packages/@aws-cdk/aws-ssm/lib/parameter.ts @@ -1,7 +1,7 @@ import iam = require('@aws-cdk/aws-iam'); import { CfnDynamicReference, CfnDynamicReferenceService, CfnParameter, - Construct, ContextProvider, Fn, IConstruct, IResource, Resource, Stack, Token + Construct, ContextProvider, Fn, IConstruct, IResource, Resource, Stack, Token, ConstructNode } from '@aws-cdk/core'; import cxapi = require('@aws-cdk/cx-api'); import ssm = require('./ssm.generated'); @@ -248,6 +248,7 @@ export class StringParameter extends ParameterBase implements IStringParameter { const stack = Stack.of(scope); const id = makeIdentityForImportedValue(parameterName); const exists = stack.node.tryFindChild(id) as IStringParameter; + if (exists) { return exists.stringValue; } return this.fromStringParameterAttributes(stack, id, { parameterName, version }).stringValue; @@ -371,7 +372,7 @@ function _assertValidValue(value: string, allowedPattern: string): void { } function makeIdentityForImportedValue(parameterName: string) { - return `SsmParameterValue:${parameterName}:C96584B6-F00A-464E-AD19-53AFF4B05118`; + return ConstructNode.sanitizeId(`SsmParameterValue:${parameterName}:C96584B6-F00A-464E-AD19-53AFF4B05118`); } function arnForParameterName(scope: IConstruct, parameterName: string): string { diff --git a/packages/@aws-cdk/aws-ssm/test/test.parameter.ts b/packages/@aws-cdk/aws-ssm/test/test.parameter.ts index 8d3416010b87c..f6b8f4a22a13e 100644 --- a/packages/@aws-cdk/aws-ssm/test/test.parameter.ts +++ b/packages/@aws-cdk/aws-ssm/test/test.parameter.ts @@ -281,7 +281,17 @@ export = { } }); test.done(); - } + }, + + 'can query actual SSM Parameter Names, multiple times'(test: Test) { + // GIVEN + const stack = new Stack(); + // WHEN + ssm.StringParameter.valueForStringParameter(stack, '/my/param/name'); + ssm.StringParameter.valueForStringParameter(stack, '/my/param/name'); + + test.done(); + }, } }; diff --git a/packages/@aws-cdk/core/lib/construct.ts b/packages/@aws-cdk/core/lib/construct.ts index ece3dd4eb4fb1..f3ad18ebed224 100644 --- a/packages/@aws-cdk/core/lib/construct.ts +++ b/packages/@aws-cdk/core/lib/construct.ts @@ -27,6 +27,14 @@ export class ConstructNode { */ public static readonly PATH_SEP = '/'; + /** + * Return a sanitized version of an arbitrary string, so it can be used as an ID + */ + public static sanitizeId(id: string) { + // Cannot have PATH_SEPs in the ID + return id.replace(/\//g, '-'); + } + /** * Synthesizes a CloudAssembly from a construct tree. * @param root The root of the construct tree. @@ -139,6 +147,10 @@ export class ConstructNode { throw new Error('Only root constructs may have an empty name'); } + if (ConstructNode.sanitizeId(id) !== id) { + throw new Error('Invalid characters in id, use ConstructNode.sanitizeId(): ' + id); + } + // Has side effect so must be very last thing in constructor scope.node.addChild(host, this.id); } else { diff --git a/packages/@aws-cdk/core/test/test.construct.ts b/packages/@aws-cdk/core/test/test.construct.ts index 6aa8cfd793209..6362c142e998d 100644 --- a/packages/@aws-cdk/core/test/test.construct.ts +++ b/packages/@aws-cdk/core/test/test.construct.ts @@ -20,6 +20,12 @@ export = { test.done(); }, + 'constructs names cannot have a slash in them'(test: Test) { + const root = new Root(); + test.throws(() => new Construct(root, 'hello/bye')); + test.done(); + }, + 'construct.name returns the name of the construct'(test: Test) { const t = createTree();