@digitraffic/common 2022.10.7-1 → 2022.10.14-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/README.md +28 -11
- package/aws/infra/canaries/canary.js +2 -2
- package/aws/infra/stack/stack-checking-aspect.d.ts +4 -3
- package/aws/infra/stack/stack-checking-aspect.js +42 -31
- package/aws/infra/stack/stack.d.ts +4 -4
- package/aws/infra/stack/stack.js +17 -15
- package/package.json +5 -6
- package/yarn-error.log +103 -0
- package/aws/index.js +0 -18
- package/aws/infra/api/index.js +0 -18
- package/types/index.js +0 -18
package/README.md
CHANGED
@@ -1,49 +1,66 @@
|
|
1
1
|
# Digitraffic-common
|
2
|
+
|
2
3
|
This is a place for common utilities and classes that can be used in other cdk-projects.
|
3
4
|
|
4
5
|
## How to use
|
6
|
+
|
5
7
|
In package.json dependencies:
|
8
|
+
|
6
9
|
```
|
7
10
|
"dependencies": {
|
8
|
-
"digitraffic
|
11
|
+
"@digitraffic/common": "*",
|
9
12
|
}
|
10
13
|
```
|
11
14
|
|
12
15
|
In code:
|
16
|
+
|
13
17
|
```
|
14
|
-
import {DigitrafficStack, StackConfiguration} from "digitraffic
|
18
|
+
import {DigitrafficStack, StackConfiguration} from "@digitraffic/common/aws/infra/stack/stack";
|
15
19
|
```
|
16
20
|
|
17
21
|
### DigitrafficStack
|
22
|
+
|
18
23
|
If you extend your stack from DigitrafficStack you get many benefits:
|
19
|
-
|
20
|
-
|
21
|
-
|
24
|
+
|
25
|
+
- Secret, VPC, Sg & alarmTopics automatically
|
26
|
+
- Stack validation with StackCheckingAspect
|
27
|
+
- Easier configuration with StackConfiguration
|
22
28
|
|
23
29
|
If you do not need those things, you should not use DigitrafficStack.
|
24
30
|
|
25
31
|
### StackConfiguration
|
26
|
-
|
32
|
+
|
33
|
+
Some commonly used parameters is predefined configuration. You can write configuration for your
|
27
34
|
environments once and use across cdk-projects.
|
28
35
|
|
29
36
|
### StackCheckingAspect
|
37
|
+
|
30
38
|
Uses cdk aspects to do some sanity checking for your cdk stack:
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
39
|
+
|
40
|
+
- Stack naming check(Test/Prod in name)
|
41
|
+
- Function configuration(memory, timeout, runtime, reservedConcurrency)
|
42
|
+
- Tags, must have Solution tag defined
|
43
|
+
- S3 Buckets, no public access
|
44
|
+
- Api Gateway resource casing(kebabCase and snake_case)
|
45
|
+
- Queue encrypting
|
46
|
+
- LogGroup Retention
|
35
47
|
|
36
48
|
You can use StackCheckingAspect for any stack, DigitrafficStack does it automatically, but you can call it manually:
|
49
|
+
|
37
50
|
```
|
38
51
|
Aspects.of(this).add(StackCheckingAspect.create(this));
|
39
52
|
```
|
40
53
|
|
54
|
+
Any resource can be whitelisted by giving it as a parameter or in the StackConfiguration
|
55
|
+
|
41
56
|
### MonitoredFunction
|
57
|
+
|
42
58
|
MonitoredFunction extends Function with alarms on memory usage and timeouts.
|
43
59
|
|
44
60
|
If you need database access in your Function, you can use MonitoredDBFunction. Creating a Function with it is this easy:
|
61
|
+
|
45
62
|
```
|
46
63
|
const lambda = MonitoredDBFunction.create(stack, 'get-metadata');
|
47
64
|
```
|
48
65
|
|
49
|
-
See the documentation for more information.
|
66
|
+
See the documentation for more information.
|
@@ -7,7 +7,7 @@ const canary_alarm_1 = require("./canary-alarm");
|
|
7
7
|
class DigitrafficCanary extends aws_synthetics_alpha_1.Canary {
|
8
8
|
constructor(scope, canaryName, role, params, environmentVariables) {
|
9
9
|
super(scope, canaryName, {
|
10
|
-
runtime: aws_synthetics_alpha_1.Runtime.
|
10
|
+
runtime: aws_synthetics_alpha_1.Runtime.SYNTHETICS_NODEJS_PUPPETEER_3_6,
|
11
11
|
role,
|
12
12
|
test: aws_synthetics_alpha_1.Test.custom({
|
13
13
|
code: new aws_synthetics_alpha_1.AssetCode("dist", {
|
@@ -26,4 +26,4 @@ class DigitrafficCanary extends aws_synthetics_alpha_1.Canary {
|
|
26
26
|
}
|
27
27
|
}
|
28
28
|
exports.DigitrafficCanary = DigitrafficCanary;
|
29
|
-
//# sourceMappingURL=data:application/json;base64,
|
29
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FuYXJ5LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2F3cy9pbmZyYS9jYW5hcmllcy9jYW5hcnkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsNkNBQXFDO0FBQ3JDLHdFQUF5RjtBQUV6RixpREFBMkM7QUFLM0MsTUFBYSxpQkFBa0IsU0FBUSw2QkFBTTtJQUN6QyxZQUNJLEtBQWdCLEVBQ2hCLFVBQWtCLEVBQ2xCLElBQVUsRUFDVixNQUF3QixFQUN4QixvQkFBdUM7UUFFdkMsS0FBSyxDQUFDLEtBQUssRUFBRSxVQUFVLEVBQUU7WUFDckIsT0FBTyxFQUFFLDhCQUFPLENBQUMsK0JBQStCO1lBQ2hELElBQUk7WUFDSixJQUFJLEVBQUUsMkJBQUksQ0FBQyxNQUFNLENBQUM7Z0JBQ2QsSUFBSSxFQUFFLElBQUksZ0NBQVMsQ0FBQyxNQUFNLEVBQUU7b0JBQ3hCLE9BQU8sRUFBRSxDQUFDLFFBQVEsQ0FBQztpQkFDdEIsQ0FBQztnQkFDRixPQUFPLEVBQUUsTUFBTSxDQUFDLE9BQU87YUFDMUIsQ0FBQztZQUNGLG9CQUFvQixFQUFFLEVBQUUsR0FBRyxvQkFBb0IsRUFBRSxHQUFHLE1BQU0sRUFBRSxTQUFTLEVBQUU7WUFDdkUsVUFBVTtZQUNWLFFBQVEsRUFBRSxNQUFNLENBQUMsUUFBUSxJQUFJLCtCQUFRLENBQUMsSUFBSSxDQUFDLHNCQUFRLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQyxDQUFDO1NBQ25FLENBQUMsQ0FBQztRQUVILElBQUksQ0FBQyxlQUFlLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQyxDQUFDO1FBRXRDLElBQUksTUFBTSxDQUFDLEtBQUssSUFBSSxJQUFJLEVBQUU7WUFDdEIsSUFBSSwwQkFBVyxDQUFDLEtBQUssRUFBRSxJQUFJLEVBQUUsTUFBTSxDQUFDLENBQUM7U0FDeEM7SUFDTCxDQUFDO0NBQ0o7QUE1QkQsOENBNEJDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtEdXJhdGlvbn0gZnJvbSBcImF3cy1jZGstbGliXCI7XG5pbXBvcnQge0Fzc2V0Q29kZSwgQ2FuYXJ5LCBSdW50aW1lLCBTY2hlZHVsZSwgVGVzdH0gZnJvbSBcIkBhd3MtY2RrL2F3cy1zeW50aGV0aWNzLWFscGhhXCI7XG5pbXBvcnQge1JvbGV9IGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtaWFtXCI7XG5pbXBvcnQge0NhbmFyeUFsYXJtfSBmcm9tIFwiLi9jYW5hcnktYWxhcm1cIjtcbmltcG9ydCB7Q2FuYXJ5UGFyYW1ldGVyc30gZnJvbSBcIi4vY2FuYXJ5LXBhcmFtZXRlcnNcIjtcbmltcG9ydCB7Q29uc3RydWN0fSBmcm9tIFwiY29uc3RydWN0c1wiO1xuaW1wb3J0IHtMYW1iZGFFbnZpcm9ubWVudH0gZnJvbSBcIi4uL3N0YWNrL2xhbWJkYS1jb25maWdzXCI7XG5cbmV4cG9ydCBjbGFzcyBEaWdpdHJhZmZpY0NhbmFyeSBleHRlbmRzIENhbmFyeSB7XG4gICAgY29uc3RydWN0b3IoXG4gICAgICAgIHNjb3BlOiBDb25zdHJ1Y3QsXG4gICAgICAgIGNhbmFyeU5hbWU6IHN0cmluZyxcbiAgICAgICAgcm9sZTogUm9sZSxcbiAgICAgICAgcGFyYW1zOiBDYW5hcnlQYXJhbWV0ZXJzLFxuICAgICAgICBlbnZpcm9ubWVudFZhcmlhYmxlczogTGFtYmRhRW52aXJvbm1lbnQsXG4gICAgKSB7XG4gICAgICAgIHN1cGVyKHNjb3BlLCBjYW5hcnlOYW1lLCB7XG4gICAgICAgICAgICBydW50aW1lOiBSdW50aW1lLlNZTlRIRVRJQ1NfTk9ERUpTX1BVUFBFVEVFUl8zXzYsXG4gICAgICAgICAgICByb2xlLFxuICAgICAgICAgICAgdGVzdDogVGVzdC5jdXN0b20oe1xuICAgICAgICAgICAgICAgIGNvZGU6IG5ldyBBc3NldENvZGUoXCJkaXN0XCIsIHtcbiAgICAgICAgICAgICAgICAgICAgZXhjbHVkZTogW1wibGFtYmRhXCJdLFxuICAgICAgICAgICAgICAgIH0pLFxuICAgICAgICAgICAgICAgIGhhbmRsZXI6IHBhcmFtcy5oYW5kbGVyLFxuICAgICAgICAgICAgfSksXG4gICAgICAgICAgICBlbnZpcm9ubWVudFZhcmlhYmxlczogeyAuLi5lbnZpcm9ubWVudFZhcmlhYmxlcywgLi4ucGFyYW1zPy5jYW5hcnlFbnYgfSxcbiAgICAgICAgICAgIGNhbmFyeU5hbWUsXG4gICAgICAgICAgICBzY2hlZHVsZTogcGFyYW1zLnNjaGVkdWxlID8/IFNjaGVkdWxlLnJhdGUoRHVyYXRpb24ubWludXRlcygxNSkpLFxuICAgICAgICB9KTtcblxuICAgICAgICB0aGlzLmFydGlmYWN0c0J1Y2tldC5ncmFudFdyaXRlKHJvbGUpO1xuXG4gICAgICAgIGlmIChwYXJhbXMuYWxhcm0gPz8gdHJ1ZSkge1xuICAgICAgICAgICAgbmV3IENhbmFyeUFsYXJtKHNjb3BlLCB0aGlzLCBwYXJhbXMpO1xuICAgICAgICB9XG4gICAgfVxufVxuIl19
|
@@ -1,9 +1,10 @@
|
|
1
1
|
import { IAspect } from "aws-cdk-lib";
|
2
|
-
import { DigitrafficStack
|
2
|
+
import { DigitrafficStack } from "./stack";
|
3
3
|
import { IConstruct } from "constructs";
|
4
4
|
export declare class StackCheckingAspect implements IAspect {
|
5
|
-
private readonly
|
6
|
-
|
5
|
+
private readonly stackShortName?;
|
6
|
+
private readonly whitelistedResources?;
|
7
|
+
constructor(stackShortName?: string, whitelistedResources?: string[]);
|
7
8
|
static create(stack: DigitrafficStack): StackCheckingAspect;
|
8
9
|
visit(node: IConstruct): void;
|
9
10
|
private isWhitelisted;
|
@@ -26,11 +26,12 @@ var ResourceType;
|
|
26
26
|
ResourceType["logGroupRetention"] = "LOG_GROUP_RETENTION";
|
27
27
|
})(ResourceType || (ResourceType = {}));
|
28
28
|
class StackCheckingAspect {
|
29
|
-
constructor(
|
30
|
-
this.
|
29
|
+
constructor(stackShortName, whitelistedResources) {
|
30
|
+
this.stackShortName = stackShortName;
|
31
|
+
this.whitelistedResources = whitelistedResources;
|
31
32
|
}
|
32
33
|
static create(stack) {
|
33
|
-
return new StackCheckingAspect(stack.configuration);
|
34
|
+
return new StackCheckingAspect(stack.configuration.shortName, stack.configuration.whitelistedResources);
|
34
35
|
}
|
35
36
|
visit(node) {
|
36
37
|
//console.info("visiting class " + node.constructor.name);
|
@@ -43,8 +44,8 @@ class StackCheckingAspect {
|
|
43
44
|
this.checkLogGroupRetention(node);
|
44
45
|
}
|
45
46
|
isWhitelisted(key) {
|
46
|
-
return this
|
47
|
-
return key.matchAll(new RegExp(wl,
|
47
|
+
return this.whitelistedResources?.some((wl) => {
|
48
|
+
return key.matchAll(new RegExp(wl, "g"));
|
48
49
|
});
|
49
50
|
}
|
50
51
|
addAnnotation(node, key, message, isError = true) {
|
@@ -62,40 +63,46 @@ class StackCheckingAspect {
|
|
62
63
|
}
|
63
64
|
checkStack(node) {
|
64
65
|
if (node instanceof stack_1.DigitrafficStack) {
|
65
|
-
if ((node.stackName.includes(
|
66
|
-
|
66
|
+
if ((node.stackName.includes("Test") ||
|
67
|
+
node.stackName.includes("Tst")) &&
|
68
|
+
node.configuration.production) {
|
69
|
+
this.addAnnotation(node, ResourceType.stackName, "Production is set for Test-stack");
|
67
70
|
}
|
68
|
-
if ((node.stackName.includes(
|
69
|
-
|
71
|
+
if ((node.stackName.includes("Prod") ||
|
72
|
+
node.stackName.includes("Prd")) &&
|
73
|
+
!node.configuration.production) {
|
74
|
+
this.addAnnotation(node, ResourceType.stackName, "Production is not set for Production-stack");
|
70
75
|
}
|
71
76
|
}
|
72
77
|
}
|
73
78
|
checkFunction(node) {
|
74
79
|
if (node instanceof aws_lambda_1.CfnFunction) {
|
75
80
|
if (!node.reservedConcurrentExecutions) {
|
76
|
-
this.addAnnotation(node, ResourceType.reservedConcurrentConcurrency,
|
81
|
+
this.addAnnotation(node, ResourceType.reservedConcurrentConcurrency, "Function must have reservedConcurrentConcurrency");
|
77
82
|
}
|
78
83
|
else if (node.reservedConcurrentExecutions > MAX_CONCURRENCY_LIMIT) {
|
79
|
-
this.addAnnotation(node, ResourceType.reservedConcurrentConcurrency,
|
84
|
+
this.addAnnotation(node, ResourceType.reservedConcurrentConcurrency, "Function reservedConcurrentConcurrency too high!");
|
80
85
|
}
|
81
86
|
if (!node.timeout) {
|
82
|
-
this.addAnnotation(node, ResourceType.functionTimeout,
|
87
|
+
this.addAnnotation(node, ResourceType.functionTimeout, "Function must have timeout");
|
83
88
|
}
|
84
89
|
if (!node.memorySize) {
|
85
|
-
this.addAnnotation(node, ResourceType.functionMemorySize,
|
90
|
+
this.addAnnotation(node, ResourceType.functionMemorySize, "Function must have memorySize");
|
86
91
|
}
|
87
92
|
if (node.runtime !== NODE_RUNTIME) {
|
88
|
-
this.addAnnotation(node, ResourceType.functionRuntime,
|
93
|
+
this.addAnnotation(node, ResourceType.functionRuntime, `Function has wrong runtime ${node.runtime}`);
|
89
94
|
}
|
90
|
-
if (this.
|
91
|
-
|
95
|
+
if (this.stackShortName &&
|
96
|
+
node.functionName &&
|
97
|
+
!node.functionName.startsWith(this.stackShortName)) {
|
98
|
+
this.addAnnotation(node, ResourceType.functionName, `Function name does not begin with ${this.stackShortName}`);
|
92
99
|
}
|
93
100
|
}
|
94
101
|
}
|
95
102
|
checkTags(node) {
|
96
103
|
if (node instanceof aws_cdk_lib_1.Stack) {
|
97
104
|
if (!node.tags.tagValues()[stack_1.SOLUTION_KEY]) {
|
98
|
-
this.addAnnotation(node, ResourceType.tagSolution,
|
105
|
+
this.addAnnotation(node, ResourceType.tagSolution, "Solution tag is missing");
|
99
106
|
}
|
100
107
|
}
|
101
108
|
}
|
@@ -103,19 +110,22 @@ class StackCheckingAspect {
|
|
103
110
|
if (node instanceof aws_s3_1.CfnBucket) {
|
104
111
|
const c = node.publicAccessBlockConfiguration;
|
105
112
|
if (c) {
|
106
|
-
if (!c.blockPublicAcls ||
|
107
|
-
|
113
|
+
if (!c.blockPublicAcls ||
|
114
|
+
!c.blockPublicPolicy ||
|
115
|
+
!c.ignorePublicAcls ||
|
116
|
+
!c.restrictPublicBuckets) {
|
117
|
+
this.addAnnotation(node, ResourceType.bucketPublicity, "Check bucket publicity");
|
108
118
|
}
|
109
119
|
}
|
110
120
|
}
|
111
121
|
}
|
112
122
|
static isValidPath(path) {
|
113
123
|
// if path includes . or { check only the trailing part of path
|
114
|
-
if (path.includes(
|
115
|
-
return this.isValidPath(path.split(
|
124
|
+
if (path.includes(".")) {
|
125
|
+
return this.isValidPath(path.split(".")[0]);
|
116
126
|
}
|
117
|
-
if (path.includes(
|
118
|
-
return this.isValidPath(path.split(
|
127
|
+
if (path.includes("{")) {
|
128
|
+
return this.isValidPath(path.split("{")[0]);
|
119
129
|
}
|
120
130
|
return (0, change_case_1.paramCase)(path) === path;
|
121
131
|
}
|
@@ -125,18 +135,19 @@ class StackCheckingAspect {
|
|
125
135
|
checkResourceCasing(node) {
|
126
136
|
if (node instanceof aws_apigateway_1.CfnResource) {
|
127
137
|
if (!StackCheckingAspect.isValidPath(node.pathPart)) {
|
128
|
-
this.addAnnotation(node, ResourceType.resourcePath,
|
138
|
+
this.addAnnotation(node, ResourceType.resourcePath, "Path part should be in kebab-case");
|
129
139
|
}
|
130
140
|
}
|
131
141
|
else if (node instanceof aws_apigateway_1.CfnMethod) {
|
132
142
|
const integration = node.integration;
|
133
143
|
if (integration && integration.requestParameters) {
|
134
|
-
Object.keys(integration.requestParameters).forEach(key => {
|
135
|
-
const split = key.split(
|
144
|
+
Object.keys(integration.requestParameters).forEach((key) => {
|
145
|
+
const split = key.split(".");
|
136
146
|
const type = split[2];
|
137
147
|
const name = split[3];
|
138
|
-
if (type ===
|
139
|
-
|
148
|
+
if (type === "querystring" &&
|
149
|
+
!StackCheckingAspect.isValidQueryString(name)) {
|
150
|
+
this.addAnnotation(node, name, "Querystring should be in snake_case");
|
140
151
|
}
|
141
152
|
});
|
142
153
|
}
|
@@ -145,7 +156,7 @@ class StackCheckingAspect {
|
|
145
156
|
checkQueueEncryption(node) {
|
146
157
|
if (node instanceof aws_sqs_1.CfnQueue) {
|
147
158
|
if (!node.kmsMasterKeyId) {
|
148
|
-
this.addAnnotation(node, ResourceType.queueEncryption,
|
159
|
+
this.addAnnotation(node, ResourceType.queueEncryption, "Queue must have encryption enabled");
|
149
160
|
}
|
150
161
|
}
|
151
162
|
}
|
@@ -154,10 +165,10 @@ class StackCheckingAspect {
|
|
154
165
|
const child = node.node.defaultChild;
|
155
166
|
const retention = child._cfnProperties.RetentionInDays;
|
156
167
|
if (!retention) {
|
157
|
-
this.addAnnotation(node, ResourceType.logGroupRetention,
|
168
|
+
this.addAnnotation(node, ResourceType.logGroupRetention, "Log group must define log group retention");
|
158
169
|
}
|
159
170
|
}
|
160
171
|
}
|
161
172
|
}
|
162
173
|
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"]}
|
174
|
+
//# 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 } 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 stackShortName?: string;\n    private readonly whitelistedResources?: string[];\n\n    constructor(stackShortName?: string, whitelistedResources?: string[]) {\n        this.stackShortName = stackShortName;\n        this.whitelistedResources = whitelistedResources;\n    }\n\n    static create(stack: DigitrafficStack) {\n        return new StackCheckingAspect(\n            stack.configuration.shortName,\n            stack.configuration.whitelistedResources\n        );\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.whitelistedResources?.some((wl) => {\n            return key.matchAll(new RegExp(wl, \"g\"));\n        });\n    }\n\n    private addAnnotation(\n        node: IConstruct,\n        key: ResourceType | string,\n        message: string,\n        isError = true\n    ) {\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 (\n                (node.stackName.includes(\"Test\") ||\n                    node.stackName.includes(\"Tst\")) &&\n                node.configuration.production\n            ) {\n                this.addAnnotation(\n                    node,\n                    ResourceType.stackName,\n                    \"Production is set for Test-stack\"\n                );\n            }\n\n            if (\n                (node.stackName.includes(\"Prod\") ||\n                    node.stackName.includes(\"Prd\")) &&\n                !node.configuration.production\n            ) {\n                this.addAnnotation(\n                    node,\n                    ResourceType.stackName,\n                    \"Production is not set for Production-stack\"\n                );\n            }\n        }\n    }\n\n    private checkFunction(node: IConstruct) {\n        if (node instanceof CfnFunction) {\n            if (!node.reservedConcurrentExecutions) {\n                this.addAnnotation(\n                    node,\n                    ResourceType.reservedConcurrentConcurrency,\n                    \"Function must have reservedConcurrentConcurrency\"\n                );\n            } else if (\n                node.reservedConcurrentExecutions > MAX_CONCURRENCY_LIMIT\n            ) {\n                this.addAnnotation(\n                    node,\n                    ResourceType.reservedConcurrentConcurrency,\n                    \"Function reservedConcurrentConcurrency too high!\"\n                );\n            }\n\n            if (!node.timeout) {\n                this.addAnnotation(\n                    node,\n                    ResourceType.functionTimeout,\n                    \"Function must have timeout\"\n                );\n            }\n\n            if (!node.memorySize) {\n                this.addAnnotation(\n                    node,\n                    ResourceType.functionMemorySize,\n                    \"Function must have memorySize\"\n                );\n            }\n\n            if (node.runtime !== NODE_RUNTIME) {\n                this.addAnnotation(\n                    node,\n                    ResourceType.functionRuntime,\n                    `Function has wrong runtime ${node.runtime!}`\n                );\n            }\n\n            if (\n                this.stackShortName &&\n                node.functionName &&\n                !node.functionName.startsWith(this.stackShortName)\n            ) {\n                this.addAnnotation(\n                    node,\n                    ResourceType.functionName,\n                    `Function name does not begin with ${this.stackShortName}`\n                );\n            }\n        }\n    }\n\n    private checkTags(node: IConstruct) {\n        if (node instanceof Stack) {\n            if (!node.tags.tagValues()[SOLUTION_KEY]) {\n                this.addAnnotation(\n                    node,\n                    ResourceType.tagSolution,\n                    \"Solution tag is missing\"\n                );\n            }\n        }\n    }\n\n    private checkBucket(node: IConstruct) {\n        if (node instanceof CfnBucket) {\n            const c =\n                node.publicAccessBlockConfiguration as CfnBucket.PublicAccessBlockConfigurationProperty;\n\n            if (c) {\n                if (\n                    !c.blockPublicAcls ||\n                    !c.blockPublicPolicy ||\n                    !c.ignorePublicAcls ||\n                    !c.restrictPublicBuckets\n                ) {\n                    this.addAnnotation(\n                        node,\n                        ResourceType.bucketPublicity,\n                        \"Check bucket publicity\"\n                    );\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(\n                    node,\n                    ResourceType.resourcePath,\n                    \"Path part should be in kebab-case\"\n                );\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 (\n                        type === \"querystring\" &&\n                        !StackCheckingAspect.isValidQueryString(name)\n                    ) {\n                        this.addAnnotation(\n                            node,\n                            name,\n                            \"Querystring should be in snake_case\"\n                        );\n                    }\n                });\n            }\n        }\n    }\n\n    private checkQueueEncryption(node: IConstruct) {\n        if (node instanceof CfnQueue) {\n            if (!node.kmsMasterKeyId) {\n                this.addAnnotation(\n                    node,\n                    ResourceType.queueEncryption,\n                    \"Queue must have encryption enabled\"\n                );\n            }\n        }\n    }\n\n    private checkLogGroupRetention(node: IConstruct) {\n        if (node instanceof LogRetention) {\n            const child = node.node.defaultChild as unknown as Record<\n                string,\n                Record<string, string>\n            >;\n            const retention = child._cfnProperties.RetentionInDays;\n\n            if (!retention) {\n                this.addAnnotation(\n                    node,\n                    ResourceType.logGroupRetention,\n                    \"Log group must define log group retention\"\n                );\n            }\n        }\n    }\n}\n"]}
|
@@ -3,14 +3,14 @@ import { IVpc } from "aws-cdk-lib/aws-ec2";
|
|
3
3
|
import { ISecurityGroup } from "aws-cdk-lib/aws-ec2/lib/security-group";
|
4
4
|
import { ITopic } from "aws-cdk-lib/aws-sns";
|
5
5
|
import { ISecret } from "aws-cdk-lib/aws-secretsmanager";
|
6
|
-
import { Function } from "aws-cdk-lib/aws-lambda";
|
6
|
+
import { Function as AWSFunction } from "aws-cdk-lib/aws-lambda";
|
7
7
|
import { Construct } from "constructs";
|
8
8
|
import { TrafficType } from "../../../types/traffictype";
|
9
9
|
import { DBLambdaEnvironment } from "./lambda-configs";
|
10
10
|
export declare const SOLUTION_KEY = "Solution";
|
11
11
|
export declare const SSM_KEY_WARNING_TOPIC: string;
|
12
12
|
export declare const SSM_KEY_ALARM_TOPIC: string;
|
13
|
-
export
|
13
|
+
export interface StackConfiguration {
|
14
14
|
readonly shortName?: string;
|
15
15
|
readonly secretId?: string;
|
16
16
|
readonly alarmTopicArn: string;
|
@@ -28,7 +28,7 @@ export declare type StackConfiguration = {
|
|
28
28
|
readonly enableDocumentation?: boolean;
|
29
29
|
};
|
30
30
|
readonly whitelistedResources?: string[];
|
31
|
-
}
|
31
|
+
}
|
32
32
|
export declare class DigitrafficStack extends Stack {
|
33
33
|
readonly vpc: IVpc;
|
34
34
|
readonly lambdaDbSg: ISecurityGroup;
|
@@ -40,5 +40,5 @@ export declare class DigitrafficStack extends Stack {
|
|
40
40
|
addAspects(): void;
|
41
41
|
createLambdaEnvironment(): DBLambdaEnvironment;
|
42
42
|
createDefaultLambdaEnvironment(dbApplication: string): DBLambdaEnvironment;
|
43
|
-
grantSecret(...lambdas:
|
43
|
+
grantSecret(...lambdas: AWSFunction[]): void;
|
44
44
|
}
|
package/aws/infra/stack/stack.js
CHANGED
@@ -7,9 +7,9 @@ const aws_sns_1 = require("aws-cdk-lib/aws-sns");
|
|
7
7
|
const aws_ssm_1 = require("aws-cdk-lib/aws-ssm");
|
8
8
|
const aws_secretsmanager_1 = require("aws-cdk-lib/aws-secretsmanager");
|
9
9
|
const stack_checking_aspect_1 = require("./stack-checking-aspect");
|
10
|
-
const SSM_ROOT =
|
11
|
-
exports.SOLUTION_KEY =
|
12
|
-
const MONITORING_ROOT =
|
10
|
+
const SSM_ROOT = "/digitraffic";
|
11
|
+
exports.SOLUTION_KEY = "Solution";
|
12
|
+
const MONITORING_ROOT = "/monitoring";
|
13
13
|
exports.SSM_KEY_WARNING_TOPIC = `${SSM_ROOT}${MONITORING_ROOT}/warning-topic`;
|
14
14
|
exports.SSM_KEY_ALARM_TOPIC = `${SSM_ROOT}${MONITORING_ROOT}/alarm-topic`;
|
15
15
|
class DigitrafficStack extends aws_cdk_lib_1.Stack {
|
@@ -17,12 +17,12 @@ class DigitrafficStack extends aws_cdk_lib_1.Stack {
|
|
17
17
|
super(scope, id, configuration.stackProps);
|
18
18
|
this.configuration = configuration;
|
19
19
|
if (configuration.secretId) {
|
20
|
-
this.secret = aws_secretsmanager_1.Secret.fromSecretNameV2(this,
|
20
|
+
this.secret = aws_secretsmanager_1.Secret.fromSecretNameV2(this, "Secret", configuration.secretId);
|
21
21
|
}
|
22
22
|
// VPC reference construction requires vpcId and availability zones
|
23
23
|
// private subnets are used in Lambda configuration
|
24
24
|
if (configuration.vpcId) {
|
25
|
-
this.vpc = aws_ec2_1.Vpc.fromVpcAttributes(this,
|
25
|
+
this.vpc = aws_ec2_1.Vpc.fromVpcAttributes(this, "vpc", {
|
26
26
|
vpcId: configuration.vpcId,
|
27
27
|
privateSubnetIds: configuration.privateSubnetIds,
|
28
28
|
availabilityZones: configuration.availabilityZones,
|
@@ -30,10 +30,10 @@ class DigitrafficStack extends aws_cdk_lib_1.Stack {
|
|
30
30
|
}
|
31
31
|
// security group that allows Lambda database access
|
32
32
|
if (configuration.lambdaDbSgId) {
|
33
|
-
this.lambdaDbSg = aws_ec2_1.SecurityGroup.fromSecurityGroupId(this,
|
33
|
+
this.lambdaDbSg = aws_ec2_1.SecurityGroup.fromSecurityGroupId(this, "LambdaDbSG", configuration.lambdaDbSgId);
|
34
34
|
}
|
35
|
-
this.alarmTopic = aws_sns_1.Topic.fromTopicArn(this,
|
36
|
-
this.warningTopic = aws_sns_1.Topic.fromTopicArn(this,
|
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
37
|
this.addAspects();
|
38
38
|
}
|
39
39
|
addAspects() {
|
@@ -43,16 +43,18 @@ class DigitrafficStack extends aws_cdk_lib_1.Stack {
|
|
43
43
|
return this.createDefaultLambdaEnvironment(this.configuration.shortName);
|
44
44
|
}
|
45
45
|
createDefaultLambdaEnvironment(dbApplication) {
|
46
|
-
return this.configuration.secretId
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
46
|
+
return this.configuration.secretId
|
47
|
+
? {
|
48
|
+
SECRET_ID: this.configuration.secretId,
|
49
|
+
DB_APPLICATION: dbApplication,
|
50
|
+
}
|
51
|
+
: {
|
52
|
+
DB_APPLICATION: dbApplication,
|
53
|
+
};
|
52
54
|
}
|
53
55
|
grantSecret(...lambdas) {
|
54
56
|
lambdas.forEach((l) => this.secret.grantRead(l));
|
55
57
|
}
|
56
58
|
}
|
57
59
|
exports.DigitrafficStack = DigitrafficStack;
|
58
|
-
//# sourceMappingURL=data:application/json;base64,
|
60
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stack.js","sourceRoot":"","sources":["../../../../src/aws/infra/stack/stack.ts"],"names":[],"mappings":";;;AAAA,6CAAyD;AACzD,iDAA+D;AAE/D,iDAAoD;AACpD,iDAAsD;AACtD,uEAAiE;AAGjE,mEAA8D;AAK9D,MAAM,QAAQ,GAAG,cAAc,CAAC;AACnB,QAAA,YAAY,GAAG,UAAU,CAAC;AACvC,MAAM,eAAe,GAAG,aAAa,CAAC;AAEzB,QAAA,qBAAqB,GAAG,GAAG,QAAQ,GAAG,eAAe,gBAAgB,CAAC;AACtE,QAAA,mBAAmB,GAAG,GAAG,QAAQ,GAAG,eAAe,cAAc,CAAC;AA2B/E,MAAa,gBAAiB,SAAQ,mBAAK;IASvC,YACI,KAAgB,EAChB,EAAU,EACV,aAAiC;QAEjC,KAAK,CAAC,KAAK,EAAE,EAAE,EAAE,aAAa,CAAC,UAAU,CAAC,CAAC;QAE3C,IAAI,CAAC,aAAa,GAAG,aAAa,CAAC;QAEnC,IAAI,aAAa,CAAC,QAAQ,EAAE;YACxB,IAAI,CAAC,MAAM,GAAG,2BAAM,CAAC,gBAAgB,CACjC,IAAI,EACJ,QAAQ,EACR,aAAa,CAAC,QAAQ,CACzB,CAAC;SACL;QAED,mEAAmE;QACnE,mDAAmD;QACnD,IAAI,aAAa,CAAC,KAAK,EAAE;YACrB,IAAI,CAAC,GAAG,GAAG,aAAG,CAAC,iBAAiB,CAAC,IAAI,EAAE,KAAK,EAAE;gBAC1C,KAAK,EAAE,aAAa,CAAC,KAAK;gBAC1B,gBAAgB,EAAE,aAAa,CAAC,gBAAgB;gBAChD,iBAAiB,EAAE,aAAa,CAAC,iBAAkB;aACtD,CAAC,CAAC;SACN;QAED,oDAAoD;QACpD,IAAI,aAAa,CAAC,YAAY,EAAE;YAC5B,IAAI,CAAC,UAAU,GAAG,uBAAa,CAAC,mBAAmB,CAC/C,IAAI,EACJ,YAAY,EACZ,aAAa,CAAC,YAAY,CAC7B,CAAC;SACL;QAED,IAAI,CAAC,UAAU,GAAG,eAAK,CAAC,YAAY,CAChC,IAAI,EACJ,YAAY,EACZ,yBAAe,CAAC,uBAAuB,CACnC,IAAI,EACJ,iBAAiB,EACjB,2BAAmB,CACtB,CAAC,WAAW,CAChB,CAAC;QACF,IAAI,CAAC,YAAY,GAAG,eAAK,CAAC,YAAY,CAClC,IAAI,EACJ,cAAc,EACd,yBAAe,CAAC,uBAAuB,CACnC,IAAI,EACJ,mBAAmB,EACnB,6BAAqB,CACxB,CAAC,WAAW,CAChB,CAAC;QAEF,IAAI,CAAC,UAAU,EAAE,CAAC;IACtB,CAAC;IAED,UAAU;QACN,qBAAO,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,2CAAmB,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC;IAC3D,CAAC;IAED,uBAAuB;QACnB,OAAO,IAAI,CAAC,8BAA8B,CACtC,IAAI,CAAC,aAAa,CAAC,SAAU,CAChC,CAAC;IACN,CAAC;IAED,8BAA8B,CAAC,aAAqB;QAChD,OAAO,IAAI,CAAC,aAAa,CAAC,QAAQ;YAC9B,CAAC,CAAC;gBACI,SAAS,EAAE,IAAI,CAAC,aAAa,CAAC,QAAQ;gBACtC,cAAc,EAAE,aAAa;aAChC;YACH,CAAC,CAAC;gBACI,cAAc,EAAE,aAAa;aAChC,CAAC;IACZ,CAAC;IAED,WAAW,CAAC,GAAG,OAAsB;QACjC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAc,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC;IAClE,CAAC;CACJ;AA3FD,4CA2FC","sourcesContent":["import { Aspects, Stack, StackProps } from \"aws-cdk-lib\";\nimport { IVpc, SecurityGroup, Vpc } from \"aws-cdk-lib/aws-ec2\";\nimport { ISecurityGroup } from \"aws-cdk-lib/aws-ec2/lib/security-group\";\nimport { ITopic, Topic } from \"aws-cdk-lib/aws-sns\";\nimport { StringParameter } from \"aws-cdk-lib/aws-ssm\";\nimport { ISecret, Secret } from \"aws-cdk-lib/aws-secretsmanager\";\nimport { Function as AWSFunction } from \"aws-cdk-lib/aws-lambda\";\n\nimport { StackCheckingAspect } from \"./stack-checking-aspect\";\nimport { Construct } from \"constructs\";\nimport { TrafficType } from \"../../../types/traffictype\";\nimport { DBLambdaEnvironment } from \"./lambda-configs\";\n\nconst SSM_ROOT = \"/digitraffic\";\nexport const SOLUTION_KEY = \"Solution\";\nconst MONITORING_ROOT = \"/monitoring\";\n\nexport const SSM_KEY_WARNING_TOPIC = `${SSM_ROOT}${MONITORING_ROOT}/warning-topic`;\nexport const SSM_KEY_ALARM_TOPIC = `${SSM_ROOT}${MONITORING_ROOT}/alarm-topic`;\n\nexport interface StackConfiguration {\n    readonly shortName?: string;\n    readonly secretId?: string;\n    readonly alarmTopicArn: string;\n    readonly warningTopicArn: string;\n    readonly logsDestinationArn?: string;\n\n    readonly vpcId?: string;\n    readonly lambdaDbSgId?: string;\n    readonly privateSubnetIds?: string[];\n    readonly availabilityZones?: string[];\n\n    readonly trafficType: TrafficType;\n    readonly production: boolean;\n    readonly stackProps: StackProps;\n\n    readonly stackFeatures?: {\n        readonly enableCanaries?: boolean;\n        readonly enableDocumentation?: boolean;\n    };\n\n    /// whitelist resources for StackCheckingAspect\n    readonly whitelistedResources?: string[];\n}\n\nexport class DigitrafficStack extends Stack {\n    readonly vpc: IVpc;\n    readonly lambdaDbSg: ISecurityGroup;\n    readonly alarmTopic: ITopic;\n    readonly warningTopic: ITopic;\n    readonly secret: ISecret;\n\n    readonly configuration: StackConfiguration;\n\n    constructor(\n        scope: Construct,\n        id: string,\n        configuration: StackConfiguration\n    ) {\n        super(scope, id, configuration.stackProps);\n\n        this.configuration = configuration;\n\n        if (configuration.secretId) {\n            this.secret = Secret.fromSecretNameV2(\n                this,\n                \"Secret\",\n                configuration.secretId\n            );\n        }\n\n        // VPC reference construction requires vpcId and availability zones\n        // private subnets are used in Lambda configuration\n        if (configuration.vpcId) {\n            this.vpc = Vpc.fromVpcAttributes(this, \"vpc\", {\n                vpcId: configuration.vpcId,\n                privateSubnetIds: configuration.privateSubnetIds,\n                availabilityZones: configuration.availabilityZones!,\n            });\n        }\n\n        // security group that allows Lambda database access\n        if (configuration.lambdaDbSgId) {\n            this.lambdaDbSg = SecurityGroup.fromSecurityGroupId(\n                this,\n                \"LambdaDbSG\",\n                configuration.lambdaDbSgId\n            );\n        }\n\n        this.alarmTopic = Topic.fromTopicArn(\n            this,\n            \"AlarmTopic\",\n            StringParameter.fromStringParameterName(\n                this,\n                \"AlarmTopicParam\",\n                SSM_KEY_ALARM_TOPIC\n            ).stringValue\n        );\n        this.warningTopic = Topic.fromTopicArn(\n            this,\n            \"WarningTopic\",\n            StringParameter.fromStringParameterName(\n                this,\n                \"WarningTopicParam\",\n                SSM_KEY_WARNING_TOPIC\n            ).stringValue\n        );\n\n        this.addAspects();\n    }\n\n    addAspects() {\n        Aspects.of(this).add(StackCheckingAspect.create(this));\n    }\n\n    createLambdaEnvironment(): DBLambdaEnvironment {\n        return this.createDefaultLambdaEnvironment(\n            this.configuration.shortName!\n        );\n    }\n\n    createDefaultLambdaEnvironment(dbApplication: string): DBLambdaEnvironment {\n        return this.configuration.secretId\n            ? {\n                  SECRET_ID: this.configuration.secretId,\n                  DB_APPLICATION: dbApplication,\n              }\n            : {\n                  DB_APPLICATION: dbApplication,\n              };\n    }\n\n    grantSecret(...lambdas: AWSFunction[]) {\n        lambdas.forEach((l: AWSFunction) => this.secret.grantRead(l));\n    }\n}\n"]}
|
package/package.json
CHANGED
@@ -1,6 +1,6 @@
|
|
1
1
|
{
|
2
2
|
"name": "@digitraffic/common",
|
3
|
-
"version": "2022.10.
|
3
|
+
"version": "2022.10.14-1",
|
4
4
|
"description": "",
|
5
5
|
"repository": {
|
6
6
|
"type": "git",
|
@@ -25,13 +25,13 @@
|
|
25
25
|
"."
|
26
26
|
],
|
27
27
|
"dependencies": {
|
28
|
-
"@aws-cdk/aws-synthetics-alpha": "
|
28
|
+
"@aws-cdk/aws-synthetics-alpha": "2.46.0-alpha.0",
|
29
29
|
"@types/geojson": "^7946.0.7",
|
30
|
-
"aws-cdk-lib": "
|
31
|
-
"aws-sdk": "
|
30
|
+
"aws-cdk-lib": "2.46.0",
|
31
|
+
"aws-sdk": "2.1232.0",
|
32
32
|
"axios": "^0.21.1",
|
33
33
|
"change-case": "4.1.2",
|
34
|
-
"constructs": "
|
34
|
+
"constructs": "10.1.131",
|
35
35
|
"geojson-validation": "^1.0.2",
|
36
36
|
"moment": "^2.29.4",
|
37
37
|
"node-ttl": "^0.2.0",
|
@@ -63,7 +63,6 @@
|
|
63
63
|
"Synthetics"
|
64
64
|
],
|
65
65
|
"lint-staged": {
|
66
|
-
"*.{js,ts}": "eslint --cache --fix",
|
67
66
|
"*.{js,ts,css,md,yml,yaml,json}": "prettier --write"
|
68
67
|
}
|
69
68
|
}
|
package/yarn-error.log
ADDED
@@ -0,0 +1,103 @@
|
|
1
|
+
Arguments:
|
2
|
+
/Users/teijoro/.asdf/installs/nodejs/16.17.1/bin/node /usr/local/bin/yarn publish
|
3
|
+
|
4
|
+
PATH:
|
5
|
+
/var/folders/_z/7m6mzqyn17d6wydv4b56w2100000gp/T/yarn--1665733050816-0.7009530622080813:/Users/teijoro/work/git/digitraffic-common-private/node_modules/.bin:/Users/teijoro/.config/yarn/link/node_modules/.bin:/Users/teijoro/.asdf/installs/nodejs/16.17.1/libexec/lib/node_modules/npm/bin/node-gyp-bin:/Users/teijoro/.asdf/installs/nodejs/16.17.1/lib/node_modules/npm/bin/node-gyp-bin:/Users/teijoro/.asdf/installs/nodejs/16.17.1/bin/node_modules/npm/bin/node-gyp-bin:/Users/teijoro/.asdf/plugins/nodejs/shims:/Users/teijoro/.asdf/installs/nodejs/16.17.1/bin:/Users/teijoro/.asdf/shims:/usr/local/opt/asdf/libexec/bin:/usr/local/bin:/usr/bin:/bin:/usr/sbin:/sbin:/Library/Apple/usr/bin
|
6
|
+
|
7
|
+
Yarn version:
|
8
|
+
1.22.18
|
9
|
+
|
10
|
+
Node version:
|
11
|
+
16.17.1
|
12
|
+
|
13
|
+
Platform:
|
14
|
+
darwin x64
|
15
|
+
|
16
|
+
Trace:
|
17
|
+
Error: https://registry.yarnpkg.com/-/user/org.couchdb.user:teijosol: failed to authenticate: Could not authenticate teijosol: bad password
|
18
|
+
at Request.params.callback [as _callback] (/usr/local/lib/node_modules/yarn/lib/cli.js:66138:18)
|
19
|
+
at Request.self.callback (/usr/local/lib/node_modules/yarn/lib/cli.js:140883:22)
|
20
|
+
at Request.emit (node:events:513:28)
|
21
|
+
at Request.<anonymous> (/usr/local/lib/node_modules/yarn/lib/cli.js:141855:10)
|
22
|
+
at Request.emit (node:events:513:28)
|
23
|
+
at IncomingMessage.<anonymous> (/usr/local/lib/node_modules/yarn/lib/cli.js:141777:12)
|
24
|
+
at Object.onceWrapper (node:events:627:28)
|
25
|
+
at IncomingMessage.emit (node:events:525:35)
|
26
|
+
at endReadableNT (node:internal/streams/readable:1358:12)
|
27
|
+
at processTicksAndRejections (node:internal/process/task_queues:83:21)
|
28
|
+
|
29
|
+
npm manifest:
|
30
|
+
{
|
31
|
+
"name": "@digitraffic/common",
|
32
|
+
"version": "2022.10.14-1",
|
33
|
+
"description": "",
|
34
|
+
"repository": {
|
35
|
+
"type": "git",
|
36
|
+
"url": "https://github.com/tmfg/digitraffic-common.git"
|
37
|
+
},
|
38
|
+
"engines": {
|
39
|
+
"node": ">=14 <17",
|
40
|
+
"yarn": ">1.2 <2"
|
41
|
+
},
|
42
|
+
"publishConfig": {
|
43
|
+
"@tmfg:registry": "https://npm.pkg.github.com"
|
44
|
+
},
|
45
|
+
"scripts": {
|
46
|
+
"build": "tsc",
|
47
|
+
"lint": "eslint **/*.ts test/**/*.ts --fix",
|
48
|
+
"test": "jest --detectOpenHandles --forceExit --coverage --coverageDirectory=output/coverage/jest",
|
49
|
+
"pub": "cp package.json README.md LICENSE dist/ && cd dist && yarn publish"
|
50
|
+
},
|
51
|
+
"main": "index.js",
|
52
|
+
"types": "index.d.ts",
|
53
|
+
"files": [
|
54
|
+
"."
|
55
|
+
],
|
56
|
+
"dependencies": {
|
57
|
+
"@aws-cdk/aws-synthetics-alpha": "2.46.0-alpha.0",
|
58
|
+
"@types/geojson": "^7946.0.7",
|
59
|
+
"aws-cdk-lib": "2.46.0",
|
60
|
+
"aws-sdk": "2.1232.0",
|
61
|
+
"axios": "^0.21.1",
|
62
|
+
"change-case": "4.1.2",
|
63
|
+
"constructs": "10.1.131",
|
64
|
+
"geojson-validation": "^1.0.2",
|
65
|
+
"moment": "^2.29.4",
|
66
|
+
"node-ttl": "^0.2.0",
|
67
|
+
"pg-native": "^3.0.1",
|
68
|
+
"pg-promise": "^10.12.0"
|
69
|
+
},
|
70
|
+
"devDependencies": {
|
71
|
+
"@types/aws-lambda": "^8.10.106",
|
72
|
+
"@types/jest": "^29.0.3",
|
73
|
+
"@types/node": "^18.7.23",
|
74
|
+
"@types/ramda": "^0.28.15",
|
75
|
+
"@types/sinon": "^10.0.13",
|
76
|
+
"@typescript-eslint/eslint-plugin": "^5.39.0",
|
77
|
+
"@typescript-eslint/parser": "^5.39.0",
|
78
|
+
"eslint": "^8.24.0",
|
79
|
+
"eslint-config-prettier": "^8.5.0",
|
80
|
+
"husky": ">=6",
|
81
|
+
"jest": "^29.1.1",
|
82
|
+
"jest-junit": "^14.0.1",
|
83
|
+
"lint-staged": ">=10",
|
84
|
+
"prettier": "^2.7.1",
|
85
|
+
"ramda": "^0.28.0",
|
86
|
+
"sinon": "^14.0.0",
|
87
|
+
"ts-jest": "^29.0.2",
|
88
|
+
"typescript": "^4.8.4"
|
89
|
+
},
|
90
|
+
"externals": [
|
91
|
+
"aws-sdk",
|
92
|
+
"Synthetics"
|
93
|
+
],
|
94
|
+
"lint-staged": {
|
95
|
+
"*.{js,ts,css,md,yml,yaml,json}": "prettier --write"
|
96
|
+
}
|
97
|
+
}
|
98
|
+
|
99
|
+
yarn manifest:
|
100
|
+
No manifest
|
101
|
+
|
102
|
+
Lockfile:
|
103
|
+
No lockfile
|
package/aws/index.js
DELETED
@@ -1,18 +0,0 @@
|
|
1
|
-
"use strict";
|
2
|
-
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
3
|
-
if (k2 === undefined) k2 = k;
|
4
|
-
var desc = Object.getOwnPropertyDescriptor(m, k);
|
5
|
-
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
6
|
-
desc = { enumerable: true, get: function() { return m[k]; } };
|
7
|
-
}
|
8
|
-
Object.defineProperty(o, k2, desc);
|
9
|
-
}) : (function(o, m, k, k2) {
|
10
|
-
if (k2 === undefined) k2 = k;
|
11
|
-
o[k2] = m[k];
|
12
|
-
}));
|
13
|
-
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
14
|
-
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
15
|
-
};
|
16
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
17
|
-
__exportStar(require("./infra/stack/stack"), exports);
|
18
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYXdzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSxzREFBb0MiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tIFwiLi9pbmZyYS9zdGFjay9zdGFja1wiO1xuIl19
|
package/aws/infra/api/index.js
DELETED
@@ -1,18 +0,0 @@
|
|
1
|
-
"use strict";
|
2
|
-
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
3
|
-
if (k2 === undefined) k2 = k;
|
4
|
-
var desc = Object.getOwnPropertyDescriptor(m, k);
|
5
|
-
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
6
|
-
desc = { enumerable: true, get: function() { return m[k]; } };
|
7
|
-
}
|
8
|
-
Object.defineProperty(o, k2, desc);
|
9
|
-
}) : (function(o, m, k, k2) {
|
10
|
-
if (k2 === undefined) k2 = k;
|
11
|
-
o[k2] = m[k];
|
12
|
-
}));
|
13
|
-
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
14
|
-
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
15
|
-
};
|
16
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
17
|
-
__exportStar(require("./integration"), exports);
|
18
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvYXdzL2luZnJhL2FwaS9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsZ0RBQThCIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSBcIi4vaW50ZWdyYXRpb25cIjtcbiJdfQ==
|
package/types/index.js
DELETED
@@ -1,18 +0,0 @@
|
|
1
|
-
"use strict";
|
2
|
-
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
3
|
-
if (k2 === undefined) k2 = k;
|
4
|
-
var desc = Object.getOwnPropertyDescriptor(m, k);
|
5
|
-
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
6
|
-
desc = { enumerable: true, get: function() { return m[k]; } };
|
7
|
-
}
|
8
|
-
Object.defineProperty(o, k2, desc);
|
9
|
-
}) : (function(o, m, k, k2) {
|
10
|
-
if (k2 === undefined) k2 = k;
|
11
|
-
o[k2] = m[k];
|
12
|
-
}));
|
13
|
-
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
14
|
-
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
15
|
-
};
|
16
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
17
|
-
__exportStar(require("./language"), exports);
|
18
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdHlwZXMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLDZDQUEyQiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gXCIuL2xhbmd1YWdlXCI7XG5cbiJdfQ==
|