@digitraffic/common 2022.10.5
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
@@ -0,0 +1,179 @@
|
|
1
|
+
"use strict";
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
3
|
+
exports.createIpRestrictionPolicyDocument = exports.createDefaultPolicyDocument = exports.createRestApi = exports.setReturnCodeForMissingAuthenticationToken = exports.add401Support = exports.add404Support = exports.DigitrafficRestApi = void 0;
|
4
|
+
const aws_apigateway_1 = require("aws-cdk-lib/aws-apigateway");
|
5
|
+
const aws_iam_1 = require("aws-cdk-lib/aws-iam");
|
6
|
+
const usage_plans_1 = require("../usage-plans");
|
7
|
+
const api_model_1 = require("../../../utils/api-model");
|
8
|
+
const mediatypes_1 = require("../../types/mediatypes");
|
9
|
+
const R = require("ramda");
|
10
|
+
class DigitrafficRestApi extends aws_apigateway_1.RestApi {
|
11
|
+
constructor(stack, apiId, apiName, allowFromIpAddresses, config) {
|
12
|
+
const policyDocument = allowFromIpAddresses == null ? createDefaultPolicyDocument() : createIpRestrictionPolicyDocument(allowFromIpAddresses);
|
13
|
+
// override default config with given extra config
|
14
|
+
const apiConfig = { ...{
|
15
|
+
deployOptions: {
|
16
|
+
loggingLevel: aws_apigateway_1.MethodLoggingLevel.ERROR,
|
17
|
+
},
|
18
|
+
restApiName: apiName,
|
19
|
+
endpointTypes: [aws_apigateway_1.EndpointType.REGIONAL],
|
20
|
+
policy: policyDocument,
|
21
|
+
}, ...config };
|
22
|
+
super(stack, apiId, apiConfig);
|
23
|
+
this.apiKeyIds = [];
|
24
|
+
add404Support(this, stack);
|
25
|
+
}
|
26
|
+
hostname() {
|
27
|
+
return `${this.restApiId}.execute-api.${this.stack.region}.amazonaws.com`;
|
28
|
+
}
|
29
|
+
createUsagePlan(apiKeyId, apiKeyName) {
|
30
|
+
const newKeyId = (0, usage_plans_1.createUsagePlan)(this, apiKeyId, apiKeyName).keyId;
|
31
|
+
this.apiKeyIds.push(newKeyId);
|
32
|
+
return newKeyId;
|
33
|
+
}
|
34
|
+
createUsagePlanV2(apiName) {
|
35
|
+
const newKeyId = (0, usage_plans_1.createDefaultUsagePlan)(this, apiName).keyId;
|
36
|
+
this.apiKeyIds.push(newKeyId);
|
37
|
+
return newKeyId;
|
38
|
+
}
|
39
|
+
addJsonModel(modelName, schema) {
|
40
|
+
return this.getModelWithReference(this.addModel(modelName, {
|
41
|
+
contentType: mediatypes_1.MediaType.APPLICATION_JSON,
|
42
|
+
modelName,
|
43
|
+
schema,
|
44
|
+
}));
|
45
|
+
}
|
46
|
+
addCSVModel(modelName) {
|
47
|
+
return this.getModelWithReference(this.addModel(modelName, {
|
48
|
+
contentType: mediatypes_1.MediaType.TEXT_CSV,
|
49
|
+
modelName,
|
50
|
+
schema: {},
|
51
|
+
}));
|
52
|
+
}
|
53
|
+
getModelWithReference(model) {
|
54
|
+
return R.assoc('modelReference', (0, api_model_1.getModelReference)(model.modelId, this.restApiId), model);
|
55
|
+
}
|
56
|
+
addDocumentationPart(resource, parameterName, resourceName, type, properties) {
|
57
|
+
const location = {
|
58
|
+
type,
|
59
|
+
path: resource.path,
|
60
|
+
name: type !== 'METHOD' ? parameterName : undefined,
|
61
|
+
};
|
62
|
+
new aws_apigateway_1.CfnDocumentationPart(this.stack, resourceName, {
|
63
|
+
restApiId: resource.api.restApiId,
|
64
|
+
location,
|
65
|
+
properties: JSON.stringify(properties),
|
66
|
+
});
|
67
|
+
}
|
68
|
+
documentResource(resource, ...documentationPart) {
|
69
|
+
documentationPart.forEach(dp => this.addDocumentationPart(resource, dp.parameterName, `${resource.path}.${dp.parameterName}.Documentation`, dp.type, dp.documentationProperties));
|
70
|
+
}
|
71
|
+
}
|
72
|
+
exports.DigitrafficRestApi = DigitrafficRestApi;
|
73
|
+
/**
|
74
|
+
* Due to AWS API design API Gateway will always return 403 'Missing Authentication Token' for requests
|
75
|
+
* with a non-existent endpoint. This function translates this response to a 404.
|
76
|
+
* Requests with an invalid or missing API key are not affected (still return 403 'Forbidden').
|
77
|
+
* @param restApi RestApi
|
78
|
+
* @param stack Construct
|
79
|
+
*/
|
80
|
+
function add404Support(restApi, stack) {
|
81
|
+
new aws_apigateway_1.GatewayResponse(stack, `MissingAuthenticationTokenResponse-${restApi.restApiName}`, {
|
82
|
+
restApi,
|
83
|
+
type: aws_apigateway_1.ResponseType.MISSING_AUTHENTICATION_TOKEN,
|
84
|
+
statusCode: '404',
|
85
|
+
templates: {
|
86
|
+
'application/json': '{"message": "Not found"}',
|
87
|
+
},
|
88
|
+
});
|
89
|
+
}
|
90
|
+
exports.add404Support = add404Support;
|
91
|
+
function add401Support(restApi, stack) {
|
92
|
+
new aws_apigateway_1.GatewayResponse(stack, `AuthenticationFailedResponse-${restApi.restApiName}`, {
|
93
|
+
restApi,
|
94
|
+
type: aws_apigateway_1.ResponseType.UNAUTHORIZED,
|
95
|
+
statusCode: "401",
|
96
|
+
responseHeaders: {
|
97
|
+
'WWW-Authenticate': "'Basic'",
|
98
|
+
},
|
99
|
+
});
|
100
|
+
}
|
101
|
+
exports.add401Support = add401Support;
|
102
|
+
/**
|
103
|
+
* Due to AWS API design API Gateway will always return 403 'Missing Authentication Token' for requests
|
104
|
+
* with a non-existent endpoint. This function converts this response to a custom one.
|
105
|
+
* Requests with an invalid or missing API key are not affected (still return 403 'Forbidden').
|
106
|
+
* @param returnCode
|
107
|
+
* @param message
|
108
|
+
* @param restApi RestApi
|
109
|
+
* @param stack Construct
|
110
|
+
*/
|
111
|
+
function setReturnCodeForMissingAuthenticationToken(returnCode, message, restApi, stack) {
|
112
|
+
new aws_apigateway_1.GatewayResponse(stack, `MissingAuthenticationTokenResponse-${restApi.restApiName}`, {
|
113
|
+
restApi,
|
114
|
+
type: aws_apigateway_1.ResponseType.MISSING_AUTHENTICATION_TOKEN,
|
115
|
+
statusCode: `${returnCode}`,
|
116
|
+
templates: {
|
117
|
+
'application/json': `{"message": ${message}}`,
|
118
|
+
},
|
119
|
+
});
|
120
|
+
}
|
121
|
+
exports.setReturnCodeForMissingAuthenticationToken = setReturnCodeForMissingAuthenticationToken;
|
122
|
+
function createRestApi(stack, apiId, apiName, allowFromIpAddresses) {
|
123
|
+
const policyDocument = allowFromIpAddresses == null ? createDefaultPolicyDocument() : createIpRestrictionPolicyDocument(allowFromIpAddresses);
|
124
|
+
const restApi = new aws_apigateway_1.RestApi(stack, apiId, {
|
125
|
+
deployOptions: {
|
126
|
+
loggingLevel: aws_apigateway_1.MethodLoggingLevel.ERROR,
|
127
|
+
},
|
128
|
+
restApiName: apiName,
|
129
|
+
endpointTypes: [aws_apigateway_1.EndpointType.REGIONAL],
|
130
|
+
policy: policyDocument,
|
131
|
+
});
|
132
|
+
add404Support(restApi, stack);
|
133
|
+
return restApi;
|
134
|
+
}
|
135
|
+
exports.createRestApi = createRestApi;
|
136
|
+
function createDefaultPolicyDocument() {
|
137
|
+
return new aws_iam_1.PolicyDocument({
|
138
|
+
statements: [
|
139
|
+
new aws_iam_1.PolicyStatement({
|
140
|
+
effect: aws_iam_1.Effect.ALLOW,
|
141
|
+
actions: [
|
142
|
+
"execute-api:Invoke",
|
143
|
+
],
|
144
|
+
resources: [
|
145
|
+
"*",
|
146
|
+
],
|
147
|
+
principals: [
|
148
|
+
new aws_iam_1.AnyPrincipal(),
|
149
|
+
],
|
150
|
+
}),
|
151
|
+
],
|
152
|
+
});
|
153
|
+
}
|
154
|
+
exports.createDefaultPolicyDocument = createDefaultPolicyDocument;
|
155
|
+
function createIpRestrictionPolicyDocument(allowFromIpAddresses) {
|
156
|
+
return new aws_iam_1.PolicyDocument({
|
157
|
+
statements: [
|
158
|
+
new aws_iam_1.PolicyStatement({
|
159
|
+
effect: aws_iam_1.Effect.ALLOW,
|
160
|
+
conditions: {
|
161
|
+
"IpAddress": {
|
162
|
+
"aws:SourceIp": allowFromIpAddresses,
|
163
|
+
},
|
164
|
+
},
|
165
|
+
actions: [
|
166
|
+
"execute-api:Invoke",
|
167
|
+
],
|
168
|
+
resources: [
|
169
|
+
"*",
|
170
|
+
],
|
171
|
+
principals: [
|
172
|
+
new aws_iam_1.AnyPrincipal(),
|
173
|
+
],
|
174
|
+
}),
|
175
|
+
],
|
176
|
+
});
|
177
|
+
}
|
178
|
+
exports.createIpRestrictionPolicyDocument = createIpRestrictionPolicyDocument;
|
179
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rest_apis.js","sourceRoot":"","sources":["../../../../../src/aws/infra/stack/rest_apis.ts"],"names":[],"mappings":";;;AAAA,+DAOoC;AACpC,iDAA0F;AAG1F,gDAAuE;AAEvE,wDAA2D;AAC3D,uDAAiD;AAGjD,2BAA4B;AAE5B,MAAa,kBAAmB,SAAQ,wBAAO;IAG3C,YACI,KAAuB,EAAE,KAAa,EAAE,OAAe,EAAE,oBAA2C,EAAE,MAA8B;QAEpI,MAAM,cAAc,GAAG,oBAAoB,IAAI,IAAI,CAAC,CAAC,CAAC,2BAA2B,EAAE,CAAC,CAAC,CAAC,iCAAiC,CAAC,oBAAoB,CAAC,CAAC;QAE9I,kDAAkD;QAClD,MAAM,SAAS,GAAG,EAAC,GAAG;gBAClB,aAAa,EAAE;oBACX,YAAY,EAAE,mCAAkB,CAAC,KAAK;iBACzC;gBACD,WAAW,EAAE,OAAO;gBACpB,aAAa,EAAE,CAAC,6BAAY,CAAC,QAAQ,CAAC;gBACtC,MAAM,EAAE,cAAc;aACzB,EAAE,GAAG,MAAM,EAAC,CAAC;QAEd,KAAK,CAAC,KAAK,EAAE,KAAK,EAAE,SAAS,CAAC,CAAC;QAE/B,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;QAEpB,aAAa,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;IAC/B,CAAC;IAED,QAAQ;QACJ,OAAO,GAAG,IAAI,CAAC,SAAS,gBAAiB,IAAI,CAAC,KAA0B,CAAC,MAAM,gBAAgB,CAAC;IACpG,CAAC;IAED,eAAe,CAAC,QAAgB,EAAE,UAAkB;QAChD,MAAM,QAAQ,GAAG,IAAA,6BAAe,EAAC,IAAI,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC,KAAK,CAAC;QAEnE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAE9B,OAAO,QAAQ,CAAC;IACpB,CAAC;IAED,iBAAiB,CAAC,OAAe;QAC7B,MAAM,QAAQ,GAAG,IAAA,oCAAsB,EAAC,IAAI,EAAE,OAAO,CAAC,CAAC,KAAK,CAAC;QAE7D,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAE9B,OAAO,QAAQ,CAAC;IACpB,CAAC;IAED,YAAY,CAAC,SAAiB,EAAE,MAAkB;QAC9C,OAAO,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,EAAE;YACvD,WAAW,EAAE,sBAAS,CAAC,gBAAgB;YACvC,SAAS;YACT,MAAM;SACT,CAAC,CAAC,CAAC;IACR,CAAC;IAED,WAAW,CAAC,SAAiB;QACzB,OAAO,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,EAAE;YACvD,WAAW,EAAE,sBAAS,CAAC,QAAQ;YAC/B,SAAS;YACT,MAAM,EAAE,EAAE;SACb,CAAC,CAAC,CAAC;IACR,CAAC;IAEO,qBAAqB,CAAC,KAAY;QACtC,OAAO,CAAC,CAAC,KAAK,CAAC,gBAAgB,EAAE,IAAA,6BAAiB,EAAC,KAAK,CAAC,OAAO,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,CAAuB,CAAC;IACpH,CAAC;IAEO,oBAAoB,CACxB,QAAkB,EAAE,aAAqB,EAAE,YAAoB,EAAE,IAAY,EAAE,UAAmC;QAElH,MAAM,QAAQ,GAA0C;YACpD,IAAI;YACJ,IAAI,EAAE,QAAQ,CAAC,IAAI;YACnB,IAAI,EAAE,IAAI,KAAK,QAAQ,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,SAAS;SACtD,CAAC;QAEF,IAAI,qCAAoB,CAAC,IAAI,CAAC,KAAK,EAAE,YAAY,EAAE;YAC/C,SAAS,EAAE,QAAQ,CAAC,GAAG,CAAC,SAAS;YACjC,QAAQ;YACR,UAAU,EAAE,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC;SACzC,CAAC,CAAC;IACP,CAAC;IAED,gBAAgB,CAAC,QAAkB,EAAE,GAAG,iBAAsC;QAC1E,iBAAiB,CAAC,OAAO,CAAC,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,oBAAoB,CACrD,QAAQ,EAAE,EAAE,CAAC,aAAa,EAAE,GAAG,QAAQ,CAAC,IAAI,IAAI,EAAE,CAAC,aAAa,gBAAgB,EAAE,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,uBAAuB,CACxH,CAAC,CAAC;IACP,CAAC;CACJ;AAtFD,gDAsFC;AAED;;;;;;GAMG;AACH,SAAgB,aAAa,CAAC,OAAgB,EAAE,KAAgB;IAC5D,IAAI,gCAAe,CAAC,KAAK,EAAE,sCAAsC,OAAO,CAAC,WAAW,EAAE,EAAE;QACpF,OAAO;QACP,IAAI,EAAE,6BAAY,CAAC,4BAA4B;QAC/C,UAAU,EAAE,KAAK;QACjB,SAAS,EAAE;YACP,kBAAkB,EAAE,0BAA0B;SACjD;KACJ,CAAC,CAAC;AACP,CAAC;AATD,sCASC;AAED,SAAgB,aAAa,CAAC,OAAgB,EAAE,KAAgB;IAC5D,IAAI,gCAAe,CAAC,KAAK,EAAE,gCAAgC,OAAO,CAAC,WAAW,EAAE,EAAE;QAC9E,OAAO;QACP,IAAI,EAAE,6BAAY,CAAC,YAAY;QAC/B,UAAU,EAAE,KAAK;QACjB,eAAe,EAAE;YACb,kBAAkB,EAAE,SAAS;SAChC;KACJ,CAAC,CAAC;AACP,CAAC;AATD,sCASC;AAED;;;;;;;;GAQG;AACH,SAAgB,0CAA0C,CAAC,UAAkB,EACzE,OAAe,EACf,OAAgB,EAChB,KAAgB;IAEhB,IAAI,gCAAe,CAAC,KAAK,EAAE,sCAAsC,OAAO,CAAC,WAAW,EAAE,EAAE;QACpF,OAAO;QACP,IAAI,EAAE,6BAAY,CAAC,4BAA4B;QAC/C,UAAU,EAAE,GAAG,UAAU,EAAE;QAC3B,SAAS,EAAE;YACP,kBAAkB,EAAE,eAAe,OAAO,GAAG;SAChD;KACJ,CAAC,CAAC;AACP,CAAC;AAbD,gGAaC;AAED,SAAgB,aAAa,CAAC,KAAgB,EAAE,KAAa,EAAE,OAAe,EAAE,oBAA2C;IACvH,MAAM,cAAc,GAAG,oBAAoB,IAAI,IAAI,CAAC,CAAC,CAAC,2BAA2B,EAAE,CAAC,CAAC,CAAC,iCAAiC,CAAC,oBAAoB,CAAC,CAAC;IAC9I,MAAM,OAAO,GAAG,IAAI,wBAAO,CAAC,KAAK,EAAE,KAAK,EAAE;QACtC,aAAa,EAAE;YACX,YAAY,EAAE,mCAAkB,CAAC,KAAK;SACzC;QACD,WAAW,EAAE,OAAO;QACpB,aAAa,EAAE,CAAC,6BAAY,CAAC,QAAQ,CAAC;QACtC,MAAM,EAAE,cAAc;KACzB,CAAC,CAAC;IACH,aAAa,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;IAC9B,OAAO,OAAO,CAAC;AACnB,CAAC;AAZD,sCAYC;AAED,SAAgB,2BAA2B;IACvC,OAAO,IAAI,wBAAc,CAAC;QACtB,UAAU,EAAE;YACR,IAAI,yBAAe,CAAC;gBAChB,MAAM,EAAE,gBAAM,CAAC,KAAK;gBACpB,OAAO,EAAE;oBACL,oBAAoB;iBACvB;gBACD,SAAS,EAAE;oBACP,GAAG;iBACN;gBACD,UAAU,EAAE;oBACR,IAAI,sBAAY,EAAE;iBACrB;aACJ,CAAC;SACL;KACJ,CAAC,CAAC;AACP,CAAC;AAjBD,kEAiBC;AAGD,SAAgB,iCAAiC,CAAC,oBAA8B;IAC5E,OAAO,IAAI,wBAAc,CAAC;QACtB,UAAU,EAAE;YACR,IAAI,yBAAe,CAAC;gBAChB,MAAM,EAAE,gBAAM,CAAC,KAAK;gBACpB,UAAU,EAAE;oBACR,WAAW,EAAE;wBACT,cAAc,EAAE,oBAAoB;qBACvC;iBACJ;gBACD,OAAO,EAAE;oBACL,oBAAoB;iBACvB;gBACD,SAAS,EAAE;oBACP,GAAG;iBACN;gBACD,UAAU,EAAE;oBACR,IAAI,sBAAY,EAAE;iBACrB;aACJ,CAAC;SACL;KACJ,CAAC,CAAC;AACP,CAAC;AAtBD,8EAsBC","sourcesContent":["import {\r\n    RestApi,\r\n    MethodLoggingLevel,\r\n    GatewayResponse,\r\n    ResponseType,\r\n    EndpointType,\r\n    RestApiProps, JsonSchema, Model, CfnDocumentationPart, Resource,\r\n} from 'aws-cdk-lib/aws-apigateway';\r\nimport {PolicyDocument, PolicyStatement, Effect, AnyPrincipal} from 'aws-cdk-lib/aws-iam';\r\nimport {Construct} from \"constructs\";\r\nimport {DigitrafficStack} from \"./stack\";\r\nimport {createDefaultUsagePlan, createUsagePlan} from \"../usage-plans\";\r\nimport {ModelWithReference} from \"../../types/model-with-reference\";\r\nimport {getModelReference} from \"../../../utils/api-model\";\r\nimport {MediaType} from \"../../types/mediatypes\";\r\nimport {DocumentationPart, DocumentationProperties} from \"../documentation\";\r\n\r\nimport R = require('ramda');\r\n\r\nexport class DigitrafficRestApi extends RestApi {\r\n    readonly apiKeyIds: string[];\r\n\r\n    constructor(\r\n        stack: DigitrafficStack, apiId: string, apiName: string, allowFromIpAddresses?: string[] | undefined, config?: Partial<RestApiProps>,\r\n    ) {\r\n        const policyDocument = allowFromIpAddresses == null ? createDefaultPolicyDocument() : createIpRestrictionPolicyDocument(allowFromIpAddresses);\r\n\r\n        // override default config with given extra config\r\n        const apiConfig = {...{\r\n            deployOptions: {\r\n                loggingLevel: MethodLoggingLevel.ERROR,\r\n            },\r\n            restApiName: apiName,\r\n            endpointTypes: [EndpointType.REGIONAL],\r\n            policy: policyDocument,\r\n        }, ...config};\r\n\r\n        super(stack, apiId, apiConfig);\r\n\r\n        this.apiKeyIds = [];\r\n\r\n        add404Support(this, stack);\r\n    }\r\n\r\n    hostname(): string {\r\n        return `${this.restApiId}.execute-api.${(this.stack as DigitrafficStack).region}.amazonaws.com`;\r\n    }\r\n\r\n    createUsagePlan(apiKeyId: string, apiKeyName: string): string {\r\n        const newKeyId = createUsagePlan(this, apiKeyId, apiKeyName).keyId;\r\n\r\n        this.apiKeyIds.push(newKeyId);\r\n\r\n        return newKeyId;\r\n    }\r\n\r\n    createUsagePlanV2(apiName: string): string {\r\n        const newKeyId = createDefaultUsagePlan(this, apiName).keyId;\r\n\r\n        this.apiKeyIds.push(newKeyId);\r\n\r\n        return newKeyId;\r\n    }\r\n\r\n    addJsonModel(modelName: string, schema: JsonSchema) {\r\n        return this.getModelWithReference(this.addModel(modelName, {\r\n            contentType: MediaType.APPLICATION_JSON,\r\n            modelName,\r\n            schema,\r\n        }));\r\n    }\r\n\r\n    addCSVModel(modelName: string) {\r\n        return this.getModelWithReference(this.addModel(modelName, {\r\n            contentType: MediaType.TEXT_CSV,\r\n            modelName,\r\n            schema: {},\r\n        }));\r\n    }\r\n\r\n    private getModelWithReference(model: Model): ModelWithReference {\r\n        return R.assoc('modelReference', getModelReference(model.modelId, this.restApiId), model) as ModelWithReference;\r\n    }\r\n\r\n    private addDocumentationPart(\r\n        resource: Resource, parameterName: string, resourceName: string, type: string, properties: DocumentationProperties,\r\n    ) {\r\n        const location: CfnDocumentationPart.LocationProperty = {\r\n            type,\r\n            path: resource.path,\r\n            name: type !== 'METHOD' ? parameterName : undefined,\r\n        };\r\n\r\n        new CfnDocumentationPart(this.stack, resourceName, {\r\n            restApiId: resource.api.restApiId,\r\n            location,\r\n            properties: JSON.stringify(properties),\r\n        });\r\n    }\r\n\r\n    documentResource(resource: Resource, ...documentationPart: DocumentationPart[]) {\r\n        documentationPart.forEach(dp => this.addDocumentationPart(\r\n            resource, dp.parameterName, `${resource.path}.${dp.parameterName}.Documentation`, dp.type, dp.documentationProperties,\r\n        ));\r\n    }\r\n}\r\n\r\n/**\r\n * Due to AWS API design API Gateway will always return 403 'Missing Authentication Token' for requests\r\n * with a non-existent endpoint. This function translates this response to a 404.\r\n * Requests with an invalid or missing API key are not affected (still return 403 'Forbidden').\r\n * @param restApi RestApi\r\n * @param stack Construct\r\n */\r\nexport function add404Support(restApi: RestApi, stack: Construct) {\r\n    new GatewayResponse(stack, `MissingAuthenticationTokenResponse-${restApi.restApiName}`, {\r\n        restApi,\r\n        type: ResponseType.MISSING_AUTHENTICATION_TOKEN,\r\n        statusCode: '404',\r\n        templates: {\r\n            'application/json': '{\"message\": \"Not found\"}',\r\n        },\r\n    });\r\n}\r\n\r\nexport function add401Support(restApi: RestApi, stack: Construct) {\r\n    new GatewayResponse(stack, `AuthenticationFailedResponse-${restApi.restApiName}`, {\r\n        restApi,\r\n        type: ResponseType.UNAUTHORIZED,\r\n        statusCode: \"401\",\r\n        responseHeaders: {\r\n            'WWW-Authenticate': \"'Basic'\",\r\n        },\r\n    });\r\n}\r\n\r\n/**\r\n * Due to AWS API design API Gateway will always return 403 'Missing Authentication Token' for requests\r\n * with a non-existent endpoint. This function converts this response to a custom one.\r\n * Requests with an invalid or missing API key are not affected (still return 403 'Forbidden').\r\n * @param returnCode\r\n * @param message\r\n * @param restApi RestApi\r\n * @param stack Construct\r\n */\r\nexport function setReturnCodeForMissingAuthenticationToken(returnCode: number,\r\n    message: string,\r\n    restApi: RestApi,\r\n    stack: Construct) {\r\n\r\n    new GatewayResponse(stack, `MissingAuthenticationTokenResponse-${restApi.restApiName}`, {\r\n        restApi,\r\n        type: ResponseType.MISSING_AUTHENTICATION_TOKEN,\r\n        statusCode: `${returnCode}`,\r\n        templates: {\r\n            'application/json': `{\"message\": ${message}}`,\r\n        },\r\n    });\r\n}\r\n\r\nexport function createRestApi(stack: Construct, apiId: string, apiName: string, allowFromIpAddresses?: string[] | undefined): RestApi {\r\n    const policyDocument = allowFromIpAddresses == null ? createDefaultPolicyDocument() : createIpRestrictionPolicyDocument(allowFromIpAddresses);\r\n    const restApi = new RestApi(stack, apiId, {\r\n        deployOptions: {\r\n            loggingLevel: MethodLoggingLevel.ERROR,\r\n        },\r\n        restApiName: apiName,\r\n        endpointTypes: [EndpointType.REGIONAL],\r\n        policy: policyDocument,\r\n    });\r\n    add404Support(restApi, stack);\r\n    return restApi;\r\n}\r\n\r\nexport function createDefaultPolicyDocument() {\r\n    return new PolicyDocument({\r\n        statements: [\r\n            new PolicyStatement({\r\n                effect: Effect.ALLOW,\r\n                actions: [\r\n                    \"execute-api:Invoke\",\r\n                ],\r\n                resources: [\r\n                    \"*\",\r\n                ],\r\n                principals: [\r\n                    new AnyPrincipal(),\r\n                ],\r\n            }),\r\n        ],\r\n    });\r\n}\r\n\r\n\r\nexport function createIpRestrictionPolicyDocument(allowFromIpAddresses: string[]): PolicyDocument {\r\n    return new PolicyDocument({\r\n        statements: [\r\n            new PolicyStatement({\r\n                effect: Effect.ALLOW,\r\n                conditions: {\r\n                    \"IpAddress\": {\r\n                        \"aws:SourceIp\": allowFromIpAddresses,\r\n                    },\r\n                },\r\n                actions: [\r\n                    \"execute-api:Invoke\",\r\n                ],\r\n                resources: [\r\n                    \"*\",\r\n                ],\r\n                principals: [\r\n                    new AnyPrincipal(),\r\n                ],\r\n            }),\r\n        ],\r\n    });\r\n}\r\n"]}
|
@@ -0,0 +1,163 @@
|
|
1
|
+
"use strict";
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
3
|
+
exports.StackCheckingAspect = void 0;
|
4
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
5
|
+
const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
|
6
|
+
const aws_s3_1 = require("aws-cdk-lib/aws-s3");
|
7
|
+
const stack_1 = require("./stack");
|
8
|
+
const aws_apigateway_1 = require("aws-cdk-lib/aws-apigateway");
|
9
|
+
const change_case_1 = require("change-case");
|
10
|
+
const aws_sqs_1 = require("aws-cdk-lib/aws-sqs");
|
11
|
+
const aws_logs_1 = require("aws-cdk-lib/aws-logs");
|
12
|
+
const MAX_CONCURRENCY_LIMIT = 100;
|
13
|
+
const NODE_RUNTIME = aws_lambda_1.Runtime.NODEJS_14_X.name;
|
14
|
+
var ResourceType;
|
15
|
+
(function (ResourceType) {
|
16
|
+
ResourceType["stackName"] = "STACK_NAME";
|
17
|
+
ResourceType["reservedConcurrentConcurrency"] = "RESERVED_CONCURRENT_CONCURRENCY";
|
18
|
+
ResourceType["functionTimeout"] = "FUNCTION_TIMEOUT";
|
19
|
+
ResourceType["functionMemorySize"] = "FUNCTION_MEMORY_SIZE";
|
20
|
+
ResourceType["functionRuntime"] = "FUNCTION_RUNTIME";
|
21
|
+
ResourceType["functionName"] = "FUNCTION_NAME";
|
22
|
+
ResourceType["tagSolution"] = "TAG_SOLUTION";
|
23
|
+
ResourceType["bucketPublicity"] = "BUCKET_PUBLICITY";
|
24
|
+
ResourceType["resourcePath"] = "RESOURCE_PATH";
|
25
|
+
ResourceType["queueEncryption"] = "QUEUE_ENCRYPTION";
|
26
|
+
ResourceType["logGroupRetention"] = "LOG_GROUP_RETENTION";
|
27
|
+
})(ResourceType || (ResourceType = {}));
|
28
|
+
class StackCheckingAspect {
|
29
|
+
constructor(configuration) {
|
30
|
+
this.configuration = configuration;
|
31
|
+
}
|
32
|
+
static create(stack) {
|
33
|
+
return new StackCheckingAspect(stack.configuration);
|
34
|
+
}
|
35
|
+
visit(node) {
|
36
|
+
//console.info("visiting class " + node.constructor.name);
|
37
|
+
this.checkStack(node);
|
38
|
+
this.checkFunction(node);
|
39
|
+
this.checkTags(node);
|
40
|
+
this.checkBucket(node);
|
41
|
+
this.checkResourceCasing(node);
|
42
|
+
this.checkQueueEncryption(node);
|
43
|
+
this.checkLogGroupRetention(node);
|
44
|
+
}
|
45
|
+
isWhitelisted(key) {
|
46
|
+
return this?.configuration?.whitelistedResources?.some(wl => {
|
47
|
+
return key.matchAll(new RegExp(wl, 'g'));
|
48
|
+
});
|
49
|
+
}
|
50
|
+
addAnnotation(node, key, message, isError = true) {
|
51
|
+
const resourceKey = `${node.node.path}/${key}`;
|
52
|
+
const isWhiteListed = this.isWhitelisted(resourceKey);
|
53
|
+
const annotationMessage = `${resourceKey}:${message}`;
|
54
|
+
// error && whitelisted -> warning
|
55
|
+
// warning && whitelisted -> nothing
|
56
|
+
if (isError && !isWhiteListed) {
|
57
|
+
aws_cdk_lib_1.Annotations.of(node).addError(annotationMessage);
|
58
|
+
}
|
59
|
+
else if ((!isError && !isWhiteListed) || (isError && isWhiteListed)) {
|
60
|
+
aws_cdk_lib_1.Annotations.of(node).addWarning(annotationMessage);
|
61
|
+
}
|
62
|
+
}
|
63
|
+
checkStack(node) {
|
64
|
+
if (node instanceof stack_1.DigitrafficStack) {
|
65
|
+
if ((node.stackName.includes('Test') || node.stackName.includes('Tst')) && node.configuration?.production) {
|
66
|
+
this.addAnnotation(node, ResourceType.stackName, 'Production is set for Test-stack');
|
67
|
+
}
|
68
|
+
if ((node.stackName.includes('Prod') || node.stackName.includes('Prd')) && !node.configuration?.production) {
|
69
|
+
this.addAnnotation(node, ResourceType.stackName, 'Production is not set for Production-stack');
|
70
|
+
}
|
71
|
+
}
|
72
|
+
}
|
73
|
+
checkFunction(node) {
|
74
|
+
if (node instanceof aws_lambda_1.CfnFunction) {
|
75
|
+
if (!node.reservedConcurrentExecutions) {
|
76
|
+
this.addAnnotation(node, ResourceType.reservedConcurrentConcurrency, 'Function must have reservedConcurrentConcurrency');
|
77
|
+
}
|
78
|
+
else if (node.reservedConcurrentExecutions > MAX_CONCURRENCY_LIMIT) {
|
79
|
+
this.addAnnotation(node, ResourceType.reservedConcurrentConcurrency, 'Function reservedConcurrentConcurrency too high!');
|
80
|
+
}
|
81
|
+
if (!node.timeout) {
|
82
|
+
this.addAnnotation(node, ResourceType.functionTimeout, 'Function must have timeout');
|
83
|
+
}
|
84
|
+
if (!node.memorySize) {
|
85
|
+
this.addAnnotation(node, ResourceType.functionMemorySize, 'Function must have memorySize');
|
86
|
+
}
|
87
|
+
if (node.runtime !== NODE_RUNTIME) {
|
88
|
+
this.addAnnotation(node, ResourceType.functionRuntime, 'wrong runtime ' + node.runtime);
|
89
|
+
}
|
90
|
+
if (this.configuration?.shortName && node.functionName && node.functionName.indexOf(this.configuration.shortName) !== 0) {
|
91
|
+
this.addAnnotation(node, ResourceType.functionName, 'Function name does not begin with ' + this.configuration.shortName);
|
92
|
+
}
|
93
|
+
}
|
94
|
+
}
|
95
|
+
checkTags(node) {
|
96
|
+
if (node instanceof aws_cdk_lib_1.Stack) {
|
97
|
+
if (!node.tags.tagValues()[stack_1.SOLUTION_KEY]) {
|
98
|
+
this.addAnnotation(node, ResourceType.tagSolution, 'Solution tag is missing');
|
99
|
+
}
|
100
|
+
}
|
101
|
+
}
|
102
|
+
checkBucket(node) {
|
103
|
+
if (node instanceof aws_s3_1.CfnBucket) {
|
104
|
+
const c = node.publicAccessBlockConfiguration;
|
105
|
+
if (c) {
|
106
|
+
if (!c.blockPublicAcls || !c.blockPublicPolicy || !c.ignorePublicAcls || !c.restrictPublicBuckets) {
|
107
|
+
this.addAnnotation(node, ResourceType.bucketPublicity, 'Check bucket publicity');
|
108
|
+
}
|
109
|
+
}
|
110
|
+
}
|
111
|
+
}
|
112
|
+
static isValidPath(path) {
|
113
|
+
// if path includes . or { check only the trailing part of path
|
114
|
+
if (path.includes('.')) {
|
115
|
+
return this.isValidPath(path.split('.')[0]);
|
116
|
+
}
|
117
|
+
if (path.includes('{')) {
|
118
|
+
return this.isValidPath(path.split('{')[0]);
|
119
|
+
}
|
120
|
+
return (0, change_case_1.paramCase)(path) === path;
|
121
|
+
}
|
122
|
+
static isValidQueryString(name) {
|
123
|
+
return (0, change_case_1.snakeCase)(name) === name;
|
124
|
+
}
|
125
|
+
checkResourceCasing(node) {
|
126
|
+
if (node instanceof aws_apigateway_1.CfnResource) {
|
127
|
+
if (!StackCheckingAspect.isValidPath(node.pathPart)) {
|
128
|
+
this.addAnnotation(node, ResourceType.resourcePath, 'Path part should be in kebab-case');
|
129
|
+
}
|
130
|
+
}
|
131
|
+
else if (node instanceof aws_apigateway_1.CfnMethod) {
|
132
|
+
const integration = node.integration;
|
133
|
+
if (integration && integration.requestParameters) {
|
134
|
+
Object.keys(integration.requestParameters).forEach(key => {
|
135
|
+
const split = key.split('.');
|
136
|
+
const type = split[2];
|
137
|
+
const name = split[3];
|
138
|
+
if (type === 'querystring' && !StackCheckingAspect.isValidQueryString(name)) {
|
139
|
+
this.addAnnotation(node, name, 'Querystring should be in snake_case');
|
140
|
+
}
|
141
|
+
});
|
142
|
+
}
|
143
|
+
}
|
144
|
+
}
|
145
|
+
checkQueueEncryption(node) {
|
146
|
+
if (node instanceof aws_sqs_1.CfnQueue) {
|
147
|
+
if (!node.kmsMasterKeyId) {
|
148
|
+
this.addAnnotation(node, ResourceType.queueEncryption, 'Queue must have encryption enabled');
|
149
|
+
}
|
150
|
+
}
|
151
|
+
}
|
152
|
+
checkLogGroupRetention(node) {
|
153
|
+
if (node instanceof aws_logs_1.LogRetention) {
|
154
|
+
const child = node.node.defaultChild;
|
155
|
+
const retention = child._cfnProperties.RetentionInDays;
|
156
|
+
if (!retention) {
|
157
|
+
this.addAnnotation(node, ResourceType.logGroupRetention, 'Log group must define log group retention');
|
158
|
+
}
|
159
|
+
}
|
160
|
+
}
|
161
|
+
}
|
162
|
+
exports.StackCheckingAspect = StackCheckingAspect;
|
163
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stack-checking-aspect.js","sourceRoot":"","sources":["../../../../../src/aws/infra/stack/stack-checking-aspect.ts"],"names":[],"mappings":"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","sourcesContent":["import {Annotations, IAspect, Stack} from \"aws-cdk-lib\";\nimport {CfnFunction, Runtime} from 'aws-cdk-lib/aws-lambda';\nimport {CfnBucket} from \"aws-cdk-lib/aws-s3\";\nimport {DigitrafficStack, SOLUTION_KEY, StackConfiguration} from \"./stack\";\nimport {IConstruct} from \"constructs\";\nimport {CfnMethod, CfnResource} from \"aws-cdk-lib/aws-apigateway\";\nimport {paramCase, snakeCase} from \"change-case\";\nimport {CfnQueue} from \"aws-cdk-lib/aws-sqs\";\nimport {LogRetention} from \"aws-cdk-lib/aws-logs\";\nimport IntegrationProperty = CfnMethod.IntegrationProperty;\n\nconst MAX_CONCURRENCY_LIMIT = 100;\nconst NODE_RUNTIME = Runtime.NODEJS_14_X.name;\n\nenum ResourceType {\n    stackName = \"STACK_NAME\",\n    reservedConcurrentConcurrency = \"RESERVED_CONCURRENT_CONCURRENCY\",\n    functionTimeout = \"FUNCTION_TIMEOUT\",\n    functionMemorySize = \"FUNCTION_MEMORY_SIZE\",\n    functionRuntime = \"FUNCTION_RUNTIME\",\n    functionName = \"FUNCTION_NAME\",\n    tagSolution = \"TAG_SOLUTION\",\n    bucketPublicity = \"BUCKET_PUBLICITY\",\n    resourcePath = \"RESOURCE_PATH\",\n    queueEncryption = \"QUEUE_ENCRYPTION\",\n    logGroupRetention = \"LOG_GROUP_RETENTION\",\n}\n\nexport class StackCheckingAspect implements IAspect {\n    private readonly configuration?: StackConfiguration;\n\n    constructor(configuration?: StackConfiguration) {\n        this.configuration = configuration;\n    }\n\n    static create(stack: DigitrafficStack) {\n        return new StackCheckingAspect(stack.configuration);\n    }\n\n    public visit(node: IConstruct): void {\n        //console.info(\"visiting class \" + node.constructor.name);\n\n        this.checkStack(node);\n        this.checkFunction(node);\n        this.checkTags(node);\n        this.checkBucket(node);\n        this.checkResourceCasing(node);\n        this.checkQueueEncryption(node);\n        this.checkLogGroupRetention(node);\n    }\n\n    private isWhitelisted(key: string) {\n        return this?.configuration?.whitelistedResources?.some(wl => {\n            return key.matchAll(new RegExp(wl, 'g'));\n        });\n    }\n\n    private addAnnotation(node: IConstruct, key: ResourceType | string, message: string, isError = true) {\n        const resourceKey = `${node.node.path}/${key}`;\n        const isWhiteListed = this.isWhitelisted(resourceKey);\n        const annotationMessage = `${resourceKey}:${message}`;\n\n        // error && whitelisted -> warning\n        // warning && whitelisted -> nothing\n        if (isError && !isWhiteListed) {\n            Annotations.of(node).addError(annotationMessage);\n        } else if ((!isError && !isWhiteListed) || (isError && isWhiteListed)) {\n            Annotations.of(node).addWarning(annotationMessage);\n        }\n    }\n\n    private checkStack(node: IConstruct) {\n        if (node instanceof DigitrafficStack) {\n            if ((node.stackName.includes('Test') || node.stackName.includes('Tst')) && node.configuration?.production) {\n                this.addAnnotation(node, ResourceType.stackName, 'Production is set for Test-stack');\n            }\n\n            if ((node.stackName.includes('Prod') || node.stackName.includes('Prd')) && !node.configuration?.production) {\n                this.addAnnotation(node, ResourceType.stackName, 'Production is not set for Production-stack');\n            }\n        }\n    }\n\n    private checkFunction(node: IConstruct) {\n        if (node instanceof CfnFunction) {\n            if (!node.reservedConcurrentExecutions) {\n                this.addAnnotation(node, ResourceType.reservedConcurrentConcurrency, 'Function must have reservedConcurrentConcurrency');\n            } else if (node.reservedConcurrentExecutions > MAX_CONCURRENCY_LIMIT) {\n                this.addAnnotation(node, ResourceType.reservedConcurrentConcurrency, 'Function reservedConcurrentConcurrency too high!');\n            }\n\n            if (!node.timeout) {\n                this.addAnnotation(node, ResourceType.functionTimeout, 'Function must have timeout');\n            }\n\n            if (!node.memorySize) {\n                this.addAnnotation(node, ResourceType.functionMemorySize, 'Function must have memorySize');\n            }\n\n            if (node.runtime !== NODE_RUNTIME) {\n                this.addAnnotation(node, ResourceType.functionRuntime,'wrong runtime ' + node.runtime);\n            }\n\n            if (this.configuration?.shortName && node.functionName && node.functionName.indexOf(this.configuration.shortName) !== 0) {\n                this.addAnnotation(node, ResourceType.functionName, 'Function name does not begin with ' + this.configuration.shortName);\n            }\n        }\n    }\n\n    private checkTags(node: IConstruct) {\n        if (node instanceof Stack) {\n            if (!node.tags.tagValues()[SOLUTION_KEY]) {\n                this.addAnnotation(node, ResourceType.tagSolution, 'Solution tag is missing');\n            }\n        }\n    }\n\n    private checkBucket(node: IConstruct) {\n        if (node instanceof CfnBucket) {\n            const c = node.publicAccessBlockConfiguration as CfnBucket.PublicAccessBlockConfigurationProperty;\n\n            if (c) {\n                if (!c.blockPublicAcls || !c.blockPublicPolicy || !c.ignorePublicAcls || !c.restrictPublicBuckets) {\n                    this.addAnnotation(node, ResourceType.bucketPublicity, 'Check bucket publicity');\n                }\n            }\n        }\n    }\n\n    private static isValidPath(path: string): boolean {\n        // if path includes . or { check only the trailing part of path\n        if (path.includes('.')) {\n            return this.isValidPath(path.split('.')[0]);\n        }\n\n        if (path.includes('{')) {\n            return this.isValidPath(path.split('{')[0]);\n        }\n\n        return paramCase(path) === path;\n    }\n\n    private static isValidQueryString(name: string) {\n        return snakeCase(name) === name;\n    }\n\n    private checkResourceCasing(node: IConstruct) {\n        if (node instanceof CfnResource) {\n            if (!StackCheckingAspect.isValidPath(node.pathPart)) {\n                this.addAnnotation(node, ResourceType.resourcePath, 'Path part should be in kebab-case');\n            }\n        } else if (node instanceof CfnMethod) {\n            const integration = node.integration as IntegrationProperty;\n\n            if (integration && integration.requestParameters) {\n                Object.keys(integration.requestParameters).forEach(key => {\n                    const split = key.split('.');\n                    const type = split[2];\n                    const name = split[3];\n\n                    if (type === 'querystring' && !StackCheckingAspect.isValidQueryString(name)) {\n                        this.addAnnotation(node, name, 'Querystring should be in snake_case');\n                    }\n                });\n            }\n        }\n    }\n\n    private checkQueueEncryption(node: IConstruct) {\n        if (node instanceof CfnQueue) {\n            if (!node.kmsMasterKeyId) {\n                this.addAnnotation(node, ResourceType.queueEncryption, 'Queue must have encryption enabled');\n            }\n        }\n    }\n\n    private checkLogGroupRetention(node: IConstruct) {\n        if (node instanceof LogRetention) {\n            const child = node.node.defaultChild as unknown as Record<string, Record<string, string>>;\n            const retention = child._cfnProperties.RetentionInDays;\n\n            if (!retention) {\n                this.addAnnotation(node, ResourceType.logGroupRetention, 'Log group must define log group retention');\n            }\n        }\n    }\n}\n"]}
|
@@ -0,0 +1,58 @@
|
|
1
|
+
"use strict";
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
3
|
+
exports.DigitrafficStack = exports.SSM_KEY_ALARM_TOPIC = exports.SSM_KEY_WARNING_TOPIC = exports.SOLUTION_KEY = void 0;
|
4
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
5
|
+
const aws_ec2_1 = require("aws-cdk-lib/aws-ec2");
|
6
|
+
const aws_sns_1 = require("aws-cdk-lib/aws-sns");
|
7
|
+
const aws_ssm_1 = require("aws-cdk-lib/aws-ssm");
|
8
|
+
const aws_secretsmanager_1 = require("aws-cdk-lib/aws-secretsmanager");
|
9
|
+
const stack_checking_aspect_1 = require("./stack-checking-aspect");
|
10
|
+
const SSM_ROOT = '/digitraffic';
|
11
|
+
exports.SOLUTION_KEY = 'Solution';
|
12
|
+
const MONITORING_ROOT = '/monitoring';
|
13
|
+
exports.SSM_KEY_WARNING_TOPIC = `${SSM_ROOT}${MONITORING_ROOT}/warning-topic`;
|
14
|
+
exports.SSM_KEY_ALARM_TOPIC = `${SSM_ROOT}${MONITORING_ROOT}/alarm-topic`;
|
15
|
+
class DigitrafficStack extends aws_cdk_lib_1.Stack {
|
16
|
+
constructor(scope, id, configuration) {
|
17
|
+
super(scope, id, configuration.stackProps);
|
18
|
+
this.configuration = configuration;
|
19
|
+
if (configuration.secretId) {
|
20
|
+
this.secret = aws_secretsmanager_1.Secret.fromSecretNameV2(this, 'Secret', configuration.secretId);
|
21
|
+
}
|
22
|
+
// VPC reference construction requires vpcId and availability zones
|
23
|
+
// private subnets are used in Lambda configuration
|
24
|
+
if (configuration.vpcId) {
|
25
|
+
this.vpc = aws_ec2_1.Vpc.fromVpcAttributes(this, 'vpc', {
|
26
|
+
vpcId: configuration.vpcId,
|
27
|
+
privateSubnetIds: configuration.privateSubnetIds,
|
28
|
+
availabilityZones: configuration.availabilityZones,
|
29
|
+
});
|
30
|
+
}
|
31
|
+
// security group that allows Lambda database access
|
32
|
+
if (configuration.lambdaDbSgId) {
|
33
|
+
this.lambdaDbSg = aws_ec2_1.SecurityGroup.fromSecurityGroupId(this, 'LambdaDbSG', configuration.lambdaDbSgId);
|
34
|
+
}
|
35
|
+
this.alarmTopic = aws_sns_1.Topic.fromTopicArn(this, 'AlarmTopic', aws_ssm_1.StringParameter.fromStringParameterName(this, 'AlarmTopicParam', exports.SSM_KEY_ALARM_TOPIC).stringValue);
|
36
|
+
this.warningTopic = aws_sns_1.Topic.fromTopicArn(this, 'WarningTopic', aws_ssm_1.StringParameter.fromStringParameterName(this, 'WarningTopicParam', exports.SSM_KEY_WARNING_TOPIC).stringValue);
|
37
|
+
this.addAspects();
|
38
|
+
}
|
39
|
+
addAspects() {
|
40
|
+
aws_cdk_lib_1.Aspects.of(this).add(stack_checking_aspect_1.StackCheckingAspect.create(this));
|
41
|
+
}
|
42
|
+
createLambdaEnvironment() {
|
43
|
+
return this.createDefaultLambdaEnvironment(this.configuration.shortName);
|
44
|
+
}
|
45
|
+
createDefaultLambdaEnvironment(dbApplication) {
|
46
|
+
return this.configuration.secretId ? {
|
47
|
+
SECRET_ID: this.configuration.secretId,
|
48
|
+
DB_APPLICATION: dbApplication,
|
49
|
+
} : {
|
50
|
+
DB_APPLICATION: dbApplication,
|
51
|
+
};
|
52
|
+
}
|
53
|
+
grantSecret(...lambdas) {
|
54
|
+
lambdas.forEach((l) => this.secret.grantRead(l));
|
55
|
+
}
|
56
|
+
}
|
57
|
+
exports.DigitrafficStack = DigitrafficStack;
|
58
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhY2suanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvYXdzL2luZnJhL3N0YWNrL3N0YWNrLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLDZDQUF1RDtBQUN2RCxpREFBNkQ7QUFFN0QsaURBQWtEO0FBQ2xELGlEQUFvRDtBQUNwRCx1RUFBK0Q7QUFHL0QsbUVBQTREO0FBSzVELE1BQU0sUUFBUSxHQUFHLGNBQWMsQ0FBQztBQUNuQixRQUFBLFlBQVksR0FBRyxVQUFVLENBQUM7QUFDdkMsTUFBTSxlQUFlLEdBQUcsYUFBYSxDQUFDO0FBRXpCLFFBQUEscUJBQXFCLEdBQUcsR0FBRyxRQUFRLEdBQUcsZUFBZSxnQkFBZ0IsQ0FBQztBQUN0RSxRQUFBLG1CQUFtQixHQUFHLEdBQUcsUUFBUSxHQUFHLGVBQWUsY0FBYyxDQUFDO0FBc0IvRSxNQUFhLGdCQUFpQixTQUFRLG1CQUFLO0lBU3ZDLFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsYUFBaUM7UUFDdkUsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLEVBQUUsYUFBYSxDQUFDLFVBQVUsQ0FBQyxDQUFDO1FBRTNDLElBQUksQ0FBQyxhQUFhLEdBQUcsYUFBYSxDQUFDO1FBRW5DLElBQUksYUFBYSxDQUFDLFFBQVEsRUFBRTtZQUN4QixJQUFJLENBQUMsTUFBTSxHQUFHLDJCQUFNLENBQUMsZ0JBQWdCLENBQUMsSUFBSSxFQUFFLFFBQVEsRUFBRSxhQUFhLENBQUMsUUFBUSxDQUFDLENBQUM7U0FDakY7UUFFRCxtRUFBbUU7UUFDbkUsbURBQW1EO1FBQ25ELElBQUksYUFBYSxDQUFDLEtBQUssRUFBRTtZQUNyQixJQUFJLENBQUMsR0FBRyxHQUFHLGFBQUcsQ0FBQyxpQkFBaUIsQ0FBQyxJQUFJLEVBQUUsS0FBSyxFQUFFO2dCQUMxQyxLQUFLLEVBQUUsYUFBYSxDQUFDLEtBQUs7Z0JBQzFCLGdCQUFnQixFQUFFLGFBQWEsQ0FBQyxnQkFBZ0I7Z0JBQ2hELGlCQUFpQixFQUFFLGFBQWEsQ0FBQyxpQkFBNkI7YUFDakUsQ0FBQyxDQUFDO1NBQ047UUFFRCxvREFBb0Q7UUFDcEQsSUFBSSxhQUFhLENBQUMsWUFBWSxFQUFFO1lBQzVCLElBQUksQ0FBQyxVQUFVLEdBQUcsdUJBQWEsQ0FBQyxtQkFBbUIsQ0FBQyxJQUFJLEVBQUUsWUFBWSxFQUFFLGFBQWEsQ0FBQyxZQUFZLENBQUMsQ0FBQztTQUN2RztRQUVELElBQUksQ0FBQyxVQUFVLEdBQUcsZUFBSyxDQUFDLFlBQVksQ0FBQyxJQUFJLEVBQ3JDLFlBQVksRUFDWix5QkFBZSxDQUFDLHVCQUF1QixDQUFDLElBQUksRUFBRSxpQkFBaUIsRUFBRSwyQkFBbUIsQ0FBQyxDQUFDLFdBQVcsQ0FBQyxDQUFDO1FBQ3ZHLElBQUksQ0FBQyxZQUFZLEdBQUcsZUFBSyxDQUFDLFlBQVksQ0FBQyxJQUFJLEVBQUUsY0FBYyxFQUN2RCx5QkFBZSxDQUFDLHVCQUF1QixDQUFDLElBQUksRUFBRSxtQkFBbUIsRUFBRSw2QkFBcUIsQ0FBQyxDQUFDLFdBQVcsQ0FBQyxDQUFDO1FBRTNHLElBQUksQ0FBQyxVQUFVLEVBQUUsQ0FBQztJQUN0QixDQUFDO0lBRUQsVUFBVTtRQUNOLHFCQUFPLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxDQUFDLEdBQUcsQ0FBQywyQ0FBbUIsQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQztJQUMzRCxDQUFDO0lBRUQsdUJBQXVCO1FBQ25CLE9BQU8sSUFBSSxDQUFDLDhCQUE4QixDQUFDLElBQUksQ0FBQyxhQUFhLENBQUMsU0FBbUIsQ0FBQyxDQUFDO0lBQ3ZGLENBQUM7SUFFRCw4QkFBOEIsQ0FBQyxhQUFxQjtRQUNoRCxPQUFPLElBQUksQ0FBQyxhQUFhLENBQUMsUUFBUSxDQUFDLENBQUMsQ0FBQztZQUNqQyxTQUFTLEVBQUUsSUFBSSxDQUFDLGFBQWEsQ0FBQyxRQUFRO1lBQ3RDLGNBQWMsRUFBRSxhQUFhO1NBQ2hDLENBQUMsQ0FBQyxDQUFDO1lBQ0EsY0FBYyxFQUFFLGFBQWE7U0FDaEMsQ0FBQztJQUNOLENBQUM7SUFFRCxXQUFXLENBQUMsR0FBRyxPQUFtQjtRQUM5QixPQUFPLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBVyxFQUFFLEVBQUUsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQy9ELENBQUM7Q0FDSjtBQTlERCw0Q0E4REMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0FzcGVjdHMsIFN0YWNrLCBTdGFja1Byb3BzfSBmcm9tIFwiYXdzLWNkay1saWJcIjtcbmltcG9ydCB7SVZwYywgU2VjdXJpdHlHcm91cCwgVnBjfSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWVjMlwiO1xuaW1wb3J0IHtJU2VjdXJpdHlHcm91cH0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1lYzIvbGliL3NlY3VyaXR5LWdyb3VwXCI7XG5pbXBvcnQge0lUb3BpYywgVG9waWN9IGZyb20gXCJhd3MtY2RrLWxpYi9hd3Mtc25zXCI7XG5pbXBvcnQge1N0cmluZ1BhcmFtZXRlcn0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1zc21cIjtcbmltcG9ydCB7SVNlY3JldCwgU2VjcmV0fSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLXNlY3JldHNtYW5hZ2VyXCI7XG5pbXBvcnQge0Z1bmN0aW9ufSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWxhbWJkYVwiO1xuXG5pbXBvcnQge1N0YWNrQ2hlY2tpbmdBc3BlY3R9IGZyb20gXCIuL3N0YWNrLWNoZWNraW5nLWFzcGVjdFwiO1xuaW1wb3J0IHtDb25zdHJ1Y3R9IGZyb20gXCJjb25zdHJ1Y3RzXCI7XG5pbXBvcnQge1RyYWZmaWNUeXBlfSBmcm9tIFwiLi4vLi4vLi4vdHlwZXMvdHJhZmZpY3R5cGVcIjtcbmltcG9ydCB7REJMYW1iZGFFbnZpcm9ubWVudH0gZnJvbSBcIi4vbGFtYmRhLWNvbmZpZ3NcIjtcblxuY29uc3QgU1NNX1JPT1QgPSAnL2RpZ2l0cmFmZmljJztcbmV4cG9ydCBjb25zdCBTT0xVVElPTl9LRVkgPSAnU29sdXRpb24nO1xuY29uc3QgTU9OSVRPUklOR19ST09UID0gJy9tb25pdG9yaW5nJztcblxuZXhwb3J0IGNvbnN0IFNTTV9LRVlfV0FSTklOR19UT1BJQyA9IGAke1NTTV9ST09UfSR7TU9OSVRPUklOR19ST09UfS93YXJuaW5nLXRvcGljYDtcbmV4cG9ydCBjb25zdCBTU01fS0VZX0FMQVJNX1RPUElDID0gYCR7U1NNX1JPT1R9JHtNT05JVE9SSU5HX1JPT1R9L2FsYXJtLXRvcGljYDtcblxuZXhwb3J0IHR5cGUgU3RhY2tDb25maWd1cmF0aW9uID0ge1xuICAgIHJlYWRvbmx5IHNob3J0TmFtZT86IHN0cmluZztcbiAgICByZWFkb25seSBzZWNyZXRJZD86IHN0cmluZztcbiAgICByZWFkb25seSBhbGFybVRvcGljQXJuOiBzdHJpbmc7XG4gICAgcmVhZG9ubHkgd2FybmluZ1RvcGljQXJuOiBzdHJpbmc7XG4gICAgcmVhZG9ubHkgZW5hYmxlQ2FuYXJpZXM6IGJvb2xlYW47XG4gICAgcmVhZG9ubHkgbG9nc0Rlc3RpbmF0aW9uQXJuPzogc3RyaW5nO1xuXG4gICAgcmVhZG9ubHkgdnBjSWQ/OiBzdHJpbmc7XG4gICAgcmVhZG9ubHkgbGFtYmRhRGJTZ0lkPzogc3RyaW5nO1xuICAgIHJlYWRvbmx5IHByaXZhdGVTdWJuZXRJZHM/OiBzdHJpbmdbXTtcbiAgICByZWFkb25seSBhdmFpbGFiaWxpdHlab25lcz86IHN0cmluZ1tdO1xuXG4gICAgcmVhZG9ubHkgdHJhZmZpY1R5cGU6IFRyYWZmaWNUeXBlO1xuICAgIHJlYWRvbmx5IHByb2R1Y3Rpb246IGJvb2xlYW47XG4gICAgcmVhZG9ubHkgc3RhY2tQcm9wczogU3RhY2tQcm9wcztcblxuICAgIHJlYWRvbmx5IHdoaXRlbGlzdGVkUmVzb3VyY2VzPzogc3RyaW5nW107XG59XG5cbmV4cG9ydCBjbGFzcyBEaWdpdHJhZmZpY1N0YWNrIGV4dGVuZHMgU3RhY2sge1xuICAgIHJlYWRvbmx5IHZwYzogSVZwYztcbiAgICByZWFkb25seSBsYW1iZGFEYlNnOiBJU2VjdXJpdHlHcm91cDtcbiAgICByZWFkb25seSBhbGFybVRvcGljOiBJVG9waWM7XG4gICAgcmVhZG9ubHkgd2FybmluZ1RvcGljOiBJVG9waWM7XG4gICAgcmVhZG9ubHkgc2VjcmV0OiBJU2VjcmV0O1xuXG4gICAgcmVhZG9ubHkgY29uZmlndXJhdGlvbjogU3RhY2tDb25maWd1cmF0aW9uO1xuXG4gICAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgY29uZmlndXJhdGlvbjogU3RhY2tDb25maWd1cmF0aW9uKSB7XG4gICAgICAgIHN1cGVyKHNjb3BlLCBpZCwgY29uZmlndXJhdGlvbi5zdGFja1Byb3BzKTtcblxuICAgICAgICB0aGlzLmNvbmZpZ3VyYXRpb24gPSBjb25maWd1cmF0aW9uO1xuXG4gICAgICAgIGlmIChjb25maWd1cmF0aW9uLnNlY3JldElkKSB7XG4gICAgICAgICAgICB0aGlzLnNlY3JldCA9IFNlY3JldC5mcm9tU2VjcmV0TmFtZVYyKHRoaXMsICdTZWNyZXQnLCBjb25maWd1cmF0aW9uLnNlY3JldElkKTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFZQQyByZWZlcmVuY2UgY29uc3RydWN0aW9uIHJlcXVpcmVzIHZwY0lkIGFuZCBhdmFpbGFiaWxpdHkgem9uZXNcbiAgICAgICAgLy8gcHJpdmF0ZSBzdWJuZXRzIGFyZSB1c2VkIGluIExhbWJkYSBjb25maWd1cmF0aW9uXG4gICAgICAgIGlmIChjb25maWd1cmF0aW9uLnZwY0lkKSB7XG4gICAgICAgICAgICB0aGlzLnZwYyA9IFZwYy5mcm9tVnBjQXR0cmlidXRlcyh0aGlzLCAndnBjJywge1xuICAgICAgICAgICAgICAgIHZwY0lkOiBjb25maWd1cmF0aW9uLnZwY0lkLFxuICAgICAgICAgICAgICAgIHByaXZhdGVTdWJuZXRJZHM6IGNvbmZpZ3VyYXRpb24ucHJpdmF0ZVN1Ym5ldElkcyxcbiAgICAgICAgICAgICAgICBhdmFpbGFiaWxpdHlab25lczogY29uZmlndXJhdGlvbi5hdmFpbGFiaWxpdHlab25lcyBhcyBzdHJpbmdbXSxcbiAgICAgICAgICAgIH0pO1xuICAgICAgICB9XG5cbiAgICAgICAgLy8gc2VjdXJpdHkgZ3JvdXAgdGhhdCBhbGxvd3MgTGFtYmRhIGRhdGFiYXNlIGFjY2Vzc1xuICAgICAgICBpZiAoY29uZmlndXJhdGlvbi5sYW1iZGFEYlNnSWQpIHtcbiAgICAgICAgICAgIHRoaXMubGFtYmRhRGJTZyA9IFNlY3VyaXR5R3JvdXAuZnJvbVNlY3VyaXR5R3JvdXBJZCh0aGlzLCAnTGFtYmRhRGJTRycsIGNvbmZpZ3VyYXRpb24ubGFtYmRhRGJTZ0lkKTtcbiAgICAgICAgfVxuXG4gICAgICAgIHRoaXMuYWxhcm1Ub3BpYyA9IFRvcGljLmZyb21Ub3BpY0Fybih0aGlzLFxuICAgICAgICAgICAgJ0FsYXJtVG9waWMnLFxuICAgICAgICAgICAgU3RyaW5nUGFyYW1ldGVyLmZyb21TdHJpbmdQYXJhbWV0ZXJOYW1lKHRoaXMsICdBbGFybVRvcGljUGFyYW0nLCBTU01fS0VZX0FMQVJNX1RPUElDKS5zdHJpbmdWYWx1ZSk7XG4gICAgICAgIHRoaXMud2FybmluZ1RvcGljID0gVG9waWMuZnJvbVRvcGljQXJuKHRoaXMsICdXYXJuaW5nVG9waWMnLFxuICAgICAgICAgICAgU3RyaW5nUGFyYW1ldGVyLmZyb21TdHJpbmdQYXJhbWV0ZXJOYW1lKHRoaXMsICdXYXJuaW5nVG9waWNQYXJhbScsIFNTTV9LRVlfV0FSTklOR19UT1BJQykuc3RyaW5nVmFsdWUpO1xuXG4gICAgICAgIHRoaXMuYWRkQXNwZWN0cygpO1xuICAgIH1cblxuICAgIGFkZEFzcGVjdHMoKSB7XG4gICAgICAgIEFzcGVjdHMub2YodGhpcykuYWRkKFN0YWNrQ2hlY2tpbmdBc3BlY3QuY3JlYXRlKHRoaXMpKTtcbiAgICB9XG5cbiAgICBjcmVhdGVMYW1iZGFFbnZpcm9ubWVudCgpOiBEQkxhbWJkYUVudmlyb25tZW50IHtcbiAgICAgICAgcmV0dXJuIHRoaXMuY3JlYXRlRGVmYXVsdExhbWJkYUVudmlyb25tZW50KHRoaXMuY29uZmlndXJhdGlvbi5zaG9ydE5hbWUgYXMgc3RyaW5nKTtcbiAgICB9XG5cbiAgICBjcmVhdGVEZWZhdWx0TGFtYmRhRW52aXJvbm1lbnQoZGJBcHBsaWNhdGlvbjogc3RyaW5nKTogREJMYW1iZGFFbnZpcm9ubWVudCB7XG4gICAgICAgIHJldHVybiB0aGlzLmNvbmZpZ3VyYXRpb24uc2VjcmV0SWQgPyB7XG4gICAgICAgICAgICBTRUNSRVRfSUQ6IHRoaXMuY29uZmlndXJhdGlvbi5zZWNyZXRJZCxcbiAgICAgICAgICAgIERCX0FQUExJQ0FUSU9OOiBkYkFwcGxpY2F0aW9uLFxuICAgICAgICB9IDoge1xuICAgICAgICAgICAgREJfQVBQTElDQVRJT046IGRiQXBwbGljYXRpb24sXG4gICAgICAgIH07XG4gICAgfVxuXG4gICAgZ3JhbnRTZWNyZXQoLi4ubGFtYmRhczogRnVuY3Rpb25bXSkge1xuICAgICAgICBsYW1iZGFzLmZvckVhY2goKGw6IEZ1bmN0aW9uKSA9PiB0aGlzLnNlY3JldC5ncmFudFJlYWQobCkpO1xuICAgIH1cbn1cbiJdfQ==
|
@@ -0,0 +1,41 @@
|
|
1
|
+
"use strict";
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
3
|
+
exports.DigitrafficLogSubscriptions = exports.createSubscription = void 0;
|
4
|
+
const aws_logs_1 = require("aws-cdk-lib/aws-logs");
|
5
|
+
/**
|
6
|
+
* Creates a subscription filter that subscribes to a Lambda Log Group and delivers the logs to another destination.
|
7
|
+
* https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-resource-logs-subscriptionfilter.html
|
8
|
+
* @param lambda The Lambda function, needed to create a dependency
|
9
|
+
* @param lambdaName The Lambda name from which the Log Group name is derived
|
10
|
+
* @param logDestinationArn Destination for streamed logs
|
11
|
+
* @param stack CloudFormation stack
|
12
|
+
*/
|
13
|
+
function createSubscription(lambda, lambdaName, logDestinationArn, stack) {
|
14
|
+
if (logDestinationArn == undefined) {
|
15
|
+
return undefined;
|
16
|
+
}
|
17
|
+
const filter = new aws_logs_1.CfnSubscriptionFilter(stack, `${lambdaName}LogsSubscription`, {
|
18
|
+
logGroupName: `/aws/lambda/${lambdaName}`,
|
19
|
+
filterPattern: '',
|
20
|
+
destinationArn: logDestinationArn,
|
21
|
+
});
|
22
|
+
filter.node.addDependency(lambda);
|
23
|
+
return filter;
|
24
|
+
}
|
25
|
+
exports.createSubscription = createSubscription;
|
26
|
+
class DigitrafficLogSubscriptions {
|
27
|
+
constructor(stack, ...lambdas) {
|
28
|
+
if (stack.configuration.logsDestinationArn) {
|
29
|
+
lambdas.forEach(lambda => {
|
30
|
+
const filter = new aws_logs_1.CfnSubscriptionFilter(stack, `${lambda.givenName}LogsSubscription`, {
|
31
|
+
logGroupName: `/aws/lambda/${lambda.givenName}`,
|
32
|
+
filterPattern: '',
|
33
|
+
destinationArn: stack.configuration.logsDestinationArn,
|
34
|
+
});
|
35
|
+
filter.node.addDependency(lambda);
|
36
|
+
});
|
37
|
+
}
|
38
|
+
}
|
39
|
+
}
|
40
|
+
exports.DigitrafficLogSubscriptions = DigitrafficLogSubscriptions;
|
41
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3Vic2NyaXB0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2F3cy9pbmZyYS9zdGFjay9zdWJzY3JpcHRpb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsbURBQTJEO0FBTTNEOzs7Ozs7O0dBT0c7QUFDSCxTQUFnQixrQkFBa0IsQ0FBQyxNQUFnQixFQUFFLFVBQWtCLEVBQUUsaUJBQXFDLEVBQUUsS0FBZ0I7SUFDNUgsSUFBSSxpQkFBaUIsSUFBSSxTQUFTLEVBQUU7UUFDaEMsT0FBTyxTQUFTLENBQUM7S0FDcEI7SUFDRCxNQUFNLE1BQU0sR0FBRyxJQUFJLGdDQUFxQixDQUFDLEtBQUssRUFBRSxHQUFHLFVBQVUsa0JBQWtCLEVBQUU7UUFDN0UsWUFBWSxFQUFFLGVBQWUsVUFBVSxFQUFFO1FBQ3pDLGFBQWEsRUFBRSxFQUFFO1FBQ2pCLGNBQWMsRUFBRSxpQkFBaUI7S0FDcEMsQ0FBQyxDQUFDO0lBRUgsTUFBTSxDQUFDLElBQUksQ0FBQyxhQUFhLENBQUMsTUFBTSxDQUFDLENBQUM7SUFFbEMsT0FBTyxNQUFNLENBQUM7QUFDbEIsQ0FBQztBQWJELGdEQWFDO0FBRUQsTUFBYSwyQkFBMkI7SUFDcEMsWUFBWSxLQUF1QixFQUFFLEdBQUcsT0FBNEI7UUFDaEUsSUFBSSxLQUFLLENBQUMsYUFBYSxDQUFDLGtCQUFrQixFQUFFO1lBQ3hDLE9BQU8sQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLEVBQUU7Z0JBQ3JCLE1BQU0sTUFBTSxHQUFHLElBQUksZ0NBQXFCLENBQUMsS0FBSyxFQUFFLEdBQUcsTUFBTSxDQUFDLFNBQVMsa0JBQWtCLEVBQUU7b0JBQ25GLFlBQVksRUFBRSxlQUFlLE1BQU0sQ0FBQyxTQUFTLEVBQUU7b0JBQy9DLGFBQWEsRUFBRSxFQUFFO29CQUNqQixjQUFjLEVBQUUsS0FBSyxDQUFDLGFBQWEsQ0FBQyxrQkFBNEI7aUJBQ25FLENBQUMsQ0FBQztnQkFFSCxNQUFNLENBQUMsSUFBSSxDQUFDLGFBQWEsQ0FBQyxNQUFNLENBQUMsQ0FBQztZQUN0QyxDQUFDLENBQUMsQ0FBQztTQUNOO0lBQ0wsQ0FBQztDQUNKO0FBZEQsa0VBY0MiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0NmblN1YnNjcmlwdGlvbkZpbHRlcn0gZnJvbSAnYXdzLWNkay1saWIvYXdzLWxvZ3MnO1xuaW1wb3J0IHtGdW5jdGlvbn0gZnJvbSAnYXdzLWNkay1saWIvYXdzLWxhbWJkYSc7XG5pbXBvcnQge0RpZ2l0cmFmZmljU3RhY2t9IGZyb20gXCIuL3N0YWNrXCI7XG5pbXBvcnQge0NvbnN0cnVjdH0gZnJvbSBcImNvbnN0cnVjdHNcIjtcbmltcG9ydCB7TW9uaXRvcmVkRnVuY3Rpb259IGZyb20gXCIuL21vbml0b3JlZGZ1bmN0aW9uXCI7XG5cbi8qKlxuICogQ3JlYXRlcyBhIHN1YnNjcmlwdGlvbiBmaWx0ZXIgdGhhdCBzdWJzY3JpYmVzIHRvIGEgTGFtYmRhIExvZyBHcm91cCBhbmQgZGVsaXZlcnMgdGhlIGxvZ3MgdG8gYW5vdGhlciBkZXN0aW5hdGlvbi5cbiAqIGh0dHBzOi8vZG9jcy5hd3MuYW1hem9uLmNvbS9BV1NDbG91ZEZvcm1hdGlvbi9sYXRlc3QvVXNlckd1aWRlL2F3cy1yZXNvdXJjZS1sb2dzLXN1YnNjcmlwdGlvbmZpbHRlci5odG1sXG4gKiBAcGFyYW0gbGFtYmRhIFRoZSBMYW1iZGEgZnVuY3Rpb24sIG5lZWRlZCB0byBjcmVhdGUgYSBkZXBlbmRlbmN5XG4gKiBAcGFyYW0gbGFtYmRhTmFtZSBUaGUgTGFtYmRhIG5hbWUgZnJvbSB3aGljaCB0aGUgTG9nIEdyb3VwIG5hbWUgaXMgZGVyaXZlZFxuICogQHBhcmFtIGxvZ0Rlc3RpbmF0aW9uQXJuIERlc3RpbmF0aW9uIGZvciBzdHJlYW1lZCBsb2dzXG4gKiBAcGFyYW0gc3RhY2sgQ2xvdWRGb3JtYXRpb24gc3RhY2tcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZVN1YnNjcmlwdGlvbihsYW1iZGE6IEZ1bmN0aW9uLCBsYW1iZGFOYW1lOiBzdHJpbmcsIGxvZ0Rlc3RpbmF0aW9uQXJuOiBzdHJpbmcgfCB1bmRlZmluZWQsIHN0YWNrOiBDb25zdHJ1Y3QpOiBDZm5TdWJzY3JpcHRpb25GaWx0ZXIgfCB1bmRlZmluZWQge1xuICAgIGlmIChsb2dEZXN0aW5hdGlvbkFybiA9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgcmV0dXJuIHVuZGVmaW5lZDtcbiAgICB9XG4gICAgY29uc3QgZmlsdGVyID0gbmV3IENmblN1YnNjcmlwdGlvbkZpbHRlcihzdGFjaywgYCR7bGFtYmRhTmFtZX1Mb2dzU3Vic2NyaXB0aW9uYCwge1xuICAgICAgICBsb2dHcm91cE5hbWU6IGAvYXdzL2xhbWJkYS8ke2xhbWJkYU5hbWV9YCxcbiAgICAgICAgZmlsdGVyUGF0dGVybjogJycsXG4gICAgICAgIGRlc3RpbmF0aW9uQXJuOiBsb2dEZXN0aW5hdGlvbkFybixcbiAgICB9KTtcblxuICAgIGZpbHRlci5ub2RlLmFkZERlcGVuZGVuY3kobGFtYmRhKTtcblxuICAgIHJldHVybiBmaWx0ZXI7XG59XG5cbmV4cG9ydCBjbGFzcyBEaWdpdHJhZmZpY0xvZ1N1YnNjcmlwdGlvbnMge1xuICAgIGNvbnN0cnVjdG9yKHN0YWNrOiBEaWdpdHJhZmZpY1N0YWNrLCAuLi5sYW1iZGFzOiBNb25pdG9yZWRGdW5jdGlvbltdKSB7XG4gICAgICAgIGlmIChzdGFjay5jb25maWd1cmF0aW9uLmxvZ3NEZXN0aW5hdGlvbkFybikge1xuICAgICAgICAgICAgbGFtYmRhcy5mb3JFYWNoKGxhbWJkYSA9PiB7XG4gICAgICAgICAgICAgICAgY29uc3QgZmlsdGVyID0gbmV3IENmblN1YnNjcmlwdGlvbkZpbHRlcihzdGFjaywgYCR7bGFtYmRhLmdpdmVuTmFtZX1Mb2dzU3Vic2NyaXB0aW9uYCwge1xuICAgICAgICAgICAgICAgICAgICBsb2dHcm91cE5hbWU6IGAvYXdzL2xhbWJkYS8ke2xhbWJkYS5naXZlbk5hbWV9YCxcbiAgICAgICAgICAgICAgICAgICAgZmlsdGVyUGF0dGVybjogJycsXG4gICAgICAgICAgICAgICAgICAgIGRlc3RpbmF0aW9uQXJuOiBzdGFjay5jb25maWd1cmF0aW9uLmxvZ3NEZXN0aW5hdGlvbkFybiBhcyBzdHJpbmcsXG4gICAgICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgICAgICBmaWx0ZXIubm9kZS5hZGREZXBlbmRlbmN5KGxhbWJkYSk7XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgfVxuICAgIH1cbn1cbiJdfQ==
|
@@ -0,0 +1,42 @@
|
|
1
|
+
"use strict";
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
3
|
+
exports.createDefaultUsagePlan = exports.createUsagePlan = void 0;
|
4
|
+
/**
|
5
|
+
* Creates an usage plan for a REST API with a single API key
|
6
|
+
* @param api The REST API
|
7
|
+
* @param apiKeyId Id for the API key, this is a surrogate id for CDK, not displayed anywhere
|
8
|
+
* @param apiKeyName Name for the API key, this is displayed in the AWS Console
|
9
|
+
* @deprecated Creates randomized API key names, use createDefaultUsagePlan instead
|
10
|
+
*/
|
11
|
+
function createUsagePlan(api, apiKeyId, apiKeyName) {
|
12
|
+
const apiKey = api.addApiKey(apiKeyId);
|
13
|
+
const plan = api.addUsagePlan(apiKeyName, {
|
14
|
+
name: apiKeyName,
|
15
|
+
});
|
16
|
+
plan.addApiStage({
|
17
|
+
stage: api.deploymentStage,
|
18
|
+
});
|
19
|
+
plan.addApiKey(apiKey);
|
20
|
+
return apiKey;
|
21
|
+
}
|
22
|
+
exports.createUsagePlan = createUsagePlan;
|
23
|
+
/**
|
24
|
+
* Creates a default usage plan for a REST API with a single API key
|
25
|
+
* @param api The REST API
|
26
|
+
* @param apiName Name of the api. Will generate key: apiName + ' API Key' and plan: apiName + ' API Usage Plan'
|
27
|
+
*/
|
28
|
+
function createDefaultUsagePlan(api, apiName) {
|
29
|
+
const apiKeyName = apiName + ' API Key';
|
30
|
+
const usagePlanName = apiName + ' API Usage Plan';
|
31
|
+
const apiKey = api.addApiKey(apiKeyName, { apiKeyName: apiKeyName });
|
32
|
+
const plan = api.addUsagePlan(usagePlanName, {
|
33
|
+
name: usagePlanName,
|
34
|
+
});
|
35
|
+
plan.addApiStage({
|
36
|
+
stage: api.deploymentStage,
|
37
|
+
});
|
38
|
+
plan.addApiKey(apiKey);
|
39
|
+
return apiKey;
|
40
|
+
}
|
41
|
+
exports.createDefaultUsagePlan = createDefaultUsagePlan;
|
42
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXNhZ2UtcGxhbnMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvYXdzL2luZnJhL3VzYWdlLXBsYW5zLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUVBOzs7Ozs7R0FNRztBQUNILFNBQWdCLGVBQWUsQ0FBQyxHQUFZLEVBQUUsUUFBZ0IsRUFBRSxVQUFrQjtJQUM5RSxNQUFNLE1BQU0sR0FBRyxHQUFHLENBQUMsU0FBUyxDQUFDLFFBQVEsQ0FBQyxDQUFDO0lBQ3ZDLE1BQU0sSUFBSSxHQUFHLEdBQUcsQ0FBQyxZQUFZLENBQUMsVUFBVSxFQUFFO1FBQ3RDLElBQUksRUFBRSxVQUFVO0tBQ25CLENBQUMsQ0FBQztJQUNILElBQUksQ0FBQyxXQUFXLENBQUM7UUFDYixLQUFLLEVBQUUsR0FBRyxDQUFDLGVBQWU7S0FDN0IsQ0FBQyxDQUFDO0lBQ0gsSUFBSSxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUV2QixPQUFPLE1BQU0sQ0FBQztBQUNsQixDQUFDO0FBWEQsMENBV0M7QUFFRDs7OztHQUlHO0FBQ0gsU0FBZ0Isc0JBQXNCLENBQUMsR0FBWSxFQUFFLE9BQWU7SUFDaEUsTUFBTSxVQUFVLEdBQUcsT0FBTyxHQUFHLFVBQVUsQ0FBQztJQUN4QyxNQUFNLGFBQWEsR0FBRyxPQUFPLEdBQUcsaUJBQWlCLENBQUM7SUFDbEQsTUFBTSxNQUFNLEdBQUcsR0FBRyxDQUFDLFNBQVMsQ0FBQyxVQUFVLEVBQUUsRUFBRSxVQUFVLEVBQUUsVUFBVSxFQUFFLENBQUMsQ0FBQztJQUNyRSxNQUFNLElBQUksR0FBRyxHQUFHLENBQUMsWUFBWSxDQUFDLGFBQWEsRUFBRTtRQUN6QyxJQUFJLEVBQUUsYUFBYTtLQUN0QixDQUFDLENBQUM7SUFDSCxJQUFJLENBQUMsV0FBVyxDQUFDO1FBQ2IsS0FBSyxFQUFFLEdBQUcsQ0FBQyxlQUFlO0tBQzdCLENBQUMsQ0FBQztJQUNILElBQUksQ0FBQyxTQUFTLENBQUMsTUFBTSxDQUFDLENBQUM7SUFFdkIsT0FBTyxNQUFNLENBQUM7QUFDbEIsQ0FBQztBQWJELHdEQWFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtJQXBpS2V5LCBSZXN0QXBpfSBmcm9tICdhd3MtY2RrLWxpYi9hd3MtYXBpZ2F0ZXdheSc7XHJcblxyXG4vKipcclxuICogQ3JlYXRlcyBhbiB1c2FnZSBwbGFuIGZvciBhIFJFU1QgQVBJIHdpdGggYSBzaW5nbGUgQVBJIGtleVxyXG4gKiBAcGFyYW0gYXBpIFRoZSBSRVNUIEFQSVxyXG4gKiBAcGFyYW0gYXBpS2V5SWQgSWQgZm9yIHRoZSBBUEkga2V5LCB0aGlzIGlzIGEgc3Vycm9nYXRlIGlkIGZvciBDREssIG5vdCBkaXNwbGF5ZWQgYW55d2hlcmVcclxuICogQHBhcmFtIGFwaUtleU5hbWUgTmFtZSBmb3IgdGhlIEFQSSBrZXksIHRoaXMgaXMgZGlzcGxheWVkIGluIHRoZSBBV1MgQ29uc29sZVxyXG4gKiBAZGVwcmVjYXRlZCBDcmVhdGVzIHJhbmRvbWl6ZWQgQVBJIGtleSBuYW1lcywgdXNlIGNyZWF0ZURlZmF1bHRVc2FnZVBsYW4gaW5zdGVhZFxyXG4gKi9cclxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZVVzYWdlUGxhbihhcGk6IFJlc3RBcGksIGFwaUtleUlkOiBzdHJpbmcsIGFwaUtleU5hbWU6IHN0cmluZyk6IElBcGlLZXkge1xyXG4gICAgY29uc3QgYXBpS2V5ID0gYXBpLmFkZEFwaUtleShhcGlLZXlJZCk7XHJcbiAgICBjb25zdCBwbGFuID0gYXBpLmFkZFVzYWdlUGxhbihhcGlLZXlOYW1lLCB7XHJcbiAgICAgICAgbmFtZTogYXBpS2V5TmFtZSxcclxuICAgIH0pO1xyXG4gICAgcGxhbi5hZGRBcGlTdGFnZSh7XHJcbiAgICAgICAgc3RhZ2U6IGFwaS5kZXBsb3ltZW50U3RhZ2UsXHJcbiAgICB9KTtcclxuICAgIHBsYW4uYWRkQXBpS2V5KGFwaUtleSk7XHJcblxyXG4gICAgcmV0dXJuIGFwaUtleTtcclxufVxyXG5cclxuLyoqXHJcbiAqIENyZWF0ZXMgYSBkZWZhdWx0IHVzYWdlIHBsYW4gZm9yIGEgUkVTVCBBUEkgd2l0aCBhIHNpbmdsZSBBUEkga2V5XHJcbiAqIEBwYXJhbSBhcGkgVGhlIFJFU1QgQVBJXHJcbiAqIEBwYXJhbSBhcGlOYW1lIE5hbWUgb2YgdGhlIGFwaS4gV2lsbCBnZW5lcmF0ZSBrZXk6IGFwaU5hbWUgKyAnIEFQSSBLZXknIGFuZCBwbGFuOiBhcGlOYW1lICsgJyBBUEkgVXNhZ2UgUGxhbidcclxuICovXHJcbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVEZWZhdWx0VXNhZ2VQbGFuKGFwaTogUmVzdEFwaSwgYXBpTmFtZTogc3RyaW5nKTogSUFwaUtleSB7XHJcbiAgICBjb25zdCBhcGlLZXlOYW1lID0gYXBpTmFtZSArICcgQVBJIEtleSc7XHJcbiAgICBjb25zdCB1c2FnZVBsYW5OYW1lID0gYXBpTmFtZSArICcgQVBJIFVzYWdlIFBsYW4nO1xyXG4gICAgY29uc3QgYXBpS2V5ID0gYXBpLmFkZEFwaUtleShhcGlLZXlOYW1lLCB7IGFwaUtleU5hbWU6IGFwaUtleU5hbWUgfSk7XHJcbiAgICBjb25zdCBwbGFuID0gYXBpLmFkZFVzYWdlUGxhbih1c2FnZVBsYW5OYW1lLCB7XHJcbiAgICAgICAgbmFtZTogdXNhZ2VQbGFuTmFtZSxcclxuICAgIH0pO1xyXG4gICAgcGxhbi5hZGRBcGlTdGFnZSh7XHJcbiAgICAgICAgc3RhZ2U6IGFwaS5kZXBsb3ltZW50U3RhZ2UsXHJcbiAgICB9KTtcclxuICAgIHBsYW4uYWRkQXBpS2V5KGFwaUtleSk7XHJcblxyXG4gICAgcmV0dXJuIGFwaUtleTtcclxufVxyXG4iXX0=
|
@@ -0,0 +1,13 @@
|
|
1
|
+
"use strict";
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
3
|
+
exports.getApiKeyFromAPIGateway = void 0;
|
4
|
+
const aws_sdk_1 = require("aws-sdk");
|
5
|
+
function getApiKeyFromAPIGateway(keyId) {
|
6
|
+
const agw = new aws_sdk_1.APIGateway();
|
7
|
+
return agw.getApiKey({
|
8
|
+
apiKey: keyId,
|
9
|
+
includeValue: true,
|
10
|
+
}).promise();
|
11
|
+
}
|
12
|
+
exports.getApiKeyFromAPIGateway = getApiKeyFromAPIGateway;
|
13
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXBpa2V5LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2F3cy9ydW50aW1lL2FwaWtleS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSxxQ0FBbUM7QUFFbkMsU0FBZ0IsdUJBQXVCLENBQUMsS0FBYTtJQUNqRCxNQUFNLEdBQUcsR0FBRyxJQUFJLG9CQUFVLEVBQUUsQ0FBQztJQUM3QixPQUFPLEdBQUcsQ0FBQyxTQUFTLENBQUM7UUFDakIsTUFBTSxFQUFFLEtBQUs7UUFDYixZQUFZLEVBQUUsSUFBSTtLQUNyQixDQUFDLENBQUMsT0FBTyxFQUFFLENBQUM7QUFDakIsQ0FBQztBQU5ELDBEQU1DIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtBUElHYXRld2F5fSBmcm9tIFwiYXdzLXNka1wiO1xuXG5leHBvcnQgZnVuY3Rpb24gZ2V0QXBpS2V5RnJvbUFQSUdhdGV3YXkoa2V5SWQ6IHN0cmluZyk6IFByb21pc2U8QVBJR2F0ZXdheS5UeXBlcy5BcGlLZXk+IHtcbiAgICBjb25zdCBhZ3cgPSBuZXcgQVBJR2F0ZXdheSgpO1xuICAgIHJldHVybiBhZ3cuZ2V0QXBpS2V5KHtcbiAgICAgICAgYXBpS2V5OiBrZXlJZCxcbiAgICAgICAgaW5jbHVkZVZhbHVlOiB0cnVlLFxuICAgIH0pLnByb21pc2UoKTtcbn1cbiJdfQ==
|
@@ -0,0 +1,26 @@
|
|
1
|
+
"use strict";
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
3
|
+
exports.DigitrafficIntegrationResponse = void 0;
|
4
|
+
const mediatypes_1 = require("../types/mediatypes");
|
5
|
+
const response_1 = require("../infra/api/response");
|
6
|
+
class DigitrafficIntegrationResponse {
|
7
|
+
static ok(mediaType) {
|
8
|
+
return this.create("200", mediaType);
|
9
|
+
}
|
10
|
+
static badRequest(mediaType) {
|
11
|
+
return this.create("400", mediaType ?? mediatypes_1.MediaType.TEXT_PLAIN);
|
12
|
+
}
|
13
|
+
static notImplemented(mediaType) {
|
14
|
+
return this.create("501", mediaType ?? mediatypes_1.MediaType.TEXT_PLAIN);
|
15
|
+
}
|
16
|
+
static create(statusCode, mediaType) {
|
17
|
+
return {
|
18
|
+
statusCode,
|
19
|
+
responseTemplates: {
|
20
|
+
[mediaType]: response_1.RESPONSE_DEFAULT_LAMBDA,
|
21
|
+
},
|
22
|
+
};
|
23
|
+
}
|
24
|
+
}
|
25
|
+
exports.DigitrafficIntegrationResponse = DigitrafficIntegrationResponse;
|
26
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGlnaXRyYWZmaWMtaW50ZWdyYXRpb24tcmVzcG9uc2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvYXdzL3J1bnRpbWUvZGlnaXRyYWZmaWMtaW50ZWdyYXRpb24tcmVzcG9uc2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQ0Esb0RBQThDO0FBQzlDLG9EQUE4RDtBQUU5RCxNQUFzQiw4QkFBOEI7SUFFaEQsTUFBTSxDQUFDLEVBQUUsQ0FBQyxTQUFvQjtRQUMxQixPQUFPLElBQUksQ0FBQyxNQUFNLENBQUMsS0FBSyxFQUFFLFNBQVMsQ0FBQyxDQUFDO0lBQ3pDLENBQUM7SUFFRCxNQUFNLENBQUMsVUFBVSxDQUFDLFNBQXFCO1FBQ25DLE9BQU8sSUFBSSxDQUFDLE1BQU0sQ0FBQyxLQUFLLEVBQUUsU0FBUyxJQUFJLHNCQUFTLENBQUMsVUFBVSxDQUFDLENBQUM7SUFDakUsQ0FBQztJQUVELE1BQU0sQ0FBQyxjQUFjLENBQUMsU0FBcUI7UUFDdkMsT0FBTyxJQUFJLENBQUMsTUFBTSxDQUFDLEtBQUssRUFBRSxTQUFTLElBQUksc0JBQVMsQ0FBQyxVQUFVLENBQUMsQ0FBQztJQUNqRSxDQUFDO0lBRUQsTUFBTSxDQUFDLE1BQU0sQ0FBQyxVQUFrQixFQUFFLFNBQW9CO1FBQ2xELE9BQU87WUFDSCxVQUFVO1lBQ1YsaUJBQWlCLEVBQUU7Z0JBQ2YsQ0FBQyxTQUFTLENBQUMsRUFBRSxrQ0FBdUI7YUFDdkM7U0FDSixDQUFDO0lBQ04sQ0FBQztDQUNKO0FBdEJELHdFQXNCQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7SW50ZWdyYXRpb25SZXNwb25zZX0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1hcGlnYXRld2F5XCI7XG5pbXBvcnQge01lZGlhVHlwZX0gZnJvbSBcIi4uL3R5cGVzL21lZGlhdHlwZXNcIjtcbmltcG9ydCB7UkVTUE9OU0VfREVGQVVMVF9MQU1CREF9IGZyb20gXCIuLi9pbmZyYS9hcGkvcmVzcG9uc2VcIjtcblxuZXhwb3J0IGFic3RyYWN0IGNsYXNzIERpZ2l0cmFmZmljSW50ZWdyYXRpb25SZXNwb25zZSB7XG5cbiAgICBzdGF0aWMgb2sobWVkaWFUeXBlOiBNZWRpYVR5cGUpOiBJbnRlZ3JhdGlvblJlc3BvbnNlIHtcbiAgICAgICAgcmV0dXJuIHRoaXMuY3JlYXRlKFwiMjAwXCIsIG1lZGlhVHlwZSk7XG4gICAgfVxuXG4gICAgc3RhdGljIGJhZFJlcXVlc3QobWVkaWFUeXBlPzogTWVkaWFUeXBlKTogSW50ZWdyYXRpb25SZXNwb25zZSB7XG4gICAgICAgIHJldHVybiB0aGlzLmNyZWF0ZShcIjQwMFwiLCBtZWRpYVR5cGUgPz8gTWVkaWFUeXBlLlRFWFRfUExBSU4pO1xuICAgIH1cblxuICAgIHN0YXRpYyBub3RJbXBsZW1lbnRlZChtZWRpYVR5cGU/OiBNZWRpYVR5cGUpOiBJbnRlZ3JhdGlvblJlc3BvbnNlIHtcbiAgICAgICAgcmV0dXJuIHRoaXMuY3JlYXRlKFwiNTAxXCIsIG1lZGlhVHlwZSA/PyBNZWRpYVR5cGUuVEVYVF9QTEFJTik7XG4gICAgfVxuXG4gICAgc3RhdGljIGNyZWF0ZShzdGF0dXNDb2RlOiBzdHJpbmcsIG1lZGlhVHlwZTogTWVkaWFUeXBlKTogSW50ZWdyYXRpb25SZXNwb25zZSB7XG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICBzdGF0dXNDb2RlLFxuICAgICAgICAgICAgcmVzcG9uc2VUZW1wbGF0ZXM6IHtcbiAgICAgICAgICAgICAgICBbbWVkaWFUeXBlXTogUkVTUE9OU0VfREVGQVVMVF9MQU1CREEsXG4gICAgICAgICAgICB9LFxuICAgICAgICB9O1xuICAgIH1cbn1cblxuIl19
|