@digitraffic/common 2022.10.5-1
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/LICENSE +291 -0
- package/aws/infra/api/integration.d.ts +21 -0
- package/aws/infra/api/integration.js +52 -0
- package/aws/infra/api/response.d.ts +22 -0
- package/aws/infra/api/response.js +61 -0
- package/aws/infra/api/responses.d.ts +39 -0
- package/aws/infra/api/responses.js +79 -0
- package/aws/infra/api/static-integration.d.ts +15 -0
- package/aws/infra/api/static-integration.js +54 -0
- package/aws/infra/canaries/canary-alarm.d.ts +6 -0
- package/aws/infra/canaries/canary-alarm.js +26 -0
- package/aws/infra/canaries/canary-parameters.d.ts +18 -0
- package/aws/infra/canaries/canary-parameters.js +3 -0
- package/aws/infra/canaries/canary-role.d.ts +6 -0
- package/aws/infra/canaries/canary-role.js +46 -0
- package/aws/infra/canaries/canary.d.ts +8 -0
- package/aws/infra/canaries/canary.js +29 -0
- package/aws/infra/canaries/database-canary.d.ts +18 -0
- package/aws/infra/canaries/database-canary.js +55 -0
- package/aws/infra/canaries/database-checker.d.ts +21 -0
- package/aws/infra/canaries/database-checker.js +109 -0
- package/aws/infra/canaries/url-canary.d.ts +19 -0
- package/aws/infra/canaries/url-canary.js +46 -0
- package/aws/infra/canaries/url-checker.d.ts +46 -0
- package/aws/infra/canaries/url-checker.js +238 -0
- package/aws/infra/documentation.d.ts +56 -0
- package/aws/infra/documentation.js +95 -0
- package/aws/infra/scheduler.d.ts +12 -0
- package/aws/infra/scheduler.js +31 -0
- package/aws/infra/security-rule.d.ts +12 -0
- package/aws/infra/security-rule.js +39 -0
- package/aws/infra/sqs-integration.d.ts +7 -0
- package/aws/infra/sqs-integration.js +93 -0
- package/aws/infra/sqs-queue.d.ts +16 -0
- package/aws/infra/sqs-queue.js +130 -0
- package/aws/infra/stack/lambda-configs.d.ts +72 -0
- package/aws/infra/stack/lambda-configs.js +93 -0
- package/aws/infra/stack/monitoredfunction.d.ts +84 -0
- package/aws/infra/stack/monitoredfunction.js +135 -0
- package/aws/infra/stack/rest_apis.d.ts +40 -0
- package/aws/infra/stack/rest_apis.js +179 -0
- package/aws/infra/stack/stack-checking-aspect.d.ts +20 -0
- package/aws/infra/stack/stack-checking-aspect.js +163 -0
- package/aws/infra/stack/stack.d.ts +41 -0
- package/aws/infra/stack/stack.js +58 -0
- package/aws/infra/stack/subscription.d.ts +17 -0
- package/aws/infra/stack/subscription.js +41 -0
- package/aws/infra/usage-plans.d.ts +15 -0
- package/aws/infra/usage-plans.js +42 -0
- package/aws/runtime/apikey.d.ts +2 -0
- package/aws/runtime/apikey.js +13 -0
- package/aws/runtime/digitraffic-integration-response.d.ts +8 -0
- package/aws/runtime/digitraffic-integration-response.js +26 -0
- package/aws/runtime/messaging.d.ts +10 -0
- package/aws/runtime/messaging.js +31 -0
- package/aws/runtime/s3.d.ts +2 -0
- package/aws/runtime/s3.js +30 -0
- package/aws/runtime/secrets/dbsecret.d.ts +54 -0
- package/aws/runtime/secrets/dbsecret.js +96 -0
- package/aws/runtime/secrets/proxy-holder.d.ts +9 -0
- package/aws/runtime/secrets/proxy-holder.js +26 -0
- package/aws/runtime/secrets/rds-holder.d.ts +9 -0
- package/aws/runtime/secrets/rds-holder.js +26 -0
- package/aws/runtime/secrets/secret-holder.d.ts +26 -0
- package/aws/runtime/secrets/secret-holder.js +73 -0
- package/aws/runtime/secrets/secret.d.ts +8 -0
- package/aws/runtime/secrets/secret.js +43 -0
- package/aws/types/errors.d.ts +4 -0
- package/aws/types/errors.js +9 -0
- package/aws/types/lambda-response.d.ts +12 -0
- package/aws/types/lambda-response.js +28 -0
- package/aws/types/mediatypes.d.ts +10 -0
- package/aws/types/mediatypes.js +15 -0
- package/aws/types/model-with-reference.d.ts +7 -0
- package/aws/types/model-with-reference.js +3 -0
- package/aws/types/proxytypes.d.ts +26 -0
- package/aws/types/proxytypes.js +3 -0
- package/aws/types/tags.d.ts +2 -0
- package/aws/types/tags.js +7 -0
- package/database/cached.d.ts +7 -0
- package/database/cached.js +32 -0
- package/database/database.d.ts +19 -0
- package/database/database.js +62 -0
- package/database/last-updated.d.ts +16 -0
- package/database/last-updated.js +54 -0
- package/index.d.ts +1 -0
- package/index.js +18 -0
- package/marine/id_utils.d.ts +3 -0
- package/marine/id_utils.js +33 -0
- package/marine/rtz.d.ts +48 -0
- package/marine/rtz.js +3 -0
- package/package.json +55 -0
- package/src/aws/infra/api/integration.js +52 -0
- package/src/aws/infra/api/response.js +61 -0
- package/src/aws/infra/api/responses.js +79 -0
- package/src/aws/infra/api/static-integration.js +54 -0
- package/src/aws/infra/canaries/canary-alarm.js +26 -0
- package/src/aws/infra/canaries/canary-parameters.js +3 -0
- package/src/aws/infra/canaries/canary-role.js +46 -0
- package/src/aws/infra/canaries/canary.js +29 -0
- package/src/aws/infra/canaries/database-canary.js +55 -0
- package/src/aws/infra/canaries/database-checker.js +109 -0
- package/src/aws/infra/canaries/url-canary.js +46 -0
- package/src/aws/infra/canaries/url-checker.js +238 -0
- package/src/aws/infra/documentation.js +95 -0
- package/src/aws/infra/scheduler.js +31 -0
- package/src/aws/infra/security-rule.js +39 -0
- package/src/aws/infra/sqs-integration.js +93 -0
- package/src/aws/infra/sqs-queue.js +130 -0
- package/src/aws/infra/stack/lambda-configs.js +93 -0
- package/src/aws/infra/stack/monitoredfunction.js +135 -0
- package/src/aws/infra/stack/rest_apis.js +179 -0
- package/src/aws/infra/stack/stack-checking-aspect.js +163 -0
- package/src/aws/infra/stack/stack.js +58 -0
- package/src/aws/infra/stack/subscription.js +41 -0
- package/src/aws/infra/usage-plans.js +42 -0
- package/src/aws/runtime/apikey.js +13 -0
- package/src/aws/runtime/digitraffic-integration-response.js +26 -0
- package/src/aws/runtime/messaging.js +31 -0
- package/src/aws/runtime/s3.js +30 -0
- package/src/aws/runtime/secrets/dbsecret.js +96 -0
- package/src/aws/runtime/secrets/proxy-holder.js +26 -0
- package/src/aws/runtime/secrets/rds-holder.js +26 -0
- package/src/aws/runtime/secrets/secret-holder.js +73 -0
- package/src/aws/runtime/secrets/secret.js +43 -0
- package/src/aws/types/errors.js +9 -0
- package/src/aws/types/lambda-response.js +28 -0
- package/src/aws/types/mediatypes.js +15 -0
- package/src/aws/types/model-with-reference.js +3 -0
- package/src/aws/types/proxytypes.js +3 -0
- package/src/aws/types/tags.js +7 -0
- package/src/database/cached.js +32 -0
- package/src/database/database.js +62 -0
- package/src/database/last-updated.js +54 -0
- package/src/marine/id_utils.js +33 -0
- package/src/marine/rtz.js +3 -0
- package/src/test/asserter.js +45 -0
- package/src/test/db-testutils.js +31 -0
- package/src/test/httpserver.js +67 -0
- package/src/test/secret.js +25 -0
- package/src/test/secrets-manager.js +59 -0
- package/src/test/testutils.js +44 -0
- package/src/types/input-error.js +7 -0
- package/src/types/language.js +10 -0
- package/src/types/traffictype.js +13 -0
- package/src/types/validator.js +14 -0
- package/src/utils/api-model.js +129 -0
- package/src/utils/base64.js +21 -0
- package/src/utils/date-utils.js +34 -0
- package/src/utils/geojson-types.js +18 -0
- package/src/utils/geometry.js +140 -0
- package/src/utils/retry.js +50 -0
- package/src/utils/slack.js +25 -0
- package/src/utils/utils.js +40 -0
- package/test/asserter.d.ts +11 -0
- package/test/asserter.js +45 -0
- package/test/db-testutils.d.ts +2 -0
- package/test/db-testutils.js +31 -0
- package/test/httpserver.d.ts +18 -0
- package/test/httpserver.js +67 -0
- package/test/marine/id_utils.test.js +69 -0
- package/test/promise/promise.test.js +125 -0
- package/test/secret.d.ts +3 -0
- package/test/secret.js +25 -0
- package/test/secrets/dbsecret.test.js +71 -0
- package/test/secrets/secret-holder.test.js +124 -0
- package/test/secrets/secret.test.js +66 -0
- package/test/secrets-manager.d.ts +9 -0
- package/test/secrets-manager.js +59 -0
- package/test/test/httpserver.test.js +87 -0
- package/test/testutils.d.ts +12 -0
- package/test/testutils.js +44 -0
- package/test/utils/date-utils.test.js +51 -0
- package/test/utils/geometry.test.js +49 -0
- package/test/utils/utils.test.js +49 -0
- package/types/input-error.d.ts +2 -0
- package/types/input-error.js +7 -0
- package/types/language.d.ts +5 -0
- package/types/language.js +10 -0
- package/types/traffictype.d.ts +8 -0
- package/types/traffictype.js +13 -0
- package/types/validator.d.ts +4 -0
- package/types/validator.js +14 -0
- package/utils/api-model.d.ts +87 -0
- package/utils/api-model.js +129 -0
- package/utils/base64.d.ts +12 -0
- package/utils/base64.js +21 -0
- package/utils/date-utils.d.ts +17 -0
- package/utils/date-utils.js +34 -0
- package/utils/geojson-types.d.ts +14 -0
- package/utils/geojson-types.js +18 -0
- package/utils/geometry.d.ts +36 -0
- package/utils/geometry.js +140 -0
- package/utils/retry.d.ts +13 -0
- package/utils/retry.js +50 -0
- package/utils/slack.d.ts +5 -0
- package/utils/slack.js +25 -0
- package/utils/utils.d.ts +22 -0
- package/utils/utils.js +40 -0
- package/yarn.lock +3200 -0
|
@@ -0,0 +1,93 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.attachQueueToApiGatewayResource = void 0;
|
|
4
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
5
|
+
const aws_apigateway_1 = require("aws-cdk-lib/aws-apigateway");
|
|
6
|
+
const aws_iam_1 = require("aws-cdk-lib/aws-iam");
|
|
7
|
+
function attachQueueToApiGatewayResource(stack, queue, resource, requestValidator, resourceName, apiKeyRequired, requestModels) {
|
|
8
|
+
// role for API Gateway
|
|
9
|
+
const apiGwRole = new aws_iam_1.Role(stack, `${resourceName}APIGatewayToSQSRole`, {
|
|
10
|
+
assumedBy: new aws_iam_1.ServicePrincipal('apigateway.amazonaws.com'),
|
|
11
|
+
});
|
|
12
|
+
// grants API Gateway the right to send SQS messages
|
|
13
|
+
apiGwRole.addToPolicy(new aws_iam_1.PolicyStatement({
|
|
14
|
+
resources: [
|
|
15
|
+
queue.queueArn,
|
|
16
|
+
],
|
|
17
|
+
actions: [
|
|
18
|
+
'sqs:SendMessage',
|
|
19
|
+
],
|
|
20
|
+
}));
|
|
21
|
+
// grants API Gateway the right write CloudWatch Logs
|
|
22
|
+
apiGwRole.addToPolicy(new aws_iam_1.PolicyStatement({
|
|
23
|
+
resources: [
|
|
24
|
+
'*',
|
|
25
|
+
],
|
|
26
|
+
actions: [
|
|
27
|
+
'logs:CreateLogGroup',
|
|
28
|
+
'logs:CreateLogStream',
|
|
29
|
+
'logs:DescribeLogGroups',
|
|
30
|
+
'logs:DescribeLogStreams',
|
|
31
|
+
'logs:PutLogEvents',
|
|
32
|
+
'logs:GetLogEvents',
|
|
33
|
+
'logs:FilterLogEvents',
|
|
34
|
+
],
|
|
35
|
+
}));
|
|
36
|
+
// create an integration between API Gateway and an SQS queue
|
|
37
|
+
const fifoMessageGroupId = queue.fifo ? '&MessageGroupId=AlwaysSameFifoGroup' : '';
|
|
38
|
+
const sqsIntegration = new aws_apigateway_1.AwsIntegration({
|
|
39
|
+
service: 'sqs',
|
|
40
|
+
integrationHttpMethod: 'POST',
|
|
41
|
+
options: {
|
|
42
|
+
passthroughBehavior: aws_apigateway_1.PassthroughBehavior.NEVER,
|
|
43
|
+
credentialsRole: apiGwRole,
|
|
44
|
+
requestParameters: {
|
|
45
|
+
// SQS requires the Content-Type of the HTTP request to be application/x-www-form-urlencoded
|
|
46
|
+
'integration.request.header.Content-Type': "'application/x-www-form-urlencoded'",
|
|
47
|
+
},
|
|
48
|
+
requestTemplates: {
|
|
49
|
+
// map the JSON request to a form parameter, FIFO needs also MessageGroupId
|
|
50
|
+
// https://docs.aws.amazon.com/AWSSimpleQueueService/latest/APIReference/API_SendMessage.html
|
|
51
|
+
'application/json': `Action=SendMessage${fifoMessageGroupId}&MessageBody=$util.urlEncode($input.body)`,
|
|
52
|
+
},
|
|
53
|
+
// these are required by SQS
|
|
54
|
+
integrationResponses: [
|
|
55
|
+
{
|
|
56
|
+
statusCode: '200',
|
|
57
|
+
responseTemplates: {
|
|
58
|
+
'text/html': 'Success',
|
|
59
|
+
},
|
|
60
|
+
},
|
|
61
|
+
{
|
|
62
|
+
statusCode: '500',
|
|
63
|
+
responseTemplates: {
|
|
64
|
+
'text/html': 'Error',
|
|
65
|
+
},
|
|
66
|
+
selectionPattern: '500',
|
|
67
|
+
},
|
|
68
|
+
],
|
|
69
|
+
},
|
|
70
|
+
path: `${aws_cdk_lib_1.Aws.ACCOUNT_ID}/${queue.queueName}`,
|
|
71
|
+
});
|
|
72
|
+
resource.addMethod('POST', sqsIntegration, {
|
|
73
|
+
requestValidator,
|
|
74
|
+
apiKeyRequired,
|
|
75
|
+
requestModels: requestModels ?? {},
|
|
76
|
+
methodResponses: [
|
|
77
|
+
{
|
|
78
|
+
statusCode: '200',
|
|
79
|
+
responseParameters: {
|
|
80
|
+
'method.response.header.Content-Type': true,
|
|
81
|
+
},
|
|
82
|
+
},
|
|
83
|
+
{
|
|
84
|
+
statusCode: '500',
|
|
85
|
+
responseParameters: {
|
|
86
|
+
'method.response.header.Content-Type': true,
|
|
87
|
+
},
|
|
88
|
+
},
|
|
89
|
+
],
|
|
90
|
+
});
|
|
91
|
+
}
|
|
92
|
+
exports.attachQueueToApiGatewayResource = attachQueueToApiGatewayResource;
|
|
93
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3FzLWludGVncmF0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2F3cy9pbmZyYS9zcXMtaW50ZWdyYXRpb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsNkNBQWdDO0FBQ2hDLCtEQUEyRztBQUUzRyxpREFBNEU7QUFJNUUsU0FBZ0IsK0JBQStCLENBQzNDLEtBQWdCLEVBQ2hCLEtBQVksRUFDWixRQUFrQixFQUNsQixnQkFBa0MsRUFDbEMsWUFBb0IsRUFDcEIsY0FBdUIsRUFDdkIsYUFBeUM7SUFFekMsdUJBQXVCO0lBQ3ZCLE1BQU0sU0FBUyxHQUFHLElBQUksY0FBSSxDQUFDLEtBQUssRUFBRSxHQUFHLFlBQVkscUJBQXFCLEVBQUU7UUFDcEUsU0FBUyxFQUFFLElBQUksMEJBQWdCLENBQUMsMEJBQTBCLENBQUM7S0FDOUQsQ0FBQyxDQUFDO0lBQ0gsb0RBQW9EO0lBQ3BELFNBQVMsQ0FBQyxXQUFXLENBQUMsSUFBSSx5QkFBZSxDQUFDO1FBQ3RDLFNBQVMsRUFBRTtZQUNQLEtBQUssQ0FBQyxRQUFRO1NBQ2pCO1FBQ0QsT0FBTyxFQUFFO1lBQ0wsaUJBQWlCO1NBQ3BCO0tBQ0osQ0FBQyxDQUFDLENBQUM7SUFDSixxREFBcUQ7SUFDckQsU0FBUyxDQUFDLFdBQVcsQ0FBQyxJQUFJLHlCQUFlLENBQUM7UUFDdEMsU0FBUyxFQUFFO1lBQ1AsR0FBRztTQUNOO1FBQ0QsT0FBTyxFQUFFO1lBQ0wscUJBQXFCO1lBQ3JCLHNCQUFzQjtZQUN0Qix3QkFBd0I7WUFDeEIseUJBQXlCO1lBQ3pCLG1CQUFtQjtZQUNuQixtQkFBbUI7WUFDbkIsc0JBQXNCO1NBQ3pCO0tBQ0osQ0FBQyxDQUFDLENBQUM7SUFDSiw2REFBNkQ7SUFDN0QsTUFBTSxrQkFBa0IsR0FBRyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxxQ0FBcUMsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDO0lBQ25GLE1BQU0sY0FBYyxHQUFHLElBQUksK0JBQWMsQ0FBQztRQUN0QyxPQUFPLEVBQUUsS0FBSztRQUNkLHFCQUFxQixFQUFFLE1BQU07UUFDN0IsT0FBTyxFQUFFO1lBQ0wsbUJBQW1CLEVBQUUsb0NBQW1CLENBQUMsS0FBSztZQUM5QyxlQUFlLEVBQUUsU0FBUztZQUMxQixpQkFBaUIsRUFBRTtnQkFDZiw0RkFBNEY7Z0JBQzVGLHlDQUF5QyxFQUFFLHFDQUFxQzthQUNuRjtZQUNELGdCQUFnQixFQUFFO2dCQUNkLDJFQUEyRTtnQkFDM0UsNkZBQTZGO2dCQUM3RixrQkFBa0IsRUFBRSxxQkFBcUIsa0JBQWtCLDJDQUEyQzthQUN6RztZQUNELDRCQUE0QjtZQUM1QixvQkFBb0IsRUFBRTtnQkFDbEI7b0JBQ0ksVUFBVSxFQUFFLEtBQUs7b0JBQ2pCLGlCQUFpQixFQUFFO3dCQUNmLFdBQVcsRUFBRSxTQUFTO3FCQUN6QjtpQkFDSjtnQkFDRDtvQkFDSSxVQUFVLEVBQUUsS0FBSztvQkFDakIsaUJBQWlCLEVBQUU7d0JBQ2YsV0FBVyxFQUFFLE9BQU87cUJBQ3ZCO29CQUNELGdCQUFnQixFQUFFLEtBQUs7aUJBQzFCO2FBRUo7U0FFSjtRQUNELElBQUksRUFBRSxHQUFHLGlCQUFHLENBQUMsVUFBVSxJQUFJLEtBQUssQ0FBQyxTQUFTLEVBQUU7S0FDL0MsQ0FBQyxDQUFDO0lBQ0gsUUFBUSxDQUFDLFNBQVMsQ0FBQyxNQUFNLEVBQUUsY0FBYyxFQUFFO1FBQ3ZDLGdCQUFnQjtRQUNoQixjQUFjO1FBQ2QsYUFBYSxFQUFFLGFBQWEsSUFBSSxFQUFFO1FBQ2xDLGVBQWUsRUFBRTtZQUNiO2dCQUNJLFVBQVUsRUFBRSxLQUFLO2dCQUNqQixrQkFBa0IsRUFBRTtvQkFDaEIscUNBQXFDLEVBQUUsSUFBSTtpQkFDOUM7YUFDSjtZQUNEO2dCQUNJLFVBQVUsRUFBRSxLQUFLO2dCQUNqQixrQkFBa0IsRUFBRTtvQkFDaEIscUNBQXFDLEVBQUUsSUFBSTtpQkFDOUM7YUFDSjtTQUNKO0tBQ0osQ0FBQyxDQUFDO0FBQ1AsQ0FBQztBQTlGRCwwRUE4RkMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0F3c30gZnJvbSBcImF3cy1jZGstbGliXCI7XG5pbXBvcnQge0F3c0ludGVncmF0aW9uLCBQYXNzdGhyb3VnaEJlaGF2aW9yLCBSZXF1ZXN0VmFsaWRhdG9yLCBSZXNvdXJjZX0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1hcGlnYXRld2F5XCI7XG5pbXBvcnQge1F1ZXVlfSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLXNxc1wiO1xuaW1wb3J0IHtQb2xpY3lTdGF0ZW1lbnQsIFJvbGUsIFNlcnZpY2VQcmluY2lwYWx9IGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtaWFtXCI7XG5pbXBvcnQge0lNb2RlbH0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1hcGlnYXRld2F5L2xpYi9tb2RlbFwiO1xuaW1wb3J0IHtDb25zdHJ1Y3R9IGZyb20gXCJjb25zdHJ1Y3RzXCI7XG5cbmV4cG9ydCBmdW5jdGlvbiBhdHRhY2hRdWV1ZVRvQXBpR2F0ZXdheVJlc291cmNlKFxuICAgIHN0YWNrOiBDb25zdHJ1Y3QsXG4gICAgcXVldWU6IFF1ZXVlLFxuICAgIHJlc291cmNlOiBSZXNvdXJjZSxcbiAgICByZXF1ZXN0VmFsaWRhdG9yOiBSZXF1ZXN0VmFsaWRhdG9yLFxuICAgIHJlc291cmNlTmFtZTogc3RyaW5nLFxuICAgIGFwaUtleVJlcXVpcmVkOiBib29sZWFuLFxuICAgIHJlcXVlc3RNb2RlbHM/OiB7W3BhcmFtOiBzdHJpbmddOiBJTW9kZWx9LFxuKSB7XG4gICAgLy8gcm9sZSBmb3IgQVBJIEdhdGV3YXlcbiAgICBjb25zdCBhcGlHd1JvbGUgPSBuZXcgUm9sZShzdGFjaywgYCR7cmVzb3VyY2VOYW1lfUFQSUdhdGV3YXlUb1NRU1JvbGVgLCB7XG4gICAgICAgIGFzc3VtZWRCeTogbmV3IFNlcnZpY2VQcmluY2lwYWwoJ2FwaWdhdGV3YXkuYW1hem9uYXdzLmNvbScpLFxuICAgIH0pO1xuICAgIC8vIGdyYW50cyBBUEkgR2F0ZXdheSB0aGUgcmlnaHQgdG8gc2VuZCBTUVMgbWVzc2FnZXNcbiAgICBhcGlHd1JvbGUuYWRkVG9Qb2xpY3kobmV3IFBvbGljeVN0YXRlbWVudCh7XG4gICAgICAgIHJlc291cmNlczogW1xuICAgICAgICAgICAgcXVldWUucXVldWVBcm4sXG4gICAgICAgIF0sXG4gICAgICAgIGFjdGlvbnM6IFtcbiAgICAgICAgICAgICdzcXM6U2VuZE1lc3NhZ2UnLFxuICAgICAgICBdLFxuICAgIH0pKTtcbiAgICAvLyBncmFudHMgQVBJIEdhdGV3YXkgdGhlIHJpZ2h0IHdyaXRlIENsb3VkV2F0Y2ggTG9nc1xuICAgIGFwaUd3Um9sZS5hZGRUb1BvbGljeShuZXcgUG9saWN5U3RhdGVtZW50KHtcbiAgICAgICAgcmVzb3VyY2VzOiBbXG4gICAgICAgICAgICAnKicsXG4gICAgICAgIF0sXG4gICAgICAgIGFjdGlvbnM6IFtcbiAgICAgICAgICAgICdsb2dzOkNyZWF0ZUxvZ0dyb3VwJyxcbiAgICAgICAgICAgICdsb2dzOkNyZWF0ZUxvZ1N0cmVhbScsXG4gICAgICAgICAgICAnbG9nczpEZXNjcmliZUxvZ0dyb3VwcycsXG4gICAgICAgICAgICAnbG9nczpEZXNjcmliZUxvZ1N0cmVhbXMnLFxuICAgICAgICAgICAgJ2xvZ3M6UHV0TG9nRXZlbnRzJyxcbiAgICAgICAgICAgICdsb2dzOkdldExvZ0V2ZW50cycsXG4gICAgICAgICAgICAnbG9nczpGaWx0ZXJMb2dFdmVudHMnLFxuICAgICAgICBdLFxuICAgIH0pKTtcbiAgICAvLyBjcmVhdGUgYW4gaW50ZWdyYXRpb24gYmV0d2VlbiBBUEkgR2F0ZXdheSBhbmQgYW4gU1FTIHF1ZXVlXG4gICAgY29uc3QgZmlmb01lc3NhZ2VHcm91cElkID0gcXVldWUuZmlmbyA/ICcmTWVzc2FnZUdyb3VwSWQ9QWx3YXlzU2FtZUZpZm9Hcm91cCcgOiAnJztcbiAgICBjb25zdCBzcXNJbnRlZ3JhdGlvbiA9IG5ldyBBd3NJbnRlZ3JhdGlvbih7XG4gICAgICAgIHNlcnZpY2U6ICdzcXMnLFxuICAgICAgICBpbnRlZ3JhdGlvbkh0dHBNZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgb3B0aW9uczoge1xuICAgICAgICAgICAgcGFzc3Rocm91Z2hCZWhhdmlvcjogUGFzc3Rocm91Z2hCZWhhdmlvci5ORVZFUixcbiAgICAgICAgICAgIGNyZWRlbnRpYWxzUm9sZTogYXBpR3dSb2xlLFxuICAgICAgICAgICAgcmVxdWVzdFBhcmFtZXRlcnM6IHtcbiAgICAgICAgICAgICAgICAvLyBTUVMgcmVxdWlyZXMgdGhlIENvbnRlbnQtVHlwZSBvZiB0aGUgSFRUUCByZXF1ZXN0IHRvIGJlIGFwcGxpY2F0aW9uL3gtd3d3LWZvcm0tdXJsZW5jb2RlZFxuICAgICAgICAgICAgICAgICdpbnRlZ3JhdGlvbi5yZXF1ZXN0LmhlYWRlci5Db250ZW50LVR5cGUnOiBcIidhcHBsaWNhdGlvbi94LXd3dy1mb3JtLXVybGVuY29kZWQnXCIsXG4gICAgICAgICAgICB9LFxuICAgICAgICAgICAgcmVxdWVzdFRlbXBsYXRlczoge1xuICAgICAgICAgICAgICAgIC8vIG1hcCB0aGUgSlNPTiByZXF1ZXN0IHRvIGEgZm9ybSBwYXJhbWV0ZXIsIEZJRk8gbmVlZHMgYWxzbyBNZXNzYWdlR3JvdXBJZFxuICAgICAgICAgICAgICAgIC8vIGh0dHBzOi8vZG9jcy5hd3MuYW1hem9uLmNvbS9BV1NTaW1wbGVRdWV1ZVNlcnZpY2UvbGF0ZXN0L0FQSVJlZmVyZW5jZS9BUElfU2VuZE1lc3NhZ2UuaHRtbFxuICAgICAgICAgICAgICAgICdhcHBsaWNhdGlvbi9qc29uJzogYEFjdGlvbj1TZW5kTWVzc2FnZSR7Zmlmb01lc3NhZ2VHcm91cElkfSZNZXNzYWdlQm9keT0kdXRpbC51cmxFbmNvZGUoJGlucHV0LmJvZHkpYCxcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAvLyB0aGVzZSBhcmUgcmVxdWlyZWQgYnkgU1FTXG4gICAgICAgICAgICBpbnRlZ3JhdGlvblJlc3BvbnNlczogW1xuICAgICAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgICAgICAgc3RhdHVzQ29kZTogJzIwMCcsXG4gICAgICAgICAgICAgICAgICAgIHJlc3BvbnNlVGVtcGxhdGVzOiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAndGV4dC9odG1sJzogJ1N1Y2Nlc3MnLFxuICAgICAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgICAgICBzdGF0dXNDb2RlOiAnNTAwJyxcbiAgICAgICAgICAgICAgICAgICAgcmVzcG9uc2VUZW1wbGF0ZXM6IHtcbiAgICAgICAgICAgICAgICAgICAgICAgICd0ZXh0L2h0bWwnOiAnRXJyb3InLFxuICAgICAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICAgICAgICBzZWxlY3Rpb25QYXR0ZXJuOiAnNTAwJyxcbiAgICAgICAgICAgICAgICB9LFxuXG4gICAgICAgICAgICBdLFxuXG4gICAgICAgIH0sXG4gICAgICAgIHBhdGg6IGAke0F3cy5BQ0NPVU5UX0lEfS8ke3F1ZXVlLnF1ZXVlTmFtZX1gLFxuICAgIH0pO1xuICAgIHJlc291cmNlLmFkZE1ldGhvZCgnUE9TVCcsIHNxc0ludGVncmF0aW9uLCB7XG4gICAgICAgIHJlcXVlc3RWYWxpZGF0b3IsXG4gICAgICAgIGFwaUtleVJlcXVpcmVkLFxuICAgICAgICByZXF1ZXN0TW9kZWxzOiByZXF1ZXN0TW9kZWxzID8/IHt9LFxuICAgICAgICBtZXRob2RSZXNwb25zZXM6IFtcbiAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgICBzdGF0dXNDb2RlOiAnMjAwJyxcbiAgICAgICAgICAgICAgICByZXNwb25zZVBhcmFtZXRlcnM6IHtcbiAgICAgICAgICAgICAgICAgICAgJ21ldGhvZC5yZXNwb25zZS5oZWFkZXIuQ29udGVudC1UeXBlJzogdHJ1ZSxcbiAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgICBzdGF0dXNDb2RlOiAnNTAwJyxcbiAgICAgICAgICAgICAgICByZXNwb25zZVBhcmFtZXRlcnM6IHtcbiAgICAgICAgICAgICAgICAgICAgJ21ldGhvZC5yZXNwb25zZS5oZWFkZXIuQ29udGVudC1UeXBlJzogdHJ1ZSxcbiAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgfSxcbiAgICAgICAgXSxcbiAgICB9KTtcbn1cbiJdfQ==
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import { Queue, QueueProps } from "aws-cdk-lib/aws-sqs";
|
|
2
|
+
import { Construct } from "constructs";
|
|
3
|
+
import { DigitrafficStack } from "./stack/stack";
|
|
4
|
+
/**
|
|
5
|
+
* Construct for creating SQS-queues.
|
|
6
|
+
*
|
|
7
|
+
* If you don't config your own deadLetterQueue, this will create a dlq for you, also a lambda function, a s3 bucket
|
|
8
|
+
* and an alarm for the queue. Anything that goes to the dlq will be written into the bucket and the alarm is activated.
|
|
9
|
+
*/
|
|
10
|
+
export declare class DigitrafficSqsQueue extends Queue {
|
|
11
|
+
constructor(scope: Construct, name: string, props: QueueProps);
|
|
12
|
+
static create(stack: DigitrafficStack, name: string, props: QueueProps): DigitrafficSqsQueue;
|
|
13
|
+
}
|
|
14
|
+
export declare class DigitrafficDLQueue {
|
|
15
|
+
static create(stack: DigitrafficStack, name: string): DigitrafficSqsQueue;
|
|
16
|
+
}
|
|
@@ -0,0 +1,130 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.DigitrafficDLQueue = exports.DigitrafficSqsQueue = void 0;
|
|
4
|
+
const aws_sqs_1 = require("aws-cdk-lib/aws-sqs");
|
|
5
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
6
|
+
const aws_s3_1 = require("aws-cdk-lib/aws-s3");
|
|
7
|
+
const aws_iam_1 = require("aws-cdk-lib/aws-iam");
|
|
8
|
+
const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
|
|
9
|
+
const aws_logs_1 = require("aws-cdk-lib/aws-logs");
|
|
10
|
+
const aws_lambda_event_sources_1 = require("aws-cdk-lib/aws-lambda-event-sources");
|
|
11
|
+
const aws_cloudwatch_1 = require("aws-cdk-lib/aws-cloudwatch");
|
|
12
|
+
const aws_cloudwatch_actions_1 = require("aws-cdk-lib/aws-cloudwatch-actions");
|
|
13
|
+
const monitoredfunction_1 = require("./stack/monitoredfunction");
|
|
14
|
+
/**
|
|
15
|
+
* Construct for creating SQS-queues.
|
|
16
|
+
*
|
|
17
|
+
* If you don't config your own deadLetterQueue, this will create a dlq for you, also a lambda function, a s3 bucket
|
|
18
|
+
* and an alarm for the queue. Anything that goes to the dlq will be written into the bucket and the alarm is activated.
|
|
19
|
+
*/
|
|
20
|
+
class DigitrafficSqsQueue extends aws_sqs_1.Queue {
|
|
21
|
+
constructor(scope, name, props) {
|
|
22
|
+
super(scope, name, props);
|
|
23
|
+
}
|
|
24
|
+
static create(stack, name, props) {
|
|
25
|
+
const queueName = `${stack.configuration.shortName}-${name}-Queue`;
|
|
26
|
+
const queueProps = { ...props, ...{
|
|
27
|
+
encryption: aws_sqs_1.QueueEncryption.KMS_MANAGED,
|
|
28
|
+
queueName,
|
|
29
|
+
deadLetterQueue: props.deadLetterQueue || {
|
|
30
|
+
maxReceiveCount: 2,
|
|
31
|
+
queue: DigitrafficDLQueue.create(stack, name),
|
|
32
|
+
},
|
|
33
|
+
} };
|
|
34
|
+
return new DigitrafficSqsQueue(stack, queueName, queueProps);
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
exports.DigitrafficSqsQueue = DigitrafficSqsQueue;
|
|
38
|
+
class DigitrafficDLQueue {
|
|
39
|
+
static create(stack, name) {
|
|
40
|
+
const dlqName = `${stack.configuration.shortName}-${name}-DLQ`;
|
|
41
|
+
const dlq = new DigitrafficSqsQueue(stack, dlqName, {
|
|
42
|
+
queueName: dlqName,
|
|
43
|
+
visibilityTimeout: aws_cdk_lib_1.Duration.seconds(60),
|
|
44
|
+
encryption: aws_sqs_1.QueueEncryption.KMS_MANAGED,
|
|
45
|
+
});
|
|
46
|
+
const dlqBucket = new aws_s3_1.Bucket(stack, `${dlqName}-Bucket`, {
|
|
47
|
+
blockPublicAccess: aws_s3_1.BlockPublicAccess.BLOCK_ALL,
|
|
48
|
+
});
|
|
49
|
+
const dlqFunctionName = `${dlqName}-Function`;
|
|
50
|
+
const lambda = monitoredfunction_1.MonitoredFunction.create(stack, dlqFunctionName, {
|
|
51
|
+
runtime: aws_lambda_1.Runtime.NODEJS_14_X,
|
|
52
|
+
logRetention: aws_logs_1.RetentionDays.ONE_YEAR,
|
|
53
|
+
functionName: dlqFunctionName,
|
|
54
|
+
code: getDlqCode(dlqBucket.bucketName),
|
|
55
|
+
timeout: aws_cdk_lib_1.Duration.seconds(10),
|
|
56
|
+
handler: 'index.handler',
|
|
57
|
+
memorySize: 128,
|
|
58
|
+
reservedConcurrentExecutions: 1,
|
|
59
|
+
});
|
|
60
|
+
const statement = new aws_iam_1.PolicyStatement();
|
|
61
|
+
statement.addActions('s3:PutObject');
|
|
62
|
+
statement.addActions('s3:PutObjectAcl');
|
|
63
|
+
statement.addResources(dlqBucket.bucketArn + '/*');
|
|
64
|
+
lambda.addToRolePolicy(statement);
|
|
65
|
+
lambda.addEventSource(new aws_lambda_event_sources_1.SqsEventSource(dlq));
|
|
66
|
+
addDLQAlarm(stack, dlqName, dlq);
|
|
67
|
+
return dlq;
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
exports.DigitrafficDLQueue = DigitrafficDLQueue;
|
|
71
|
+
function addDLQAlarm(stack, dlqName, dlq) {
|
|
72
|
+
const alarmName = `${dlqName}-Alarm`;
|
|
73
|
+
dlq.metricNumberOfMessagesReceived({
|
|
74
|
+
period: aws_cdk_lib_1.Duration.minutes(5),
|
|
75
|
+
}).createAlarm(stack, alarmName, {
|
|
76
|
+
alarmName,
|
|
77
|
+
threshold: 0,
|
|
78
|
+
evaluationPeriods: 1,
|
|
79
|
+
treatMissingData: aws_cloudwatch_1.TreatMissingData.NOT_BREACHING,
|
|
80
|
+
comparisonOperator: aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_THRESHOLD,
|
|
81
|
+
}).addAlarmAction(new aws_cloudwatch_actions_1.SnsAction(stack.warningTopic));
|
|
82
|
+
}
|
|
83
|
+
function getDlqCode(bName) {
|
|
84
|
+
const functionBody = DLQ_LAMBDA_CODE
|
|
85
|
+
.replace("__bucketName__", bName)
|
|
86
|
+
.replace("__upload__", uploadToS3.toString())
|
|
87
|
+
.replace("__doUpload__", doUpload.toString())
|
|
88
|
+
.replace("__handler__", createHandler().toString().substring(23)); // remove function handler() from signature
|
|
89
|
+
return new aws_lambda_1.InlineCode(functionBody);
|
|
90
|
+
}
|
|
91
|
+
async function uploadToS3(s3, bName, body, objectName) {
|
|
92
|
+
try {
|
|
93
|
+
console.info('writing %s to %s', objectName, bName);
|
|
94
|
+
await doUpload(s3, bName, body, objectName);
|
|
95
|
+
}
|
|
96
|
+
catch (error) {
|
|
97
|
+
console.warn(error);
|
|
98
|
+
console.warn('method=uploadToS3 retrying upload to bucket %s', bName);
|
|
99
|
+
try {
|
|
100
|
+
await doUpload(s3, bName, body, objectName);
|
|
101
|
+
}
|
|
102
|
+
catch (e2) {
|
|
103
|
+
console.error('method=uploadToS3 failed retrying upload to bucket %s', bName);
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
function doUpload(s3, bName, Body, Key) {
|
|
108
|
+
return s3.upload({
|
|
109
|
+
Bucket: bName, Body, Key,
|
|
110
|
+
}).promise();
|
|
111
|
+
}
|
|
112
|
+
// bucketName is unused, will be overridden in the actual lambda code below
|
|
113
|
+
const bucketName = '';
|
|
114
|
+
function createHandler() {
|
|
115
|
+
return async function handler(event) {
|
|
116
|
+
// eslint-disable-next-line @typescript-eslint/no-var-requires
|
|
117
|
+
const AWS = require('aws-sdk');
|
|
118
|
+
const millis = new Date().getTime();
|
|
119
|
+
await Promise.all(event.Records.map((e, idx) => uploadToS3(new AWS.S3(), bucketName, e.body, `dlq-${millis}-${idx}.json`)));
|
|
120
|
+
};
|
|
121
|
+
}
|
|
122
|
+
const DLQ_LAMBDA_CODE = `const AWS = require('aws-sdk');
|
|
123
|
+
const bucketName = "__bucketName__";
|
|
124
|
+
|
|
125
|
+
__upload__
|
|
126
|
+
__doUpload__
|
|
127
|
+
|
|
128
|
+
exports.handler = async (event) => __handler__
|
|
129
|
+
`;
|
|
130
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sqs-queue.js","sourceRoot":"","sources":["../../../src/aws/infra/sqs-queue.ts"],"names":[],"mappings":"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","sourcesContent":["import {Queue, QueueEncryption, QueueProps} from \"aws-cdk-lib/aws-sqs\";\nimport {Duration} from \"aws-cdk-lib\";\nimport {BlockPublicAccess, Bucket} from \"aws-cdk-lib/aws-s3\";\nimport {PolicyStatement} from \"aws-cdk-lib/aws-iam\";\nimport {InlineCode, Runtime} from \"aws-cdk-lib/aws-lambda\";\nimport {RetentionDays} from \"aws-cdk-lib/aws-logs\";\nimport {SqsEventSource} from \"aws-cdk-lib/aws-lambda-event-sources\";\nimport {ComparisonOperator, TreatMissingData} from \"aws-cdk-lib/aws-cloudwatch\";\nimport {SnsAction} from \"aws-cdk-lib/aws-cloudwatch-actions\";\nimport {ManagedUpload} from \"aws-sdk/clients/s3\";\nimport {S3} from \"aws-sdk\";\nimport {SQSEvent, SQSHandler, SQSRecord} from \"aws-lambda\";\nimport {Construct} from \"constructs\";\nimport {DigitrafficStack} from \"./stack/stack\";\nimport {MonitoredFunction} from \"./stack/monitoredfunction\";\n\n/**\n * Construct for creating SQS-queues.\n *\n * If you don't config your own deadLetterQueue, this will create a dlq for you, also a lambda function, a s3 bucket\n * and an alarm for the queue.  Anything that goes to the dlq will be written into the bucket and the alarm is activated.\n */\nexport class DigitrafficSqsQueue extends Queue {\n    constructor(scope: Construct, name: string, props: QueueProps) {\n        super(scope, name, props);\n    }\n\n    static create(stack: DigitrafficStack, name: string, props: QueueProps): DigitrafficSqsQueue {\n        const queueName = `${stack.configuration.shortName}-${name}-Queue`;\n        const queueProps = {...props, ...{\n            encryption: QueueEncryption.KMS_MANAGED,\n            queueName,\n            deadLetterQueue: props.deadLetterQueue || {\n                maxReceiveCount: 2,\n                queue: DigitrafficDLQueue.create(stack, name),\n            },\n        }};\n\n        return new DigitrafficSqsQueue(stack, queueName, queueProps);\n    }\n}\n\nexport class DigitrafficDLQueue {\n    static create(stack: DigitrafficStack, name: string): DigitrafficSqsQueue {\n        const dlqName = `${stack.configuration.shortName}-${name}-DLQ`;\n\n        const dlq = new DigitrafficSqsQueue(stack, dlqName, {\n            queueName: dlqName,\n            visibilityTimeout: Duration.seconds(60),\n            encryption: QueueEncryption.KMS_MANAGED,\n        });\n\n        const dlqBucket = new Bucket(stack, `${dlqName}-Bucket`, {\n            blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n        });\n\n        const dlqFunctionName = `${dlqName}-Function`;\n        const lambda = MonitoredFunction.create(stack, dlqFunctionName, {\n            runtime: Runtime.NODEJS_14_X,\n            logRetention: RetentionDays.ONE_YEAR,\n            functionName: dlqFunctionName,\n            code: getDlqCode(dlqBucket.bucketName),\n            timeout: Duration.seconds(10),\n            handler: 'index.handler',\n            memorySize: 128,\n            reservedConcurrentExecutions: 1,\n        });\n\n        const statement = new PolicyStatement();\n        statement.addActions('s3:PutObject');\n        statement.addActions('s3:PutObjectAcl');\n        statement.addResources(dlqBucket.bucketArn + '/*');\n\n        lambda.addToRolePolicy(statement);\n        lambda.addEventSource(new SqsEventSource(dlq));\n\n        addDLQAlarm(stack, dlqName, dlq);\n\n        return dlq;\n    }\n}\n\nfunction addDLQAlarm(stack: DigitrafficStack, dlqName: string, dlq: Queue) {\n    const alarmName = `${dlqName}-Alarm`;\n    dlq.metricNumberOfMessagesReceived({\n        period: Duration.minutes(5),\n    }).createAlarm(stack, alarmName, {\n        alarmName,\n        threshold: 0,\n        evaluationPeriods: 1,\n        treatMissingData: TreatMissingData.NOT_BREACHING,\n        comparisonOperator: ComparisonOperator.GREATER_THAN_THRESHOLD,\n    }).addAlarmAction(new SnsAction(stack.warningTopic));\n}\n\nfunction getDlqCode(bName: string): InlineCode {\n    const functionBody = DLQ_LAMBDA_CODE\n        .replace(\"__bucketName__\", bName)\n        .replace(\"__upload__\", uploadToS3.toString())\n        .replace(\"__doUpload__\", doUpload.toString())\n        .replace(\"__handler__\", createHandler().toString().substring(23)); // remove function handler() from signature\n\n    return new InlineCode(functionBody);\n}\n\nasync function uploadToS3(s3: S3, bName: string, body: string, objectName: string): Promise<void> {\n    try {\n        console.info('writing %s to %s', objectName, bName);\n        await doUpload(s3, bName, body, objectName);\n    } catch (error) {\n        console.warn(error);\n        console.warn('method=uploadToS3 retrying upload to bucket %s', bName);\n        try {\n            await doUpload(s3, bName, body, objectName);\n        } catch (e2) {\n            console.error('method=uploadToS3 failed retrying upload to bucket %s', bName);\n        }\n    }\n}\n\nfunction doUpload(s3: S3, bName: string, Body: string, Key: string): Promise<ManagedUpload.SendData> {\n    return s3.upload({\n        Bucket: bName, Body, Key,\n    }).promise();\n}\n\n// bucketName is unused, will be overridden in the actual lambda code below\nconst bucketName = '';\n\nfunction createHandler(): SQSHandler {\n    return async function handler(event: SQSEvent): Promise<void> {\n        // eslint-disable-next-line @typescript-eslint/no-var-requires\n        const AWS = require('aws-sdk');\n\n        const millis = new Date().getTime();\n        await Promise.all(event.Records.map((e: SQSRecord, idx: number) =>\n            uploadToS3(new AWS.S3(), bucketName, e.body, `dlq-${millis}-${idx}.json`)));\n    };\n}\n\nconst DLQ_LAMBDA_CODE = `const AWS = require('aws-sdk');\nconst bucketName = \"__bucketName__\";\n\n__upload__\n__doUpload__\n\nexports.handler = async (event) => __handler__\n`;\n"]}
|
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
import { Architecture, Code, FunctionProps, Runtime } from 'aws-cdk-lib/aws-lambda';
|
|
2
|
+
import { ISecurityGroup, IVpc, SubnetSelection } from "aws-cdk-lib/aws-ec2";
|
|
3
|
+
import { Role } from 'aws-cdk-lib/aws-iam';
|
|
4
|
+
import { DigitrafficStack } from "./stack";
|
|
5
|
+
import { MonitoredFunctionAlarmProps } from "./monitoredfunction";
|
|
6
|
+
export declare type LambdaEnvironment = Record<string, string>;
|
|
7
|
+
export declare type DBLambdaEnvironment = LambdaEnvironment & {
|
|
8
|
+
SECRET_ID?: string;
|
|
9
|
+
DB_APPLICATION: string;
|
|
10
|
+
};
|
|
11
|
+
export interface LambdaConfiguration {
|
|
12
|
+
vpcId: string;
|
|
13
|
+
allowFromIpAddresses?: string[];
|
|
14
|
+
privateSubnetIds: string[];
|
|
15
|
+
availabilityZones: string[];
|
|
16
|
+
lambdaDbSgId: string;
|
|
17
|
+
dbProps?: DbProps;
|
|
18
|
+
defaultLambdaDurationSeconds?: number;
|
|
19
|
+
logsDestinationArn: string;
|
|
20
|
+
memorySize?: number;
|
|
21
|
+
runtime?: Runtime;
|
|
22
|
+
}
|
|
23
|
+
declare interface DbProps {
|
|
24
|
+
username: string;
|
|
25
|
+
password: string;
|
|
26
|
+
uri?: string;
|
|
27
|
+
ro_uri?: string;
|
|
28
|
+
}
|
|
29
|
+
export declare function databaseFunctionProps(stack: DigitrafficStack, environment: LambdaEnvironment, lambdaName: string, simpleLambdaName: string, config?: FunctionParameters): FunctionProps;
|
|
30
|
+
export declare function lambdaFunctionProps(stack: DigitrafficStack, environment: LambdaEnvironment, lambdaName: string, simpleLambdaName: string, config?: FunctionParameters): FunctionProps;
|
|
31
|
+
/**
|
|
32
|
+
* Creates a base configuration for a Lambda that uses an RDS database
|
|
33
|
+
* @param vpc "Private" Lambdas are associated with a VPC
|
|
34
|
+
* @param lambdaDbSg Security Group shared by Lambda and RDS
|
|
35
|
+
* @param props Database connection properties for the Lambda
|
|
36
|
+
* @param config Lambda configuration
|
|
37
|
+
*/
|
|
38
|
+
export declare function dbLambdaConfiguration(vpc: IVpc, lambdaDbSg: ISecurityGroup, props: LambdaConfiguration, config: FunctionParameters): FunctionProps;
|
|
39
|
+
export declare function defaultLambdaConfiguration(config: FunctionParameters): FunctionProps;
|
|
40
|
+
export interface FunctionParameters {
|
|
41
|
+
memorySize?: number;
|
|
42
|
+
timeout?: number;
|
|
43
|
+
functionName?: string;
|
|
44
|
+
code?: Code;
|
|
45
|
+
handler?: string;
|
|
46
|
+
readOnly?: boolean;
|
|
47
|
+
environment?: {
|
|
48
|
+
[key: string]: string;
|
|
49
|
+
};
|
|
50
|
+
reservedConcurrentExecutions?: number;
|
|
51
|
+
role?: Role;
|
|
52
|
+
vpc?: IVpc;
|
|
53
|
+
vpcSubnets?: SubnetSelection;
|
|
54
|
+
runtime?: Runtime;
|
|
55
|
+
architecture?: Architecture;
|
|
56
|
+
singleLambda?: boolean;
|
|
57
|
+
}
|
|
58
|
+
export declare type MonitoredFunctionParameters = {
|
|
59
|
+
readonly memorySize?: number;
|
|
60
|
+
readonly timeout?: number;
|
|
61
|
+
readonly functionName?: string;
|
|
62
|
+
readonly reservedConcurrentExecutions?: number;
|
|
63
|
+
readonly role?: Role;
|
|
64
|
+
readonly runtime?: Runtime;
|
|
65
|
+
readonly architecture?: Architecture;
|
|
66
|
+
readonly singleLambda?: boolean;
|
|
67
|
+
readonly durationAlarmProps?: MonitoredFunctionAlarmProps;
|
|
68
|
+
readonly durationWarningProps?: MonitoredFunctionAlarmProps;
|
|
69
|
+
readonly errorAlarmProps?: MonitoredFunctionAlarmProps;
|
|
70
|
+
readonly throttleAlarmProps?: MonitoredFunctionAlarmProps;
|
|
71
|
+
};
|
|
72
|
+
export {};
|
|
@@ -0,0 +1,93 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.defaultLambdaConfiguration = exports.dbLambdaConfiguration = exports.lambdaFunctionProps = exports.databaseFunctionProps = void 0;
|
|
4
|
+
const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
|
|
5
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
6
|
+
const aws_logs_1 = require("aws-cdk-lib/aws-logs");
|
|
7
|
+
function databaseFunctionProps(stack, environment, lambdaName, simpleLambdaName, config) {
|
|
8
|
+
const vpcSubnets = stack.vpc ? {
|
|
9
|
+
subnets: stack.vpc.privateSubnets,
|
|
10
|
+
} : undefined;
|
|
11
|
+
return { ...lambdaFunctionProps(stack, environment, lambdaName, simpleLambdaName, config), ...{
|
|
12
|
+
vpc: stack.vpc || undefined,
|
|
13
|
+
vpcSubnets,
|
|
14
|
+
securityGroup: stack.lambdaDbSg || undefined,
|
|
15
|
+
} };
|
|
16
|
+
}
|
|
17
|
+
exports.databaseFunctionProps = databaseFunctionProps;
|
|
18
|
+
function lambdaFunctionProps(stack, environment, lambdaName, simpleLambdaName, config) {
|
|
19
|
+
return {
|
|
20
|
+
runtime: config?.runtime || aws_lambda_1.Runtime.NODEJS_14_X,
|
|
21
|
+
architecture: config?.architecture || aws_lambda_1.Architecture.ARM_64,
|
|
22
|
+
memorySize: config?.memorySize || 128,
|
|
23
|
+
functionName: lambdaName,
|
|
24
|
+
role: config?.role,
|
|
25
|
+
timeout: aws_cdk_lib_1.Duration.seconds(config?.timeout || 60),
|
|
26
|
+
logRetention: aws_logs_1.RetentionDays.ONE_YEAR,
|
|
27
|
+
reservedConcurrentExecutions: config?.reservedConcurrentExecutions || 2,
|
|
28
|
+
code: getAssetCode(simpleLambdaName, config),
|
|
29
|
+
handler: `${simpleLambdaName}.handler`,
|
|
30
|
+
environment,
|
|
31
|
+
};
|
|
32
|
+
}
|
|
33
|
+
exports.lambdaFunctionProps = lambdaFunctionProps;
|
|
34
|
+
function getAssetCode(simpleLambdaName, config) {
|
|
35
|
+
const lambdaPath = config?.singleLambda ? `dist/lambda/` : `dist/lambda/${simpleLambdaName}`;
|
|
36
|
+
return new aws_lambda_1.AssetCode(lambdaPath);
|
|
37
|
+
}
|
|
38
|
+
/**
|
|
39
|
+
* Creates a base configuration for a Lambda that uses an RDS database
|
|
40
|
+
* @param vpc "Private" Lambdas are associated with a VPC
|
|
41
|
+
* @param lambdaDbSg Security Group shared by Lambda and RDS
|
|
42
|
+
* @param props Database connection properties for the Lambda
|
|
43
|
+
* @param config Lambda configuration
|
|
44
|
+
*/
|
|
45
|
+
function dbLambdaConfiguration(vpc, lambdaDbSg, props, config) {
|
|
46
|
+
return {
|
|
47
|
+
runtime: props.runtime || aws_lambda_1.Runtime.NODEJS_14_X,
|
|
48
|
+
memorySize: props.memorySize || config.memorySize || 1024,
|
|
49
|
+
functionName: config.functionName,
|
|
50
|
+
code: config.code,
|
|
51
|
+
role: config.role,
|
|
52
|
+
handler: config.handler,
|
|
53
|
+
timeout: aws_cdk_lib_1.Duration.seconds(config.timeout || props.defaultLambdaDurationSeconds || 60),
|
|
54
|
+
environment: config.environment || {
|
|
55
|
+
DB_USER: props.dbProps?.username,
|
|
56
|
+
DB_PASS: props.dbProps?.password,
|
|
57
|
+
DB_URI: (config.readOnly ? props.dbProps?.ro_uri : props.dbProps?.uri),
|
|
58
|
+
},
|
|
59
|
+
logRetention: aws_logs_1.RetentionDays.ONE_YEAR,
|
|
60
|
+
vpc: vpc,
|
|
61
|
+
vpcSubnets: {
|
|
62
|
+
subnets: vpc.privateSubnets,
|
|
63
|
+
},
|
|
64
|
+
securityGroups: [lambdaDbSg],
|
|
65
|
+
reservedConcurrentExecutions: config.reservedConcurrentExecutions || 3,
|
|
66
|
+
};
|
|
67
|
+
}
|
|
68
|
+
exports.dbLambdaConfiguration = dbLambdaConfiguration;
|
|
69
|
+
function defaultLambdaConfiguration(config) {
|
|
70
|
+
const props = {
|
|
71
|
+
runtime: aws_lambda_1.Runtime.NODEJS_14_X,
|
|
72
|
+
memorySize: config.memorySize ?? 128,
|
|
73
|
+
functionName: config.functionName,
|
|
74
|
+
handler: config.handler,
|
|
75
|
+
environment: config.environment ?? {},
|
|
76
|
+
logRetention: aws_logs_1.RetentionDays.ONE_YEAR,
|
|
77
|
+
reservedConcurrentExecutions: config.reservedConcurrentExecutions,
|
|
78
|
+
code: config.code,
|
|
79
|
+
role: config.role,
|
|
80
|
+
timeout: aws_cdk_lib_1.Duration.seconds(config.timeout || 10),
|
|
81
|
+
};
|
|
82
|
+
if (config.vpc) {
|
|
83
|
+
return { ...props, ...{
|
|
84
|
+
vpc: config.vpc,
|
|
85
|
+
vpcSubnets: {
|
|
86
|
+
subnets: config.vpc?.privateSubnets,
|
|
87
|
+
},
|
|
88
|
+
} };
|
|
89
|
+
}
|
|
90
|
+
return props;
|
|
91
|
+
}
|
|
92
|
+
exports.defaultLambdaConfiguration = defaultLambdaConfiguration;
|
|
93
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lambda-configs.js","sourceRoot":"","sources":["../../../../src/aws/infra/stack/lambda-configs.ts"],"names":[],"mappings":"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","sourcesContent":["import {Architecture, AssetCode, Code, FunctionProps, Runtime} from 'aws-cdk-lib/aws-lambda';\r\nimport {Duration} from \"aws-cdk-lib\";\r\nimport {ISecurityGroup, IVpc, SubnetSelection} from \"aws-cdk-lib/aws-ec2\";\r\nimport {RetentionDays} from 'aws-cdk-lib/aws-logs';\r\nimport {Role} from 'aws-cdk-lib/aws-iam';\r\nimport {DigitrafficStack} from \"./stack\";\r\nimport {MonitoredFunctionAlarmProps} from \"./monitoredfunction\";\r\n\r\nexport type LambdaEnvironment = Record<string, string>;\r\n\r\nexport type DBLambdaEnvironment = LambdaEnvironment & {\r\n    SECRET_ID?: string\r\n    DB_APPLICATION: string\r\n}\r\n\r\nexport interface LambdaConfiguration {\r\n    vpcId: string;\r\n    allowFromIpAddresses?: string[];\r\n    privateSubnetIds: string[];\r\n    availabilityZones: string[];\r\n    lambdaDbSgId: string;\r\n    dbProps?: DbProps;\r\n    defaultLambdaDurationSeconds?: number;\r\n    logsDestinationArn: string;\r\n    memorySize?: number,\r\n    runtime?: Runtime;\r\n}\r\n\r\ndeclare interface DbProps {\r\n    username: string;\r\n    password: string;\r\n    uri?: string;\r\n    ro_uri?: string;\r\n}\r\n\r\nexport function databaseFunctionProps(\r\n    stack: DigitrafficStack, environment: LambdaEnvironment, lambdaName: string, simpleLambdaName: string, config?: FunctionParameters,\r\n): FunctionProps {\r\n    const vpcSubnets = stack.vpc ? {\r\n        subnets: stack.vpc.privateSubnets,\r\n    } : undefined;\r\n\r\n    return {...lambdaFunctionProps(\r\n        stack, environment, lambdaName, simpleLambdaName, config,\r\n    ), ...{\r\n        vpc: stack.vpc || undefined,\r\n        vpcSubnets,\r\n        securityGroup: stack.lambdaDbSg || undefined,\r\n    }};\r\n}\r\n\r\nexport function lambdaFunctionProps(\r\n    stack: DigitrafficStack, environment: LambdaEnvironment, lambdaName: string, simpleLambdaName: string, config?: FunctionParameters,\r\n): FunctionProps {\r\n    return {\r\n        runtime: config?.runtime || Runtime.NODEJS_14_X,\r\n        architecture: config?.architecture || Architecture.ARM_64,\r\n        memorySize: config?.memorySize || 128,\r\n        functionName: lambdaName,\r\n        role: config?.role,\r\n        timeout: Duration.seconds(config?.timeout || 60),\r\n        logRetention: RetentionDays.ONE_YEAR,\r\n        reservedConcurrentExecutions: config?.reservedConcurrentExecutions || 2,\r\n        code: getAssetCode(simpleLambdaName, config),\r\n        handler: `${simpleLambdaName}.handler`,\r\n        environment,\r\n    };\r\n}\r\n\r\nfunction getAssetCode(simpleLambdaName: string, config?: FunctionParameters): AssetCode {\r\n    const lambdaPath = config?.singleLambda ? `dist/lambda/` : `dist/lambda/${simpleLambdaName}`;\r\n\r\n    return new AssetCode(lambdaPath);\r\n}\r\n\r\n/**\r\n * Creates a base configuration for a Lambda that uses an RDS database\r\n * @param vpc \"Private\" Lambdas are associated with a VPC\r\n * @param lambdaDbSg Security Group shared by Lambda and RDS\r\n * @param props Database connection properties for the Lambda\r\n * @param config Lambda configuration\r\n */\r\nexport function dbLambdaConfiguration(vpc: IVpc,\r\n    lambdaDbSg: ISecurityGroup,\r\n    props: LambdaConfiguration,\r\n    config: FunctionParameters): FunctionProps {\r\n\r\n    return {\r\n        runtime: props.runtime || Runtime.NODEJS_14_X,\r\n        memorySize: props.memorySize || config.memorySize || 1024,\r\n        functionName: config.functionName,\r\n        code: config.code as Code,\r\n        role: config.role,\r\n        handler: config.handler as string,\r\n        timeout: Duration.seconds(config.timeout || props.defaultLambdaDurationSeconds || 60),\r\n        environment: config.environment || {\r\n            DB_USER: props.dbProps?.username as string,\r\n            DB_PASS: props.dbProps?.password as string,\r\n            DB_URI: (config.readOnly ? props.dbProps?.ro_uri : props.dbProps?.uri) as string,\r\n        },\r\n        logRetention: RetentionDays.ONE_YEAR,\r\n        vpc: vpc,\r\n        vpcSubnets: {\r\n            subnets: vpc.privateSubnets,\r\n        },\r\n        securityGroups: [lambdaDbSg],\r\n        reservedConcurrentExecutions: config.reservedConcurrentExecutions || 3,\r\n    };\r\n}\r\n\r\nexport function defaultLambdaConfiguration(config: FunctionParameters): FunctionProps {\r\n    const props: FunctionProps = {\r\n        runtime: Runtime.NODEJS_14_X,\r\n        memorySize: config.memorySize ?? 128,\r\n        functionName: config.functionName,\r\n        handler: config.handler as string,\r\n        environment: config.environment ?? {},\r\n        logRetention: RetentionDays.ONE_YEAR,\r\n        reservedConcurrentExecutions: config.reservedConcurrentExecutions,\r\n        code: config.code as Code,\r\n        role: config.role,\r\n        timeout: Duration.seconds(config.timeout || 10),\r\n    };\r\n    if (config.vpc) {\r\n        return {...props, ...{\r\n            vpc: config.vpc,\r\n            vpcSubnets: {\r\n                subnets: config.vpc?.privateSubnets,\r\n            },\r\n        }};\r\n    }\r\n    return props;\r\n}\r\n\r\nexport interface FunctionParameters {\r\n    memorySize?: number;\r\n    timeout?: number;\r\n    functionName?: string;\r\n    code?: Code;\r\n    handler?: string;\r\n    readOnly?: boolean;\r\n    environment?: {\r\n        [key: string]: string;\r\n    };\r\n    reservedConcurrentExecutions?: number;\r\n    role?: Role;\r\n    vpc?: IVpc;\r\n    vpcSubnets?: SubnetSelection;\r\n    runtime?: Runtime;\r\n    architecture?: Architecture;\r\n    singleLambda?: boolean;\r\n}\r\n\r\nexport type MonitoredFunctionParameters = {\r\n    readonly memorySize?: number;\r\n    readonly timeout?: number;\r\n    readonly functionName?: string;\r\n    readonly reservedConcurrentExecutions?: number;\r\n    readonly role?: Role;\r\n    readonly runtime?: Runtime;\r\n    readonly architecture?: Architecture;\r\n    readonly singleLambda?: boolean;\r\n\r\n    readonly durationAlarmProps?: MonitoredFunctionAlarmProps\r\n    readonly durationWarningProps?: MonitoredFunctionAlarmProps\r\n    readonly errorAlarmProps?: MonitoredFunctionAlarmProps\r\n    readonly throttleAlarmProps?: MonitoredFunctionAlarmProps\r\n}\r\n"]}
|
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
import { Function, FunctionProps } from 'aws-cdk-lib/aws-lambda';
|
|
2
|
+
import { Stack } from "aws-cdk-lib";
|
|
3
|
+
import { ComparisonOperator } from "aws-cdk-lib/aws-cloudwatch";
|
|
4
|
+
import { DigitrafficStack } from "../stack/stack";
|
|
5
|
+
import { ITopic } from "aws-cdk-lib/aws-sns";
|
|
6
|
+
import { LambdaEnvironment, MonitoredFunctionParameters } from "../stack/lambda-configs";
|
|
7
|
+
import { TrafficType } from "../../../types/traffictype";
|
|
8
|
+
/**
|
|
9
|
+
* Allows customization of CloudWatch Alarm properties
|
|
10
|
+
*/
|
|
11
|
+
export declare type MonitoredFunctionAlarmProps = {
|
|
12
|
+
/**
|
|
13
|
+
* Setting this to false will not create a CloudWatch alarm
|
|
14
|
+
*/
|
|
15
|
+
readonly create: boolean;
|
|
16
|
+
readonly threshold?: number;
|
|
17
|
+
readonly evaluationPeriods?: number;
|
|
18
|
+
readonly datapointsToAlarm?: number;
|
|
19
|
+
readonly comparisonOperator?: ComparisonOperator;
|
|
20
|
+
};
|
|
21
|
+
export declare type MonitoredFunctionProps = {
|
|
22
|
+
readonly durationAlarmProps?: MonitoredFunctionAlarmProps;
|
|
23
|
+
readonly durationWarningProps?: MonitoredFunctionAlarmProps;
|
|
24
|
+
readonly errorAlarmProps?: MonitoredFunctionAlarmProps;
|
|
25
|
+
readonly throttleAlarmProps?: MonitoredFunctionAlarmProps;
|
|
26
|
+
};
|
|
27
|
+
/**
|
|
28
|
+
* Creates a Lambda function that monitors default CloudWatch Lambda metrics with CloudWatch Alarms.
|
|
29
|
+
*/
|
|
30
|
+
export declare class MonitoredFunction extends Function {
|
|
31
|
+
readonly givenName: string;
|
|
32
|
+
/** disable all alarms */
|
|
33
|
+
static readonly DISABLE_ALARMS: MonitoredFunctionProps;
|
|
34
|
+
/**
|
|
35
|
+
* Create new MonitoredFunction. Use topics from given DigitrafficStack.
|
|
36
|
+
*
|
|
37
|
+
* @param stack DigitrafficStack
|
|
38
|
+
* @param id Lambda construct Id
|
|
39
|
+
* @param functionProps Lambda function properties
|
|
40
|
+
* @param props Monitored function properties
|
|
41
|
+
*/
|
|
42
|
+
static create(stack: DigitrafficStack, id: string, functionProps: FunctionProps, props?: MonitoredFunctionProps): MonitoredFunction;
|
|
43
|
+
/**
|
|
44
|
+
* Create new MonitoredFunction. Use topics from given DigitrafficStack. Generate names from given name and configuration shortName.
|
|
45
|
+
*
|
|
46
|
+
* For example, shortName FOO and given name update-things will create function FOO-UpdateThings and use code from lambda/update-things/update-things.ts method handler.
|
|
47
|
+
*
|
|
48
|
+
* @param stack DigitrafficStack
|
|
49
|
+
* @param name param-case name
|
|
50
|
+
* @param environment Lambda environment
|
|
51
|
+
* @param functionParameters Lambda function parameters
|
|
52
|
+
*/
|
|
53
|
+
static createV2(stack: DigitrafficStack, name: string, environment: LambdaEnvironment, functionParameters?: MonitoredFunctionParameters): MonitoredFunction;
|
|
54
|
+
/**
|
|
55
|
+
* @param scope Stack
|
|
56
|
+
* @param id Lambda construct Id
|
|
57
|
+
* @param functionProps Lambda function properties
|
|
58
|
+
* @param alarmSnsTopic SNS topic for alarms
|
|
59
|
+
* @param warningSnsTopic SNS topic for warnings
|
|
60
|
+
* @param production Is the stack a production stack, used for determining the alarm topic
|
|
61
|
+
* @param trafficType Traffic type, used for alarm names. Set to null if Lambda is not related to any traffic type.
|
|
62
|
+
* @param props Monitored function properties
|
|
63
|
+
*/
|
|
64
|
+
constructor(scope: Stack, id: string, functionProps: FunctionProps, alarmSnsTopic: ITopic, warningSnsTopic: ITopic, production: boolean, trafficType: TrafficType | null, props?: MonitoredFunctionProps);
|
|
65
|
+
private createAlarm;
|
|
66
|
+
private getAlarmActionForEnv;
|
|
67
|
+
}
|
|
68
|
+
export declare class MonitoredDBFunction {
|
|
69
|
+
/**
|
|
70
|
+
* Create new MonitoredDBFunction. Use topics from given DigitrafficStack. Generate names from given name and configuration shortName.
|
|
71
|
+
* Grant secret and create log subscription.
|
|
72
|
+
*
|
|
73
|
+
* For example, shortName FOO and given name update-things will create function FOO-UpdateThings and use code from lambda/update-things/update-things.ts method handler.
|
|
74
|
+
*
|
|
75
|
+
* If you don't need to pass any extra arguments to lambda-environment, you can leave environment out and this function will create the
|
|
76
|
+
* default Lambda Environment with SECRET_ID and DB_APPLICATION.
|
|
77
|
+
*
|
|
78
|
+
* @param stack DigitrafficStack
|
|
79
|
+
* @param name param-case name
|
|
80
|
+
* @param environment Lambda environment
|
|
81
|
+
* @param functionParameters Lambda function parameters
|
|
82
|
+
*/
|
|
83
|
+
static create(stack: DigitrafficStack, name: string, environment?: LambdaEnvironment, functionParameters?: MonitoredFunctionParameters): MonitoredFunction;
|
|
84
|
+
}
|