@digitraffic/common 2022.10.10-1 → 2022.10.25-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 CHANGED
@@ -1,8 +1,11 @@
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
11
  "@digitraffic/common": "*",
@@ -10,33 +13,40 @@ In package.json dependencies:
10
13
  ```
11
14
 
12
15
  In code:
16
+
13
17
  ```
14
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
- * Secret, VPC, Sg & alarmTopics automatically
20
- * Stack validation with StackCheckingAspect
21
- * Easier configuration with StackConfiguration
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
- Some commonly used parameters is predefined configuration. You can write configuration for your
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
- * Stack naming check(Test/Prod in name)
32
- * Function configuration(memory, timeout, runtime, reservedConcurrency)
33
- * Tags, must have Solution tag defined
34
- * S3 Buckets, no public access
35
- * Api Gateway resource casing(kebabCase and snake_case)
36
- * Queue encrypting
37
- * LogGroup Retention
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
38
47
 
39
48
  You can use StackCheckingAspect for any stack, DigitrafficStack does it automatically, but you can call it manually:
49
+
40
50
  ```
41
51
  Aspects.of(this).add(StackCheckingAspect.create(this));
42
52
  ```
@@ -44,9 +54,11 @@ Aspects.of(this).add(StackCheckingAspect.create(this));
44
54
  Any resource can be whitelisted by giving it as a parameter or in the StackConfiguration
45
55
 
46
56
  ### MonitoredFunction
57
+
47
58
  MonitoredFunction extends Function with alarms on memory usage and timeouts.
48
59
 
49
60
  If you need database access in your Function, you can use MonitoredDBFunction. Creating a Function with it is this easy:
61
+
50
62
  ```
51
63
  const lambda = MonitoredDBFunction.create(stack, 'get-metadata');
52
64
  ```
@@ -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.SYNTHETICS_NODEJS_PUPPETEER_3_5,
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", {
@@ -15,7 +15,10 @@ class DigitrafficCanary extends aws_synthetics_alpha_1.Canary {
15
15
  }),
16
16
  handler: params.handler,
17
17
  }),
18
- environmentVariables: { ...environmentVariables, ...params?.canaryEnv },
18
+ environmentVariables: {
19
+ ...environmentVariables,
20
+ ...params?.canaryEnv,
21
+ },
19
22
  canaryName,
20
23
  schedule: params.schedule ?? aws_synthetics_alpha_1.Schedule.rate(aws_cdk_lib_1.Duration.minutes(15)),
21
24
  });
@@ -26,4 +29,4 @@ class DigitrafficCanary extends aws_synthetics_alpha_1.Canary {
26
29
  }
27
30
  }
28
31
  exports.DigitrafficCanary = DigitrafficCanary;
