Skip to content

Commit

Permalink
fix(rds): fail with a descriptive error if Cluster's instance count i…
Browse files Browse the repository at this point in the history
…s a deploy-time value (#13765)

Added a condition to check whether the `instanceCount` is a token or not. If it's not a token then an exception will be thrown.

Fixes #13558 

----

*By submitting this pull request, I confirm that my contribution is made under the terms of the Apache-2.0 license*
  • Loading branch information
BLasan authored Mar 26, 2021
1 parent 2edca31 commit dd22e8f
Show file tree
Hide file tree
Showing 2 changed files with 17 additions and 36 deletions.
3 changes: 3 additions & 0 deletions packages/@aws-cdk/aws-rds/lib/cluster.ts
Original file line number Diff line number Diff line change
Expand Up @@ -651,6 +651,9 @@ interface InstanceConfig {
*/
function createInstances(cluster: DatabaseClusterNew, props: DatabaseClusterBaseProps, subnetGroup: ISubnetGroup): InstanceConfig {
const instanceCount = props.instances != null ? props.instances : 2;
if (Token.isUnresolved(instanceCount)) {
throw new Error('The number of instances an RDS Cluster consists of cannot be provided as a deploy-time only value!');
}
if (instanceCount < 1) {
throw new Error('At least one instance is required');
}
Expand Down
50 changes: 14 additions & 36 deletions packages/@aws-cdk/aws-rds/test/cluster.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -50,8 +50,22 @@ describe('cluster', () => {
DeletionPolicy: 'Delete',
UpdateReplacePolicy: 'Delete',
}, ResourcePart.CompleteDefinition);
});

test('validates that the number of instances is not a deploy-time value', () => {
const stack = testStack();
const vpc = new ec2.Vpc(stack, 'VPC');
const parameter = new cdk.CfnParameter(stack, 'Param', { type: 'Number' });

expect(() => {
new DatabaseCluster(stack, 'Database', {
instances: parameter.valueAsNumber,
engine: DatabaseClusterEngine.AURORA,
instanceProps: {
vpc,
},
});
}).toThrow('The number of instances an RDS Cluster consists of cannot be provided as a deploy-time only value!');
});

test('can create a cluster with a single instance', () => {
Expand Down Expand Up @@ -81,8 +95,6 @@ describe('cluster', () => {
MasterUserPassword: 'tooshort',
VpcSecurityGroupIds: [{ 'Fn::GetAtt': ['DatabaseSecurityGroup5C91FDCB', 'GroupId'] }],
});


});

test('can create a cluster with imported vpc and security group', () => {
Expand Down Expand Up @@ -116,8 +128,6 @@ describe('cluster', () => {
MasterUserPassword: 'tooshort',
VpcSecurityGroupIds: ['SecurityGroupId12345'],
});


});

test('cluster with parameter group', () => {
Expand Down Expand Up @@ -150,8 +160,6 @@ describe('cluster', () => {
expect(stack).toHaveResource('AWS::RDS::DBCluster', {
DBClusterParameterGroupName: { Ref: 'ParamsA8366201' },
});


});

test("sets the retention policy of the SubnetGroup to 'Retain' if the Cluster is created with 'Retain'", () => {
Expand All @@ -172,8 +180,6 @@ describe('cluster', () => {
DeletionPolicy: 'Retain',
UpdateReplacePolicy: 'Retain',
}, ResourcePart.CompleteDefinition);


});

test('creates a secret when master credentials are not specified', () => {
Expand Down Expand Up @@ -230,8 +236,6 @@ describe('cluster', () => {
SecretStringTemplate: '{"username":"admin"}',
},
});


});

test('create an encrypted cluster with custom KMS key', () => {
Expand Down Expand Up @@ -261,8 +265,6 @@ describe('cluster', () => {
],
},
});


});

test('cluster with instance parameter group', () => {
Expand Down Expand Up @@ -294,8 +296,6 @@ describe('cluster', () => {
Ref: 'ParameterGroup5E32DECB',
},
});


});

describe('performance insights', () => {
Expand Down Expand Up @@ -323,8 +323,6 @@ describe('cluster', () => {
PerformanceInsightsRetentionPeriod: 731,
PerformanceInsightsKMSKeyId: { 'Fn::GetAtt': ['Key961B73FD', 'Arn'] },
});


});

test('setting performance insights fields enables performance insights', () => {
Expand All @@ -348,8 +346,6 @@ describe('cluster', () => {
EnablePerformanceInsights: true,
PerformanceInsightsRetentionPeriod: 731,
});


});

test('throws if performance insights fields are set but performance insights is disabled', () => {
Expand All @@ -370,8 +366,6 @@ describe('cluster', () => {
},
});
}).toThrow(/`enablePerformanceInsights` disabled, but `performanceInsightRetention` or `performanceInsightEncryptionKey` was set/);


});
});

Expand All @@ -392,8 +386,6 @@ describe('cluster', () => {
expect(stack).toHaveResource('AWS::RDS::DBInstance', {
AutoMinorVersionUpgrade: false,
});


});

test('cluster with allow upgrade of major version', () => {
Expand All @@ -413,8 +405,6 @@ describe('cluster', () => {
expect(stack).toHaveResourceLike('AWS::RDS::DBInstance', {
AllowMajorVersionUpgrade: true,
});


});

test('cluster with disallow remove backups', () => {
Expand All @@ -434,8 +424,6 @@ describe('cluster', () => {
expect(stack).toHaveResourceLike('AWS::RDS::DBInstance', {
DeleteAutomatedBackups: false,
});


});

test('create a cluster using a specific version of MySQL', () => {
Expand All @@ -462,8 +450,6 @@ describe('cluster', () => {
Engine: 'aurora-mysql',
EngineVersion: '5.7.mysql_aurora.2.04.4',
});


});

test('create a cluster using a specific version of Postgresql', () => {
Expand Down Expand Up @@ -513,8 +499,6 @@ describe('cluster', () => {

// THEN
expect(stack.resolve(cluster.clusterEndpoint)).not.toEqual(stack.resolve(cluster.clusterReadEndpoint));


});

test('imported cluster with imported security group honors allowAllOutbound', () => {
Expand All @@ -540,8 +524,6 @@ describe('cluster', () => {
expect(stack).toHaveResource('AWS::EC2::SecurityGroupEgress', {
GroupId: 'sg-123456789',
});


});

test('can import a cluster with minimal attributes', () => {
Expand All @@ -567,8 +549,6 @@ describe('cluster', () => {
expect(() => cluster.clusterReadEndpoint).toThrow(/Cannot access `clusterReadEndpoint` of an imported cluster/);
expect(() => cluster.instanceIdentifiers).toThrow(/Cannot access `instanceIdentifiers` of an imported cluster/);
expect(() => cluster.instanceEndpoints).toThrow(/Cannot access `instanceEndpoints` of an imported cluster/);


});

test('imported cluster can access properties if attributes are provided', () => {
Expand All @@ -590,8 +570,6 @@ describe('cluster', () => {
expect(cluster.clusterReadEndpoint.socketAddress).toEqual('reader-address:3306');
expect(cluster.instanceIdentifiers).toEqual(['identifier']);
expect(cluster.instanceEndpoints.map(endpoint => endpoint.socketAddress)).toEqual(['instance-addr:3306']);


});

test('cluster supports metrics', () => {
Expand Down

0 comments on commit dd22e8f

Please sign in to comment.