@aws-solutions-constructs/aws-lambda-kinesisstreams 2.30.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/.eslintignore +5 -0
- package/.jsii +3655 -0
- package/README.md +121 -0
- package/architecture.png +0 -0
- package/lib/index.d.ts +90 -0
- package/lib/index.js +64 -0
- package/package.json +92 -0
- package/test/integ.existingLambda.d.ts +13 -0
- package/test/integ.existingLambda.expected.json +232 -0
- package/test/integ.existingLambda.js +31 -0
- package/test/integ.existingStream.d.ts +13 -0
- package/test/integ.existingStream.expected.json +231 -0
- package/test/integ.existingStream.js +37 -0
- package/test/integ.existingStreamWithCmk.d.ts +13 -0
- package/test/integ.existingStreamWithCmk.expected.json +300 -0
- package/test/integ.existingStreamWithCmk.js +39 -0
- package/test/integ.existingVpc.d.ts +13 -0
- package/test/integ.existingVpc.expected.json +1068 -0
- package/test/integ.existingVpc.js +34 -0
- package/test/integ.newStreamFromProps.d.ts +13 -0
- package/test/integ.newStreamFromProps.expected.json +231 -0
- package/test/integ.newStreamFromProps.js +34 -0
- package/test/integ.newVpc.d.ts +13 -0
- package/test/integ.newVpc.expected.json +674 -0
- package/test/integ.newVpc.js +31 -0
- package/test/integ.newVpcFromProps.d.ts +13 -0
- package/test/integ.newVpcFromProps.expected.json +560 -0
- package/test/integ.newVpcFromProps.js +34 -0
- package/test/integ.noArguments.d.ts +13 -0
- package/test/integ.noArguments.expected.json +232 -0
- package/test/integ.noArguments.js +30 -0
- package/test/lambda/index.mjs +15 -0
- package/test/lambda-kinesisstream.test.d.ts +13 -0
- package/test/lambda-kinesisstream.test.js +472 -0
|
@@ -0,0 +1,472 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
|
4
|
+
*
|
|
5
|
+
* Licensed under the Apache License, Version 2.0 (the "License"). You may not use this file except in compliance
|
|
6
|
+
* with the License. A copy of the License is located at
|
|
7
|
+
*
|
|
8
|
+
* http://www.apache.org/licenses/LICENSE-2.0
|
|
9
|
+
*
|
|
10
|
+
* or in the 'license' file accompanying this file. This file is distributed on an 'AS IS' BASIS, WITHOUT WARRANTIES
|
|
11
|
+
* OR CONDITIONS OF ANY KIND, express or implied. See the License for the specific language governing permissions
|
|
12
|
+
* and limitations under the License.
|
|
13
|
+
*/
|
|
14
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
15
|
+
const lib_1 = require("../lib");
|
|
16
|
+
const lambda = require("aws-cdk-lib/aws-lambda");
|
|
17
|
+
const cdk = require("aws-cdk-lib");
|
|
18
|
+
const ec2 = require("aws-cdk-lib/aws-ec2");
|
|
19
|
+
const kinesis = require("aws-cdk-lib/aws-kinesis");
|
|
20
|
+
const kms = require("aws-cdk-lib/aws-kms");
|
|
21
|
+
require("@aws-cdk/assert/jest");
|
|
22
|
+
const assertions_1 = require("aws-cdk-lib/assertions");
|
|
23
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
24
|
+
test('Default construct has all expected properties', () => {
|
|
25
|
+
const stack = new cdk.Stack();
|
|
26
|
+
const construct = new lib_1.LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {
|
|
27
|
+
lambdaFunctionProps: {
|
|
28
|
+
code: lambda.Code.fromAsset(`${__dirname}/lambda`),
|
|
29
|
+
runtime: lambda.Runtime.NODEJS_18_X,
|
|
30
|
+
handler: 'index.handler'
|
|
31
|
+
}
|
|
32
|
+
});
|
|
33
|
+
expect(construct.lambdaFunction).toBeDefined();
|
|
34
|
+
expect(construct.kinesisStream).toBeDefined();
|
|
35
|
+
expect(construct.cloudwatchAlarms).toBeDefined();
|
|
36
|
+
// By default, we don't create a VPC
|
|
37
|
+
expect(construct.vpc).toBeUndefined();
|
|
38
|
+
});
|
|
39
|
+
test('New VPC is created when deployVpc flag is true', () => {
|
|
40
|
+
const stack = new cdk.Stack();
|
|
41
|
+
new lib_1.LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {
|
|
42
|
+
lambdaFunctionProps: {
|
|
43
|
+
code: lambda.Code.fromAsset(`${__dirname}/lambda`),
|
|
44
|
+
runtime: lambda.Runtime.NODEJS_18_X,
|
|
45
|
+
handler: 'index.handler'
|
|
46
|
+
},
|
|
47
|
+
deployVpc: true
|
|
48
|
+
});
|
|
49
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
50
|
+
// Get the VPC created by the construct
|
|
51
|
+
const vpcResource = template.findResources('AWS::EC2::VPC');
|
|
52
|
+
const [vpcId] = Object.keys(vpcResource);
|
|
53
|
+
verifyLambdaFunctionVpcProps(stack, template, vpcId);
|
|
54
|
+
});
|
|
55
|
+
test('Existing VPC is used when specified', () => {
|
|
56
|
+
const stack = new cdk.Stack();
|
|
57
|
+
const existingVpc = new ec2.Vpc(stack, 'test-vpc', {
|
|
58
|
+
vpcName: 'my-vpc-name'
|
|
59
|
+
});
|
|
60
|
+
new lib_1.LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {
|
|
61
|
+
lambdaFunctionProps: {
|
|
62
|
+
code: lambda.Code.fromAsset(`${__dirname}/lambda`),
|
|
63
|
+
runtime: lambda.Runtime.NODEJS_18_X,
|
|
64
|
+
handler: 'index.handler'
|
|
65
|
+
},
|
|
66
|
+
existingVpc
|
|
67
|
+
});
|
|
68
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
69
|
+
// Get the VPC by its name
|
|
70
|
+
const vpcResource = template.findResources('AWS::EC2::VPC', {
|
|
71
|
+
Properties: {
|
|
72
|
+
Name: 'my-vpc-name'
|
|
73
|
+
}
|
|
74
|
+
});
|
|
75
|
+
const [vpcId] = Object.keys(vpcResource);
|
|
76
|
+
verifyLambdaFunctionVpcProps(stack, template, vpcId);
|
|
77
|
+
});
|
|
78
|
+
test('New VPC is created from user-provided vpcProps', () => {
|
|
79
|
+
const stack = new cdk.Stack();
|
|
80
|
+
new lib_1.LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {
|
|
81
|
+
lambdaFunctionProps: {
|
|
82
|
+
code: lambda.Code.fromAsset(`${__dirname}/lambda`),
|
|
83
|
+
runtime: lambda.Runtime.NODEJS_18_X,
|
|
84
|
+
handler: 'index.handler'
|
|
85
|
+
},
|
|
86
|
+
vpcProps: {
|
|
87
|
+
vpcName: 'my-vpc-name'
|
|
88
|
+
},
|
|
89
|
+
deployVpc: true
|
|
90
|
+
});
|
|
91
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
92
|
+
// Get the VPC by its name
|
|
93
|
+
const vpcResource = template.findResources('AWS::EC2::VPC', {
|
|
94
|
+
Properties: {
|
|
95
|
+
Name: 'my-vpc-name'
|
|
96
|
+
}
|
|
97
|
+
});
|
|
98
|
+
const [vpcId] = Object.keys(vpcResource);
|
|
99
|
+
verifyLambdaFunctionVpcProps(stack, template, vpcId);
|
|
100
|
+
});
|
|
101
|
+
test('Lambda Function has default stream environment variable', () => {
|
|
102
|
+
const stack = new cdk.Stack();
|
|
103
|
+
new lib_1.LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {
|
|
104
|
+
lambdaFunctionProps: {
|
|
105
|
+
code: lambda.Code.fromAsset(`${__dirname}/lambda`),
|
|
106
|
+
runtime: lambda.Runtime.NODEJS_18_X,
|
|
107
|
+
handler: 'index.handler'
|
|
108
|
+
}
|
|
109
|
+
});
|
|
110
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
111
|
+
// Get Kinesis Data Stream created by the construct
|
|
112
|
+
const kinesisStream = template.findResources('AWS::Kinesis::Stream');
|
|
113
|
+
const [kinesisStreamId] = Object.keys(kinesisStream);
|
|
114
|
+
expect(stack).toHaveResourceLike('AWS::Lambda::Function', {
|
|
115
|
+
Environment: {
|
|
116
|
+
Variables: {
|
|
117
|
+
KINESIS_DATASTREAM_NAME: {
|
|
118
|
+
Ref: kinesisStreamId
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
});
|
|
123
|
+
});
|
|
124
|
+
test('Lambda Function stream name environment variable can be overridden', () => {
|
|
125
|
+
const stack = new cdk.Stack();
|
|
126
|
+
new lib_1.LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {
|
|
127
|
+
lambdaFunctionProps: {
|
|
128
|
+
code: lambda.Code.fromAsset(`${__dirname}/lambda`),
|
|
129
|
+
runtime: lambda.Runtime.NODEJS_18_X,
|
|
130
|
+
handler: 'index.handler'
|
|
131
|
+
},
|
|
132
|
+
streamEnvironmentVariableName: 'CUSTOM_ENV_VAR_NAME'
|
|
133
|
+
});
|
|
134
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
135
|
+
// Get Kinesis Data Stream created by the construct
|
|
136
|
+
const kinesisStream = template.findResources('AWS::Kinesis::Stream');
|
|
137
|
+
const [kinesisStreamId] = Object.keys(kinesisStream);
|
|
138
|
+
expect(stack).toHaveResourceLike('AWS::Lambda::Function', {
|
|
139
|
+
Environment: {
|
|
140
|
+
Variables: {
|
|
141
|
+
CUSTOM_ENV_VAR_NAME: {
|
|
142
|
+
Ref: kinesisStreamId
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
});
|
|
147
|
+
});
|
|
148
|
+
test('Kinesis Stream is encrypted with AWS-managed CMK by default', () => {
|
|
149
|
+
const stack = new cdk.Stack();
|
|
150
|
+
new lib_1.LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {
|
|
151
|
+
lambdaFunctionProps: {
|
|
152
|
+
code: lambda.Code.fromAsset(`${__dirname}/lambda`),
|
|
153
|
+
runtime: lambda.Runtime.NODEJS_18_X,
|
|
154
|
+
handler: 'index.handler'
|
|
155
|
+
}
|
|
156
|
+
});
|
|
157
|
+
expect(stack).toHaveResourceLike('AWS::Kinesis::Stream', {
|
|
158
|
+
StreamEncryption: {
|
|
159
|
+
EncryptionType: 'KMS',
|
|
160
|
+
KeyId: 'alias/aws/kinesis'
|
|
161
|
+
}
|
|
162
|
+
});
|
|
163
|
+
});
|
|
164
|
+
test('CloudWatch Alarms are created for Kinesis Stream by default', () => {
|
|
165
|
+
const stack = new cdk.Stack();
|
|
166
|
+
new lib_1.LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {
|
|
167
|
+
lambdaFunctionProps: {
|
|
168
|
+
code: lambda.Code.fromAsset(`${__dirname}/lambda`),
|
|
169
|
+
runtime: lambda.Runtime.NODEJS_18_X,
|
|
170
|
+
handler: 'index.handler'
|
|
171
|
+
}
|
|
172
|
+
});
|
|
173
|
+
expect(stack).toHaveResourceLike('AWS::CloudWatch::Alarm', {
|
|
174
|
+
Namespace: 'AWS/Kinesis',
|
|
175
|
+
MetricName: 'GetRecords.IteratorAgeMilliseconds'
|
|
176
|
+
});
|
|
177
|
+
expect(stack).toHaveResourceLike('AWS::CloudWatch::Alarm', {
|
|
178
|
+
Namespace: 'AWS/Kinesis',
|
|
179
|
+
MetricName: 'ReadProvisionedThroughputExceeded'
|
|
180
|
+
});
|
|
181
|
+
});
|
|
182
|
+
test('CloudWatch Alarms are not created when createCloudWatchAlarms property is set to false', () => {
|
|
183
|
+
const stack = new cdk.Stack();
|
|
184
|
+
new lib_1.LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {
|
|
185
|
+
lambdaFunctionProps: {
|
|
186
|
+
code: lambda.Code.fromAsset(`${__dirname}/lambda`),
|
|
187
|
+
runtime: lambda.Runtime.NODEJS_18_X,
|
|
188
|
+
handler: 'index.handler'
|
|
189
|
+
},
|
|
190
|
+
createCloudWatchAlarms: false
|
|
191
|
+
});
|
|
192
|
+
expect(stack).toCountResources('AWS::CloudWatch::Alarm', 0);
|
|
193
|
+
});
|
|
194
|
+
test('Error is thrown when vpc is specified and existing lambda function is not associated with it', () => {
|
|
195
|
+
const stack = new aws_cdk_lib_1.Stack();
|
|
196
|
+
const existingLambdaObj = new lambda.Function(stack, 'test-lamba', {
|
|
197
|
+
code: lambda.Code.fromAsset(`${__dirname}/lambda`),
|
|
198
|
+
runtime: lambda.Runtime.NODEJS_18_X,
|
|
199
|
+
handler: 'index.handler'
|
|
200
|
+
});
|
|
201
|
+
const existingVpc = new ec2.Vpc(stack, 'test-vpc', {});
|
|
202
|
+
const app = () => {
|
|
203
|
+
new lib_1.LambdaToKinesisStreams(stack, 'lambda-to-sqs-stack', {
|
|
204
|
+
existingLambdaObj,
|
|
205
|
+
existingVpc
|
|
206
|
+
});
|
|
207
|
+
};
|
|
208
|
+
expect(app).toThrowError();
|
|
209
|
+
});
|
|
210
|
+
test('Construct uses existing Lambda Function', () => {
|
|
211
|
+
const stack = new aws_cdk_lib_1.Stack();
|
|
212
|
+
const existingLambdaObj = new lambda.Function(stack, 'test-lamba', {
|
|
213
|
+
code: lambda.Code.fromAsset(`${__dirname}/lambda`),
|
|
214
|
+
runtime: lambda.Runtime.NODEJS_18_X,
|
|
215
|
+
handler: 'index.handler',
|
|
216
|
+
functionName: 'my-lambda-function'
|
|
217
|
+
});
|
|
218
|
+
new lib_1.LambdaToKinesisStreams(stack, 'lambda-to-sqs-stack', {
|
|
219
|
+
existingLambdaObj
|
|
220
|
+
});
|
|
221
|
+
expect(stack).toHaveResourceLike('AWS::Lambda::Function', {
|
|
222
|
+
FunctionName: 'my-lambda-function'
|
|
223
|
+
});
|
|
224
|
+
});
|
|
225
|
+
test('Construct uses existing Kinesis Stream', () => {
|
|
226
|
+
const stack = new aws_cdk_lib_1.Stack();
|
|
227
|
+
const existingStreamObj = new kinesis.Stream(stack, 'test-stream', {
|
|
228
|
+
streamName: 'my-stream',
|
|
229
|
+
});
|
|
230
|
+
new lib_1.LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {
|
|
231
|
+
lambdaFunctionProps: {
|
|
232
|
+
code: lambda.Code.fromAsset(`${__dirname}/lambda`),
|
|
233
|
+
runtime: lambda.Runtime.NODEJS_18_X,
|
|
234
|
+
handler: 'index.handler'
|
|
235
|
+
},
|
|
236
|
+
existingStreamObj
|
|
237
|
+
});
|
|
238
|
+
expect(stack).toHaveResourceLike('AWS::Kinesis::Stream', {
|
|
239
|
+
Name: 'my-stream'
|
|
240
|
+
});
|
|
241
|
+
expect(stack).toCountResources('AWS::Kinesis::Stream', 1);
|
|
242
|
+
});
|
|
243
|
+
test('Construct uses unencrypted existing stream', () => {
|
|
244
|
+
const stack = new aws_cdk_lib_1.Stack();
|
|
245
|
+
const existingStreamObj = new kinesis.Stream(stack, 'test-stream', {
|
|
246
|
+
streamName: 'my-stream',
|
|
247
|
+
encryption: kinesis.StreamEncryption.UNENCRYPTED
|
|
248
|
+
});
|
|
249
|
+
new lib_1.LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {
|
|
250
|
+
lambdaFunctionProps: {
|
|
251
|
+
code: lambda.Code.fromAsset(`${__dirname}/lambda`),
|
|
252
|
+
runtime: lambda.Runtime.NODEJS_18_X,
|
|
253
|
+
handler: 'index.handler'
|
|
254
|
+
},
|
|
255
|
+
existingStreamObj
|
|
256
|
+
});
|
|
257
|
+
expect(stack).not.toHaveResourceLike('AWS::Kinesis::Stream', {
|
|
258
|
+
StreamEncryption: {
|
|
259
|
+
EncryptionType: 'KMS',
|
|
260
|
+
KeyId: 'alias/aws/kinesis'
|
|
261
|
+
}
|
|
262
|
+
});
|
|
263
|
+
expect(stack).toCountResources('AWS::Kinesis::Stream', 1);
|
|
264
|
+
});
|
|
265
|
+
test('Construct uses unencrypted streams from stream props', () => {
|
|
266
|
+
const stack = new aws_cdk_lib_1.Stack();
|
|
267
|
+
new lib_1.LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {
|
|
268
|
+
lambdaFunctionProps: {
|
|
269
|
+
code: lambda.Code.fromAsset(`${__dirname}/lambda`),
|
|
270
|
+
runtime: lambda.Runtime.NODEJS_18_X,
|
|
271
|
+
handler: 'index.handler'
|
|
272
|
+
},
|
|
273
|
+
kinesisStreamProps: {
|
|
274
|
+
streamName: 'my-stream',
|
|
275
|
+
encryption: kinesis.StreamEncryption.UNENCRYPTED
|
|
276
|
+
}
|
|
277
|
+
});
|
|
278
|
+
expect(stack).not.toHaveResourceLike('AWS::Kinesis::Stream', {
|
|
279
|
+
StreamEncryption: {
|
|
280
|
+
EncryptionType: 'KMS',
|
|
281
|
+
KeyId: 'alias/aws/kinesis'
|
|
282
|
+
}
|
|
283
|
+
});
|
|
284
|
+
expect(stack).toCountResources('AWS::Kinesis::Stream', 1);
|
|
285
|
+
});
|
|
286
|
+
test('Construct grants PutRecord permission for the Lambda Function to write to the Kinesis Stream', () => {
|
|
287
|
+
const stack = new aws_cdk_lib_1.Stack();
|
|
288
|
+
new lib_1.LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {
|
|
289
|
+
lambdaFunctionProps: {
|
|
290
|
+
code: lambda.Code.fromAsset(`${__dirname}/lambda`),
|
|
291
|
+
runtime: lambda.Runtime.NODEJS_18_X,
|
|
292
|
+
handler: 'index.handler'
|
|
293
|
+
}
|
|
294
|
+
});
|
|
295
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
296
|
+
// Get Kinesis Data Stream created by the construct
|
|
297
|
+
const kinesisStream = template.findResources('AWS::Kinesis::Stream');
|
|
298
|
+
const [kinesisStreamId] = Object.keys(kinesisStream);
|
|
299
|
+
expect(stack).toHaveResourceLike('AWS::IAM::Policy', {
|
|
300
|
+
PolicyDocument: {
|
|
301
|
+
Statement: [
|
|
302
|
+
{
|
|
303
|
+
Action: [
|
|
304
|
+
'xray:PutTraceSegments',
|
|
305
|
+
'xray:PutTelemetryRecords'
|
|
306
|
+
],
|
|
307
|
+
Effect: 'Allow',
|
|
308
|
+
Resource: '*'
|
|
309
|
+
},
|
|
310
|
+
{
|
|
311
|
+
Action: [
|
|
312
|
+
'kinesis:ListShards',
|
|
313
|
+
'kinesis:PutRecord',
|
|
314
|
+
'kinesis:PutRecords'
|
|
315
|
+
],
|
|
316
|
+
Effect: 'Allow',
|
|
317
|
+
Resource: {
|
|
318
|
+
'Fn::GetAtt': [
|
|
319
|
+
kinesisStreamId,
|
|
320
|
+
'Arn'
|
|
321
|
+
]
|
|
322
|
+
}
|
|
323
|
+
}
|
|
324
|
+
]
|
|
325
|
+
}
|
|
326
|
+
});
|
|
327
|
+
});
|
|
328
|
+
test('When a Customer-manged CMK is used on an existing stream, construct grants the Lambda Function permission to use the encryption key so it can publish messages to it', () => {
|
|
329
|
+
const stack = new aws_cdk_lib_1.Stack();
|
|
330
|
+
const encryptionKey = new kms.Key(stack, 'Key', {
|
|
331
|
+
description: 'my-key-description'
|
|
332
|
+
});
|
|
333
|
+
const existingStreamObj = new kinesis.Stream(stack, 'test-stream', {
|
|
334
|
+
streamName: 'my-stream',
|
|
335
|
+
encryptionKey
|
|
336
|
+
});
|
|
337
|
+
new lib_1.LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {
|
|
338
|
+
lambdaFunctionProps: {
|
|
339
|
+
code: lambda.Code.fromAsset(`${__dirname}/lambda`),
|
|
340
|
+
runtime: lambda.Runtime.NODEJS_18_X,
|
|
341
|
+
handler: 'index.handler'
|
|
342
|
+
},
|
|
343
|
+
existingStreamObj
|
|
344
|
+
});
|
|
345
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
346
|
+
const resource = template.findResources('AWS::KMS::Key', {
|
|
347
|
+
Properties: {
|
|
348
|
+
Description: assertions_1.Match.exact('my-key-description')
|
|
349
|
+
}
|
|
350
|
+
});
|
|
351
|
+
const [kmsKey] = Object.keys(resource);
|
|
352
|
+
expect(stack).toHaveResourceLike('AWS::IAM::Policy', {
|
|
353
|
+
PolicyDocument: {
|
|
354
|
+
Statement: [
|
|
355
|
+
{
|
|
356
|
+
Action: [
|
|
357
|
+
'xray:PutTraceSegments',
|
|
358
|
+
'xray:PutTelemetryRecords'
|
|
359
|
+
],
|
|
360
|
+
Effect: 'Allow',
|
|
361
|
+
Resource: '*'
|
|
362
|
+
},
|
|
363
|
+
{
|
|
364
|
+
Action: [
|
|
365
|
+
'kinesis:ListShards',
|
|
366
|
+
'kinesis:PutRecord',
|
|
367
|
+
'kinesis:PutRecords'
|
|
368
|
+
],
|
|
369
|
+
Effect: 'Allow',
|
|
370
|
+
Resource: {
|
|
371
|
+
'Fn::GetAtt': [
|
|
372
|
+
'teststream04374A09',
|
|
373
|
+
'Arn'
|
|
374
|
+
]
|
|
375
|
+
}
|
|
376
|
+
},
|
|
377
|
+
{
|
|
378
|
+
Action: [
|
|
379
|
+
'kms:Encrypt',
|
|
380
|
+
'kms:ReEncrypt*',
|
|
381
|
+
'kms:GenerateDataKey*'
|
|
382
|
+
],
|
|
383
|
+
Effect: 'Allow',
|
|
384
|
+
Resource: {
|
|
385
|
+
'Fn::GetAtt': [
|
|
386
|
+
kmsKey,
|
|
387
|
+
'Arn'
|
|
388
|
+
]
|
|
389
|
+
}
|
|
390
|
+
},
|
|
391
|
+
{
|
|
392
|
+
Action: [
|
|
393
|
+
'kms:Decrypt',
|
|
394
|
+
'kms:GenerateDataKey*'
|
|
395
|
+
],
|
|
396
|
+
Effect: 'Allow',
|
|
397
|
+
Resource: {
|
|
398
|
+
'Fn::GetAtt': [
|
|
399
|
+
kmsKey,
|
|
400
|
+
'Arn'
|
|
401
|
+
]
|
|
402
|
+
}
|
|
403
|
+
}
|
|
404
|
+
]
|
|
405
|
+
}
|
|
406
|
+
});
|
|
407
|
+
});
|
|
408
|
+
function verifyLambdaFunctionVpcProps(stack, template, vpcId) {
|
|
409
|
+
// Get the Security Group associated with the VPC
|
|
410
|
+
const securityGroupResource = template.findResources('AWS::EC2::SecurityGroup', {
|
|
411
|
+
Properties: {
|
|
412
|
+
VpcId: {
|
|
413
|
+
Ref: assertions_1.Match.exact(vpcId)
|
|
414
|
+
}
|
|
415
|
+
}
|
|
416
|
+
});
|
|
417
|
+
const [securityGroupId] = Object.keys(securityGroupResource);
|
|
418
|
+
// Get the Subnets associated with the VPC
|
|
419
|
+
const subnetResources = template.findResources('AWS::EC2::Subnet', {
|
|
420
|
+
Properties: {
|
|
421
|
+
VpcId: {
|
|
422
|
+
Ref: assertions_1.Match.exact(vpcId)
|
|
423
|
+
}
|
|
424
|
+
}
|
|
425
|
+
});
|
|
426
|
+
const [subnet1, subnet2] = Object.keys(subnetResources);
|
|
427
|
+
// Verify the Lambda Function has the same Security Group
|
|
428
|
+
expect(stack).toHaveResourceLike('AWS::Lambda::Function', {
|
|
429
|
+
VpcConfig: {
|
|
430
|
+
SecurityGroupIds: [
|
|
431
|
+
{
|
|
432
|
+
'Fn::GetAtt': [
|
|
433
|
+
securityGroupId,
|
|
434
|
+
'GroupId'
|
|
435
|
+
]
|
|
436
|
+
}
|
|
437
|
+
],
|
|
438
|
+
SubnetIds: [
|
|
439
|
+
{
|
|
440
|
+
Ref: subnet1
|
|
441
|
+
},
|
|
442
|
+
{
|
|
443
|
+
Ref: subnet2
|
|
444
|
+
}
|
|
445
|
+
]
|
|
446
|
+
}
|
|
447
|
+
});
|
|
448
|
+
// Verify the VPC has an interface endpoint for Kinesis Streams
|
|
449
|
+
expect(stack).toHaveResourceLike('AWS::EC2::VPCEndpoint', {
|
|
450
|
+
ServiceName: {
|
|
451
|
+
'Fn::Join': [
|
|
452
|
+
'',
|
|
453
|
+
[
|
|
454
|
+
'com.amazonaws.',
|
|
455
|
+
{
|
|
456
|
+
Ref: 'AWS::Region'
|
|
457
|
+
},
|
|
458
|
+
'.kinesis-streams'
|
|
459
|
+
]
|
|
460
|
+
]
|
|
461
|
+
},
|
|
462
|
+
VpcId: {
|
|
463
|
+
Ref: vpcId
|
|
464
|
+
},
|
|
465
|
+
});
|
|
466
|
+
// Verify the VPC has dns hostnames and support enabled
|
|
467
|
+
expect(stack).toHaveResource('AWS::EC2::VPC', {
|
|
468
|
+
EnableDnsHostnames: true,
|
|
469
|
+
EnableDnsSupport: true,
|
|
470
|
+
});
|
|
471
|
+
}
|
|
472
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lambda-kinesisstream.test.js","sourceRoot":"","sources":["lambda-kinesisstream.test.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n *  Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\n *\n *  Licensed under the Apache License, Version 2.0 (the \"License\"). You may not use this file except in compliance\n *  with the License. A copy of the License is located at\n *\n *      http://www.apache.org/licenses/LICENSE-2.0\n *\n *  or in the 'license' file accompanying this file. This file is distributed on an 'AS IS' BASIS, WITHOUT WARRANTIES\n *  OR CONDITIONS OF ANY KIND, express or implied. See the License for the specific language governing permissions\n *  and limitations under the License.\n */\n\nimport { LambdaToKinesisStreams } from \"../lib\";\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport * as cdk from \"aws-cdk-lib\";\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\";\nimport * as kinesis from 'aws-cdk-lib/aws-kinesis';\nimport * as kms from 'aws-cdk-lib/aws-kms';\nimport '@aws-cdk/assert/jest';\nimport { Match, Template } from \"aws-cdk-lib/assertions\";\nimport { Stack } from \"aws-cdk-lib\";\n\ntest('Default construct has all expected properties', () => {\n  const stack = new cdk.Stack();\n\n  const construct = new LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {\n    lambdaFunctionProps: {\n      code: lambda.Code.fromAsset(`${__dirname}/lambda`),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      handler: 'index.handler'\n    }\n  });\n\n  expect(construct.lambdaFunction).toBeDefined();\n  expect(construct.kinesisStream).toBeDefined();\n  expect(construct.cloudwatchAlarms).toBeDefined();\n\n  // By default, we don't create a VPC\n  expect(construct.vpc).toBeUndefined();\n});\n\ntest('New VPC is created when deployVpc flag is true', () => {\n  const stack = new cdk.Stack();\n\n  new LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {\n    lambdaFunctionProps: {\n      code: lambda.Code.fromAsset(`${__dirname}/lambda`),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      handler: 'index.handler'\n    },\n    deployVpc: true\n  });\n\n  const template = Template.fromStack(stack);\n\n  // Get the VPC created by the construct\n  const vpcResource = template.findResources('AWS::EC2::VPC');\n  const [ vpcId ] = Object.keys(vpcResource);\n\n  verifyLambdaFunctionVpcProps(stack, template, vpcId);\n});\n\ntest('Existing VPC is used when specified', () => {\n  const stack = new cdk.Stack();\n\n  const existingVpc = new ec2.Vpc(stack, 'test-vpc', {\n    vpcName: 'my-vpc-name'\n  });\n\n  new LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {\n    lambdaFunctionProps: {\n      code: lambda.Code.fromAsset(`${__dirname}/lambda`),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      handler: 'index.handler'\n    },\n    existingVpc\n  });\n\n  const template = Template.fromStack(stack);\n\n  // Get the VPC by its name\n  const vpcResource = template.findResources('AWS::EC2::VPC', {\n    Properties: {\n      Name: 'my-vpc-name'\n    }\n  });\n  const [ vpcId ] = Object.keys(vpcResource);\n\n  verifyLambdaFunctionVpcProps(stack, template, vpcId);\n});\n\ntest('New VPC is created from user-provided vpcProps', () => {\n  const stack = new cdk.Stack();\n\n  new LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {\n    lambdaFunctionProps: {\n      code: lambda.Code.fromAsset(`${__dirname}/lambda`),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      handler: 'index.handler'\n    },\n    vpcProps: {\n      vpcName: 'my-vpc-name'\n    },\n    deployVpc: true\n  });\n\n  const template = Template.fromStack(stack);\n\n  // Get the VPC by its name\n  const vpcResource = template.findResources('AWS::EC2::VPC', {\n    Properties: {\n      Name: 'my-vpc-name'\n    }\n  });\n  const [ vpcId ] = Object.keys(vpcResource);\n\n  verifyLambdaFunctionVpcProps(stack, template, vpcId);\n});\n\ntest('Lambda Function has default stream environment variable', () => {\n  const stack = new cdk.Stack();\n\n  new LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {\n    lambdaFunctionProps: {\n      code: lambda.Code.fromAsset(`${__dirname}/lambda`),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      handler: 'index.handler'\n    }\n  });\n\n  const template = Template.fromStack(stack);\n\n  // Get Kinesis Data Stream created by the construct\n  const kinesisStream = template.findResources('AWS::Kinesis::Stream');\n  const [ kinesisStreamId ] = Object.keys(kinesisStream);\n\n  expect(stack).toHaveResourceLike('AWS::Lambda::Function', {\n    Environment: {\n      Variables: {\n        KINESIS_DATASTREAM_NAME: {\n          Ref: kinesisStreamId\n        }\n      }\n    }\n  });\n});\n\ntest('Lambda Function stream name environment variable can be overridden', () => {\n  const stack = new cdk.Stack();\n\n  new LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {\n    lambdaFunctionProps: {\n      code: lambda.Code.fromAsset(`${__dirname}/lambda`),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      handler: 'index.handler'\n    },\n    streamEnvironmentVariableName: 'CUSTOM_ENV_VAR_NAME'\n  });\n\n  const template = Template.fromStack(stack);\n\n  // Get Kinesis Data Stream created by the construct\n  const kinesisStream = template.findResources('AWS::Kinesis::Stream');\n  const [ kinesisStreamId ] = Object.keys(kinesisStream);\n\n  expect(stack).toHaveResourceLike('AWS::Lambda::Function', {\n    Environment: {\n      Variables: {\n        CUSTOM_ENV_VAR_NAME: {\n          Ref: kinesisStreamId\n        }\n      }\n    }\n  });\n});\n\ntest('Kinesis Stream is encrypted with AWS-managed CMK by default', () => {\n  const stack = new cdk.Stack();\n\n  new LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {\n    lambdaFunctionProps: {\n      code: lambda.Code.fromAsset(`${__dirname}/lambda`),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      handler: 'index.handler'\n    }\n  });\n\n  expect(stack).toHaveResourceLike('AWS::Kinesis::Stream', {\n    StreamEncryption: {\n      EncryptionType: 'KMS',\n      KeyId: 'alias/aws/kinesis'\n    }\n  });\n});\n\ntest('CloudWatch Alarms are created for Kinesis Stream by default', () => {\n  const stack = new cdk.Stack();\n\n  new LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {\n    lambdaFunctionProps: {\n      code: lambda.Code.fromAsset(`${__dirname}/lambda`),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      handler: 'index.handler'\n    }\n  });\n\n  expect(stack).toHaveResourceLike('AWS::CloudWatch::Alarm', {\n    Namespace: 'AWS/Kinesis',\n    MetricName: 'GetRecords.IteratorAgeMilliseconds'\n  });\n\n  expect(stack).toHaveResourceLike('AWS::CloudWatch::Alarm', {\n    Namespace: 'AWS/Kinesis',\n    MetricName: 'ReadProvisionedThroughputExceeded'\n  });\n});\n\ntest('CloudWatch Alarms are not created when createCloudWatchAlarms property is set to false', () => {\n  const stack = new cdk.Stack();\n\n  new LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {\n    lambdaFunctionProps: {\n      code: lambda.Code.fromAsset(`${__dirname}/lambda`),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      handler: 'index.handler'\n    },\n    createCloudWatchAlarms: false\n  });\n\n  expect(stack).toCountResources('AWS::CloudWatch::Alarm', 0);\n});\n\ntest('Error is thrown when vpc is specified and existing lambda function is not associated with it', () => {\n  const stack = new Stack();\n\n  const existingLambdaObj = new lambda.Function(stack, 'test-lamba', {\n    code: lambda.Code.fromAsset(`${__dirname}/lambda`),\n    runtime: lambda.Runtime.NODEJS_18_X,\n    handler: 'index.handler'\n  });\n\n  const existingVpc = new ec2.Vpc(stack, 'test-vpc', {});\n\n  const app = () => {\n    new LambdaToKinesisStreams(stack, 'lambda-to-sqs-stack', {\n      existingLambdaObj,\n      existingVpc\n    });\n  };\n\n  expect(app).toThrowError();\n});\n\ntest('Construct uses existing Lambda Function', () => {\n  const stack = new Stack();\n\n  const existingLambdaObj = new lambda.Function(stack, 'test-lamba', {\n    code: lambda.Code.fromAsset(`${__dirname}/lambda`),\n    runtime: lambda.Runtime.NODEJS_18_X,\n    handler: 'index.handler',\n    functionName: 'my-lambda-function'\n  });\n\n  new LambdaToKinesisStreams(stack, 'lambda-to-sqs-stack', {\n    existingLambdaObj\n  });\n\n  expect(stack).toHaveResourceLike('AWS::Lambda::Function', {\n    FunctionName: 'my-lambda-function'\n  });\n});\n\ntest('Construct uses existing Kinesis Stream', () => {\n  const stack = new Stack();\n\n  const existingStreamObj = new kinesis.Stream(stack, 'test-stream', {\n    streamName: 'my-stream',\n  });\n\n  new LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {\n    lambdaFunctionProps: {\n      code: lambda.Code.fromAsset(`${__dirname}/lambda`),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      handler: 'index.handler'\n    },\n    existingStreamObj\n  });\n\n  expect(stack).toHaveResourceLike('AWS::Kinesis::Stream', {\n    Name: 'my-stream'\n  });\n\n  expect(stack).toCountResources('AWS::Kinesis::Stream', 1);\n});\n\ntest('Construct uses unencrypted existing stream', () => {\n  const stack = new Stack();\n\n  const existingStreamObj = new kinesis.Stream(stack, 'test-stream', {\n    streamName: 'my-stream',\n    encryption: kinesis.StreamEncryption.UNENCRYPTED\n  });\n\n  new LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {\n    lambdaFunctionProps: {\n      code: lambda.Code.fromAsset(`${__dirname}/lambda`),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      handler: 'index.handler'\n    },\n    existingStreamObj\n  });\n\n  expect(stack).not.toHaveResourceLike('AWS::Kinesis::Stream', {\n    StreamEncryption: {\n      EncryptionType: 'KMS',\n      KeyId: 'alias/aws/kinesis'\n    }\n  });\n\n  expect(stack).toCountResources('AWS::Kinesis::Stream', 1);\n});\n\ntest('Construct uses unencrypted streams from stream props', () => {\n  const stack = new Stack();\n\n  new LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {\n    lambdaFunctionProps: {\n      code: lambda.Code.fromAsset(`${__dirname}/lambda`),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      handler: 'index.handler'\n    },\n    kinesisStreamProps: {\n      streamName: 'my-stream',\n      encryption: kinesis.StreamEncryption.UNENCRYPTED\n    }\n  });\n\n  expect(stack).not.toHaveResourceLike('AWS::Kinesis::Stream', {\n    StreamEncryption: {\n      EncryptionType: 'KMS',\n      KeyId: 'alias/aws/kinesis'\n    }\n  });\n\n  expect(stack).toCountResources('AWS::Kinesis::Stream', 1);\n});\n\ntest('Construct grants PutRecord permission for the Lambda Function to write to the Kinesis Stream', () => {\n  const stack = new Stack();\n\n  new LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {\n    lambdaFunctionProps: {\n      code: lambda.Code.fromAsset(`${__dirname}/lambda`),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      handler: 'index.handler'\n    }\n  });\n\n  const template = Template.fromStack(stack);\n\n  // Get Kinesis Data Stream created by the construct\n  const kinesisStream = template.findResources('AWS::Kinesis::Stream');\n  const [ kinesisStreamId ] = Object.keys(kinesisStream);\n\n  expect(stack).toHaveResourceLike('AWS::IAM::Policy', {\n    PolicyDocument: {\n      Statement: [\n        {\n          Action: [\n            'xray:PutTraceSegments',\n            'xray:PutTelemetryRecords'\n          ],\n          Effect: 'Allow',\n          Resource: '*'\n        },\n        {\n          Action: [\n            'kinesis:ListShards',\n            'kinesis:PutRecord',\n            'kinesis:PutRecords'\n          ],\n          Effect: 'Allow',\n          Resource: {\n            'Fn::GetAtt': [\n              kinesisStreamId,\n              'Arn'\n            ]\n          }\n        }\n      ]\n    }\n  });\n});\n\ntest('When a Customer-manged CMK is used on an existing stream, construct grants the Lambda Function permission to use the encryption key so it can publish messages to it', () => {\n  const stack = new Stack();\n\n  const encryptionKey = new kms.Key(stack, 'Key', {\n    description: 'my-key-description'\n  });\n\n  const existingStreamObj = new kinesis.Stream(stack, 'test-stream', {\n    streamName: 'my-stream',\n    encryptionKey\n  });\n\n  new LambdaToKinesisStreams(stack, 'test-lambda-kinesisstreams', {\n    lambdaFunctionProps: {\n      code: lambda.Code.fromAsset(`${__dirname}/lambda`),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      handler: 'index.handler'\n    },\n    existingStreamObj\n  });\n\n  const template = Template.fromStack(stack);\n  const resource = template.findResources('AWS::KMS::Key', {\n    Properties: {\n      Description: Match.exact('my-key-description')\n    }\n  });\n  const [ kmsKey ] = Object.keys(resource);\n\n  expect(stack).toHaveResourceLike('AWS::IAM::Policy', {\n    PolicyDocument: {\n      Statement: [\n        {\n          Action: [\n            'xray:PutTraceSegments',\n            'xray:PutTelemetryRecords'\n          ],\n          Effect: 'Allow',\n          Resource: '*'\n        },\n        {\n          Action: [\n            'kinesis:ListShards',\n            'kinesis:PutRecord',\n            'kinesis:PutRecords'\n          ],\n          Effect: 'Allow',\n          Resource: {\n            'Fn::GetAtt': [\n              'teststream04374A09',\n              'Arn'\n            ]\n          }\n        },\n        {\n          Action: [\n            'kms:Encrypt',\n            'kms:ReEncrypt*',\n            'kms:GenerateDataKey*'\n          ],\n          Effect: 'Allow',\n          Resource: {\n            'Fn::GetAtt': [\n              kmsKey,\n              'Arn'\n            ]\n          }\n        },\n        {\n          Action: [\n            'kms:Decrypt',\n            'kms:GenerateDataKey*'\n          ],\n          Effect: 'Allow',\n          Resource: {\n            'Fn::GetAtt': [\n              kmsKey,\n              'Arn'\n            ]\n          }\n        }\n      ]\n    }\n  });\n});\n\nfunction verifyLambdaFunctionVpcProps(stack: Stack, template: Template, vpcId: string) {\n  // Get the Security Group associated with the VPC\n  const securityGroupResource = template.findResources('AWS::EC2::SecurityGroup', {\n    Properties: {\n      VpcId: {\n        Ref: Match.exact(vpcId)\n      }\n    }\n  });\n  const [ securityGroupId ] = Object.keys(securityGroupResource);\n\n  // Get the Subnets associated with the VPC\n  const subnetResources = template.findResources('AWS::EC2::Subnet', {\n    Properties: {\n      VpcId: {\n        Ref: Match.exact(vpcId)\n      }\n    }\n  });\n  const [subnet1, subnet2] = Object.keys(subnetResources);\n\n  // Verify the Lambda Function has the same Security Group\n  expect(stack).toHaveResourceLike('AWS::Lambda::Function', {\n    VpcConfig: {\n      SecurityGroupIds: [\n        {\n          'Fn::GetAtt': [\n            securityGroupId,\n            'GroupId'\n          ]\n        }\n      ],\n      SubnetIds: [\n        {\n          Ref: subnet1\n        },\n        {\n          Ref: subnet2\n        }\n      ]\n    }\n  });\n\n  // Verify the VPC has an interface endpoint for Kinesis Streams\n  expect(stack).toHaveResourceLike('AWS::EC2::VPCEndpoint', {\n    ServiceName: {\n      'Fn::Join': [\n        '',\n        [\n          'com.amazonaws.',\n          {\n            Ref: 'AWS::Region'\n          },\n          '.kinesis-streams'\n        ]\n      ]\n    },\n    VpcId: {\n      Ref: vpcId\n    },\n  });\n\n  // Verify the VPC has dns hostnames and support enabled\n  expect(stack).toHaveResource('AWS::EC2::VPC', {\n    EnableDnsHostnames: true,\n    EnableDnsSupport: true,\n  });\n}"]}
|