29
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FuYXJ5LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2F3cy9pbmZyYS9jYW5hcmllcy9jYW5hcnkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsNkNBQXFDO0FBQ3JDLHdFQUF5RjtBQUV6RixpREFBMkM7QUFLM0MsTUFBYSxpQkFBa0IsU0FBUSw2QkFBTTtJQUN6QyxZQUNJLEtBQWdCLEVBQ2hCLFVBQWtCLEVBQ2xCLElBQVUsRUFDVixNQUF3QixFQUN4QixvQkFBdUM7UUFFdkMsS0FBSyxDQUFDLEtBQUssRUFBRSxVQUFVLEVBQUU7WUFDckIsT0FBTyxFQUFFLDhCQUFPLENBQUMsK0JBQStCO1lBQ2hELElBQUk7WUFDSixJQUFJLEVBQUUsMkJBQUksQ0FBQyxNQUFNLENBQUM7Z0JBQ2QsSUFBSSxFQUFFLElBQUksZ0NBQVMsQ0FBQyxNQUFNLEVBQUU7b0JBQ3hCLE9BQU8sRUFBRSxDQUFDLFFBQVEsQ0FBQztpQkFDdEIsQ0FBQztnQkFDRixPQUFPLEVBQUUsTUFBTSxDQUFDLE9BQU87YUFDMUIsQ0FBQztZQUNGLG9CQUFvQixFQUFFLEVBQUUsR0FBRyxvQkFBb0IsRUFBRSxHQUFHLE1BQU0sRUFBRSxTQUFTLEVBQUU7WUFDdkUsVUFBVTtZQUNWLFFBQVEsRUFBRSxNQUFNLENBQUMsUUFBUSxJQUFJLCtCQUFRLENBQUMsSUFBSSxDQUFDLHNCQUFRLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQyxDQUFDO1NBQ25FLENBQUMsQ0FBQztRQUVILElBQUksQ0FBQyxlQUFlLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQyxDQUFDO1FBRXRDLElBQUksTUFBTSxDQUFDLEtBQUssSUFBSSxJQUFJLEVBQUU7WUFDdEIsSUFBSSwwQkFBVyxDQUFDLEtBQUssRUFBRSxJQUFJLEVBQUUsTUFBTSxDQUFDLENBQUM7U0FDeEM7SUFDTCxDQUFDO0NBQ0o7QUE1QkQsOENBNEJDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtEdXJhdGlvbn0gZnJvbSBcImF3cy1jZGstbGliXCI7XG5pbXBvcnQge0Fzc2V0Q29kZSwgQ2FuYXJ5LCBSdW50aW1lLCBTY2hlZHVsZSwgVGVzdH0gZnJvbSBcIkBhd3MtY2RrL2F3cy1zeW50aGV0aWNzLWFscGhhXCI7XG5pbXBvcnQge1JvbGV9IGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtaWFtXCI7XG5pbXBvcnQge0NhbmFyeUFsYXJtfSBmcm9tIFwiLi9jYW5hcnktYWxhcm1cIjtcbmltcG9ydCB7Q2FuYXJ5UGFyYW1ldGVyc30gZnJvbSBcIi4vY2FuYXJ5LXBhcmFtZXRlcnNcIjtcbmltcG9ydCB7Q29uc3RydWN0fSBmcm9tIFwiY29uc3RydWN0c1wiO1xuaW1wb3J0IHtMYW1iZGFFbnZpcm9ubWVudH0gZnJvbSBcIi4uL3N0YWNrL2xhbWJkYS1jb25maWdzXCI7XG5cbmV4cG9ydCBjbGFzcyBEaWdpdHJhZmZpY0NhbmFyeSBleHRlbmRzIENhbmFyeSB7XG4gICAgY29uc3RydWN0b3IoXG4gICAgICAgIHNjb3BlOiBDb25zdHJ1Y3QsXG4gICAgICAgIGNhbmFyeU5hbWU6IHN0cmluZyxcbiAgICAgICAgcm9sZTogUm9sZSxcbiAgICAgICAgcGFyYW1zOiBDYW5hcnlQYXJhbWV0ZXJzLFxuICAgICAgICBlbnZpcm9ubWVudFZhcmlhYmxlczogTGFtYmRhRW52aXJvbm1lbnQsXG4gICAgKSB7XG4gICAgICAgIHN1cGVyKHNjb3BlLCBjYW5hcnlOYW1lLCB7XG4gICAgICAgICAgICBydW50aW1lOiBSdW50aW1lLlNZTlRIRVRJQ1NfTk9ERUpTX1BVUFBFVEVFUl8zXzUsXG4gICAgICAgICAgICByb2xlLFxuICAgICAgICAgICAgdGVzdDogVGVzdC5jdXN0b20oe1xuICAgICAgICAgICAgICAgIGNvZGU6IG5ldyBBc3NldENvZGUoXCJkaXN0XCIsIHtcbiAgICAgICAgICAgICAgICAgICAgZXhjbHVkZTogW1wibGFtYmRhXCJdLFxuICAgICAgICAgICAgICAgIH0pLFxuICAgICAgICAgICAgICAgIGhhbmRsZXI6IHBhcmFtcy5oYW5kbGVyLFxuICAgICAgICAgICAgfSksXG4gICAgICAgICAgICBlbnZpcm9ubWVudFZhcmlhYmxlczogeyAuLi5lbnZpcm9ubWVudFZhcmlhYmxlcywgLi4ucGFyYW1zPy5jYW5hcnlFbnYgfSxcbiAgICAgICAgICAgIGNhbmFyeU5hbWUsXG4gICAgICAgICAgICBzY2hlZHVsZTogcGFyYW1zLnNjaGVkdWxlID8/IFNjaGVkdWxlLnJhdGUoRHVyYXRpb24ubWludXRlcygxNSkpLFxuICAgICAgICB9KTtcblxuICAgICAgICB0aGlzLmFydGlmYWN0c0J1Y2tldC5ncmFudFdyaXRlKHJvbGUpO1xuXG4gICAgICAgIGlmIChwYXJhbXMuYWxhcm0gPz8gdHJ1ZSkge1xuICAgICAgICAgICAgbmV3IENhbmFyeUFsYXJtKHNjb3BlLCB0aGlzLCBwYXJhbXMpO1xuICAgICAgICB9XG4gICAgfVxufVxuIl19
32
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FuYXJ5LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2F3cy9pbmZyYS9jYW5hcmllcy9jYW5hcnkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsNkNBQXVDO0FBQ3ZDLHdFQU11QztBQUV2QyxpREFBNkM7QUFLN0MsTUFBYSxpQkFBa0IsU0FBUSw2QkFBTTtJQUN6QyxZQUNJLEtBQWdCLEVBQ2hCLFVBQWtCLEVBQ2xCLElBQVUsRUFDVixNQUF3QixFQUN4QixvQkFBdUM7UUFFdkMsS0FBSyxDQUFDLEtBQUssRUFBRSxVQUFVLEVBQUU7WUFDckIsT0FBTyxFQUFFLDhCQUFPLENBQUMsK0JBQStCO1lBQ2hELElBQUk7WUFDSixJQUFJLEVBQUUsMkJBQUksQ0FBQyxNQUFNLENBQUM7Z0JBQ2QsSUFBSSxFQUFFLElBQUksZ0NBQVMsQ0FBQyxNQUFNLEVBQUU7b0JBQ3hCLE9BQU8sRUFBRSxDQUFDLFFBQVEsQ0FBQztpQkFDdEIsQ0FBQztnQkFDRixPQUFPLEVBQUUsTUFBTSxDQUFDLE9BQU87YUFDMUIsQ0FBQztZQUNGLG9CQUFvQixFQUFFO2dCQUNsQixHQUFHLG9CQUFvQjtnQkFDdkIsR0FBRyxNQUFNLEVBQUUsU0FBUzthQUN2QjtZQUNELFVBQVU7WUFDVixRQUFRLEVBQUUsTUFBTSxDQUFDLFFBQVEsSUFBSSwrQkFBUSxDQUFDLElBQUksQ0FBQyxzQkFBUSxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUMsQ0FBQztTQUNuRSxDQUFDLENBQUM7UUFFSCxJQUFJLENBQUMsZUFBZSxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUV0QyxJQUFJLE1BQU0sQ0FBQyxLQUFLLElBQUksSUFBSSxFQUFFO1lBQ3RCLElBQUksMEJBQVcsQ0FBQyxLQUFLLEVBQUUsSUFBSSxFQUFFLE1BQU0sQ0FBQyxDQUFDO1NBQ3hDO0lBQ0wsQ0FBQztDQUNKO0FBL0JELDhDQStCQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IER1cmF0aW9uIH0gZnJvbSBcImF3cy1jZGstbGliXCI7XG5pbXBvcnQge1xuICAgIEFzc2V0Q29kZSxcbiAgICBDYW5hcnksXG4gICAgUnVudGltZSxcbiAgICBTY2hlZHVsZSxcbiAgICBUZXN0LFxufSBmcm9tIFwiQGF3cy1jZGsvYXdzLXN5bnRoZXRpY3MtYWxwaGFcIjtcbmltcG9ydCB7IFJvbGUgfSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWlhbVwiO1xuaW1wb3J0IHsgQ2FuYXJ5QWxhcm0gfSBmcm9tIFwiLi9jYW5hcnktYWxhcm1cIjtcbmltcG9ydCB7IENhbmFyeVBhcmFtZXRlcnMgfSBmcm9tIFwiLi9jYW5hcnktcGFyYW1ldGVyc1wiO1xuaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSBcImNvbnN0cnVjdHNcIjtcbmltcG9ydCB7IExhbWJkYUVudmlyb25tZW50IH0gZnJvbSBcIi4uL3N0YWNrL2xhbWJkYS1jb25maWdzXCI7XG5cbmV4cG9ydCBjbGFzcyBEaWdpdHJhZmZpY0NhbmFyeSBleHRlbmRzIENhbmFyeSB7XG4gICAgY29uc3RydWN0b3IoXG4gICAgICAgIHNjb3BlOiBDb25zdHJ1Y3QsXG4gICAgICAgIGNhbmFyeU5hbWU6IHN0cmluZyxcbiAgICAgICAgcm9sZTogUm9sZSxcbiAgICAgICAgcGFyYW1zOiBDYW5hcnlQYXJhbWV0ZXJzLFxuICAgICAgICBlbnZpcm9ubWVudFZhcmlhYmxlczogTGFtYmRhRW52aXJvbm1lbnRcbiAgICApIHtcbiAgICAgICAgc3VwZXIoc2NvcGUsIGNhbmFyeU5hbWUsIHtcbiAgICAgICAgICAgIHJ1bnRpbWU6IFJ1bnRpbWUuU1lOVEhFVElDU19OT0RFSlNfUFVQUEVURUVSXzNfNixcbiAgICAgICAgICAgIHJvbGUsXG4gICAgICAgICAgICB0ZXN0OiBUZXN0LmN1c3RvbSh7XG4gICAgICAgICAgICAgICAgY29kZTogbmV3IEFzc2V0Q29kZShcImRpc3RcIiwge1xuICAgICAgICAgICAgICAgICAgICBleGNsdWRlOiBbXCJsYW1iZGFcIl0sXG4gICAgICAgICAgICAgICAgfSksXG4gICAgICAgICAgICAgICAgaGFuZGxlcjogcGFyYW1zLmhhbmRsZXIsXG4gICAgICAgICAgICB9KSxcbiAgICAgICAgICAgIGVudmlyb25tZW50VmFyaWFibGVzOiB7XG4gICAgICAgICAgICAgICAgLi4uZW52aXJvbm1lbnRWYXJpYWJsZXMsXG4gICAgICAgICAgICAgICAgLi4ucGFyYW1zPy5jYW5hcnlFbnYsXG4gICAgICAgICAgICB9LFxuICAgICAgICAgICAgY2FuYXJ5TmFtZSxcbiAgICAgICAgICAgIHNjaGVkdWxlOiBwYXJhbXMuc2NoZWR1bGUgPz8gU2NoZWR1bGUucmF0ZShEdXJhdGlvbi5taW51dGVzKDE1KSksXG4gICAgICAgIH0pO1xuXG4gICAgICAgIHRoaXMuYXJ0aWZhY3RzQnVja2V0LmdyYW50V3JpdGUocm9sZSk7XG5cbiAgICAgICAgaWYgKHBhcmFtcy5hbGFybSA/PyB0cnVlKSB7XG4gICAgICAgICAgICBuZXcgQ2FuYXJ5QWxhcm0oc2NvcGUsIHRoaXMsIHBhcmFtcyk7XG4gICAgICAgIH1cbiAgICB9XG59XG4iXX0=
@@ -44,8 +44,8 @@ class StackCheckingAspect {
44
44
  this.checkLogGroupRetention(node);
45
45
  }
