@digitraffic/common 2022.10.6-1 → 2022.10.10-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.
Files changed (95) hide show
  1. package/README.md +54 -0
  2. package/aws/index.d.ts +1 -0
  3. package/aws/index.js +18 -0
  4. package/aws/infra/api/index.d.ts +1 -0
  5. package/aws/infra/api/index.js +18 -0
  6. package/aws/infra/api/responses.js +1 -1
  7. package/aws/infra/stack/lambda-configs.js +1 -1
  8. package/aws/infra/stack/rest_apis.d.ts +1 -0
  9. package/aws/infra/stack/rest_apis.js +8 -2
  10. package/aws/infra/stack/stack-checking-aspect.d.ts +4 -3
  11. package/aws/infra/stack/stack-checking-aspect.js +8 -7
  12. package/aws/infra/stack/stack.d.ts +4 -1
  13. package/aws/infra/stack/stack.js +1 -1
  14. package/aws/infra/usage-plans.js +1 -1
  15. package/aws/types/errors.js +1 -1
  16. package/aws/types/tags.js +1 -1
  17. package/database/last-updated.js +10 -10
  18. package/package.json +3 -2
  19. package/test/httpserver.js +1 -1
  20. package/types/index.d.ts +1 -0
  21. package/types/index.js +18 -0
  22. package/types/language.js +1 -1
  23. package/utils/geometry.js +1 -1
  24. package/src/aws/infra/api/integration.js +0 -52
  25. package/src/aws/infra/api/response.js +0 -61
  26. package/src/aws/infra/api/responses.js +0 -79
  27. package/src/aws/infra/api/static-integration.js +0 -54
  28. package/src/aws/infra/canaries/canary-alarm.js +0 -26
  29. package/src/aws/infra/canaries/canary-parameters.js +0 -3
  30. package/src/aws/infra/canaries/canary-role.js +0 -46
  31. package/src/aws/infra/canaries/canary.js +0 -29
  32. package/src/aws/infra/canaries/database-canary.js +0 -55
  33. package/src/aws/infra/canaries/database-checker.js +0 -109
  34. package/src/aws/infra/canaries/url-canary.js +0 -46
  35. package/src/aws/infra/canaries/url-checker.js +0 -238
  36. package/src/aws/infra/documentation.js +0 -95
  37. package/src/aws/infra/scheduler.js +0 -31
  38. package/src/aws/infra/security-rule.js +0 -39
  39. package/src/aws/infra/sqs-integration.js +0 -93
  40. package/src/aws/infra/sqs-queue.js +0 -130
  41. package/src/aws/infra/stack/lambda-configs.js +0 -93
  42. package/src/aws/infra/stack/monitoredfunction.js +0 -135
  43. package/src/aws/infra/stack/rest_apis.js +0 -179
  44. package/src/aws/infra/stack/stack-checking-aspect.js +0 -163
  45. package/src/aws/infra/stack/stack.js +0 -58
  46. package/src/aws/infra/stack/subscription.js +0 -41
  47. package/src/aws/infra/usage-plans.js +0 -42
  48. package/src/aws/runtime/apikey.js +0 -13
  49. package/src/aws/runtime/digitraffic-integration-response.js +0 -26
  50. package/src/aws/runtime/messaging.js +0 -31
  51. package/src/aws/runtime/s3.js +0 -30
  52. package/src/aws/runtime/secrets/dbsecret.js +0 -96
  53. package/src/aws/runtime/secrets/proxy-holder.js +0 -26
  54. package/src/aws/runtime/secrets/rds-holder.js +0 -26
  55. package/src/aws/runtime/secrets/secret-holder.js +0 -73
  56. package/src/aws/runtime/secrets/secret.js +0 -43
  57. package/src/aws/types/errors.js +0 -9
  58. package/src/aws/types/lambda-response.js +0 -28
  59. package/src/aws/types/mediatypes.js +0 -15
  60. package/src/aws/types/model-with-reference.js +0 -3
  61. package/src/aws/types/proxytypes.js +0 -3
  62. package/src/aws/types/tags.js +0 -7
  63. package/src/database/cached.js +0 -32
  64. package/src/database/database.js +0 -62
  65. package/src/database/last-updated.js +0 -54
  66. package/src/marine/id_utils.js +0 -33
  67. package/src/marine/rtz.js +0 -3
  68. package/src/test/asserter.js +0 -45
  69. package/src/test/db-testutils.js +0 -31
  70. package/src/test/httpserver.js +0 -67
  71. package/src/test/secret.js +0 -25
  72. package/src/test/secrets-manager.js +0 -59
  73. package/src/test/testutils.js +0 -44
  74. package/src/types/input-error.js +0 -7
  75. package/src/types/language.js +0 -10
  76. package/src/types/traffictype.js +0 -13
  77. package/src/types/validator.js +0 -14
  78. package/src/utils/api-model.js +0 -129
  79. package/src/utils/base64.js +0 -21
  80. package/src/utils/date-utils.js +0 -34
  81. package/src/utils/geojson-types.js +0 -18
  82. package/src/utils/geometry.js +0 -140
  83. package/src/utils/retry.js +0 -50
  84. package/src/utils/slack.js +0 -25
  85. package/src/utils/utils.js +0 -40
  86. package/test/marine/id_utils.test.js +0 -69
  87. package/test/promise/promise.test.js +0 -125
  88. package/test/secrets/dbsecret.test.js +0 -71
  89. package/test/secrets/secret-holder.test.js +0 -124
  90. package/test/secrets/secret.test.js +0 -66
  91. package/test/test/httpserver.test.js +0 -87
  92. package/test/utils/date-utils.test.js +0 -51
  93. package/test/utils/geometry.test.js +0 -49
  94. package/test/utils/utils.test.js +0 -49
  95. package/yarn.lock +0 -3200
@@ -1,135 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.MonitoredDBFunction = exports.MonitoredFunction = void 0;
4
- const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
5
- const aws_cloudwatch_actions_1 = require("aws-cdk-lib/aws-cloudwatch-actions");
6
- const aws_cloudwatch_1 = require("aws-cdk-lib/aws-cloudwatch");
7
- const lambda_configs_1 = require("../stack/lambda-configs");
8
- const change_case_1 = require("change-case");
9
- const subscription_1 = require("../stack/subscription");
10
- /**
11
- * Creates a Lambda function that monitors default CloudWatch Lambda metrics with CloudWatch Alarms.
12
- */
13
- class MonitoredFunction extends aws_lambda_1.Function {
14
- /**
15
- * @param scope Stack
16
- * @param id Lambda construct Id
17
- * @param functionProps Lambda function properties
18
- * @param alarmSnsTopic SNS topic for alarms
19
- * @param warningSnsTopic SNS topic for warnings
20
- * @param production Is the stack a production stack, used for determining the alarm topic
21
- * @param trafficType Traffic type, used for alarm names. Set to null if Lambda is not related to any traffic type.
22
- * @param props Monitored function properties
23
- */
24
- constructor(scope, id, functionProps, alarmSnsTopic, warningSnsTopic, production, trafficType, props) {
25
- super(scope, id, functionProps);
26
- this.givenName = functionProps.functionName;
27
- const alarmSnsAction = new aws_cloudwatch_actions_1.SnsAction(alarmSnsTopic);
28
- const warningSnsAction = new aws_cloudwatch_actions_1.SnsAction(warningSnsTopic);
29
- if (props?.durationAlarmProps?.create !== false) {
30
- if (!functionProps.timeout) {
31
- throw new Error('Timeout needs to be explicitly set');
32
- }
33
- this.createAlarm(scope, this.metricDuration().with({ statistic: 'max' }), 'Duration', 'Duration alarm', `Duration has exceeded ${functionProps.timeout.toSeconds()} seconds`, trafficType, this.getAlarmActionForEnv(alarmSnsAction, warningSnsAction, production), functionProps.timeout.toMilliseconds(), 1, 1, aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD, props?.durationAlarmProps);
34
- }
35
- if (props?.durationWarningProps?.create !== false) {
36
- if (!functionProps.timeout) {
37
- throw new Error('Timeout needs to be explicitly set');
38
- }
39
- this.createAlarm(scope, this.metricDuration().with({ statistic: 'max' }), 'Duration-Warning', 'Duration warning', `Duration is 85 % of max ${functionProps.timeout.toSeconds()} seconds`, trafficType, warningSnsAction, functionProps.timeout.toMilliseconds() * 0.85, 1, 1, aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD, props?.durationWarningProps);
40
- }
41
- if (props?.errorAlarmProps?.create !== false) {
42
- this.createAlarm(scope, this.metricErrors(), 'Errors', 'Errors alarm', 'Invocations did not succeed', trafficType, this.getAlarmActionForEnv(alarmSnsAction, warningSnsAction, production), 1, 1, 1, aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD, props?.errorAlarmProps);
43
- }
44
- if (props?.throttleAlarmProps?.create !== false) {
45
- this.createAlarm(scope, this.metricThrottles(), 'Throttles', 'Throttles alarm', 'Has throttled', trafficType, this.getAlarmActionForEnv(alarmSnsAction, warningSnsAction, production), 0, 1, 1, aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_THRESHOLD, props?.throttleAlarmProps);
46
- }
47
- }
48
- /**
49
- * Create new MonitoredFunction. Use topics from given DigitrafficStack.
50
- *
51
- * @param stack DigitrafficStack
52
- * @param id Lambda construct Id
53
- * @param functionProps Lambda function properties
54
- * @param props Monitored function properties
55
- */
56
- static create(stack, id, functionProps, props) {
57
- if (props === MonitoredFunction.DISABLE_ALARMS && stack.configuration.production) {
58
- throw new Error(`Function ${functionProps.functionName} has DISABLE_ALARMS. Remove before installing to production or define your own properties!`);
59
- }
60
- return new MonitoredFunction(stack, id, functionProps, stack.alarmTopic, stack.warningTopic, stack.configuration.production, stack.configuration.trafficType, props);
61
- }
62
- /**
63
- * Create new MonitoredFunction. Use topics from given DigitrafficStack. Generate names from given name and configuration shortName.
64
- *
65
- * For example, shortName FOO and given name update-things will create function FOO-UpdateThings and use code from lambda/update-things/update-things.ts method handler.
66
- *
67
- * @param stack DigitrafficStack
68
- * @param name param-case name
69
- * @param environment Lambda environment
70
- * @param functionParameters Lambda function parameters
71
- */
72
- static createV2(stack, name, environment, functionParameters) {
73
- const functionName = functionParameters?.functionName || `${stack.configuration.shortName}-${(0, change_case_1.pascalCase)(name)}`;
74
- const functionProps = (0, lambda_configs_1.databaseFunctionProps)(stack, environment, functionName, name, functionParameters);
75
- return MonitoredFunction.create(stack, functionName, functionProps, functionParameters);
76
- }
77
- createAlarm(stack, metric, alarmId, alarmName, alarmDescription, trafficType, alarmSnsAction, threshold, evaluationPeriods, datapointsToAlarm, comparisonOperator, alarmProps) {
78
- metric.createAlarm(stack, `${this.node.id}-${alarmId}`, {
79
- alarmName: `${trafficType ?? ''} ${stack.stackName} ${this.functionName} ${alarmName}`.trim(),
80
- alarmDescription,
81
- threshold: alarmProps?.threshold ?? threshold,
82
- evaluationPeriods: alarmProps?.evaluationPeriods ?? evaluationPeriods,
83
- datapointsToAlarm: alarmProps?.datapointsToAlarm ?? datapointsToAlarm,
84
- comparisonOperator: alarmProps?.comparisonOperator ?? comparisonOperator,
85
- }).addAlarmAction(alarmSnsAction);
86
- }
87
- getAlarmActionForEnv(alarmAction, warningAction, production) {
88
- return production ? alarmAction : warningAction;
89
- }
90
- }
91
- exports.MonitoredFunction = MonitoredFunction;
92
- /** disable all alarms */
93
- MonitoredFunction.DISABLE_ALARMS = {
94
- durationAlarmProps: {
95
- create: false,
96
- },
97
- durationWarningProps: {
98
- create: false,
99
- },
100
- errorAlarmProps: {
101
- create: false,
102
- },
103
- throttleAlarmProps: {
104
- create: false,
105
- },
106
- };
107
- class MonitoredDBFunction {
108
- /**
109
- * Create new MonitoredDBFunction. Use topics from given DigitrafficStack. Generate names from given name and configuration shortName.
110
- * Grant secret and create log subscription.
111
- *
112
- * For example, shortName FOO and given name update-things will create function FOO-UpdateThings and use code from lambda/update-things/update-things.ts method handler.
113
- *
114
- * If you don't need to pass any extra arguments to lambda-environment, you can leave environment out and this function will create the
115
- * default Lambda Environment with SECRET_ID and DB_APPLICATION.
116
- *
117
- * @param stack DigitrafficStack
118
- * @param name param-case name
119
- * @param environment Lambda environment
120
- * @param functionParameters Lambda function parameters
121
- */
122
- static create(stack, name, environment, functionParameters) {
123
- const functionName = functionParameters?.functionName || `${stack.configuration.shortName}-${(0, change_case_1.pascalCase)(name)}`;
124
- const env = environment ? environment : stack.createLambdaEnvironment();
125
- const functionProps = (0, lambda_configs_1.databaseFunctionProps)(stack, env, functionName, name, functionParameters);
126
- const mf = MonitoredFunction.create(stack, functionName, functionProps, functionParameters);
127
- stack.grantSecret(mf);
128
- if (stack.configuration.logsDestinationArn) {
129
- new subscription_1.DigitrafficLogSubscriptions(stack, mf);
130
- }
131
- return mf;
132
- }
133
- }
134
- exports.MonitoredDBFunction = MonitoredDBFunction;
135
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"monitoredfunction.js","sourceRoot":"","sources":["../../../../../src/aws/infra/stack/monitoredfunction.ts"],"names":[],"mappings":";;;AAAA,uDAA+D;AAE/D,+EAA6D;AAC7D,+DAAsE;AAGtE,4DAA8G;AAC9G,6CAAuC;AACvC,wDAAkE;AA+BlE;;GAEG;AACH,MAAa,iBAAkB,SAAQ,qBAAQ;IAsE3C;;;;;;;;;OASG;IACH,YACI,KAAY,EACZ,EAAU,EACV,aAA4B,EAC5B,aAAqB,EACrB,eAAuB,EACvB,UAAmB,EACnB,WAA+B,EAC/B,KAA8B;QAE9B,KAAK,CAAC,KAAK,EAAE,EAAE,EAAE,aAAa,CAAC,CAAC;QAEhC,IAAI,CAAC,SAAS,GAAG,aAAa,CAAC,YAAsB,CAAC;QAEtD,MAAM,cAAc,GAAG,IAAI,kCAAS,CAAC,aAAa,CAAC,CAAC;QACpD,MAAM,gBAAgB,GAAG,IAAI,kCAAS,CAAC,eAAe,CAAC,CAAC;QAExD,IAAI,KAAK,EAAE,kBAAkB,EAAE,MAAM,KAAK,KAAK,EAAE;YAC7C,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE;gBACxB,MAAM,IAAI,KAAK,CAAC,oCAAoC,CAAC,CAAC;aACzD;YAED,IAAI,CAAC,WAAW,CACZ,KAAK,EACL,IAAI,CAAC,cAAc,EAAE,CAAC,IAAI,CAAC,EAAC,SAAS,EAAE,KAAK,EAAC,CAAC,EAC9C,UAAU,EACV,gBAAgB,EAChB,yBAAyB,aAAa,CAAC,OAAO,CAAC,SAAS,EAAE,UAAU,EACpE,WAAW,EACX,IAAI,CAAC,oBAAoB,CAAC,cAAc,EAAE,gBAAgB,EAAE,UAAU,CAAC,EACvE,aAAa,CAAC,OAAO,CAAC,cAAc,EAAE,EACtC,CAAC,EACD,CAAC,EACD,mCAAkB,CAAC,kCAAkC,EACrD,KAAK,EAAE,kBAAkB,CAC5B,CAAC;SACL;QACD,IAAI,KAAK,EAAE,oBAAoB,EAAE,MAAM,KAAK,KAAK,EAAE;YAC/C,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE;gBACxB,MAAM,IAAI,KAAK,CAAC,oCAAoC,CAAC,CAAC;aACzD;YAED,IAAI,CAAC,WAAW,CACZ,KAAK,EACL,IAAI,CAAC,cAAc,EAAE,CAAC,IAAI,CAAC,EAAC,SAAS,EAAE,KAAK,EAAC,CAAC,EAC9C,kBAAkB,EAClB,kBAAkB,EAClB,2BAA2B,aAAa,CAAC,OAAO,CAAC,SAAS,EAAE,UAAU,EACtE,WAAW,EACX,gBAAgB,EAChB,aAAa,CAAC,OAAO,CAAC,cAAc,EAAE,GAAG,IAAI,EAC7C,CAAC,EACD,CAAC,EACD,mCAAkB,CAAC,kCAAkC,EACrD,KAAK,EAAE,oBAAoB,CAC9B,CAAC;SACL;QAED,IAAI,KAAK,EAAE,eAAe,EAAE,MAAM,KAAK,KAAK,EAAE;YAC1C,IAAI,CAAC,WAAW,CACZ,KAAK,EACL,IAAI,CAAC,YAAY,EAAE,EACnB,QAAQ,EACR,cAAc,EACd,6BAA6B,EAC7B,WAAW,EACX,IAAI,CAAC,oBAAoB,CAAC,cAAc,EAAE,gBAAgB,EAAE,UAAU,CAAC,EACvE,CAAC,EACD,CAAC,EACD,CAAC,EACD,mCAAkB,CAAC,kCAAkC,EACrD,KAAK,EAAE,eAAe,CACzB,CAAC;SACL;QAED,IAAI,KAAK,EAAE,kBAAkB,EAAE,MAAM,KAAK,KAAK,EAAE;YAC7C,IAAI,CAAC,WAAW,CACZ,KAAK,EACL,IAAI,CAAC,eAAe,EAAE,EACtB,WAAW,EACX,iBAAiB,EACjB,eAAe,EACf,WAAW,EACX,IAAI,CAAC,oBAAoB,CAAC,cAAc,EAAE,gBAAgB,EAAE,UAAU,CAAC,EACvE,CAAC,EACD,CAAC,EACD,CAAC,EACD,mCAAkB,CAAC,sBAAsB,EACzC,KAAK,EAAE,kBAAkB,CAC5B,CAAC;SACL;IACL,CAAC;IAxJD;;;;;;;OAOG;IACH,MAAM,CAAC,MAAM,CAAC,KAAuB,EACjC,EAAU,EACV,aAA4B,EAC5B,KAA8B;QAE9B,IAAI,KAAK,KAAK,iBAAiB,CAAC,cAAc,IAAI,KAAK,CAAC,aAAa,CAAC,UAAU,EAAE;YAC9E,MAAM,IAAI,KAAK,CAAC,YAAY,aAAa,CAAC,YAAY,6FAA6F,CAAC,CAAC;SACxJ;QAED,OAAO,IAAI,iBAAiB,CACxB,KAAK,EACL,EAAE,EACF,aAAa,EACb,KAAK,CAAC,UAAU,EAChB,KAAK,CAAC,YAAY,EAClB,KAAK,CAAC,aAAa,CAAC,UAAU,EAC9B,KAAK,CAAC,aAAa,CAAC,WAAW,EAC/B,KAAK,CACR,CAAC;IACN,CAAC;IAED;;;;;;;;;OASG;IACH,MAAM,CAAC,QAAQ,CAAC,KAAuB,EACnC,IAAY,EACZ,WAA8B,EAC9B,kBAAgD;QAChD,MAAM,YAAY,GAAG,kBAAkB,EAAE,YAAY,IAAI,GAAG,KAAK,CAAC,aAAa,CAAC,SAAS,IAAI,IAAA,wBAAU,EAAC,IAAI,CAAC,EAAE,CAAC;QAChH,MAAM,aAAa,GAAG,IAAA,sCAAqB,EACvC,KAAK,EAAE,WAAW,EAAE,YAAY,EAAE,IAAI,EAAE,kBAAkB,CAC7D,CAAC;QAEF,OAAO,iBAAiB,CAAC,MAAM,CAAC,KAAK,EAAE,YAAY,EAAE,aAAa,EAAE,kBAAkB,CAAC,CAAC;IAC5F,CAAC;IAyGO,WAAW,CACf,KAAY,EACZ,MAAc,EACd,OAAe,EACf,SAAiB,EACjB,gBAAwB,EACxB,WAA+B,EAC/B,cAAyB,EACzB,SAAiB,EACjB,iBAAyB,EACzB,iBAAyB,EACzB,kBAAsC,EACtC,UAAwC;QAExC,MAAM,CAAC,WAAW,CAAC,KAAK,EAAE,GAAG,IAAI,CAAC,IAAI,CAAC,EAAE,IAAI,OAAO,EAAE,EAAE;YACpD,SAAS,EAAE,GAAG,WAAW,IAAI,EAAE,IAAI,KAAK,CAAC,SAAS,IAAI,IAAI,CAAC,YAAY,IAAI,SAAS,EAAE,CAAC,IAAI,EAAE;YAC7F,gBAAgB;YAChB,SAAS,EAAE,UAAU,EAAE,SAAS,IAAI,SAAS;YAC7C,iBAAiB,EAAE,UAAU,EAAE,iBAAiB,IAAI,iBAAiB;YACrE,iBAAiB,EAAE,UAAU,EAAE,iBAAiB,IAAI,iBAAiB;YACrE,kBAAkB,EAAE,UAAU,EAAE,kBAAkB,IAAI,kBAAkB;SAC3E,CAAC,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;IACtC,CAAC;IAEO,oBAAoB,CAAC,WAAsB,EAC/C,aAAwB,EACxB,UAAmB;QAEnB,OAAO,UAAU,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,aAAa,CAAC;IACpD,CAAC;;AA1ML,8CA2MC;AAxMG,yBAAyB;AACF,gCAAc,GAA2B;IAC5D,kBAAkB,EAAE;QAChB,MAAM,EAAE,KAAK;KAChB;IACD,oBAAoB,EAAE;QAClB,MAAM,EAAE,KAAK;KAChB;IACD,eAAe,EAAE;QACb,MAAM,EAAE,KAAK;KAChB;IACD,kBAAkB,EAAE;QAChB,MAAM,EAAE,KAAK;KAChB;CACJ,CAAC;AA4LN,MAAa,mBAAmB;IAC5B;;;;;;;;;;;;;OAaG;IACH,MAAM,CAAC,MAAM,CAAC,KAAuB,EACjC,IAAY,EACZ,WAA+B,EAC/B,kBAAgD;QAChD,MAAM,YAAY,GAAG,kBAAkB,EAAE,YAAY,IAAI,GAAG,KAAK,CAAC,aAAa,CAAC,SAAS,IAAI,IAAA,wBAAU,EAAC,IAAI,CAAC,EAAE,CAAC;QAChH,MAAM,GAAG,GAAG,WAAW,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,KAAK,CAAC,uBAAuB,EAAE,CAAC;QACxE,MAAM,aAAa,GAAG,IAAA,sCAAqB,EACvC,KAAK,EAAE,GAAG,EAAE,YAAY,EAAE,IAAI,EAAE,kBAAkB,CACrD,CAAC;QAEF,MAAM,EAAE,GAAG,iBAAiB,CAAC,MAAM,CAAC,KAAK,EAAE,YAAY,EAAE,aAAa,EAAE,kBAAkB,CAAC,CAAC;QAE5F,KAAK,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;QAEtB,IAAI,KAAK,CAAC,aAAa,CAAC,kBAAkB,EAAE;YACxC,IAAI,0CAA2B,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;SAC9C;QAED,OAAO,EAAE,CAAC;IACd,CAAC;CACJ;AAnCD,kDAmCC","sourcesContent":["import {Function, FunctionProps} from 'aws-cdk-lib/aws-lambda';\nimport {Stack} from \"aws-cdk-lib\";\nimport {SnsAction} from \"aws-cdk-lib/aws-cloudwatch-actions\";\nimport {ComparisonOperator, Metric} from \"aws-cdk-lib/aws-cloudwatch\";\nimport {DigitrafficStack} from \"../stack/stack\";\nimport {ITopic} from \"aws-cdk-lib/aws-sns\";\nimport {databaseFunctionProps, LambdaEnvironment, MonitoredFunctionParameters} from \"../stack/lambda-configs\";\nimport {pascalCase} from \"change-case\";\nimport {DigitrafficLogSubscriptions} from \"../stack/subscription\";\nimport {TrafficType} from \"../../../types/traffictype\";\n\n/**\n * Allows customization of CloudWatch Alarm properties\n */\nexport type MonitoredFunctionAlarmProps = {\n    /**\n     * Setting this to false will not create a CloudWatch alarm\n     */\n    readonly create: boolean\n\n    readonly threshold?: number\n\n    readonly evaluationPeriods?: number\n\n    readonly datapointsToAlarm?: number\n\n    readonly comparisonOperator?: ComparisonOperator\n}\n\nexport type MonitoredFunctionProps = {\n    readonly durationAlarmProps?: MonitoredFunctionAlarmProps\n\n    readonly durationWarningProps?: MonitoredFunctionAlarmProps\n\n    readonly errorAlarmProps?: MonitoredFunctionAlarmProps\n\n    readonly throttleAlarmProps?: MonitoredFunctionAlarmProps\n}\n\n/**\n * Creates a Lambda function that monitors default CloudWatch Lambda metrics with CloudWatch Alarms.\n */\nexport class MonitoredFunction extends Function {\n    readonly givenName: string;\n\n    /** disable all alarms */\n    public static readonly DISABLE_ALARMS: MonitoredFunctionProps = {\n        durationAlarmProps: {\n            create: false,\n        },\n        durationWarningProps: {\n            create: false,\n        },\n        errorAlarmProps: {\n            create: false,\n        },\n        throttleAlarmProps: {\n            create: false,\n        },\n    };\n\n    /**\n     * Create new MonitoredFunction.  Use topics from given DigitrafficStack.\n     *\n     * @param stack DigitrafficStack\n     * @param id Lambda construct Id\n     * @param functionProps Lambda function properties\n     * @param props Monitored function properties\n     */\n    static create(stack: DigitrafficStack,\n        id: string,\n        functionProps: FunctionProps,\n        props?: MonitoredFunctionProps): MonitoredFunction {\n\n        if (props === MonitoredFunction.DISABLE_ALARMS && stack.configuration.production) {\n            throw new Error(`Function ${functionProps.functionName} has DISABLE_ALARMS.  Remove before installing to production or define your own properties!`);\n        }\n\n        return new MonitoredFunction(\n            stack,\n            id,\n            functionProps,\n            stack.alarmTopic,\n            stack.warningTopic,\n            stack.configuration.production,\n            stack.configuration.trafficType,\n            props,\n        );\n    }\n\n    /**\n     * Create new MonitoredFunction.  Use topics from given DigitrafficStack.  Generate names from given name and configuration shortName.\n     *\n     * For example, shortName FOO and given name update-things will create function FOO-UpdateThings and use code from lambda/update-things/update-things.ts method handler.\n     *\n     * @param stack DigitrafficStack\n     * @param name param-case name\n     * @param environment Lambda environment\n     * @param functionParameters Lambda function parameters\n     */\n    static createV2(stack: DigitrafficStack,\n        name: string,\n        environment: LambdaEnvironment,\n        functionParameters?: MonitoredFunctionParameters): MonitoredFunction {\n        const functionName = functionParameters?.functionName || `${stack.configuration.shortName}-${pascalCase(name)}`;\n        const functionProps = databaseFunctionProps(\n            stack, environment, functionName, name, functionParameters,\n        );\n\n        return MonitoredFunction.create(stack, functionName, functionProps, functionParameters);\n    }\n\n    /**\n     * @param scope Stack\n     * @param id Lambda construct Id\n     * @param functionProps Lambda function properties\n     * @param alarmSnsTopic SNS topic for alarms\n     * @param warningSnsTopic SNS topic for warnings\n     * @param production Is the stack a production stack, used for determining the alarm topic\n     * @param trafficType Traffic type, used for alarm names. Set to null if Lambda is not related to any traffic type.\n     * @param props Monitored function properties\n     */\n    constructor(\n        scope: Stack,\n        id: string,\n        functionProps: FunctionProps,\n        alarmSnsTopic: ITopic,\n        warningSnsTopic: ITopic,\n        production: boolean,\n        trafficType: TrafficType | null,\n        props?: MonitoredFunctionProps,\n    ) {\n        super(scope, id, functionProps);\n\n        this.givenName = functionProps.functionName as string;\n\n        const alarmSnsAction = new SnsAction(alarmSnsTopic);\n        const warningSnsAction = new SnsAction(warningSnsTopic);\n\n        if (props?.durationAlarmProps?.create !== false) {\n            if (!functionProps.timeout) {\n                throw new Error('Timeout needs to be explicitly set');\n            }\n\n            this.createAlarm(\n                scope,\n                this.metricDuration().with({statistic: 'max'}),\n                'Duration',\n                'Duration alarm',\n                `Duration has exceeded ${functionProps.timeout.toSeconds()} seconds`,\n                trafficType,\n                this.getAlarmActionForEnv(alarmSnsAction, warningSnsAction, production),\n                functionProps.timeout.toMilliseconds(),\n                1,\n                1,\n                ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n                props?.durationAlarmProps,\n            );\n        }\n        if (props?.durationWarningProps?.create !== false) {\n            if (!functionProps.timeout) {\n                throw new Error('Timeout needs to be explicitly set');\n            }\n\n            this.createAlarm(\n                scope,\n                this.metricDuration().with({statistic: 'max'}),\n                'Duration-Warning',\n                'Duration warning',\n                `Duration is 85 % of max ${functionProps.timeout.toSeconds()} seconds`,\n                trafficType,\n                warningSnsAction,\n                functionProps.timeout.toMilliseconds() * 0.85,\n                1,\n                1,\n                ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n                props?.durationWarningProps,\n            );\n        }\n\n        if (props?.errorAlarmProps?.create !== false) {\n            this.createAlarm(\n                scope,\n                this.metricErrors(),\n                'Errors',\n                'Errors alarm',\n                'Invocations did not succeed',\n                trafficType,\n                this.getAlarmActionForEnv(alarmSnsAction, warningSnsAction, production),\n                1,\n                1,\n                1,\n                ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n                props?.errorAlarmProps,\n            );\n        }\n\n        if (props?.throttleAlarmProps?.create !== false) {\n            this.createAlarm(\n                scope,\n                this.metricThrottles(),\n                'Throttles',\n                'Throttles alarm',\n                'Has throttled',\n                trafficType,\n                this.getAlarmActionForEnv(alarmSnsAction, warningSnsAction, production),\n                0,\n                1,\n                1,\n                ComparisonOperator.GREATER_THAN_THRESHOLD,\n                props?.throttleAlarmProps,\n            );\n        }\n    }\n\n    private createAlarm(\n        stack: Stack,\n        metric: Metric,\n        alarmId: string,\n        alarmName: string,\n        alarmDescription: string,\n        trafficType: TrafficType | null,\n        alarmSnsAction: SnsAction,\n        threshold: number,\n        evaluationPeriods: number,\n        datapointsToAlarm: number,\n        comparisonOperator: ComparisonOperator,\n        alarmProps?: MonitoredFunctionAlarmProps,\n    ) {\n        metric.createAlarm(stack, `${this.node.id}-${alarmId}`, {\n            alarmName: `${trafficType ?? ''} ${stack.stackName} ${this.functionName} ${alarmName}`.trim(),\n            alarmDescription,\n            threshold: alarmProps?.threshold ?? threshold,\n            evaluationPeriods: alarmProps?.evaluationPeriods ?? evaluationPeriods,\n            datapointsToAlarm: alarmProps?.datapointsToAlarm ?? datapointsToAlarm,\n            comparisonOperator: alarmProps?.comparisonOperator ?? comparisonOperator,\n        }).addAlarmAction(alarmSnsAction);\n    }\n\n    private getAlarmActionForEnv(alarmAction: SnsAction,\n        warningAction: SnsAction,\n        production: boolean): SnsAction {\n\n        return production ? alarmAction : warningAction;\n    }\n}\n\nexport class MonitoredDBFunction {\n    /**\n     * Create new MonitoredDBFunction.  Use topics from given DigitrafficStack.  Generate names from given name and configuration shortName.\n     * Grant secret and create log subscription.\n     *\n     * For example, shortName FOO and given name update-things will create function FOO-UpdateThings and use code from lambda/update-things/update-things.ts method handler.\n     *\n     * If you don't need to pass any extra arguments to lambda-environment, you can leave environment out and this function will create the\n     * default Lambda Environment with SECRET_ID and DB_APPLICATION.\n     *\n     * @param stack DigitrafficStack\n     * @param name param-case name\n     * @param environment Lambda environment\n     * @param functionParameters Lambda function parameters\n     */\n    static create(stack: DigitrafficStack,\n        name: string,\n        environment?: LambdaEnvironment,\n        functionParameters?: MonitoredFunctionParameters): MonitoredFunction {\n        const functionName = functionParameters?.functionName || `${stack.configuration.shortName}-${pascalCase(name)}`;\n        const env = environment ? environment : stack.createLambdaEnvironment();\n        const functionProps = databaseFunctionProps(\n            stack, env, functionName, name, functionParameters,\n        );\n\n        const mf = MonitoredFunction.create(stack, functionName, functionProps, functionParameters);\n\n        stack.grantSecret(mf);\n\n        if (stack.configuration.logsDestinationArn) {\n            new DigitrafficLogSubscriptions(stack, mf);\n        }\n\n        return mf;\n    }\n}\n"]}
@@ -1,179 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.createIpRestrictionPolicyDocument = exports.createDefaultPolicyDocument = exports.createRestApi = exports.setReturnCodeForMissingAuthenticationToken = exports.add401Support = exports.add404Support = exports.DigitrafficRestApi = void 0;
4
- const aws_apigateway_1 = require("aws-cdk-lib/aws-apigateway");
5
- const aws_iam_1 = require("aws-cdk-lib/aws-iam");
6
- const usage_plans_1 = require("../usage-plans");
7
- const api_model_1 = require("../../../utils/api-model");
8
- const mediatypes_1 = require("../../types/mediatypes");
9
- const R = require("ramda");
10
- class DigitrafficRestApi extends aws_apigateway_1.RestApi {
11
- constructor(stack, apiId, apiName, allowFromIpAddresses, config) {
12
- const policyDocument = allowFromIpAddresses == null ? createDefaultPolicyDocument() : createIpRestrictionPolicyDocument(allowFromIpAddresses);
13
- // override default config with given extra config
14
- const apiConfig = { ...{
15
- deployOptions: {
16
- loggingLevel: aws_apigateway_1.MethodLoggingLevel.ERROR,
17
- },
18
- restApiName: apiName,
19
- endpointTypes: [aws_apigateway_1.EndpointType.REGIONAL],
20
- policy: policyDocument,
21
- }, ...config };
22
- super(stack, apiId, apiConfig);
23
- this.apiKeyIds = [];
24
- add404Support(this, stack);
25
- }
26
- hostname() {
27
- return `${this.restApiId}.execute-api.${this.stack.region}.amazonaws.com`;
28
- }
29
- createUsagePlan(apiKeyId, apiKeyName) {
30
- const newKeyId = (0, usage_plans_1.createUsagePlan)(this, apiKeyId, apiKeyName).keyId;
31
- this.apiKeyIds.push(newKeyId);
32
- return newKeyId;
33
- }
34
- createUsagePlanV2(apiName) {
35
- const newKeyId = (0, usage_plans_1.createDefaultUsagePlan)(this, apiName).keyId;
36
- this.apiKeyIds.push(newKeyId);
37
- return newKeyId;
38
- }
39
- addJsonModel(modelName, schema) {
40
- return this.getModelWithReference(this.addModel(modelName, {
41
- contentType: mediatypes_1.MediaType.APPLICATION_JSON,
42
- modelName,
43
- schema,
44
- }));
45
- }
46
- addCSVModel(modelName) {
47
- return this.getModelWithReference(this.addModel(modelName, {
48
- contentType: mediatypes_1.MediaType.TEXT_CSV,
49
- modelName,
50
- schema: {},
51
- }));
52
- }
53
- getModelWithReference(model) {
54
- return R.assoc('modelReference', (0, api_model_1.getModelReference)(model.modelId, this.restApiId), model);
55
- }
56
- addDocumentationPart(resource, parameterName, resourceName, type, properties) {
57
- const location = {
58
- type,
59
- path: resource.path,
60
- name: type !== 'METHOD' ? parameterName : undefined,
61
- };
62
- new aws_apigateway_1.CfnDocumentationPart(this.stack, resourceName, {
63
- restApiId: resource.api.restApiId,
64
- location,
65
- properties: JSON.stringify(properties),
66
- });
67
- }
68
- documentResource(resource, ...documentationPart) {
69
- documentationPart.forEach(dp => this.addDocumentationPart(resource, dp.parameterName, `${resource.path}.${dp.parameterName}.Documentation`, dp.type, dp.documentationProperties));
70
- }
71
- }
72
- exports.DigitrafficRestApi = DigitrafficRestApi;
73
- /**
74
- * Due to AWS API design API Gateway will always return 403 'Missing Authentication Token' for requests
75
- * with a non-existent endpoint. This function translates this response to a 404.
76
- * Requests with an invalid or missing API key are not affected (still return 403 'Forbidden').
77
- * @param restApi RestApi
78
- * @param stack Construct
79
- */
80
- function add404Support(restApi, stack) {
81
- new aws_apigateway_1.GatewayResponse(stack, `MissingAuthenticationTokenResponse-${restApi.restApiName}`, {
82
- restApi,
83
- type: aws_apigateway_1.ResponseType.MISSING_AUTHENTICATION_TOKEN,
84
- statusCode: '404',
85
- templates: {
86
- 'application/json': '{"message": "Not found"}',
87
- },
88
- });
89
- }
90
- exports.add404Support = add404Support;
91
- function add401Support(restApi, stack) {
92
- new aws_apigateway_1.GatewayResponse(stack, `AuthenticationFailedResponse-${restApi.restApiName}`, {
93
- restApi,
94
- type: aws_apigateway_1.ResponseType.UNAUTHORIZED,
95
- statusCode: "401",
96
- responseHeaders: {
97
- 'WWW-Authenticate': "'Basic'",
98
- },
99
- });
100
- }
101
- exports.add401Support = add401Support;
102
- /**
103
- * Due to AWS API design API Gateway will always return 403 'Missing Authentication Token' for requests
104
- * with a non-existent endpoint. This function converts this response to a custom one.
105
- * Requests with an invalid or missing API key are not affected (still return 403 'Forbidden').
106
- * @param returnCode
107
- * @param message
108
- * @param restApi RestApi
109
- * @param stack Construct
110
- */
111
- function setReturnCodeForMissingAuthenticationToken(returnCode, message, restApi, stack) {
112
- new aws_apigateway_1.GatewayResponse(stack, `MissingAuthenticationTokenResponse-${restApi.restApiName}`, {
113
- restApi,
114
- type: aws_apigateway_1.ResponseType.MISSING_AUTHENTICATION_TOKEN,
115
- statusCode: `${returnCode}`,
116
- templates: {
117
- 'application/json': `{"message": ${message}}`,
118
- },
119
- });
120
- }
121
- exports.setReturnCodeForMissingAuthenticationToken = setReturnCodeForMissingAuthenticationToken;
122
- function createRestApi(stack, apiId, apiName, allowFromIpAddresses) {
123
- const policyDocument = allowFromIpAddresses == null ? createDefaultPolicyDocument() : createIpRestrictionPolicyDocument(allowFromIpAddresses);
124
- const restApi = new aws_apigateway_1.RestApi(stack, apiId, {
125
- deployOptions: {
126
- loggingLevel: aws_apigateway_1.MethodLoggingLevel.ERROR,
127
- },
128
- restApiName: apiName,
129
- endpointTypes: [aws_apigateway_1.EndpointType.REGIONAL],
130
- policy: policyDocument,
131
- });
132
- add404Support(restApi, stack);
133
- return restApi;
134
- }
135
- exports.createRestApi = createRestApi;
136
- function createDefaultPolicyDocument() {
137
- return new aws_iam_1.PolicyDocument({
138
- statements: [
139
- new aws_iam_1.PolicyStatement({
140
- effect: aws_iam_1.Effect.ALLOW,
141
- actions: [
142
- "execute-api:Invoke",
143
- ],
144
- resources: [
145
- "*",
146
- ],
147
- principals: [
148
- new aws_iam_1.AnyPrincipal(),
149
- ],
150
- }),
151
- ],
152
- });
153
- }
154
- exports.createDefaultPolicyDocument = createDefaultPolicyDocument;
155
- function createIpRestrictionPolicyDocument(allowFromIpAddresses) {
156
- return new aws_iam_1.PolicyDocument({
157
- statements: [
158
- new aws_iam_1.PolicyStatement({
159
- effect: aws_iam_1.Effect.ALLOW,
160
- conditions: {
161
- "IpAddress": {
162
- "aws:SourceIp": allowFromIpAddresses,
163
- },
164
- },
165
- actions: [
166
- "execute-api:Invoke",
167
- ],
168
- resources: [
169
- "*",
170
- ],
171
- principals: [
172
- new aws_iam_1.AnyPrincipal(),
173
- ],
174
- }),
175
- ],
176
- });
177
- }
178
- exports.createIpRestrictionPolicyDocument = createIpRestrictionPolicyDocument;
179
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rest_apis.js","sourceRoot":"","sources":["../../../../../src/aws/infra/stack/rest_apis.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n    RestApi,\r\n    MethodLoggingLevel,\r\n    GatewayResponse,\r\n    ResponseType,\r\n    EndpointType,\r\n    RestApiProps, JsonSchema, Model, CfnDocumentationPart, Resource,\r\n} from 'aws-cdk-lib/aws-apigateway';\r\nimport {PolicyDocument, PolicyStatement, Effect, AnyPrincipal} from 'aws-cdk-lib/aws-iam';\r\nimport {Construct} from \"constructs\";\r\nimport {DigitrafficStack} from \"./stack\";\r\nimport {createDefaultUsagePlan, createUsagePlan} from \"../usage-plans\";\r\nimport {ModelWithReference} from \"../../types/model-with-reference\";\r\nimport {getModelReference} from \"../../../utils/api-model\";\r\nimport {MediaType} from \"../../types/mediatypes\";\r\nimport {DocumentationPart, DocumentationProperties} from \"../documentation\";\r\n\r\nimport R = require('ramda');\r\n\r\nexport class DigitrafficRestApi extends RestApi {\r\n    readonly apiKeyIds: string[];\r\n\r\n    constructor(\r\n        stack: DigitrafficStack, apiId: string, apiName: string, allowFromIpAddresses?: string[] | undefined, config?: Partial<RestApiProps>,\r\n    ) {\r\n        const policyDocument = allowFromIpAddresses == null ? createDefaultPolicyDocument() : createIpRestrictionPolicyDocument(allowFromIpAddresses);\r\n\r\n        // override default config with given extra config\r\n        const apiConfig = {...{\r\n            deployOptions: {\r\n                loggingLevel: MethodLoggingLevel.ERROR,\r\n            },\r\n            restApiName: apiName,\r\n            endpointTypes: [EndpointType.REGIONAL],\r\n            policy: policyDocument,\r\n        }, ...config};\r\n\r\n        super(stack, apiId, apiConfig);\r\n\r\n        this.apiKeyIds = [];\r\n\r\n        add404Support(this, stack);\r\n    }\r\n\r\n    hostname(): string {\r\n        return `${this.restApiId}.execute-api.${(this.stack as DigitrafficStack).region}.amazonaws.com`;\r\n    }\r\n\r\n    createUsagePlan(apiKeyId: string, apiKeyName: string): string {\r\n        const newKeyId = createUsagePlan(this, apiKeyId, apiKeyName).keyId;\r\n\r\n        this.apiKeyIds.push(newKeyId);\r\n\r\n        return newKeyId;\r\n    }\r\n\r\n    createUsagePlanV2(apiName: string): string {\r\n        const newKeyId = createDefaultUsagePlan(this, apiName).keyId;\r\n\r\n        this.apiKeyIds.push(newKeyId);\r\n\r\n        return newKeyId;\r\n    }\r\n\r\n    addJsonModel(modelName: string, schema: JsonSchema) {\r\n        return this.getModelWithReference(this.addModel(modelName, {\r\n            contentType: MediaType.APPLICATION_JSON,\r\n            modelName,\r\n            schema,\r\n        }));\r\n    }\r\n\r\n    addCSVModel(modelName: string) {\r\n        return this.getModelWithReference(this.addModel(modelName, {\r\n            contentType: MediaType.TEXT_CSV,\r\n            modelName,\r\n            schema: {},\r\n        }));\r\n    }\r\n\r\n    private getModelWithReference(model: Model): ModelWithReference {\r\n        return R.assoc('modelReference', getModelReference(model.modelId, this.restApiId), model) as ModelWithReference;\r\n    }\r\n\r\n    private addDocumentationPart(\r\n        resource: Resource, parameterName: string, resourceName: string, type: string, properties: DocumentationProperties,\r\n    ) {\r\n        const location: CfnDocumentationPart.LocationProperty = {\r\n            type,\r\n            path: resource.path,\r\n            name: type !== 'METHOD' ? parameterName : undefined,\r\n        };\r\n\r\n        new CfnDocumentationPart(this.stack, resourceName, {\r\n            restApiId: resource.api.restApiId,\r\n            location,\r\n            properties: JSON.stringify(properties),\r\n        });\r\n    }\r\n\r\n    documentResource(resource: Resource, ...documentationPart: DocumentationPart[]) {\r\n        documentationPart.forEach(dp => this.addDocumentationPart(\r\n            resource, dp.parameterName, `${resource.path}.${dp.parameterName}.Documentation`, dp.type, dp.documentationProperties,\r\n        ));\r\n    }\r\n}\r\n\r\n/**\r\n * Due to AWS API design API Gateway will always return 403 'Missing Authentication Token' for requests\r\n * with a non-existent endpoint. This function translates this response to a 404.\r\n * Requests with an invalid or missing API key are not affected (still return 403 'Forbidden').\r\n * @param restApi RestApi\r\n * @param stack Construct\r\n */\r\nexport function add404Support(restApi: RestApi, stack: Construct) {\r\n    new GatewayResponse(stack, `MissingAuthenticationTokenResponse-${restApi.restApiName}`, {\r\n        restApi,\r\n        type: ResponseType.MISSING_AUTHENTICATION_TOKEN,\r\n        statusCode: '404',\r\n        templates: {\r\n            'application/json': '{\"message\": \"Not found\"}',\r\n        },\r\n    });\r\n}\r\n\r\nexport function add401Support(restApi: RestApi, stack: Construct) {\r\n    new GatewayResponse(stack, `AuthenticationFailedResponse-${restApi.restApiName}`, {\r\n        restApi,\r\n        type: ResponseType.UNAUTHORIZED,\r\n        statusCode: \"401\",\r\n        responseHeaders: {\r\n            'WWW-Authenticate': \"'Basic'\",\r\n        },\r\n    });\r\n}\r\n\r\n/**\r\n * Due to AWS API design API Gateway will always return 403 'Missing Authentication Token' for requests\r\n * with a non-existent endpoint. This function converts this response to a custom one.\r\n * Requests with an invalid or missing API key are not affected (still return 403 'Forbidden').\r\n * @param returnCode\r\n * @param message\r\n * @param restApi RestApi\r\n * @param stack Construct\r\n */\r\nexport function setReturnCodeForMissingAuthenticationToken(returnCode: number,\r\n    message: string,\r\n    restApi: RestApi,\r\n    stack: Construct) {\r\n\r\n    new GatewayResponse(stack, `MissingAuthenticationTokenResponse-${restApi.restApiName}`, {\r\n        restApi,\r\n        type: ResponseType.MISSING_AUTHENTICATION_TOKEN,\r\n        statusCode: `${returnCode}`,\r\n        templates: {\r\n            'application/json': `{\"message\": ${message}}`,\r\n        },\r\n    });\r\n}\r\n\r\nexport function createRestApi(stack: Construct, apiId: string, apiName: string, allowFromIpAddresses?: string[] | undefined): RestApi {\r\n    const policyDocument = allowFromIpAddresses == null ? createDefaultPolicyDocument() : createIpRestrictionPolicyDocument(allowFromIpAddresses);\r\n    const restApi = new RestApi(stack, apiId, {\r\n        deployOptions: {\r\n            loggingLevel: MethodLoggingLevel.ERROR,\r\n        },\r\n        restApiName: apiName,\r\n        endpointTypes: [EndpointType.REGIONAL],\r\n        policy: policyDocument,\r\n    });\r\n    add404Support(restApi, stack);\r\n    return restApi;\r\n}\r\n\r\nexport function createDefaultPolicyDocument() {\r\n    return new PolicyDocument({\r\n        statements: [\r\n            new PolicyStatement({\r\n                effect: Effect.ALLOW,\r\n                actions: [\r\n                    \"execute-api:Invoke\",\r\n                ],\r\n                resources: [\r\n                    \"*\",\r\n                ],\r\n                principals: [\r\n                    new AnyPrincipal(),\r\n                ],\r\n            }),\r\n        ],\r\n    });\r\n}\r\n\r\n\r\nexport function createIpRestrictionPolicyDocument(allowFromIpAddresses: string[]): PolicyDocument {\r\n    return new PolicyDocument({\r\n        statements: [\r\n            new PolicyStatement({\r\n                effect: Effect.ALLOW,\r\n                conditions: {\r\n                    \"IpAddress\": {\r\n                        \"aws:SourceIp\": allowFromIpAddresses,\r\n                    },\r\n                },\r\n                actions: [\r\n                    \"execute-api:Invoke\",\r\n                ],\r\n                resources: [\r\n                    \"*\",\r\n                ],\r\n                principals: [\r\n                    new AnyPrincipal(),\r\n                ],\r\n            }),\r\n        ],\r\n    });\r\n}\r\n"]}
@@ -1,163 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.StackCheckingAspect = void 0;
4
- const aws_cdk_lib_1 = require("aws-cdk-lib");
5
- const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
6
- const aws_s3_1 = require("aws-cdk-lib/aws-s3");
7
- const stack_1 = require("./stack");
8
- const aws_apigateway_1 = require("aws-cdk-lib/aws-apigateway");
9
- const change_case_1 = require("change-case");
10
- const aws_sqs_1 = require("aws-cdk-lib/aws-sqs");
11
- const aws_logs_1 = require("aws-cdk-lib/aws-logs");
12
- const MAX_CONCURRENCY_LIMIT = 100;
13
- const NODE_RUNTIME = aws_lambda_1.Runtime.NODEJS_14_X.name;
14
- var ResourceType;
15
- (function (ResourceType) {
16
- ResourceType["stackName"] = "STACK_NAME";
17
- ResourceType["reservedConcurrentConcurrency"] = "RESERVED_CONCURRENT_CONCURRENCY";
18
- ResourceType["functionTimeout"] = "FUNCTION_TIMEOUT";
19
- ResourceType["functionMemorySize"] = "FUNCTION_MEMORY_SIZE";
20
- ResourceType["functionRuntime"] = "FUNCTION_RUNTIME";
21
- ResourceType["functionName"] = "FUNCTION_NAME";
22
- ResourceType["tagSolution"] = "TAG_SOLUTION";
23
- ResourceType["bucketPublicity"] = "BUCKET_PUBLICITY";
24
- ResourceType["resourcePath"] = "RESOURCE_PATH";
25
- ResourceType["queueEncryption"] = "QUEUE_ENCRYPTION";
26
- ResourceType["logGroupRetention"] = "LOG_GROUP_RETENTION";
27
- })(ResourceType || (ResourceType = {}));
28
- class StackCheckingAspect {
29
- constructor(configuration) {
30
- this.configuration = configuration;
31
- }
32
- static create(stack) {
33
- return new StackCheckingAspect(stack.configuration);
34
- }
35
- visit(node) {
36
- //console.info("visiting class " + node.constructor.name);
37
- this.checkStack(node);
38
- this.checkFunction(node);
39
- this.checkTags(node);
40
- this.checkBucket(node);
41
- this.checkResourceCasing(node);
42
- this.checkQueueEncryption(node);
43
- this.checkLogGroupRetention(node);
44
- }
45
- isWhitelisted(key) {
46
- return this?.configuration?.whitelistedResources?.some(wl => {
47
- return key.matchAll(new RegExp(wl, 'g'));
48
- });
49
- }
50
- addAnnotation(node, key, message, isError = true) {
51
- const resourceKey = `${node.node.path}/${key}`;
52
- const isWhiteListed = this.isWhitelisted(resourceKey);
53
- const annotationMessage = `${resourceKey}:${message}`;
54
- // error && whitelisted -> warning
55
- // warning && whitelisted -> nothing
56
- if (isError && !isWhiteListed) {
57
- aws_cdk_lib_1.Annotations.of(node).addError(annotationMessage);
58
- }
59
- else if ((!isError && !isWhiteListed) || (isError && isWhiteListed)) {
60
- aws_cdk_lib_1.Annotations.of(node).addWarning(annotationMessage);
61
- }
62
- }
63
- checkStack(node) {
64
- if (node instanceof stack_1.DigitrafficStack) {
65
- if ((node.stackName.includes('Test') || node.stackName.includes('Tst')) && node.configuration?.production) {
66
- this.addAnnotation(node, ResourceType.stackName, 'Production is set for Test-stack');
67
- }
68
- if ((node.stackName.includes('Prod') || node.stackName.includes('Prd')) && !node.configuration?.production) {
69
- this.addAnnotation(node, ResourceType.stackName, 'Production is not set for Production-stack');
70
- }
71
- }
72
- }
73
- checkFunction(node) {
74
- if (node instanceof aws_lambda_1.CfnFunction) {
75
- if (!node.reservedConcurrentExecutions) {
76
- this.addAnnotation(node, ResourceType.reservedConcurrentConcurrency, 'Function must have reservedConcurrentConcurrency');
77
- }
78
- else if (node.reservedConcurrentExecutions > MAX_CONCURRENCY_LIMIT) {
79
- this.addAnnotation(node, ResourceType.reservedConcurrentConcurrency, 'Function reservedConcurrentConcurrency too high!');
80
- }
81
- if (!node.timeout) {
82
- this.addAnnotation(node, ResourceType.functionTimeout, 'Function must have timeout');
83
- }
84
- if (!node.memorySize) {
85
- this.addAnnotation(node, ResourceType.functionMemorySize, 'Function must have memorySize');
86
- }
87
- if (node.runtime !== NODE_RUNTIME) {
88
- this.addAnnotation(node, ResourceType.functionRuntime, 'wrong runtime ' + node.runtime);
89
- }
90
- if (this.configuration?.shortName && node.functionName && node.functionName.indexOf(this.configuration.shortName) !== 0) {
91
- this.addAnnotation(node, ResourceType.functionName, 'Function name does not begin with ' + this.configuration.shortName);
92
- }
93
- }
94
- }
95
- checkTags(node) {
96
- if (node instanceof aws_cdk_lib_1.Stack) {
97
- if (!node.tags.tagValues()[stack_1.SOLUTION_KEY]) {
98
- this.addAnnotation(node, ResourceType.tagSolution, 'Solution tag is missing');
99
- }
100
- }
101
- }
102
- checkBucket(node) {
103
- if (node instanceof aws_s3_1.CfnBucket) {
104
- const c = node.publicAccessBlockConfiguration;
105
- if (c) {
106
- if (!c.blockPublicAcls || !c.blockPublicPolicy || !c.ignorePublicAcls || !c.restrictPublicBuckets) {
107
- this.addAnnotation(node, ResourceType.bucketPublicity, 'Check bucket publicity');
108
- }
109
- }
110
- }
111
- }
112
- static isValidPath(path) {
113
- // if path includes . or { check only the trailing part of path
114
- if (path.includes('.')) {
115
- return this.isValidPath(path.split('.')[0]);
116
- }
117
- if (path.includes('{')) {
118
- return this.isValidPath(path.split('{')[0]);
119
- }
120
- return (0, change_case_1.paramCase)(path) === path;
121
- }
122
- static isValidQueryString(name) {
123
- return (0, change_case_1.snakeCase)(name) === name;
124
- }
125
- checkResourceCasing(node) {
126
- if (node instanceof aws_apigateway_1.CfnResource) {
127
- if (!StackCheckingAspect.isValidPath(node.pathPart)) {
128
- this.addAnnotation(node, ResourceType.resourcePath, 'Path part should be in kebab-case');
129
- }
130
- }
131
- else if (node instanceof aws_apigateway_1.CfnMethod) {
132
- const integration = node.integration;
133
- if (integration && integration.requestParameters) {
134
- Object.keys(integration.requestParameters).forEach(key => {
135
- const split = key.split('.');
136
- const type = split[2];
137
- const name = split[3];
138
- if (type === 'querystring' && !StackCheckingAspect.isValidQueryString(name)) {
139
- this.addAnnotation(node, name, 'Querystring should be in snake_case');
140
- }
141
- });
142
- }
143
- }
144
- }
145
- checkQueueEncryption(node) {
146
- if (node instanceof aws_sqs_1.CfnQueue) {
147
- if (!node.kmsMasterKeyId) {
148
- this.addAnnotation(node, ResourceType.queueEncryption, 'Queue must have encryption enabled');
149
- }
150
- }
151
- }
152
- checkLogGroupRetention(node) {
153
- if (node instanceof aws_logs_1.LogRetention) {
154
- const child = node.node.defaultChild;
155
- const retention = child._cfnProperties.RetentionInDays;
156
- if (!retention) {
157
- this.addAnnotation(node, ResourceType.logGroupRetention, 'Log group must define log group retention');
158
- }
159
- }
160
- }
161
- }
162
- exports.StackCheckingAspect = StackCheckingAspect;
163
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stack-checking-aspect.js","sourceRoot":"","sources":["../../../../../src/aws/infra/stack/stack-checking-aspect.ts"],"names":[],"mappings":"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","sourcesContent":["import {Annotations, IAspect, Stack} from \"aws-cdk-lib\";\nimport {CfnFunction, Runtime} from 'aws-cdk-lib/aws-lambda';\nimport {CfnBucket} from \"aws-cdk-lib/aws-s3\";\nimport {DigitrafficStack, SOLUTION_KEY, StackConfiguration} from \"./stack\";\nimport {IConstruct} from \"constructs\";\nimport {CfnMethod, CfnResource} from \"aws-cdk-lib/aws-apigateway\";\nimport {paramCase, snakeCase} from \"change-case\";\nimport {CfnQueue} from \"aws-cdk-lib/aws-sqs\";\nimport {LogRetention} from \"aws-cdk-lib/aws-logs\";\nimport IntegrationProperty = CfnMethod.IntegrationProperty;\n\nconst MAX_CONCURRENCY_LIMIT = 100;\nconst NODE_RUNTIME = Runtime.NODEJS_14_X.name;\n\nenum ResourceType {\n    stackName = \"STACK_NAME\",\n    reservedConcurrentConcurrency = \"RESERVED_CONCURRENT_CONCURRENCY\",\n    functionTimeout = \"FUNCTION_TIMEOUT\",\n    functionMemorySize = \"FUNCTION_MEMORY_SIZE\",\n    functionRuntime = \"FUNCTION_RUNTIME\",\n    functionName = \"FUNCTION_NAME\",\n    tagSolution = \"TAG_SOLUTION\",\n    bucketPublicity = \"BUCKET_PUBLICITY\",\n    resourcePath = \"RESOURCE_PATH\",\n    queueEncryption = \"QUEUE_ENCRYPTION\",\n    logGroupRetention = \"LOG_GROUP_RETENTION\",\n}\n\nexport class StackCheckingAspect implements IAspect {\n    private readonly configuration?: StackConfiguration;\n\n    constructor(configuration?: StackConfiguration) {\n        this.configuration = configuration;\n    }\n\n    static create(stack: DigitrafficStack) {\n        return new StackCheckingAspect(stack.configuration);\n    }\n\n    public visit(node: IConstruct): void {\n        //console.info(\"visiting class \" + node.constructor.name);\n\n        this.checkStack(node);\n        this.checkFunction(node);\n        this.checkTags(node);\n        this.checkBucket(node);\n        this.checkResourceCasing(node);\n        this.checkQueueEncryption(node);\n        this.checkLogGroupRetention(node);\n    }\n\n    private isWhitelisted(key: string) {\n        return this?.configuration?.whitelistedResources?.some(wl => {\n            return key.matchAll(new RegExp(wl, 'g'));\n        });\n    }\n\n    private addAnnotation(node: IConstruct, key: ResourceType | string, message: string, isError = true) {\n        const resourceKey = `${node.node.path}/${key}`;\n        const isWhiteListed = this.isWhitelisted(resourceKey);\n        const annotationMessage = `${resourceKey}:${message}`;\n\n        // error && whitelisted -> warning\n        // warning && whitelisted -> nothing\n        if (isError && !isWhiteListed) {\n            Annotations.of(node).addError(annotationMessage);\n        } else if ((!isError && !isWhiteListed) || (isError && isWhiteListed)) {\n            Annotations.of(node).addWarning(annotationMessage);\n        }\n    }\n\n    private checkStack(node: IConstruct) {\n        if (node instanceof DigitrafficStack) {\n            if ((node.stackName.includes('Test') || node.stackName.includes('Tst')) && node.configuration?.production) {\n                this.addAnnotation(node, ResourceType.stackName, 'Production is set for Test-stack');\n            }\n\n            if ((node.stackName.includes('Prod') || node.stackName.includes('Prd')) && !node.configuration?.production) {\n                this.addAnnotation(node, ResourceType.stackName, 'Production is not set for Production-stack');\n            }\n        }\n    }\n\n    private checkFunction(node: IConstruct) {\n        if (node instanceof CfnFunction) {\n            if (!node.reservedConcurrentExecutions) {\n                this.addAnnotation(node, ResourceType.reservedConcurrentConcurrency, 'Function must have reservedConcurrentConcurrency');\n            } else if (node.reservedConcurrentExecutions > MAX_CONCURRENCY_LIMIT) {\n                this.addAnnotation(node, ResourceType.reservedConcurrentConcurrency, 'Function reservedConcurrentConcurrency too high!');\n            }\n\n            if (!node.timeout) {\n                this.addAnnotation(node, ResourceType.functionTimeout, 'Function must have timeout');\n            }\n\n            if (!node.memorySize) {\n                this.addAnnotation(node, ResourceType.functionMemorySize, 'Function must have memorySize');\n            }\n\n            if (node.runtime !== NODE_RUNTIME) {\n                this.addAnnotation(node, ResourceType.functionRuntime,'wrong runtime ' + node.runtime);\n            }\n\n            if (this.configuration?.shortName && node.functionName && node.functionName.indexOf(this.configuration.shortName) !== 0) {\n                this.addAnnotation(node, ResourceType.functionName, 'Function name does not begin with ' + this.configuration.shortName);\n            }\n        }\n    }\n\n    private checkTags(node: IConstruct) {\n        if (node instanceof Stack) {\n            if (!node.tags.tagValues()[SOLUTION_KEY]) {\n                this.addAnnotation(node, ResourceType.tagSolution, 'Solution tag is missing');\n            }\n        }\n    }\n\n    private checkBucket(node: IConstruct) {\n        if (node instanceof CfnBucket) {\n            const c = node.publicAccessBlockConfiguration as CfnBucket.PublicAccessBlockConfigurationProperty;\n\n            if (c) {\n                if (!c.blockPublicAcls || !c.blockPublicPolicy || !c.ignorePublicAcls || !c.restrictPublicBuckets) {\n                    this.addAnnotation(node, ResourceType.bucketPublicity, 'Check bucket publicity');\n                }\n            }\n        }\n    }\n\n    private static isValidPath(path: string): boolean {\n        // if path includes . or { check only the trailing part of path\n        if (path.includes('.')) {\n            return this.isValidPath(path.split('.')[0]);\n        }\n\n        if (path.includes('{')) {\n            return this.isValidPath(path.split('{')[0]);\n        }\n\n        return paramCase(path) === path;\n    }\n\n    private static isValidQueryString(name: string) {\n        return snakeCase(name) === name;\n    }\n\n    private checkResourceCasing(node: IConstruct) {\n        if (node instanceof CfnResource) {\n            if (!StackCheckingAspect.isValidPath(node.pathPart)) {\n                this.addAnnotation(node, ResourceType.resourcePath, 'Path part should be in kebab-case');\n            }\n        } else if (node instanceof CfnMethod) {\n            const integration = node.integration as IntegrationProperty;\n\n            if (integration && integration.requestParameters) {\n                Object.keys(integration.requestParameters).forEach(key => {\n                    const split = key.split('.');\n                    const type = split[2];\n                    const name = split[3];\n\n                    if (type === 'querystring' && !StackCheckingAspect.isValidQueryString(name)) {\n                        this.addAnnotation(node, name, 'Querystring should be in snake_case');\n                    }\n                });\n            }\n        }\n    }\n\n    private checkQueueEncryption(node: IConstruct) {\n        if (node instanceof CfnQueue) {\n            if (!node.kmsMasterKeyId) {\n                this.addAnnotation(node, ResourceType.queueEncryption, 'Queue must have encryption enabled');\n            }\n        }\n    }\n\n    private checkLogGroupRetention(node: IConstruct) {\n        if (node instanceof LogRetention) {\n            const child = node.node.defaultChild as unknown as Record<string, Record<string, string>>;\n            const retention = child._cfnProperties.RetentionInDays;\n\n            if (!retention) {\n                this.addAnnotation(node, ResourceType.logGroupRetention, 'Log group must define log group retention');\n            }\n        }\n    }\n}\n"]}