46
46
  isWhitelisted(key) {
47
- return this.whitelistedResources?.some(wl => {
48
- return key.matchAll(new RegExp(wl, 'g'));
47
+ return this.whitelistedResources?.some((wl) => {
48
+ return key.matchAll(new RegExp(wl, "g"));
49
49
  });
50
50
  }
51
51
  addAnnotation(node, key, message, isError = true) {
@@ -63,40 +63,46 @@ class StackCheckingAspect {
63
63
  }
64
64
  checkStack(node) {
65
65
  if (node instanceof stack_1.DigitrafficStack) {
66
- if ((node.stackName.includes('Test') || node.stackName.includes('Tst')) && node.configuration?.production) {
67
- this.addAnnotation(node, ResourceType.stackName, 'Production is set for Test-stack');
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");
68
70
  }
69
- if ((node.stackName.includes('Prod') || node.stackName.includes('Prd')) && !node.configuration?.production) {
70
- this.addAnnotation(node, ResourceType.stackName, 'Production is not set for Production-stack');
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");
71
75
  }
72
76
  }
73
77
  }
74
78
  checkFunction(node) {
75
79
  if (node instanceof aws_lambda_1.CfnFunction) {
76
80
  if (!node.reservedConcurrentExecutions) {
77
- this.addAnnotation(node, ResourceType.reservedConcurrentConcurrency, 'Function must have reservedConcurrentConcurrency');
81
+ this.addAnnotation(node, ResourceType.reservedConcurrentConcurrency, "Function must have reservedConcurrentConcurrency");
78
82
  }
79
83
  else if (node.reservedConcurrentExecutions > MAX_CONCURRENCY_LIMIT) {
80
- this.addAnnotation(node, ResourceType.reservedConcurrentConcurrency, 'Function reservedConcurrentConcurrency too high!');
84
+ this.addAnnotation(node, ResourceType.reservedConcurrentConcurrency, "Function reservedConcurrentConcurrency too high!");
81
85
  }
82
86
  if (!node.timeout) {
83
- this.addAnnotation(node, ResourceType.functionTimeout, 'Function must have timeout');
87
+ this.addAnnotation(node, ResourceType.functionTimeout, "Function must have timeout");
84
88
  }
85
89
  if (!node.memorySize) {
86
- this.addAnnotation(node, ResourceType.functionMemorySize, 'Function must have memorySize');
90
+ this.addAnnotation(node, ResourceType.functionMemorySize, "Function must have memorySize");
87
91
  }
88
92
  if (node.runtime !== NODE_RUNTIME) {
89
- this.addAnnotation(node, ResourceType.functionRuntime, 'wrong runtime ' + node.runtime);
93
+ this.addAnnotation(node, ResourceType.functionRuntime, `Function has wrong runtime ${node.runtime}`);
90
94
  }
91
- if (this.stackShortName && node.functionName && node.functionName.indexOf(this.stackShortName) !== 0) {
92
- this.addAnnotation(node, ResourceType.functionName, 'Function name does not begin with ' + this.stackShortName);
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}`);
93
99
  }
94
100
  }
95
101
  }
96
102
  checkTags(node) {
97
103
  if (node instanceof aws_cdk_lib_1.Stack) {
98
104
  if (!node.tags.tagValues()[stack_1.SOLUTION_KEY]) {
99
- this.addAnnotation(node, ResourceType.tagSolution, 'Solution tag is missing');
105
+ this.addAnnotation(node, ResourceType.tagSolution, "Solution tag is missing");
100
106
  }
101
107
  }
102
108
  }
@@ -104,19 +110,22 @@ class StackCheckingAspect {
104
110
  if (node instanceof aws_s3_1.CfnBucket) {
105
111
  const c = node.publicAccessBlockConfiguration;
106
112
  if (c) {
107
- if (!c.blockPublicAcls || !c.blockPublicPolicy || !c.ignorePublicAcls || !c.restrictPublicBuckets) {
108
- this.addAnnotation(node, ResourceType.bucketPublicity, 'Check bucket publicity');
113
+ if (!c.blockPublicAcls ||
114
+ !c.blockPublicPolicy ||
115
+ !c.ignorePublicAcls ||
116
+ !c.restrictPublicBuckets) {
117
+ this.addAnnotation(node, ResourceType.bucketPublicity, "Check bucket publicity");
109
118
  }
110
119
  }
111
120
  }
112
121
  }
113
122
  static isValidPath(path) {
114
123
  // if path includes . or { check only the trailing part of path
115
- if (path.includes('.')) {
116
- return this.isValidPath(path.split('.')[0]);
124
+ if (path.includes(".")) {
125
+ return this.isValidPath(path.split(".")[0]);
117
126
  }
118
- if (path.includes('{')) {
119
- return this.isValidPath(path.split('{')[0]);
127
+ if (path.includes("{")) {
128
+ return this.isValidPath(path.split("{")[0]);
120
129
  }
121
130
  return (0, change_case_1.paramCase)(path) === path;
122
131
  }
@@ -126,18 +135,19 @@ class StackCheckingAspect {
126
135
  checkResourceCasing(node) {
127
136
  if (node instanceof aws_apigateway_1.CfnResource) {
128
137
  if (!StackCheckingAspect.isValidPath(node.pathPart)) {
129
- this.addAnnotation(node, ResourceType.resourcePath, 'Path part should be in kebab-case');
138
+ this.addAnnotation(node, ResourceType.resourcePath, "Path part should be in kebab-case");
130
139
  }
131
140
  }
132
141
  else if (node instanceof aws_apigateway_1.CfnMethod) {
133
142
  const integration = node.integration;
134
143
  if (integration && integration.requestParameters) {
135
- Object.keys(integration.requestParameters).forEach(key => {
136
- const split = key.split('.');
144
+ Object.keys(integration.requestParameters).forEach((key) => {
145
+ const split = key.split(".");
137
146
  const type = split[2];
138
147
  const name = split[3];
139
- if (type === 'querystring' && !StackCheckingAspect.isValidQueryString(name)) {
140
- this.addAnnotation(node, name, 'Querystring should be in snake_case');
148
+ if (type === "querystring" &&
149
+ !StackCheckingAspect.isValidQueryString(name)) {
150
+ this.addAnnotation(node, name, "Querystring should be in snake_case");
141
151
  }
142
152
  });
143
153
  }
@@ -146,7 +156,7 @@ class StackCheckingAspect {
146
156
  checkQueueEncryption(node) {
147
157
  if (node instanceof aws_sqs_1.CfnQueue) {
148
158
  if (!node.kmsMasterKeyId) {
149
- this.addAnnotation(node, ResourceType.queueEncryption, 'Queue must have encryption enabled');
159
+ this.addAnnotation(node, ResourceType.queueEncryption, "Queue must have encryption enabled");
150
160
  }
151
161
  }
152
162
  }
@@ -155,10 +165,10 @@ class StackCheckingAspect {
155
165
  const child = node.node.defaultChild;
156
166
  const retention = child._cfnProperties.RetentionInDays;
157
167
  if (!retention) {
158
- this.addAnnotation(node, ResourceType.logGroupRetention, 'Log group must define log group retention');
168
+ this.addAnnotation(node, ResourceType.logGroupRetention, "Log group must define log group retention");
159
169
  }
160
170
  }
161
171
  }
162
172
  }
163
173
  exports.StackCheckingAspect = StackCheckingAspect;
164
- //# 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(stack.configuration.shortName, stack.configuration.whitelistedResources);\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(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.stackShortName && node.functionName && node.functionName.indexOf(this.stackShortName) !== 0) {\n                this.addAnnotation(node, ResourceType.functionName, 'Function name does not begin with ' + this.stackShortName);\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 declare type StackConfiguration = {
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: Function[]): void;
43
+ grantSecret(...lambdas: AWSFunction[]): void;
44
44
  }
@@ -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 = '/digitraffic';
11
- exports.SOLUTION_KEY = 'Solution';
12
- const MONITORING_ROOT = '/monitoring';
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, 'Secret', configuration.secretId);
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, 'vpc', {
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, 'LambdaDbSG', configuration.lambdaDbSgId);
33
+ this.lambdaDbSg = aws_ec2_1.SecurityGroup.fromSecurityGroupId(this, "LambdaDbSG", configuration.lambdaDbSgId);
34
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);
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
- SECRET_ID: this.configuration.secretId,
48
- DB_APPLICATION: dbApplication,
49
- } : {
50
- DB_APPLICATION: dbApplication,
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,{"version":3,"file":"stack.js","sourceRoot":"","sources":["../../../../src/aws/infra/stack/stack.ts"],"names":[],"mappings":";;;AAAA,6CAAuD;AACvD,iDAA6D;AAE7D,iDAAkD;AAClD,iDAAoD;AACpD,uEAA+D;AAG/D,mEAA4D;AAK5D,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,YAAY,KAAgB,EAAE,EAAU,EAAE,aAAiC;QACvE,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,CAAC,IAAI,EAAE,QAAQ,EAAE,aAAa,CAAC,QAAQ,CAAC,CAAC;SACjF;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,iBAA6B;aACjE,CAAC,CAAC;SACN;QAED,oDAAoD;QACpD,IAAI,aAAa,CAAC,YAAY,EAAE;YAC5B,IAAI,CAAC,UAAU,GAAG,uBAAa,CAAC,mBAAmB,CAAC,IAAI,EAAE,YAAY,EAAE,aAAa,CAAC,YAAY,CAAC,CAAC;SACvG;QAED,IAAI,CAAC,UAAU,GAAG,eAAK,CAAC,YAAY,CAAC,IAAI,EACrC,YAAY,EACZ,yBAAe,CAAC,uBAAuB,CAAC,IAAI,EAAE,iBAAiB,EAAE,2BAAmB,CAAC,CAAC,WAAW,CAAC,CAAC;QACvG,IAAI,CAAC,YAAY,GAAG,eAAK,CAAC,YAAY,CAAC,IAAI,EAAE,cAAc,EACvD,yBAAe,CAAC,uBAAuB,CAAC,IAAI,EAAE,mBAAmB,EAAE,6BAAqB,CAAC,CAAC,WAAW,CAAC,CAAC;QAE3G,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,CAAC,IAAI,CAAC,aAAa,CAAC,SAAmB,CAAC,CAAC;IACvF,CAAC;IAED,8BAA8B,CAAC,aAAqB;QAChD,OAAO,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC,CAAC;YACjC,SAAS,EAAE,IAAI,CAAC,aAAa,CAAC,QAAQ;YACtC,cAAc,EAAE,aAAa;SAChC,CAAC,CAAC,CAAC;YACA,cAAc,EAAE,aAAa;SAChC,CAAC;IACN,CAAC;IAED,WAAW,CAAC,GAAG,OAAmB;QAC9B,OAAO,CAAC,OAAO,CAAC,CAAC,CAAW,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC;IAC/D,CAAC;CACJ;AA9DD,4CA8DC","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} 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 type 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(scope: Construct, id: string, configuration: StackConfiguration) {\n        super(scope, id, configuration.stackProps);\n\n        this.configuration = configuration;\n\n        if (configuration.secretId) {\n            this.secret = Secret.fromSecretNameV2(this, 'Secret', configuration.secretId);\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 as string[],\n            });\n        }\n\n        // security group that allows Lambda database access\n        if (configuration.lambdaDbSgId) {\n            this.lambdaDbSg = SecurityGroup.fromSecurityGroupId(this, 'LambdaDbSG', configuration.lambdaDbSgId);\n        }\n\n        this.alarmTopic = Topic.fromTopicArn(this,\n            'AlarmTopic',\n            StringParameter.fromStringParameterName(this, 'AlarmTopicParam', SSM_KEY_ALARM_TOPIC).stringValue);\n        this.warningTopic = Topic.fromTopicArn(this, 'WarningTopic',\n            StringParameter.fromStringParameterName(this, 'WarningTopicParam', SSM_KEY_WARNING_TOPIC).stringValue);\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(this.configuration.shortName as string);\n    }\n\n    createDefaultLambdaEnvironment(dbApplication: string): DBLambdaEnvironment {\n        return this.configuration.secretId ? {\n            SECRET_ID: this.configuration.secretId,\n            DB_APPLICATION: dbApplication,\n        } : {\n            DB_APPLICATION: dbApplication,\n        };\n    }\n\n    grantSecret(...lambdas: Function[]) {\n        lambdas.forEach((l: Function) => this.secret.grantRead(l));\n    }\n}\n"]}
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"]}
@@ -0,0 +1 @@
1
+ export declare function envValue(key: string): string;
@@ -0,0 +1,12 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.envValue = void 0;
4
+ function envValue(key) {
5
+ const value = process.env[key];
6
+ if (value == null) {
7
+ throw new Error(`Missing environment value ${key}`);
8
+ }
9
+ return value;
10
+ }
11
+ exports.envValue = envValue;
12
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZW52aXJvbm1lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvYXdzL3J1bnRpbWUvZW52aXJvbm1lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsU0FBZ0IsUUFBUSxDQUFDLEdBQVc7SUFDaEMsTUFBTSxLQUFLLEdBQUcsT0FBTyxDQUFDLEdBQUcsQ0FBQyxHQUFHLENBQUMsQ0FBQztJQUUvQixJQUFJLEtBQUssSUFBSSxJQUFJLEVBQUU7UUFDZixNQUFNLElBQUksS0FBSyxDQUFDLDZCQUE2QixHQUFHLEVBQUUsQ0FBQyxDQUFDO0tBQ3ZEO0lBRUQsT0FBTyxLQUFLLENBQUM7QUFDakIsQ0FBQztBQVJELDRCQVFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGZ1bmN0aW9uIGVudlZhbHVlKGtleTogc3RyaW5nKTogc3RyaW5nIHtcbiAgICBjb25zdCB2YWx1ZSA9IHByb2Nlc3MuZW52W2tleV07XG5cbiAgICBpZiAodmFsdWUgPT0gbnVsbCkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYE1pc3NpbmcgZW52aXJvbm1lbnQgdmFsdWUgJHtrZXl9YCk7XG4gICAgfVxuXG4gICAgcmV0dXJuIHZhbHVlO1xufVxuIl19
@@ -0,0 +1,6 @@
1
+ export interface Countable {
2
+ count: number;
3
+ }
4
+ export interface Identifiable<T> {
5
+ id: T;
6
+ }
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibW9kZWxzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2RhdGFiYXNlL21vZGVscy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGludGVyZmFjZSBDb3VudGFibGUge1xuICAgIGNvdW50OiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgSWRlbnRpZmlhYmxlPFQ+IHtcbiAgICBpZDogVDtcbn1cblxuaW50ZXJmYWNlIFVwZGF0ZWREYXRlIHtcbiAgICB1cGRhdGVkX2RhdGU6IERhdGU7XG59XG4iXX0=
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@digitraffic/common",
3
- "version": "2022.10.10-1",
3
+ "version": "2022.10.25-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": "^2.27.0-alpha.0",
28
+ "@aws-cdk/aws-synthetics-alpha": "2.46.0-alpha.0",
29
29
  "@types/geojson": "^7946.0.7",
30
- "aws-cdk-lib": "^2.43.1",
31
- "aws-sdk": "^2.1225.0",
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": "^10.1.116",
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
  }
@@ -1,2 +1,2 @@
1
1
  import { DTDatabase } from "../database/database";
2
- export declare function dbTestBase(fn: (db: DTDatabase) => void, truncateFn: (db: DTDatabase) => void, dbUser: string, dbPass: string, dbUri: string): () => void;
2
+ export declare function dbTestBase(fn: (db: DTDatabase) => void, truncateFn: (db: DTDatabase) => Promise<void>, dbUser: string, dbPass: string, dbUri: string): () => void;
@@ -7,7 +7,7 @@ function dbTestBase(fn, truncateFn, dbUser, dbPass, dbUri) {
7
7
  const theDbUri = process.env.DB_URI ?? dbUri;
8
8
  console.log(`Test database URI: ${theDbUri}`);
9
9
  return () => {
10
- const db = (0, database_1.initDbConnection)(dbUser, dbPass, 'test', theDbUri, {
10
+ const db = (0, database_1.initDbConnection)(dbUser, dbPass, "test", theDbUri, {
11
11
  noWarnings: true, // ignore duplicate connection warning for tests
12
12
  });
13
13
  beforeAll(async () => {
@@ -28,4 +28,4 @@ function dbTestBase(fn, truncateFn, dbUser, dbPass, dbUri) {
28
28
  };
29
29
  }
30
30
  exports.dbTestBase = dbTestBase;
31
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGItdGVzdHV0aWxzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3Rlc3QvZGItdGVzdHV0aWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLG1EQUFrRTtBQUNsRSw4REFBd0U7QUFFeEUsU0FBZ0IsVUFBVSxDQUN0QixFQUE0QixFQUM1QixVQUFvQyxFQUNwQyxNQUFjLEVBQ2QsTUFBYyxFQUNkLEtBQWE7SUFHYixNQUFNLFFBQVEsR0FBRyxPQUFPLENBQUMsR0FBRyxDQUFDLE1BQU0sSUFBSSxLQUFLLENBQUM7SUFDN0MsT0FBTyxDQUFDLEdBQUcsQ0FBQyxzQkFBc0IsUUFBUSxFQUFFLENBQUMsQ0FBQztJQUU5QyxPQUFPLEdBQUcsRUFBRTtRQUNSLE1BQU0sRUFBRSxHQUFlLElBQUEsMkJBQWdCLEVBQ25DLE1BQU0sRUFBRSxNQUFNLEVBQUUsTUFBTSxFQUFFLFFBQVEsRUFBRTtZQUM5QixVQUFVLEVBQUUsSUFBSSxFQUFFLGdEQUFnRDtTQUNyRSxDQUNKLENBQUM7UUFFRixTQUFTLENBQUMsS0FBSyxJQUFJLEVBQUU7WUFDakIsT0FBTyxDQUFDLEdBQUcsQ0FBQyxrQ0FBdUIsQ0FBQyxPQUFPLENBQUMsR0FBRyxNQUFNLENBQUM7WUFDdEQsT0FBTyxDQUFDLEdBQUcsQ0FBQyxrQ0FBdUIsQ0FBQyxPQUFPLENBQUMsR0FBRyxNQUFNLENBQUM7WUFDdEQsT0FBTyxDQUFDLEdBQUcsQ0FBQyxrQ0FBdUIsQ0FBQyxNQUFNLENBQUMsR0FBRyxRQUFRLENBQUM7WUFDdkQsT0FBTyxDQUFDLEdBQUcsQ0FBQyxrQ0FBdUIsQ0FBQyxTQUFTLENBQUMsR0FBRyxRQUFRLENBQUM7WUFDMUQsTUFBTSxVQUFVLENBQUMsRUFBRSxDQUFDLENBQUM7UUFDekIsQ0FBQyxDQUFDLENBQUM7UUFFSCxRQUFRLENBQUMsS0FBSyxJQUFJLEVBQUU7WUFDaEIsTUFBTSxVQUFVLENBQUMsRUFBRSxDQUFDLENBQUM7WUFDckIsTUFBTSxFQUFFLENBQUMsS0FBSyxDQUFDLEdBQUcsRUFBRSxDQUFDO1FBQ3pCLENBQUMsQ0FBQyxDQUFDO1FBRUgsVUFBVSxDQUFDLEtBQUssSUFBSSxFQUFFO1lBQ2xCLE1BQU0sVUFBVSxDQUFDLEVBQUUsQ0FBQyxDQUFDO1FBQ3pCLENBQUMsQ0FBQyxDQUFDO1FBRUgsRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDO0lBQ1gsQ0FBQyxDQUFDO0FBQ04sQ0FBQztBQXJDRCxnQ0FxQ0MiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0RURGF0YWJhc2UsIGluaXREYkNvbm5lY3Rpb259IGZyb20gXCIuLi9kYXRhYmFzZS9kYXRhYmFzZVwiO1xuaW1wb3J0IHtEYXRhYmFzZUVudmlyb25tZW50S2V5c30gZnJvbSBcIi4uL2F3cy9ydW50aW1lL3NlY3JldHMvZGJzZWNyZXRcIjtcblxuZXhwb3J0IGZ1bmN0aW9uIGRiVGVzdEJhc2UoXG4gICAgZm46IChkYjogRFREYXRhYmFzZSkgPT4gdm9pZCxcbiAgICB0cnVuY2F0ZUZuOiAoZGI6IERURGF0YWJhc2UpID0+IHZvaWQsXG4gICAgZGJVc2VyOiBzdHJpbmcsXG4gICAgZGJQYXNzOiBzdHJpbmcsXG4gICAgZGJVcmk6IHN0cmluZyxcbik6ICgpID0+IHZvaWQge1xuXG4gICAgY29uc3QgdGhlRGJVcmkgPSBwcm9jZXNzLmVudi5EQl9VUkkgPz8gZGJVcmk7XG4gICAgY29uc29sZS5sb2coYFRlc3QgZGF0YWJhc2UgVVJJOiAke3RoZURiVXJpfWApO1xuXG4gICAgcmV0dXJuICgpID0+IHtcbiAgICAgICAgY29uc3QgZGI6IERURGF0YWJhc2UgPSBpbml0RGJDb25uZWN0aW9uKFxuICAgICAgICAgICAgZGJVc2VyLCBkYlBhc3MsICd0ZXN0JywgdGhlRGJVcmksIHtcbiAgICAgICAgICAgICAgICBub1dhcm5pbmdzOiB0cnVlLCAvLyBpZ25vcmUgZHVwbGljYXRlIGNvbm5lY3Rpb24gd2FybmluZyBmb3IgdGVzdHNcbiAgICAgICAgICAgIH0sXG4gICAgICAgICk7XG5cbiAgICAgICAgYmVmb3JlQWxsKGFzeW5jICgpID0+IHtcbiAgICAgICAgICAgIHByb2Nlc3MuZW52W0RhdGFiYXNlRW52aXJvbm1lbnRLZXlzLkRCX1VTRVJdID0gZGJVc2VyO1xuICAgICAgICAgICAgcHJvY2Vzcy5lbnZbRGF0YWJhc2VFbnZpcm9ubWVudEtleXMuREJfUEFTU10gPSBkYlBhc3M7XG4gICAgICAgICAgICBwcm9jZXNzLmVudltEYXRhYmFzZUVudmlyb25tZW50S2V5cy5EQl9VUkldID0gdGhlRGJVcmk7XG4gICAgICAgICAgICBwcm9jZXNzLmVudltEYXRhYmFzZUVudmlyb25tZW50S2V5cy5EQl9ST19VUkldID0gdGhlRGJVcmk7XG4gICAgICAgICAgICBhd2FpdCB0cnVuY2F0ZUZuKGRiKTtcbiAgICAgICAgfSk7XG5cbiAgICAgICAgYWZ0ZXJBbGwoYXN5bmMgKCkgPT4ge1xuICAgICAgICAgICAgYXdhaXQgdHJ1bmNhdGVGbihkYik7XG4gICAgICAgICAgICBhd2FpdCBkYi4kcG9vbC5lbmQoKTtcbiAgICAgICAgfSk7XG5cbiAgICAgICAgYmVmb3JlRWFjaChhc3luYyAoKSA9PiB7XG4gICAgICAgICAgICBhd2FpdCB0cnVuY2F0ZUZuKGRiKTtcbiAgICAgICAgfSk7XG5cbiAgICAgICAgZm4oZGIpO1xuICAgIH07XG59XG4iXX0=
31
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGItdGVzdHV0aWxzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3Rlc3QvZGItdGVzdHV0aWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLG1EQUFvRTtBQUNwRSw4REFBMEU7QUFFMUUsU0FBZ0IsVUFBVSxDQUN0QixFQUE0QixFQUM1QixVQUE2QyxFQUM3QyxNQUFjLEVBQ2QsTUFBYyxFQUNkLEtBQWE7SUFFYixNQUFNLFFBQVEsR0FBRyxPQUFPLENBQUMsR0FBRyxDQUFDLE1BQU0sSUFBSSxLQUFLLENBQUM7SUFDN0MsT0FBTyxDQUFDLEdBQUcsQ0FBQyxzQkFBc0IsUUFBUSxFQUFFLENBQUMsQ0FBQztJQUU5QyxPQUFPLEdBQUcsRUFBRTtRQUNSLE1BQU0sRUFBRSxHQUFlLElBQUEsMkJBQWdCLEVBQ25DLE1BQU0sRUFDTixNQUFNLEVBQ04sTUFBTSxFQUNOLFFBQVEsRUFDUjtZQUNJLFVBQVUsRUFBRSxJQUFJLEVBQUUsZ0RBQWdEO1NBQ3JFLENBQ0osQ0FBQztRQUVGLFNBQVMsQ0FBQyxLQUFLLElBQUksRUFBRTtZQUNqQixPQUFPLENBQUMsR0FBRyxDQUFDLGtDQUF1QixDQUFDLE9BQU8sQ0FBQyxHQUFHLE1BQU0sQ0FBQztZQUN0RCxPQUFPLENBQUMsR0FBRyxDQUFDLGtDQUF1QixDQUFDLE9BQU8sQ0FBQyxHQUFHLE1BQU0sQ0FBQztZQUN0RCxPQUFPLENBQUMsR0FBRyxDQUFDLGtDQUF1QixDQUFDLE1BQU0sQ0FBQyxHQUFHLFFBQVEsQ0FBQztZQUN2RCxPQUFPLENBQUMsR0FBRyxDQUFDLGtDQUF1QixDQUFDLFNBQVMsQ0FBQyxHQUFHLFFBQVEsQ0FBQztZQUMxRCxNQUFNLFVBQVUsQ0FBQyxFQUFFLENBQUMsQ0FBQztRQUN6QixDQUFDLENBQUMsQ0FBQztRQUVILFFBQVEsQ0FBQyxLQUFLLElBQUksRUFBRTtZQUNoQixNQUFNLFVBQVUsQ0FBQyxFQUFFLENBQUMsQ0FBQztZQUNyQixNQUFNLEVBQUUsQ0FBQyxLQUFLLENBQUMsR0FBRyxFQUFFLENBQUM7UUFDekIsQ0FBQyxDQUFDLENBQUM7UUFFSCxVQUFVLENBQUMsS0FBSyxJQUFJLEVBQUU7WUFDbEIsTUFBTSxVQUFVLENBQUMsRUFBRSxDQUFDLENBQUM7UUFDekIsQ0FBQyxDQUFDLENBQUM7UUFFSCxFQUFFLENBQUMsRUFBRSxDQUFDLENBQUM7SUFDWCxDQUFDLENBQUM7QUFDTixDQUFDO0FBeENELGdDQXdDQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERURGF0YWJhc2UsIGluaXREYkNvbm5lY3Rpb24gfSBmcm9tIFwiLi4vZGF0YWJhc2UvZGF0YWJhc2VcIjtcbmltcG9ydCB7IERhdGFiYXNlRW52aXJvbm1lbnRLZXlzIH0gZnJvbSBcIi4uL2F3cy9ydW50aW1lL3NlY3JldHMvZGJzZWNyZXRcIjtcblxuZXhwb3J0IGZ1bmN0aW9uIGRiVGVzdEJhc2UoXG4gICAgZm46IChkYjogRFREYXRhYmFzZSkgPT4gdm9pZCxcbiAgICB0cnVuY2F0ZUZuOiAoZGI6IERURGF0YWJhc2UpID0+IFByb21pc2U8dm9pZD4sXG4gICAgZGJVc2VyOiBzdHJpbmcsXG4gICAgZGJQYXNzOiBzdHJpbmcsXG4gICAgZGJVcmk6IHN0cmluZ1xuKTogKCkgPT4gdm9pZCB7XG4gICAgY29uc3QgdGhlRGJVcmkgPSBwcm9jZXNzLmVudi5EQl9VUkkgPz8gZGJVcmk7XG4gICAgY29uc29sZS5sb2coYFRlc3QgZGF0YWJhc2UgVVJJOiAke3RoZURiVXJpfWApO1xuXG4gICAgcmV0dXJuICgpID0+IHtcbiAgICAgICAgY29uc3QgZGI6IERURGF0YWJhc2UgPSBpbml0RGJDb25uZWN0aW9uKFxuICAgICAgICAgICAgZGJVc2VyLFxuICAgICAgICAgICAgZGJQYXNzLFxuICAgICAgICAgICAgXCJ0ZXN0XCIsXG4gICAgICAgICAgICB0aGVEYlVyaSxcbiAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgICBub1dhcm5pbmdzOiB0cnVlLCAvLyBpZ25vcmUgZHVwbGljYXRlIGNvbm5lY3Rpb24gd2FybmluZyBmb3IgdGVzdHNcbiAgICAgICAgICAgIH1cbiAgICAgICAgKTtcblxuICAgICAgICBiZWZvcmVBbGwoYXN5bmMgKCkgPT4ge1xuICAgICAgICAgICAgcHJvY2Vzcy5lbnZbRGF0YWJhc2VFbnZpcm9ubWVudEtleXMuREJfVVNFUl0gPSBkYlVzZXI7XG4gICAgICAgICAgICBwcm9jZXNzLmVudltEYXRhYmFzZUVudmlyb25tZW50S2V5cy5EQl9QQVNTXSA9IGRiUGFzcztcbiAgICAgICAgICAgIHByb2Nlc3MuZW52W0RhdGFiYXNlRW52aXJvbm1lbnRLZXlzLkRCX1VSSV0gPSB0aGVEYlVyaTtcbiAgICAgICAgICAgIHByb2Nlc3MuZW52W0RhdGFiYXNlRW52aXJvbm1lbnRLZXlzLkRCX1JPX1VSSV0gPSB0aGVEYlVyaTtcbiAgICAgICAgICAgIGF3YWl0IHRydW5jYXRlRm4oZGIpO1xuICAgICAgICB9KTtcblxuICAgICAgICBhZnRlckFsbChhc3luYyAoKSA9PiB7XG4gICAgICAgICAgICBhd2FpdCB0cnVuY2F0ZUZuKGRiKTtcbiAgICAgICAgICAgIGF3YWl0IGRiLiRwb29sLmVuZCgpO1xuICAgICAgICB9KTtcblxuICAgICAgICBiZWZvcmVFYWNoKGFzeW5jICgpID0+IHtcbiAgICAgICAgICAgIGF3YWl0IHRydW5jYXRlRm4oZGIpO1xuICAgICAgICB9KTtcblxuICAgICAgICBmbihkYik7XG4gICAgfTtcbn1cbiJdfQ==
package/utils/utils.d.ts CHANGED
@@ -20,3 +20,11 @@
20
20
  * @param b second array to compare
21
21
  */
22
22
  export declare function bothArraysHasSameValues(a: null | undefined | unknown[], b: null | undefined | unknown[]): boolean;
23
+ /**
24
+ * Returns the last item on the array. If the array is empty, throws an error!
25
+ */
26
+ export declare function getLast<T>(array: T[], sortFunction?: (a: T) => number): T;
27
+ /**
28
+ * Returns the first item on the array. If the array is empty, throws an error!
29
+ */
30
+ export declare function getFirst<T>(array: T[], sortFunction?: (a: T) => number): T;
package/utils/utils.js CHANGED
@@ -1,6 +1,6 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.bothArraysHasSameValues = void 0;
3
+ exports.getFirst = exports.getLast = exports.bothArraysHasSameValues = void 0;
4
4
  /**
5
5
  * Check if arrays have only elements that also exists also in other array.
6
6
  * Individual element count doesn't matter.
@@ -34,7 +34,31 @@ function bothArraysHasSameValues(a, b) {
34
34
  if (aSet.size !== bSet.size) {
35
35
  return false;
36
36
  }
37
- return Array.from(aSet).every(value => bSet.has(value));
37
+ return Array.from(aSet).every((value) => bSet.has(value));
38
38
  }
39
39
  exports.bothArraysHasSameValues = bothArraysHasSameValues;
40
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdXRpbHMvdXRpbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBb0JHO0FBQ0gsU0FBZ0IsdUJBQXVCLENBQUMsQ0FBMkIsRUFBRSxDQUEyQjtJQUM1RixJQUFJLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsRUFBRTtRQUN4QixPQUFPLEtBQUssQ0FBQztLQUNoQjtTQUFNLElBQUksQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDLEVBQUU7UUFDakIsT0FBTyxJQUFJLENBQUM7S0FDZjtJQUNELE1BQU0sSUFBSSxHQUFHLElBQUksR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ3hCLE1BQU0sSUFBSSxHQUFHLElBQUksR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ3hCLElBQUksSUFBSSxDQUFDLElBQUksS0FBSyxJQUFJLENBQUMsSUFBSSxFQUFFO1FBQ3pCLE9BQU8sS0FBSyxDQUFDO0tBQ2hCO0lBQ0QsT0FBTyxLQUFLLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDLEtBQUssQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQztBQUM1RCxDQUFDO0FBWkQsMERBWUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIENoZWNrIGlmIGFycmF5cyBoYXZlIG9ubHkgZWxlbWVudHMgdGhhdCBhbHNvIGV4aXN0cyBhbHNvIGluIG90aGVyIGFycmF5LlxuICogSW5kaXZpZHVhbCBlbGVtZW50IGNvdW50IGRvZXNuJ3QgbWF0dGVyLlxuICogRnVuY3Rpb24gd29ya3Mgb25seSBmb3IgcHJpbWl0aXZlIHR5cGVzIGFuZCBmb3Igb3RoZXIgaXQganVzdCBjaGVja3MgdGhlIHJlZmVyZW5jZSB0byBvYmplY3QuXG4gKlxuICogU29tZSBleGFtcGxlc1xuICogYm90aEFycmF5c0hhc1NhbWVWYWx1ZXMoIFthLCBiXSwgW2IsIGFdICkgICAgPT4gdHJ1ZVxuICogYm90aEFycmF5c0hhc1NhbWVWYWx1ZXMoIFthLCBhXSwgW2EsIGEsIGFdICkgPT4gdHJ1ZVxuICogYm90aEFycmF5c0hhc1NhbWVWYWx1ZXMoIFthLCBiXSwgW2FdICkgICAgICAgPT4gZmFsc2VcbiAqXG4gKiBPYmplY3QgcmVmZXJlbmNlczpcbiAqIGNvbnN0IG8xID0geyBhOiAxLCBiOiAyfTtcbiAqIGNvbnN0IG8yID0geyBhOiAxLCBiOiAyfTtcbiAqIC8vIEFycmF5cyBoYXMgcmVmZXJlbmNlcyB0byBzYW1lIG9iamVjdHNcbiAqIGJvdGhBcnJheXNIYXNTYW1lVmFsdWVzKFtvMV0sIFtvMV0pKSAgICAgICAgID0+IHRydWVcbiAqIEFycmF5cyBoYXZlIHJlZmVyZW5jZXMgdG8gZGlmZmVyZW50IG9iamVjdHNcbiAqIGJvdGhBcnJheXNIYXNTYW1lVmFsdWVzKFtvMV0sIFtvMl0pKSAgICAgICAgID0+IGZhbHNlXG4gKlxuICogQHBhcmFtIGEgZmlyc3QgYXJyYXkgdG8gY29tcGFyZVxuICogQHBhcmFtIGIgc2Vjb25kIGFycmF5IHRvIGNvbXBhcmVcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGJvdGhBcnJheXNIYXNTYW1lVmFsdWVzKGE6IG51bGx8dW5kZWZpbmVkfHVua25vd25bXSwgYjogbnVsbHx1bmRlZmluZWR8dW5rbm93bltdKTogYm9vbGVhbiB7XG4gICAgaWYgKChhICYmICFiKSB8fCAoIWEgJiYgYikpIHtcbiAgICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH0gZWxzZSBpZiAoIWEgJiYgIWIpIHtcbiAgICAgICAgcmV0dXJuIHRydWU7XG4gICAgfVxuICAgIGNvbnN0IGFTZXQgPSBuZXcgU2V0KGEpO1xuICAgIGNvbnN0IGJTZXQgPSBuZXcgU2V0KGIpO1xuICAgIGlmIChhU2V0LnNpemUgIT09IGJTZXQuc2l6ZSkge1xuICAgICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuICAgIHJldHVybiBBcnJheS5mcm9tKGFTZXQpLmV2ZXJ5KHZhbHVlID0+IGJTZXQuaGFzKHZhbHVlKSk7XG59XG4iXX0=
40
+ /**
41
+ * Returns the last item on the array. If the array is empty, throws an error!
42
+ */
43
+ function getLast(array, sortFunction) {
44
+ return getFirstOrLast(false, array, sortFunction);
45
+ }
46
+ exports.getLast = getLast;
47
+ /**
48
+ * Returns the first item on the array. If the array is empty, throws an error!
49
+ */
50
+ function getFirst(array, sortFunction) {
51
+ return getFirstOrLast(true, array, sortFunction);
52
+ }
53
+ exports.getFirst = getFirst;
54
+ function getFirstOrLast(getFirst, array, sortFunction) {
55
+ if (array.length == 0) {
56
+ throw new Error(`can't get ${getFirst ? "first" : "last"} from empty array!`);
57
+ }
58
+ const index = getFirst ? 0 : array.length - 1;
59
+ if (sortFunction) {
60
+ return array.sort(sortFunction)[index];
61
+ }
62
+ return array[index];
63
+ }
64
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdXRpbHMvdXRpbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBb0JHO0FBQ0gsU0FBZ0IsdUJBQXVCLENBQ25DLENBQStCLEVBQy9CLENBQStCO0lBRS9CLElBQUksQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxFQUFFO1FBQ3hCLE9BQU8sS0FBSyxDQUFDO0tBQ2hCO1NBQU0sSUFBSSxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsRUFBRTtRQUNqQixPQUFPLElBQUksQ0FBQztLQUNmO0lBQ0QsTUFBTSxJQUFJLEdBQUcsSUFBSSxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDeEIsTUFBTSxJQUFJLEdBQUcsSUFBSSxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDeEIsSUFBSSxJQUFJLENBQUMsSUFBSSxLQUFLLElBQUksQ0FBQyxJQUFJLEVBQUU7UUFDekIsT0FBTyxLQUFLLENBQUM7S0FDaEI7SUFDRCxPQUFPLEtBQUssQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUMsS0FBSyxDQUFDLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUM7QUFDOUQsQ0FBQztBQWZELDBEQWVDO0FBRUQ7O0dBRUc7QUFDSCxTQUFnQixPQUFPLENBQUksS0FBVSxFQUFFLFlBQStCO0lBQ2xFLE9BQU8sY0FBYyxDQUFDLEtBQUssRUFBRSxLQUFLLEVBQUUsWUFBWSxDQUFDLENBQUM7QUFDdEQsQ0FBQztBQUZELDBCQUVDO0FBRUQ7O0dBRUc7QUFDSCxTQUFnQixRQUFRLENBQUksS0FBVSxFQUFFLFlBQStCO0lBQ25FLE9BQU8sY0FBYyxDQUFDLElBQUksRUFBRSxLQUFLLEVBQUUsWUFBWSxDQUFDLENBQUM7QUFDckQsQ0FBQztBQUZELDRCQUVDO0FBRUQsU0FBUyxjQUFjLENBQ25CLFFBQWlCLEVBQ2pCLEtBQVUsRUFDVixZQUErQjtJQUUvQixJQUFJLEtBQUssQ0FBQyxNQUFNLElBQUksQ0FBQyxFQUFFO1FBQ25CLE1BQU0sSUFBSSxLQUFLLENBQ1gsYUFBYSxRQUFRLENBQUMsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsTUFBTSxvQkFBb0IsQ0FDL0QsQ0FBQztLQUNMO0lBRUQsTUFBTSxLQUFLLEdBQUcsUUFBUSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDO0lBRTlDLElBQUksWUFBWSxFQUFFO1FBQ2QsT0FBTyxLQUFLLENBQUMsSUFBSSxDQUFDLFlBQVksQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDO0tBQzFDO0lBRUQsT0FBTyxLQUFLLENBQUMsS0FBSyxDQUFDLENBQUM7QUFDeEIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQ2hlY2sgaWYgYXJyYXlzIGhhdmUgb25seSBlbGVtZW50cyB0aGF0IGFsc28gZXhpc3RzIGFsc28gaW4gb3RoZXIgYXJyYXkuXG4gKiBJbmRpdmlkdWFsIGVsZW1lbnQgY291bnQgZG9lc24ndCBtYXR0ZXIuXG4gKiBGdW5jdGlvbiB3b3JrcyBvbmx5IGZvciBwcmltaXRpdmUgdHlwZXMgYW5kIGZvciBvdGhlciBpdCBqdXN0IGNoZWNrcyB0aGUgcmVmZXJlbmNlIHRvIG9iamVjdC5cbiAqXG4gKiBTb21lIGV4YW1wbGVzXG4gKiBib3RoQXJyYXlzSGFzU2FtZVZhbHVlcyggW2EsIGJdLCBbYiwgYV0gKSAgICA9PiB0cnVlXG4gKiBib3RoQXJyYXlzSGFzU2FtZVZhbHVlcyggW2EsIGFdLCBbYSwgYSwgYV0gKSA9PiB0cnVlXG4gKiBib3RoQXJyYXlzSGFzU2FtZVZhbHVlcyggW2EsIGJdLCBbYV0gKSAgICAgICA9PiBmYWxzZVxuICpcbiAqIE9iamVjdCByZWZlcmVuY2VzOlxuICogY29uc3QgbzEgPSB7IGE6IDEsIGI6IDJ9O1xuICogY29uc3QgbzIgPSB7IGE6IDEsIGI6IDJ9O1xuICogLy8gQXJyYXlzIGhhcyByZWZlcmVuY2VzIHRvIHNhbWUgb2JqZWN0c1xuICogYm90aEFycmF5c0hhc1NhbWVWYWx1ZXMoW28xXSwgW28xXSkpICAgICAgICAgPT4gdHJ1ZVxuICogQXJyYXlzIGhhdmUgcmVmZXJlbmNlcyB0byBkaWZmZXJlbnQgb2JqZWN0c1xuICogYm90aEFycmF5c0hhc1NhbWVWYWx1ZXMoW28xXSwgW28yXSkpICAgICAgICAgPT4gZmFsc2VcbiAqXG4gKiBAcGFyYW0gYSBmaXJzdCBhcnJheSB0byBjb21wYXJlXG4gKiBAcGFyYW0gYiBzZWNvbmQgYXJyYXkgdG8gY29tcGFyZVxuICovXG5leHBvcnQgZnVuY3Rpb24gYm90aEFycmF5c0hhc1NhbWVWYWx1ZXMoXG4gICAgYTogbnVsbCB8IHVuZGVmaW5lZCB8IHVua25vd25bXSxcbiAgICBiOiBudWxsIHwgdW5kZWZpbmVkIHwgdW5rbm93bltdXG4pOiBib29sZWFuIHtcbiAgICBpZiAoKGEgJiYgIWIpIHx8ICghYSAmJiBiKSkge1xuICAgICAgICByZXR1cm4gZmFsc2U7XG4gICAgfSBlbHNlIGlmICghYSAmJiAhYikge1xuICAgICAgICByZXR1cm4gdHJ1ZTtcbiAgICB9XG4gICAgY29uc3QgYVNldCA9IG5ldyBTZXQoYSk7XG4gICAgY29uc3QgYlNldCA9IG5ldyBTZXQoYik7XG4gICAgaWYgKGFTZXQuc2l6ZSAhPT0gYlNldC5zaXplKSB7XG4gICAgICAgIHJldHVybiBmYWxzZTtcbiAgICB9XG4gICAgcmV0dXJuIEFycmF5LmZyb20oYVNldCkuZXZlcnkoKHZhbHVlKSA9PiBiU2V0Lmhhcyh2YWx1ZSkpO1xufVxuXG4vKipcbiAqIFJldHVybnMgdGhlIGxhc3QgaXRlbSBvbiB0aGUgYXJyYXkuICBJZiB0aGUgYXJyYXkgaXMgZW1wdHksIHRocm93cyBhbiBlcnJvciFcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGdldExhc3Q8VD4oYXJyYXk6IFRbXSwgc29ydEZ1bmN0aW9uPzogKGE6IFQpID0+IG51bWJlcik6IFQge1xuICAgIHJldHVybiBnZXRGaXJzdE9yTGFzdChmYWxzZSwgYXJyYXksIHNvcnRGdW5jdGlvbik7XG59XG5cbi8qKlxuICogUmV0dXJucyB0aGUgZmlyc3QgaXRlbSBvbiB0aGUgYXJyYXkuICBJZiB0aGUgYXJyYXkgaXMgZW1wdHksIHRocm93cyBhbiBlcnJvciFcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGdldEZpcnN0PFQ+KGFycmF5OiBUW10sIHNvcnRGdW5jdGlvbj86IChhOiBUKSA9PiBudW1iZXIpOiBUIHtcbiAgICByZXR1cm4gZ2V0Rmlyc3RPckxhc3QodHJ1ZSwgYXJyYXksIHNvcnRGdW5jdGlvbik7XG59XG5cbmZ1bmN0aW9uIGdldEZpcnN0T3JMYXN0PFQ+KFxuICAgIGdldEZpcnN0OiBib29sZWFuLFxuICAgIGFycmF5OiBUW10sXG4gICAgc29ydEZ1bmN0aW9uPzogKGE6IFQpID0+IG51bWJlclxuKTogVCB7XG4gICAgaWYgKGFycmF5Lmxlbmd0aCA9PSAwKSB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgICAgIGBjYW4ndCBnZXQgJHtnZXRGaXJzdCA/IFwiZmlyc3RcIiA6IFwibGFzdFwifSBmcm9tIGVtcHR5IGFycmF5IWBcbiAgICAgICAgKTtcbiAgICB9XG5cbiAgICBjb25zdCBpbmRleCA9IGdldEZpcnN0ID8gMCA6IGFycmF5Lmxlbmd0aCAtIDE7XG5cbiAgICBpZiAoc29ydEZ1bmN0aW9uKSB7XG4gICAgICAgIHJldHVybiBhcnJheS5zb3J0KHNvcnRGdW5jdGlvbilbaW5kZXhdO1xuICAgIH1cblxuICAgIHJldHVybiBhcnJheVtpbmRleF07XG59XG4iXX0=
package/aws/index.d.ts DELETED
@@ -1 +0,0 @@
1
- export * from "./infra/stack/stack";
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
@@ -1 +0,0 @@
1
- export * from "./integration";
@@ -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.d.ts DELETED
@@ -1 +0,0 @@
1
- export * from "./language";
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==