@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,29 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.DigitrafficCanary = void 0;
4
- const aws_cdk_lib_1 = require("aws-cdk-lib");
5
- const aws_synthetics_alpha_1 = require("@aws-cdk/aws-synthetics-alpha");
6
- const canary_alarm_1 = require("./canary-alarm");
7
- class DigitrafficCanary extends aws_synthetics_alpha_1.Canary {
8
- constructor(scope, canaryName, role, params, environmentVariables) {
9
- super(scope, canaryName, {
10
- runtime: aws_synthetics_alpha_1.Runtime.SYNTHETICS_NODEJS_PUPPETEER_3_5,
11
- role,
12
- test: aws_synthetics_alpha_1.Test.custom({
13
- code: new aws_synthetics_alpha_1.AssetCode("dist", {
14
- exclude: ["lambda"],
15
- }),
16
- handler: params.handler,
17
- }),
18
- environmentVariables: { ...environmentVariables, ...params?.canaryEnv },
19
- canaryName,
20
- schedule: params.schedule ?? aws_synthetics_alpha_1.Schedule.rate(aws_cdk_lib_1.Duration.minutes(15)),
21
- });
22
- this.artifactsBucket.grantWrite(role);
23
- if (params.alarm ?? true) {
24
- new canary_alarm_1.CanaryAlarm(scope, this, params);
25
- }
26
- }
27
- }
28
- exports.DigitrafficCanary = DigitrafficCanary;
29
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FuYXJ5LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2F3cy9pbmZyYS9jYW5hcmllcy9jYW5hcnkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsNkNBQXFDO0FBQ3JDLHdFQUF5RjtBQUV6RixpREFBMkM7QUFLM0MsTUFBYSxpQkFBa0IsU0FBUSw2QkFBTTtJQUN6QyxZQUNJLEtBQWdCLEVBQ2hCLFVBQWtCLEVBQ2xCLElBQVUsRUFDVixNQUF3QixFQUN4QixvQkFBdUM7UUFFdkMsS0FBSyxDQUFDLEtBQUssRUFBRSxVQUFVLEVBQUU7WUFDckIsT0FBTyxFQUFFLDhCQUFPLENBQUMsK0JBQStCO1lBQ2hELElBQUk7WUFDSixJQUFJLEVBQUUsMkJBQUksQ0FBQyxNQUFNLENBQUM7Z0JBQ2QsSUFBSSxFQUFFLElBQUksZ0NBQVMsQ0FBQyxNQUFNLEVBQUU7b0JBQ3hCLE9BQU8sRUFBRSxDQUFDLFFBQVEsQ0FBQztpQkFDdEIsQ0FBQztnQkFDRixPQUFPLEVBQUUsTUFBTSxDQUFDLE9BQU87YUFDMUIsQ0FBQztZQUNGLG9CQUFvQixFQUFFLEVBQUUsR0FBRyxvQkFBb0IsRUFBRSxHQUFHLE1BQU0sRUFBRSxTQUFTLEVBQUU7WUFDdkUsVUFBVTtZQUNWLFFBQVEsRUFBRSxNQUFNLENBQUMsUUFBUSxJQUFJLCtCQUFRLENBQUMsSUFBSSxDQUFDLHNCQUFRLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQyxDQUFDO1NBQ25FLENBQUMsQ0FBQztRQUVILElBQUksQ0FBQyxlQUFlLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQyxDQUFDO1FBRXRDLElBQUksTUFBTSxDQUFDLEtBQUssSUFBSSxJQUFJLEVBQUU7WUFDdEIsSUFBSSwwQkFBVyxDQUFDLEtBQUssRUFBRSxJQUFJLEVBQUUsTUFBTSxDQUFDLENBQUM7U0FDeEM7SUFDTCxDQUFDO0NBQ0o7QUE1QkQsOENBNEJDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtEdXJhdGlvbn0gZnJvbSBcImF3cy1jZGstbGliXCI7XG5pbXBvcnQge0Fzc2V0Q29kZSwgQ2FuYXJ5LCBSdW50aW1lLCBTY2hlZHVsZSwgVGVzdH0gZnJvbSBcIkBhd3MtY2RrL2F3cy1zeW50aGV0aWNzLWFscGhhXCI7XG5pbXBvcnQge1JvbGV9IGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtaWFtXCI7XG5pbXBvcnQge0NhbmFyeUFsYXJtfSBmcm9tIFwiLi9jYW5hcnktYWxhcm1cIjtcbmltcG9ydCB7Q2FuYXJ5UGFyYW1ldGVyc30gZnJvbSBcIi4vY2FuYXJ5LXBhcmFtZXRlcnNcIjtcbmltcG9ydCB7Q29uc3RydWN0fSBmcm9tIFwiY29uc3RydWN0c1wiO1xuaW1wb3J0IHtMYW1iZGFFbnZpcm9ubWVudH0gZnJvbSBcIi4uL3N0YWNrL2xhbWJkYS1jb25maWdzXCI7XG5cbmV4cG9ydCBjbGFzcyBEaWdpdHJhZmZpY0NhbmFyeSBleHRlbmRzIENhbmFyeSB7XG4gICAgY29uc3RydWN0b3IoXG4gICAgICAgIHNjb3BlOiBDb25zdHJ1Y3QsXG4gICAgICAgIGNhbmFyeU5hbWU6IHN0cmluZyxcbiAgICAgICAgcm9sZTogUm9sZSxcbiAgICAgICAgcGFyYW1zOiBDYW5hcnlQYXJhbWV0ZXJzLFxuICAgICAgICBlbnZpcm9ubWVudFZhcmlhYmxlczogTGFtYmRhRW52aXJvbm1lbnQsXG4gICAgKSB7XG4gICAgICAgIHN1cGVyKHNjb3BlLCBjYW5hcnlOYW1lLCB7XG4gICAgICAgICAgICBydW50aW1lOiBSdW50aW1lLlNZTlRIRVRJQ1NfTk9ERUpTX1BVUFBFVEVFUl8zXzUsXG4gICAgICAgICAgICByb2xlLFxuICAgICAgICAgICAgdGVzdDogVGVzdC5jdXN0b20oe1xuICAgICAgICAgICAgICAgIGNvZGU6IG5ldyBBc3NldENvZGUoXCJkaXN0XCIsIHtcbiAgICAgICAgICAgICAgICAgICAgZXhjbHVkZTogW1wibGFtYmRhXCJdLFxuICAgICAgICAgICAgICAgIH0pLFxuICAgICAgICAgICAgICAgIGhhbmRsZXI6IHBhcmFtcy5oYW5kbGVyLFxuICAgICAgICAgICAgfSksXG4gICAgICAgICAgICBlbnZpcm9ubWVudFZhcmlhYmxlczogeyAuLi5lbnZpcm9ubWVudFZhcmlhYmxlcywgLi4ucGFyYW1zPy5jYW5hcnlFbnYgfSxcbiAgICAgICAgICAgIGNhbmFyeU5hbWUsXG4gICAgICAgICAgICBzY2hlZHVsZTogcGFyYW1zLnNjaGVkdWxlID8/IFNjaGVkdWxlLnJhdGUoRHVyYXRpb24ubWludXRlcygxNSkpLFxuICAgICAgICB9KTtcblxuICAgICAgICB0aGlzLmFydGlmYWN0c0J1Y2tldC5ncmFudFdyaXRlKHJvbGUpO1xuXG4gICAgICAgIGlmIChwYXJhbXMuYWxhcm0gPz8gdHJ1ZSkge1xuICAgICAgICAgICAgbmV3IENhbmFyeUFsYXJtKHNjb3BlLCB0aGlzLCBwYXJhbXMpO1xuICAgICAgICB9XG4gICAgfVxufVxuIl19
@@ -1,55 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.DatabaseCanary = void 0;
4
- const aws_synthetics_1 = require("aws-cdk-lib/aws-synthetics");
5
- const aws_events_1 = require("aws-cdk-lib/aws-events");
6
- const aws_cdk_lib_1 = require("aws-cdk-lib");
7
- const canary_1 = require("./canary");
8
- class DatabaseCanary extends canary_1.DigitrafficCanary {
9
- constructor(stack, role, secret, params) {
10
- const canaryName = `${params.name}-db`;
11
- const environmentVariables = stack.createDefaultLambdaEnvironment(`Synthetics-${canaryName}`);
12
- // the handler code is defined at the actual project using this
13
- super(stack, canaryName, role, params, environmentVariables);
14
- this.artifactsBucket.grantWrite(this.role);
15
- secret.grantRead(this.role);
16
- // need to override vpc and security group, can't do this with cdk
17
- if (this.node.defaultChild instanceof aws_synthetics_1.CfnCanary) {
18
- const subnetIds = stack.vpc.privateSubnets.map(subnet => subnet.subnetId);
19
- this.node.defaultChild.vpcConfig = {
20
- vpcId: stack.vpc.vpcId,
21
- securityGroupIds: [stack.lambdaDbSg.securityGroupId],
22
- subnetIds: subnetIds,
23
- };
24
- }
25
- }
26
- static create(stack, role, params) {
27
- return new DatabaseCanary(stack, role, stack.secret, { ...{
28
- secret: stack.configuration.secretId,
29
- schedule: aws_events_1.Schedule.rate(aws_cdk_lib_1.Duration.hours(1)),
30
- handler: `${params.name}.handler`,
31
- }, ...params });
32
- }
33
- /**
34
- *
35
- * @param stack
36
- * @param role
37
- * @param name name of the typescipt file without -db -suffix. Max len is 10 char if @param canaryName is not given.
38
- * @param params
39
- * @param canaryName Optional name for canary if multiple canaries is made from same ${name}-db.ts canary file.
40
- */
41
- static createV2(stack, role, name, params = {}, canaryName = name) {
42
- return new DatabaseCanary(stack, role, stack.secret, { ...{
43
- secret: stack.configuration.secretId,
44
- schedule: aws_events_1.Schedule.rate(aws_cdk_lib_1.Duration.hours(1)),
45
- handler: `${name}-db.handler`,
46
- name: canaryName,
47
- alarm: {
48
- alarmName: (canaryName === name ? `${stack.configuration.shortName}-DB-Alarm` : `${canaryName}-DB-Alarm`),
49
- topicArn: stack.configuration.alarmTopicArn,
50
- },
51
- }, ...params });
52
- }
53
- }
54
- exports.DatabaseCanary = DatabaseCanary;
55
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF0YWJhc2UtY2FuYXJ5LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2F3cy9pbmZyYS9jYW5hcmllcy9kYXRhYmFzZS1jYW5hcnkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBRUEsK0RBQXFEO0FBQ3JELHVEQUFnRDtBQUNoRCw2Q0FBcUM7QUFHckMscUNBQTJDO0FBRzNDLE1BQWEsY0FBZSxTQUFRLDBCQUFpQjtJQUNqRCxZQUFZLEtBQXVCLEVBQy9CLElBQVUsRUFDVixNQUFlLEVBQ2YsTUFBd0I7UUFDeEIsTUFBTSxVQUFVLEdBQUcsR0FBRyxNQUFNLENBQUMsSUFBSSxLQUFLLENBQUM7UUFDdkMsTUFBTSxvQkFBb0IsR0FBRyxLQUFLLENBQUMsOEJBQThCLENBQUMsY0FBYyxVQUFVLEVBQUUsQ0FBQyxDQUFDO1FBRTlGLCtEQUErRDtRQUMvRCxLQUFLLENBQ0QsS0FBSyxFQUFFLFVBQVUsRUFBRSxJQUFJLEVBQ3ZCLE1BQU0sRUFBRSxvQkFBb0IsQ0FDL0IsQ0FBQztRQUVGLElBQUksQ0FBQyxlQUFlLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUMzQyxNQUFNLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUU1QixrRUFBa0U7UUFDbEUsSUFBSSxJQUFJLENBQUMsSUFBSSxDQUFDLFlBQVksWUFBWSwwQkFBUyxFQUFFO1lBQzdDLE1BQU0sU0FBUyxHQUFHLEtBQUssQ0FBQyxHQUFHLENBQUMsY0FBYyxDQUFDLEdBQUcsQ0FBQyxNQUFNLENBQUMsRUFBRSxDQUFDLE1BQU0sQ0FBQyxRQUFRLENBQUMsQ0FBQztZQUUxRSxJQUFJLENBQUMsSUFBSSxDQUFDLFlBQVksQ0FBQyxTQUFTLEdBQUc7Z0JBQy9CLEtBQUssRUFBRSxLQUFLLENBQUMsR0FBRyxDQUFDLEtBQUs7Z0JBQ3RCLGdCQUFnQixFQUFFLENBQUMsS0FBSyxDQUFDLFVBQVUsQ0FBQyxlQUFlLENBQUM7Z0JBQ3BELFNBQVMsRUFBRSxTQUFTO2FBQ3ZCLENBQUM7U0FDTDtJQUNMLENBQUM7SUFFRCxNQUFNLENBQUMsTUFBTSxDQUFDLEtBQXVCLEVBQ2pDLElBQVUsRUFDVixNQUF3QjtRQUN4QixPQUFPLElBQUksY0FBYyxDQUFDLEtBQUssRUFBRSxJQUFJLEVBQUUsS0FBSyxDQUFDLE1BQU0sRUFDL0MsRUFBQyxHQUFHO2dCQUNBLE1BQU0sRUFBRSxLQUFLLENBQUMsYUFBYSxDQUFDLFFBQVE7Z0JBQ3BDLFFBQVEsRUFBRSxxQkFBUSxDQUFDLElBQUksQ0FBQyxzQkFBUSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztnQkFDMUMsT0FBTyxFQUFFLEdBQUcsTUFBTSxDQUFDLElBQUksVUFBVTthQUNwQyxFQUFFLEdBQUcsTUFBTSxFQUFDLENBQUMsQ0FBQztJQUN2QixDQUFDO0lBRUQ7Ozs7Ozs7T0FPRztJQUNILE1BQU0sQ0FBQyxRQUFRLENBQ1gsS0FBdUIsRUFDdkIsSUFBVSxFQUNWLElBQVksRUFDWixTQUFvQyxFQUFFLEVBQ3RDLFVBQVUsR0FBQyxJQUFJO1FBRWYsT0FBTyxJQUFJLGNBQWMsQ0FBQyxLQUFLLEVBQUUsSUFBSSxFQUFFLEtBQUssQ0FBQyxNQUFNLEVBQy9DLEVBQUMsR0FBRztnQkFDQSxNQUFNLEVBQUUsS0FBSyxDQUFDLGFBQWEsQ0FBQyxRQUFRO2dCQUNwQyxRQUFRLEVBQUUscUJBQVEsQ0FBQyxJQUFJLENBQUMsc0JBQVEsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7Z0JBQzFDLE9BQU8sRUFBRSxHQUFHLElBQUksYUFBYTtnQkFDN0IsSUFBSSxFQUFHLFVBQVU7Z0JBQ2pCLEtBQUssRUFBRTtvQkFDSCxTQUFTLEVBQUUsQ0FBQyxVQUFVLEtBQUssSUFBSSxDQUFDLENBQUMsQ0FBQyxHQUFHLEtBQUssQ0FBQyxhQUFhLENBQUMsU0FBUyxXQUFXLENBQUMsQ0FBQyxDQUFDLEdBQUcsVUFBVSxXQUFXLENBQUM7b0JBQ3pHLFFBQVEsRUFBRSxLQUFLLENBQUMsYUFBYSxDQUFDLGFBQWE7aUJBQzlDO2FBQ0osRUFBRSxHQUFHLE1BQU0sRUFBQyxDQUFDLENBQUM7SUFDdkIsQ0FBQztDQUNKO0FBbkVELHdDQW1FQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7Um9sZX0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1pYW1cIjtcbmltcG9ydCB7SVNlY3JldH0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1zZWNyZXRzbWFuYWdlclwiO1xuaW1wb3J0IHtDZm5DYW5hcnl9IGZyb20gXCJhd3MtY2RrLWxpYi9hd3Mtc3ludGhldGljc1wiO1xuaW1wb3J0IHtTY2hlZHVsZX0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1ldmVudHNcIjtcbmltcG9ydCB7RHVyYXRpb259IGZyb20gXCJhd3MtY2RrLWxpYlwiO1xuXG5pbXBvcnQge0NhbmFyeVBhcmFtZXRlcnN9IGZyb20gXCIuL2NhbmFyeS1wYXJhbWV0ZXJzXCI7XG5pbXBvcnQge0RpZ2l0cmFmZmljQ2FuYXJ5fSBmcm9tIFwiLi9jYW5hcnlcIjtcbmltcG9ydCB7RGlnaXRyYWZmaWNTdGFja30gZnJvbSBcIi4uL3N0YWNrL3N0YWNrXCI7XG5cbmV4cG9ydCBjbGFzcyBEYXRhYmFzZUNhbmFyeSBleHRlbmRzIERpZ2l0cmFmZmljQ2FuYXJ5IHtcbiAgICBjb25zdHJ1Y3RvcihzdGFjazogRGlnaXRyYWZmaWNTdGFjayxcbiAgICAgICAgcm9sZTogUm9sZSxcbiAgICAgICAgc2VjcmV0OiBJU2VjcmV0LFxuICAgICAgICBwYXJhbXM6IENhbmFyeVBhcmFtZXRlcnMpIHtcbiAgICAgICAgY29uc3QgY2FuYXJ5TmFtZSA9IGAke3BhcmFtcy5uYW1lfS1kYmA7XG4gICAgICAgIGNvbnN0IGVudmlyb25tZW50VmFyaWFibGVzID0gc3RhY2suY3JlYXRlRGVmYXVsdExhbWJkYUVudmlyb25tZW50KGBTeW50aGV0aWNzLSR7Y2FuYXJ5TmFtZX1gKTtcblxuICAgICAgICAvLyB0aGUgaGFuZGxlciBjb2RlIGlzIGRlZmluZWQgYXQgdGhlIGFjdHVhbCBwcm9qZWN0IHVzaW5nIHRoaXNcbiAgICAgICAgc3VwZXIoXG4gICAgICAgICAgICBzdGFjaywgY2FuYXJ5TmFtZSwgcm9sZSxcbiAgICAgICAgICAgIHBhcmFtcywgZW52aXJvbm1lbnRWYXJpYWJsZXMsXG4gICAgICAgICk7XG5cbiAgICAgICAgdGhpcy5hcnRpZmFjdHNCdWNrZXQuZ3JhbnRXcml0ZSh0aGlzLnJvbGUpO1xuICAgICAgICBzZWNyZXQuZ3JhbnRSZWFkKHRoaXMucm9sZSk7XG5cbiAgICAgICAgLy8gbmVlZCB0byBvdmVycmlkZSB2cGMgYW5kIHNlY3VyaXR5IGdyb3VwLCBjYW4ndCBkbyB0aGlzIHdpdGggY2RrXG4gICAgICAgIGlmICh0aGlzLm5vZGUuZGVmYXVsdENoaWxkIGluc3RhbmNlb2YgQ2ZuQ2FuYXJ5KSB7XG4gICAgICAgICAgICBjb25zdCBzdWJuZXRJZHMgPSBzdGFjay52cGMucHJpdmF0ZVN1Ym5ldHMubWFwKHN1Ym5ldCA9PiBzdWJuZXQuc3VibmV0SWQpO1xuXG4gICAgICAgICAgICB0aGlzLm5vZGUuZGVmYXVsdENoaWxkLnZwY0NvbmZpZyA9IHtcbiAgICAgICAgICAgICAgICB2cGNJZDogc3RhY2sudnBjLnZwY0lkLFxuICAgICAgICAgICAgICAgIHNlY3VyaXR5R3JvdXBJZHM6IFtzdGFjay5sYW1iZGFEYlNnLnNlY3VyaXR5R3JvdXBJZF0sXG4gICAgICAgICAgICAgICAgc3VibmV0SWRzOiBzdWJuZXRJZHMsXG4gICAgICAgICAgICB9O1xuICAgICAgICB9XG4gICAgfVxuXG4gICAgc3RhdGljIGNyZWF0ZShzdGFjazogRGlnaXRyYWZmaWNTdGFjayxcbiAgICAgICAgcm9sZTogUm9sZSxcbiAgICAgICAgcGFyYW1zOiBDYW5hcnlQYXJhbWV0ZXJzKTogRGF0YWJhc2VDYW5hcnkge1xuICAgICAgICByZXR1cm4gbmV3IERhdGFiYXNlQ2FuYXJ5KHN0YWNrLCByb2xlLCBzdGFjay5zZWNyZXQsXG4gICAgICAgICAgICB7Li4ue1xuICAgICAgICAgICAgICAgIHNlY3JldDogc3RhY2suY29uZmlndXJhdGlvbi5zZWNyZXRJZCxcbiAgICAgICAgICAgICAgICBzY2hlZHVsZTogU2NoZWR1bGUucmF0ZShEdXJhdGlvbi5ob3VycygxKSksXG4gICAgICAgICAgICAgICAgaGFuZGxlcjogYCR7cGFyYW1zLm5hbWV9LmhhbmRsZXJgLFxuICAgICAgICAgICAgfSwgLi4ucGFyYW1zfSk7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICpcbiAgICAgKiBAcGFyYW0gc3RhY2tcbiAgICAgKiBAcGFyYW0gcm9sZVxuICAgICAqIEBwYXJhbSBuYW1lIG5hbWUgb2YgdGhlIHR5cGVzY2lwdCBmaWxlIHdpdGhvdXQgLWRiIC1zdWZmaXguIE1heCBsZW4gaXMgMTAgY2hhciBpZiBAcGFyYW0gY2FuYXJ5TmFtZSBpcyBub3QgZ2l2ZW4uXG4gICAgICogQHBhcmFtIHBhcmFtc1xuICAgICAqIEBwYXJhbSBjYW5hcnlOYW1lIE9wdGlvbmFsIG5hbWUgZm9yIGNhbmFyeSBpZiBtdWx0aXBsZSBjYW5hcmllcyBpcyBtYWRlIGZyb20gc2FtZSAke25hbWV9LWRiLnRzIGNhbmFyeSBmaWxlLlxuICAgICAqL1xuICAgIHN0YXRpYyBjcmVhdGVWMihcbiAgICAgICAgc3RhY2s6IERpZ2l0cmFmZmljU3RhY2ssXG4gICAgICAgIHJvbGU6IFJvbGUsXG4gICAgICAgIG5hbWU6IHN0cmluZyxcbiAgICAgICAgcGFyYW1zOiBQYXJ0aWFsPENhbmFyeVBhcmFtZXRlcnM+ID0ge30sXG4gICAgICAgIGNhbmFyeU5hbWU9bmFtZSxcbiAgICApOiBEYXRhYmFzZUNhbmFyeSB7XG4gICAgICAgIHJldHVybiBuZXcgRGF0YWJhc2VDYW5hcnkoc3RhY2ssIHJvbGUsIHN0YWNrLnNlY3JldCxcbiAgICAgICAgICAgIHsuLi57XG4gICAgICAgICAgICAgICAgc2VjcmV0OiBzdGFjay5jb25maWd1cmF0aW9uLnNlY3JldElkLFxuICAgICAgICAgICAgICAgIHNjaGVkdWxlOiBTY2hlZHVsZS5yYXRlKER1cmF0aW9uLmhvdXJzKDEpKSxcbiAgICAgICAgICAgICAgICBoYW5kbGVyOiBgJHtuYW1lfS1kYi5oYW5kbGVyYCxcbiAgICAgICAgICAgICAgICBuYW1lIDogY2FuYXJ5TmFtZSxcbiAgICAgICAgICAgICAgICBhbGFybToge1xuICAgICAgICAgICAgICAgICAgICBhbGFybU5hbWU6IChjYW5hcnlOYW1lID09PSBuYW1lID8gYCR7c3RhY2suY29uZmlndXJhdGlvbi5zaG9ydE5hbWV9LURCLUFsYXJtYCA6IGAke2NhbmFyeU5hbWV9LURCLUFsYXJtYCksXG4gICAgICAgICAgICAgICAgICAgIHRvcGljQXJuOiBzdGFjay5jb25maWd1cmF0aW9uLmFsYXJtVG9waWNBcm4sXG4gICAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIH0sIC4uLnBhcmFtc30pO1xuICAgIH1cbn1cbiJdfQ==
@@ -1,109 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.DatabaseChecker = void 0;
4
- // eslint-disable-next-line @typescript-eslint/no-var-requires
5
- const database_1 = require("../../../database/database");
6
- const proxy_holder_1 = require("../../runtime/secrets/proxy-holder");
7
- const rds_holder_1 = require("../../runtime/secrets/rds-holder");
8
- // eslint-disable-next-line @typescript-eslint/no-var-requires
9
- const synthetics = require('Synthetics');
10
- class DatabaseCheck {
11
- constructor(name, sql) {
12
- this.name = name;
13
- this.sql = sql;
14
- this.failed = false;
15
- }
16
- }
17
- class BaseResponse {
18
- }
19
- class CountResponse extends BaseResponse {
20
- }
21
- class CountDatabaseCheck extends DatabaseCheck {
22
- constructor(name, sql, minCount, maxCount) {
23
- super(name, sql);
24
- if (minCount == null && maxCount == null) {
25
- throw new Error('no max or min given!');
26
- }
27
- this.minCount = minCount;
28
- this.maxCount = maxCount;
29
- }
30
- check(value) {
31
- if (!value) {
32
- this.failed = true;
33
- throw new Error('no return value');
34
- }
35
- else {
36
- if ('count' in value) {
37
- if (this.minCount && value.count < this.minCount) {
38
- this.failed = true;
39
- throw new Error(`count was ${value.count}, minimum is ${this.minCount}`);
40
- }
41
- if (this.maxCount && value.count > this.maxCount) {
42
- this.failed = true;
43
- throw new Error(`count was ${value.count}, max is ${this.maxCount}`);
44
- }
45
- }
46
- else {
47
- this.failed = true;
48
- console.info("received " + JSON.stringify(value));
49
- throw new Error('no count available');
50
- }
51
- }
52
- }
53
- }
54
- const stepConfig = {
55
- 'continueOnStepFailure': true,
56
- 'screenshotOnStepStart': false,
57
- 'screenshotOnStepSuccess': false,
58
- 'screenshotOnStepFailure': false,
59
- };
60
- class DatabaseChecker {
61
- constructor(credentialsFunction) {
62
- this.credentialsFunction = credentialsFunction;
63
- this.checks = [];
64
- synthetics.getConfiguration()
65
- .disableRequestMetrics();
66
- synthetics.getConfiguration()
67
- .withFailedCanaryMetric(true);
68
- }
69
- static createForProxy() {
70
- return new DatabaseChecker(() => new proxy_holder_1.ProxyHolder(process.env.SECRET_ID).setCredentials());
71
- }
72
- static createForRds() {
73
- return new DatabaseChecker(() => new rds_holder_1.RdsHolder(process.env.SECRET_ID).setCredentials());
74
- }
75
- one(name, sql) {
76
- this.checks.push(new CountDatabaseCheck(name, sql, 1, 1));
77
- return this;
78
- }
79
- empty(name, sql) {
80
- this.checks.push(new CountDatabaseCheck(name, sql, null, 0));
81
- return this;
82
- }
83
- notEmpty(name, sql) {
84
- this.checks.push(new CountDatabaseCheck(name, sql, 1, null));
85
- return this;
86
- }
87
- async expect() {
88
- if (!this.checks.length) {
89
- throw new Error('No checks');
90
- }
91
- await this.credentialsFunction();
92
- await (0, database_1.inDatabaseReadonly)(async (db) => {
93
- for (const check of this.checks) {
94
- console.info("canary checking sql " + check.sql);
95
- const value = await db.oneOrNone(check.sql);
96
- const checkFunction = () => {
97
- check.check(value);
98
- };
99
- synthetics.executeStep(check.name, checkFunction, stepConfig);
100
- }
101
- });
102
- if (this.checks.some(check => check.failed)) {
103
- throw new Error('Failed');
104
- }
105
- return 'OK';
106
- }
107
- }
108
- exports.DatabaseChecker = DatabaseChecker;
109
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"database-checker.js","sourceRoot":"","sources":["../../../../../src/aws/infra/canaries/database-checker.ts"],"names":[],"mappings":"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","sourcesContent":["// eslint-disable-next-line @typescript-eslint/no-var-requires\nimport {DTDatabase, inDatabaseReadonly} from \"../../../database/database\";\nimport {ProxyHolder} from \"../../runtime/secrets/proxy-holder\";\nimport {RdsHolder} from \"../../runtime/secrets/rds-holder\";\n\n// eslint-disable-next-line @typescript-eslint/no-var-requires\nconst synthetics = require('Synthetics');\n\nabstract class DatabaseCheck<T> {\n    readonly name: string;\n    readonly sql: string;\n    failed: boolean;\n\n    protected constructor(name: string, sql: string) {\n        this.name = name;\n        this.sql = sql;\n        this.failed = false;\n    }\n\n    abstract check(value: T): void;\n}\n\nclass BaseResponse {}\n\nclass CountResponse extends BaseResponse {\n    count: number;\n}\n\nclass CountDatabaseCheck extends DatabaseCheck<CountResponse> {\n    readonly minCount: number|null;\n    readonly maxCount: number|null;\n\n    constructor(name: string,\n        sql: string,\n        minCount: number|null,\n        maxCount: number|null) {\n        super(name, sql);\n\n        if (minCount == null && maxCount == null) {\n            throw new Error('no max or min given!');\n        }\n\n        this.minCount = minCount;\n        this.maxCount = maxCount;\n    }\n\n    check(value: CountResponse) {\n        if (!value) {\n            this.failed = true;\n            throw new Error('no return value');\n        } else {\n            if ('count' in value) {\n                if (this.minCount && value.count < this.minCount) {\n                    this.failed = true;\n                    throw new Error(`count was ${value.count}, minimum is ${this.minCount}`);\n                }\n                if (this.maxCount && value.count > this.maxCount) {\n                    this.failed = true;\n                    throw new Error(`count was ${value.count}, max is ${this.maxCount}`);\n                }\n            } else {\n                this.failed = true;\n\n                console.info(\"received \" + JSON.stringify(value));\n\n                throw new Error('no count available');\n            }\n        }\n    }\n}\n\nconst stepConfig = {\n    'continueOnStepFailure': true,\n    'screenshotOnStepStart': false,\n    'screenshotOnStepSuccess': false,\n    'screenshotOnStepFailure': false,\n};\n\nexport class DatabaseChecker {\n    credentialsFunction: (() => Promise<void>);\n    checks: DatabaseCheck<BaseResponse>[];\n\n    private constructor(credentialsFunction: (() => Promise<void>)) {\n        this.credentialsFunction = credentialsFunction;\n        this.checks = [];\n\n        synthetics.getConfiguration()\n            .disableRequestMetrics();\n\n        synthetics.getConfiguration()\n            .withFailedCanaryMetric(true);\n    }\n\n    static createForProxy() {\n        return new DatabaseChecker(() => new ProxyHolder(process.env.SECRET_ID as string).setCredentials());\n    }\n\n    static createForRds() {\n        return new DatabaseChecker(() => new RdsHolder(process.env.SECRET_ID as string).setCredentials());\n    }\n\n    one(name: string, sql: string) {\n        this.checks.push(new CountDatabaseCheck(name,\n            sql,\n            1,\n            1));\n\n        return this;\n    }\n\n    empty(name: string, sql: string) {\n        this.checks.push(new CountDatabaseCheck(name,\n            sql,\n            null,\n            0));\n\n        return this;\n    }\n\n    notEmpty(name: string, sql: string) {\n        this.checks.push(new CountDatabaseCheck(name,\n            sql,\n            1,\n            null));\n\n        return this;\n    }\n\n    async expect() {\n        if (!this.checks.length) {\n            throw new Error('No checks');\n        }\n\n        await this.credentialsFunction();\n        await inDatabaseReadonly(async (db: DTDatabase) => {\n            for (const check of this.checks) {\n                console.info(\"canary checking sql \" + check.sql);\n\n                const value = await db.oneOrNone(check.sql);\n                const checkFunction = () => {\n                    check.check(value);\n                };\n\n                synthetics.executeStep(check.name,\n                    checkFunction,\n                    stepConfig);\n            }\n        });\n\n        if (this.checks.some(check => check.failed)) {\n            throw new Error('Failed');\n        }\n\n        return 'OK';\n    }\n}\n"]}
@@ -1,46 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.UrlCanary = exports.ENV_SECRET = exports.ENV_HOSTNAME = exports.ENV_API_KEY = void 0;
4
- const canary_1 = require("./canary");
5
- exports.ENV_API_KEY = "apiKeyId";
6
- exports.ENV_HOSTNAME = "hostname";
7
- exports.ENV_SECRET = "secret";
8
- class UrlCanary extends canary_1.DigitrafficCanary {
9
- constructor(stack, role, params, secret) {
10
- const canaryName = `${params.name}-url`;
11
- const environmentVariables = {};
12
- environmentVariables[exports.ENV_HOSTNAME] = params.hostname;
13
- if (params.secret) {
14
- environmentVariables[exports.ENV_SECRET] = params.secret;
15
- }
16
- if (params.apiKeyId) {
17
- environmentVariables[exports.ENV_API_KEY] = params.apiKeyId;
18
- }
19
- if (secret) {
20
- secret.grantRead(role);
21
- }
22
- // the handler code is defined at the actual project using this
23
- super(stack, canaryName, role, params, environmentVariables);
24
- }
25
- static create(stack, role, publicApi, params) {
26
- return new UrlCanary(stack, role, { ...{
27
- handler: `${params.name}.handler`,
28
- hostname: publicApi.hostname(),
29
- apiKeyId: this.getApiKey(publicApi),
30
- }, ...params });
31
- }
32
- static getApiKey(publicApi) {
33
- const apiKeys = publicApi.apiKeyIds;
34
- if (apiKeys.length > 1) {
35
- console.info("rest api has more than one api key");
36
- }
37
- if (apiKeys.length === 0) {
38
- console.info("rest api has no api keys");
39
- return undefined;
40
- }
41
- // always use first api key
42
- return publicApi.apiKeyIds[0];
43
- }
44
- }
45
- exports.UrlCanary = UrlCanary;
46
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXJsLWNhbmFyeS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9hd3MvaW5mcmEvY2FuYXJpZXMvdXJsLWNhbmFyeS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFHQSxxQ0FBMkM7QUFNOUIsUUFBQSxXQUFXLEdBQUcsVUFBVSxDQUFDO0FBQ3pCLFFBQUEsWUFBWSxHQUFHLFVBQVUsQ0FBQztBQUMxQixRQUFBLFVBQVUsR0FBRyxRQUFRLENBQUM7QUFPbkMsTUFBYSxTQUFVLFNBQVEsMEJBQWlCO0lBQzVDLFlBQVksS0FBZ0IsRUFDeEIsSUFBVSxFQUNWLE1BQTJCLEVBQzNCLE1BQWdCO1FBQ2hCLE1BQU0sVUFBVSxHQUFHLEdBQUcsTUFBTSxDQUFDLElBQUksTUFBTSxDQUFDO1FBQ3hDLE1BQU0sb0JBQW9CLEdBQXNCLEVBQUUsQ0FBQztRQUNuRCxvQkFBb0IsQ0FBQyxvQkFBWSxDQUFDLEdBQUcsTUFBTSxDQUFDLFFBQVEsQ0FBQztRQUVyRCxJQUFJLE1BQU0sQ0FBQyxNQUFNLEVBQUU7WUFDZixvQkFBb0IsQ0FBQyxrQkFBVSxDQUFDLEdBQUcsTUFBTSxDQUFDLE1BQU0sQ0FBQztTQUNwRDtRQUVELElBQUksTUFBTSxDQUFDLFFBQVEsRUFBRTtZQUNqQixvQkFBb0IsQ0FBQyxtQkFBVyxDQUFDLEdBQUcsTUFBTSxDQUFDLFFBQVEsQ0FBQztTQUN2RDtRQUVELElBQUksTUFBTSxFQUFFO1lBQ1IsTUFBTSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsQ0FBQztTQUMxQjtRQUVELCtEQUErRDtRQUMvRCxLQUFLLENBQ0QsS0FBSyxFQUFFLFVBQVUsRUFBRSxJQUFJLEVBQUUsTUFBTSxFQUFFLG9CQUFvQixDQUN4RCxDQUFDO0lBQ04sQ0FBQztJQUVELE1BQU0sQ0FBQyxNQUFNLENBQUMsS0FBdUIsRUFDakMsSUFBVSxFQUNWLFNBQTZCLEVBQzdCLE1BQW9DO1FBQ3BDLE9BQU8sSUFBSSxTQUFTLENBQUMsS0FBSyxFQUFFLElBQUksRUFBRSxFQUFDLEdBQUc7Z0JBQ2xDLE9BQU8sRUFBRSxHQUFHLE1BQU0sQ0FBQyxJQUFJLFVBQVU7Z0JBQ2pDLFFBQVEsRUFBRSxTQUFTLENBQUMsUUFBUSxFQUFFO2dCQUM5QixRQUFRLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxTQUFTLENBQUM7YUFDdEMsRUFBRSxHQUFHLE1BQU0sRUFBd0IsQ0FBQyxDQUFDO0lBQzFDLENBQUM7SUFFRCxNQUFNLENBQUMsU0FBUyxDQUFDLFNBQTZCO1FBQzFDLE1BQU0sT0FBTyxHQUFHLFNBQVMsQ0FBQyxTQUFTLENBQUM7UUFFcEMsSUFBSSxPQUFPLENBQUMsTUFBTSxHQUFHLENBQUMsRUFBRTtZQUNwQixPQUFPLENBQUMsSUFBSSxDQUFDLG9DQUFvQyxDQUFDLENBQUM7U0FDdEQ7UUFFRCxJQUFJLE9BQU8sQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFO1lBQ3RCLE9BQU8sQ0FBQyxJQUFJLENBQUMsMEJBQTBCLENBQUMsQ0FBQztZQUN6QyxPQUFPLFNBQVMsQ0FBQztTQUNwQjtRQUVELDJCQUEyQjtRQUMzQixPQUFPLFNBQVMsQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDbEMsQ0FBQztDQUNKO0FBckRELDhCQXFEQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7Q29uc3RydWN0fSBmcm9tIFwiY29uc3RydWN0c1wiO1xuaW1wb3J0IHtDYW5hcnlQYXJhbWV0ZXJzfSBmcm9tIFwiLi9jYW5hcnktcGFyYW1ldGVyc1wiO1xuaW1wb3J0IHtSb2xlfSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWlhbVwiO1xuaW1wb3J0IHtEaWdpdHJhZmZpY0NhbmFyeX0gZnJvbSBcIi4vY2FuYXJ5XCI7XG5pbXBvcnQge0lTZWNyZXR9IGZyb20gXCJhd3MtY2RrLWxpYi9hd3Mtc2VjcmV0c21hbmFnZXJcIjtcbmltcG9ydCB7RGlnaXRyYWZmaWNTdGFja30gZnJvbSBcIi4uL3N0YWNrL3N0YWNrXCI7XG5pbXBvcnQge0xhbWJkYUVudmlyb25tZW50fSBmcm9tIFwiLi4vc3RhY2svbGFtYmRhLWNvbmZpZ3NcIjtcbmltcG9ydCB7RGlnaXRyYWZmaWNSZXN0QXBpfSBmcm9tIFwiLi4vc3RhY2svcmVzdF9hcGlzXCI7XG5cbmV4cG9ydCBjb25zdCBFTlZfQVBJX0tFWSA9IFwiYXBpS2V5SWRcIjtcbmV4cG9ydCBjb25zdCBFTlZfSE9TVE5BTUUgPSBcImhvc3RuYW1lXCI7XG5leHBvcnQgY29uc3QgRU5WX1NFQ1JFVCA9IFwic2VjcmV0XCI7XG5cbmV4cG9ydCBpbnRlcmZhY2UgVXJsQ2FuYXJ5UGFyYW1ldGVycyBleHRlbmRzIENhbmFyeVBhcmFtZXRlcnMge1xuICAgIHJlYWRvbmx5IGhvc3RuYW1lOiBzdHJpbmc7XG4gICAgcmVhZG9ubHkgYXBpS2V5SWQ/OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBjbGFzcyBVcmxDYW5hcnkgZXh0ZW5kcyBEaWdpdHJhZmZpY0NhbmFyeSB7XG4gICAgY29uc3RydWN0b3Ioc3RhY2s6IENvbnN0cnVjdCxcbiAgICAgICAgcm9sZTogUm9sZSxcbiAgICAgICAgcGFyYW1zOiBVcmxDYW5hcnlQYXJhbWV0ZXJzLFxuICAgICAgICBzZWNyZXQ/OiBJU2VjcmV0KSB7XG4gICAgICAgIGNvbnN0IGNhbmFyeU5hbWUgPSBgJHtwYXJhbXMubmFtZX0tdXJsYDtcbiAgICAgICAgY29uc3QgZW52aXJvbm1lbnRWYXJpYWJsZXM6IExhbWJkYUVudmlyb25tZW50ID0ge307XG4gICAgICAgIGVudmlyb25tZW50VmFyaWFibGVzW0VOVl9IT1NUTkFNRV0gPSBwYXJhbXMuaG9zdG5hbWU7XG5cbiAgICAgICAgaWYgKHBhcmFtcy5zZWNyZXQpIHtcbiAgICAgICAgICAgIGVudmlyb25tZW50VmFyaWFibGVzW0VOVl9TRUNSRVRdID0gcGFyYW1zLnNlY3JldDtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmIChwYXJhbXMuYXBpS2V5SWQpIHtcbiAgICAgICAgICAgIGVudmlyb25tZW50VmFyaWFibGVzW0VOVl9BUElfS0VZXSA9IHBhcmFtcy5hcGlLZXlJZDtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmIChzZWNyZXQpIHtcbiAgICAgICAgICAgIHNlY3JldC5ncmFudFJlYWQocm9sZSk7XG4gICAgICAgIH1cblxuICAgICAgICAvLyB0aGUgaGFuZGxlciBjb2RlIGlzIGRlZmluZWQgYXQgdGhlIGFjdHVhbCBwcm9qZWN0IHVzaW5nIHRoaXNcbiAgICAgICAgc3VwZXIoXG4gICAgICAgICAgICBzdGFjaywgY2FuYXJ5TmFtZSwgcm9sZSwgcGFyYW1zLCBlbnZpcm9ubWVudFZhcmlhYmxlcyxcbiAgICAgICAgKTtcbiAgICB9XG5cbiAgICBzdGF0aWMgY3JlYXRlKHN0YWNrOiBEaWdpdHJhZmZpY1N0YWNrLFxuICAgICAgICByb2xlOiBSb2xlLFxuICAgICAgICBwdWJsaWNBcGk6IERpZ2l0cmFmZmljUmVzdEFwaSxcbiAgICAgICAgcGFyYW1zOiBQYXJ0aWFsPFVybENhbmFyeVBhcmFtZXRlcnM+KTogVXJsQ2FuYXJ5IHtcbiAgICAgICAgcmV0dXJuIG5ldyBVcmxDYW5hcnkoc3RhY2ssIHJvbGUsIHsuLi57XG4gICAgICAgICAgICBoYW5kbGVyOiBgJHtwYXJhbXMubmFtZX0uaGFuZGxlcmAsXG4gICAgICAgICAgICBob3N0bmFtZTogcHVibGljQXBpLmhvc3RuYW1lKCksXG4gICAgICAgICAgICBhcGlLZXlJZDogdGhpcy5nZXRBcGlLZXkocHVibGljQXBpKSxcbiAgICAgICAgfSwgLi4ucGFyYW1zfSBhcyBVcmxDYW5hcnlQYXJhbWV0ZXJzKTtcbiAgICB9XG5cbiAgICBzdGF0aWMgZ2V0QXBpS2V5KHB1YmxpY0FwaTogRGlnaXRyYWZmaWNSZXN0QXBpKTogc3RyaW5nIHwgdW5kZWZpbmVkIHtcbiAgICAgICAgY29uc3QgYXBpS2V5cyA9IHB1YmxpY0FwaS5hcGlLZXlJZHM7XG5cbiAgICAgICAgaWYgKGFwaUtleXMubGVuZ3RoID4gMSkge1xuICAgICAgICAgICAgY29uc29sZS5pbmZvKFwicmVzdCBhcGkgaGFzIG1vcmUgdGhhbiBvbmUgYXBpIGtleVwiKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmIChhcGlLZXlzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgICAgICAgY29uc29sZS5pbmZvKFwicmVzdCBhcGkgaGFzIG5vIGFwaSBrZXlzXCIpO1xuICAgICAgICAgICAgcmV0dXJuIHVuZGVmaW5lZDtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIGFsd2F5cyB1c2UgZmlyc3QgYXBpIGtleVxuICAgICAgICByZXR1cm4gcHVibGljQXBpLmFwaUtleUlkc1swXTtcbiAgICB9XG59XG4iXX0=
@@ -1,238 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.HeaderChecker = exports.GeoJsonChecker = exports.ContentTypeChecker = exports.ContentChecker = exports.ResponseChecker = exports.UrlChecker = exports.API_KEY_HEADER = void 0;
4
- const http2_1 = require("http2");
5
- const asserter_1 = require("digitraffic-common/test/asserter");
6
- // eslint-disable-next-line @typescript-eslint/no-var-requires
7
- const synthetics = require('Synthetics');
8
- const zlib = require("zlib");
9
- const mediatypes_1 = require("../../types/mediatypes");
10
- const apikey_1 = require("../../runtime/apikey");
11
- const geometry_1 = require("../../../utils/geometry");
12
- exports.API_KEY_HEADER = "x-api-key";
13
- const baseHeaders = {
14
- "Digitraffic-User": "internal-digitraffic-canary",
15
- "Accept-Encoding": "gzip",
16
- "Accept": "*/*",
17
- };
18
- class UrlChecker {
19
- constructor(hostname, apiKey) {
20
- const headers = { ...baseHeaders };
21
- if (apiKey) {
22
- headers[exports.API_KEY_HEADER] = apiKey;
23
- }
24
- this.requestOptions = {
25
- hostname,
26
- method: 'GET',
27
- protocol: 'https:',
28
- headers,
29
- };
30
- synthetics.getConfiguration()
31
- .disableRequestMetrics();
32
- synthetics.getConfiguration()
33
- .withIncludeRequestBody(false)
34
- .withIncludeRequestHeaders(false)
35
- .withIncludeResponseBody(false)
36
- .withIncludeResponseHeaders(false)
37
- .withFailedCanaryMetric(true);
38
- }
39
- static create(hostname, apiKeyId) {
40
- return (0, apikey_1.getApiKeyFromAPIGateway)(apiKeyId).then(apiKey => {
41
- return new UrlChecker(hostname, apiKey.value);
42
- });
43
- }
44
- static createV2() {
45
- return this.create(process.env.hostname, process.env.apiKeyId);
46
- }
47
- expectStatus(statusCode, url, callback) {
48
- const requestOptions = { ...this.requestOptions, ...{
49
- path: url,
50
- } };
51
- return synthetics.executeHttpStep(`Verify ${statusCode} for ${url.replace(/auth=.*/, '')}`, requestOptions, callback);
52
- }
53
- expect200(url, ...callbacks) {
54
- const callback = async (json, body, res) => {
55
- await Promise.allSettled(callbacks.map(c => c(json, body, res)));
56
- };
57
- return this.expectStatus(200, url, callback);
58
- }
59
- expect404(url) {
60
- const requestOptions = { ...this.requestOptions, ...{
61
- path: url,
62
- } };
63
- return synthetics.executeHttpStep(`Verify 404 for ${url}`, requestOptions, validateStatusCodeAndContentType(404, mediatypes_1.MediaType.TEXT_PLAIN));
64
- }
65
- expect400(url) {
66
- const requestOptions = { ...this.requestOptions, ...{
67
- path: url,
68
- } };
69
- return synthetics.executeHttpStep(`Verify 400 for ${url}`, requestOptions, validateStatusCodeAndContentType(400, mediatypes_1.MediaType.TEXT_PLAIN));
70
- }
71
- expect403WithoutApiKey(url, mediaType) {
72
- if (!this.requestOptions.headers || !this.requestOptions.headers[exports.API_KEY_HEADER]) {
73
- console.error("No api key defined");
74
- }
75
- const requestOptions = { ...this.requestOptions, ...{
76
- path: url,
77
- headers: baseHeaders,
78
- } };
79
- return synthetics.executeHttpStep(`Verify 403 for ${url}`, requestOptions, validateStatusCodeAndContentType(403, mediaType || mediatypes_1.MediaType.APPLICATION_JSON));
80
- }
81
- done() {
82
- return "Canary successful";
83
- }
84
- }
85
- exports.UrlChecker = UrlChecker;
86
- async function getResponseBody(response) {
87
- const body = await getBodyFromResponse(response);
88
- if (response.headers[http2_1.constants.HTTP2_HEADER_CONTENT_ENCODING] === 'gzip') {
89
- try {
90
- return zlib.gunzipSync(body).toString();
91
- }
92
- catch (e) {
93
- console.info("error " + JSON.stringify(e));
94
- }
95
- }
96
- return body.toString();
97
- }
98
- function getBodyFromResponse(response) {
99
- return new Promise((resolve) => {
100
- const buffers = [];
101
- response.on('data', (data) => {
102
- buffers.push(data);
103
- });
104
- response.on('end', () => {
105
- resolve(Buffer.concat(buffers).toString());
106
- });
107
- });
108
- }
109
- /**
110
- * Returns function, that validates that the status code and content-type from response are the given values
111
- * @param statusCode
112
- * @param contentType
113
- */
114
- function validateStatusCodeAndContentType(statusCode, contentType) {
115
- return (res) => {
116
- return new Promise(resolve => {
117
- if (res.statusCode !== statusCode) {
118
- throw new Error(`${res.statusCode} ${res.statusMessage}`);
119
- }
120
- if (res.headers[http2_1.constants.HTTP2_HEADER_CONTENT_TYPE] !== contentType) {
121
- throw new Error('Wrong content-type ' + res.headers[http2_1.constants.HTTP2_HEADER_CONTENT_TYPE]);
122
- }
123
- resolve();
124
- });
125
- };
126
- }
127
- // DEPRECATED
128
- class ResponseChecker {
129
- constructor(contentType) {
130
- this.checkCors = true;
131
- this.contentType = contentType;
132
- }
133
- static forJson() {
134
- return new ResponseChecker(mediatypes_1.MediaType.APPLICATION_JSON);
135
- }
136
- static forCSV() {
137
- return new ResponseChecker(mediatypes_1.MediaType.TEXT_CSV);
138
- }
139
- static forGeojson() {
140
- return new ResponseChecker(mediatypes_1.MediaType.APPLICATION_GEOJSON);
141
- }
142
- static forJpeg() {
143
- return new ResponseChecker(mediatypes_1.MediaType.IMAGE_JPEG);
144
- }
145
- check() {
146
- return this.responseChecker(() => {
147
- // no need to do anything
148
- });
149
- }
150
- checkJson(fn) {
151
- return this.responseChecker((body, res) => {
152
- fn(JSON.parse(body), body, res);
153
- });
154
- }
155
- responseChecker(fn) {
156
- return async (res) => {
157
- if (!res.statusCode) {
158
- throw new Error('statusCode missing');
159
- }
160
- if (res.statusCode < 200 || res.statusCode > 299) {
161
- throw new Error(`${res.statusCode} ${res.statusMessage}`);
162
- }
163
- if (this.checkCors && !res.headers[http2_1.constants.HTTP2_HEADER_ACCESS_CONTROL_ALLOW_ORIGIN]) {
164
- throw new Error('CORS missing');
165
- }
166
- if (res.headers[http2_1.constants.HTTP2_HEADER_CONTENT_TYPE] !== this.contentType) {
167
- throw new Error('Wrong content-type ' + res.headers[http2_1.constants.HTTP2_HEADER_CONTENT_TYPE]);
168
- }
169
- const body = await getResponseBody(res);
170
- fn(body, res);
171
- };
172
- }
173
- }
174
- exports.ResponseChecker = ResponseChecker;
175
- class ContentChecker {
176
- static checkJson(fn) {
177
- return async (res) => {
178
- const body = await getResponseBody(res);
179
- fn(JSON.parse(body), body, res);
180
- };
181
- }
182
- static checkResponse(fn) {
183
- return async (res) => {
184
- const body = await getResponseBody(res);
185
- fn(body, res);
186
- };
187
- }
188
- }
189
- exports.ContentChecker = ContentChecker;
190
- class ContentTypeChecker {
191
- static checkContentType(contentType) {
192
- return (res) => {
193
- if (!res.statusCode) {
194
- throw new Error('statusCode missing');
195
- }
196
- if (res.statusCode < 200 || res.statusCode > 299) {
197
- throw new Error(`${res.statusCode} ${res.statusMessage}`);
198
- }
199
- if (!res.headers[http2_1.constants.HTTP2_HEADER_ACCESS_CONTROL_ALLOW_ORIGIN]) {
200
- throw new Error('CORS missing');
201
- }
202
- if (res.headers[http2_1.constants.HTTP2_HEADER_CONTENT_TYPE] !== contentType) {
203
- throw new Error('Wrong content-type ' + res.headers[http2_1.constants.HTTP2_HEADER_CONTENT_TYPE]);
204
- }
205
- };
206
- }
207
- }
208
- exports.ContentTypeChecker = ContentTypeChecker;
209
- class GeoJsonChecker {
210
- static validFeatureCollection(fn) {
211
- return ResponseChecker.forGeojson().checkJson((json) => {
212
- asserter_1.Asserter.assertEquals(json.type, 'FeatureCollection');
213
- asserter_1.Asserter.assertTrue((0, geometry_1.isValidGeoJson)(json));
214
- if (fn) {
215
- fn(json);
216
- }
217
- });
218
- }
219
- }
220
- exports.GeoJsonChecker = GeoJsonChecker;
221
- class HeaderChecker {
222
- static checkHeaderExists(headerName) {
223
- return (res) => {
224
- if (!res.headers[headerName]) {
225
- throw new Error('Missing header: ' + headerName);
226
- }
227
- };
228
- }
229
- static checkHeaderMissing(headerName) {
230
- return (res) => {
231
- if (res.headers[headerName]) {
232
- throw new Error('Header should not exist: ' + headerName);
233
- }
234
- };
235
- }
236
- }
237
- exports.HeaderChecker = HeaderChecker;
238
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"url-checker.js","sourceRoot":"","sources":["../../../../../src/aws/infra/canaries/url-checker.ts"],"names":[],"mappings":"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","sourcesContent":["import {constants} from \"http2\";\nimport {IncomingMessage, RequestOptions} from \"http\";\nimport {Asserter} from \"digitraffic-common/test/asserter\";\n\n// eslint-disable-next-line @typescript-eslint/no-var-requires\nconst synthetics = require('Synthetics');\nimport zlib = require('zlib');\nimport {MediaType} from \"../../types/mediatypes\";\nimport {getApiKeyFromAPIGateway} from \"../../runtime/apikey\";\nimport {FeatureCollection} from \"geojson\";\nimport {isValidGeoJson} from \"../../../utils/geometry\";\n\nexport const API_KEY_HEADER = \"x-api-key\";\n\nconst baseHeaders = {\n    \"Digitraffic-User\" : \"internal-digitraffic-canary\",\n    \"Accept-Encoding\" : \"gzip\",\n    \"Accept\": \"*/*\",\n} as Record<string, string>;\n\ntype CheckerFunction = (Res: IncomingMessage) => void;\ntype JsonCheckerFunction<T> = (json: T, body: string, message: IncomingMessage) => void;\n\nexport class UrlChecker {\n    private readonly requestOptions: RequestOptions;\n\n    constructor(hostname: string, apiKey?: string) {\n        const headers = {...baseHeaders};\n\n        if (apiKey) {\n            headers[API_KEY_HEADER] = apiKey;\n        }\n\n        this.requestOptions = {\n            hostname,\n            method: 'GET',\n            protocol: 'https:',\n            headers,\n        };\n\n        synthetics.getConfiguration()\n            .disableRequestMetrics();\n\n        synthetics.getConfiguration()\n            .withIncludeRequestBody(false)\n            .withIncludeRequestHeaders(false)\n            .withIncludeResponseBody(false)\n            .withIncludeResponseHeaders(false)\n            .withFailedCanaryMetric(true);\n    }\n\n    static create(hostname: string, apiKeyId: string): Promise<UrlChecker> {\n        return getApiKeyFromAPIGateway(apiKeyId).then(apiKey => {\n            return new UrlChecker(hostname, apiKey.value);\n        });\n    }\n\n    static createV2(): Promise<UrlChecker> {\n        return this.create(process.env.hostname as string, process.env.apiKeyId as string);\n    }\n\n    expectStatus<T>(statusCode: number, url: string, callback: JsonCheckerFunction<T>): Promise<void> {\n        const requestOptions = {...this.requestOptions, ...{\n            path: url,\n        }};\n\n        return synthetics.executeHttpStep(`Verify ${statusCode} for ${url.replace(/auth=.*/, '')}`,\n            requestOptions,\n            callback);\n    }\n\n    expect200<T>(url: string, ...callbacks: JsonCheckerFunction<T>[]): Promise<void> {\n        const callback = async (json: T, body: string, res: IncomingMessage) => {\n            await Promise.allSettled(callbacks.map(c => c(json, body, res)));\n        };\n\n        return this.expectStatus(200, url, callback);\n    }\n\n    expect404(url: string): Promise<void> {\n        const requestOptions = {...this.requestOptions, ...{\n            path: url,\n        }};\n\n        return synthetics.executeHttpStep(`Verify 404 for ${url}`, requestOptions, validateStatusCodeAndContentType(404, MediaType.TEXT_PLAIN));\n    }\n\n    expect400(url: string): Promise<void> {\n        const requestOptions = {...this.requestOptions, ...{\n            path: url,\n        }};\n\n        return synthetics.executeHttpStep(`Verify 400 for ${url}`, requestOptions, validateStatusCodeAndContentType(400, MediaType.TEXT_PLAIN));\n    }\n\n    expect403WithoutApiKey(url: string, mediaType?: MediaType): Promise<void> {\n        if (!this.requestOptions.headers || !this.requestOptions.headers[API_KEY_HEADER]) {\n            console.error(\"No api key defined\");\n        }\n\n        const requestOptions = {...this.requestOptions, ...{\n            path: url,\n            headers: baseHeaders,\n        }};\n\n        return synthetics.executeHttpStep(`Verify 403 for ${url}`,\n            requestOptions,\n            validateStatusCodeAndContentType(403, mediaType || MediaType.APPLICATION_JSON));\n    }\n\n    done(): string {\n        return \"Canary successful\";\n    }\n}\n\nasync function getResponseBody(response: IncomingMessage): Promise<string> {\n    const body = await getBodyFromResponse(response);\n\n    if (response.headers[constants.HTTP2_HEADER_CONTENT_ENCODING] === 'gzip') {\n        try {\n            return zlib.gunzipSync(body).toString();\n        } catch (e) {\n            console.info(\"error \" + JSON.stringify(e));\n        }\n    }\n\n    return body.toString();\n}\n\nfunction getBodyFromResponse(response: IncomingMessage): Promise<string> {\n    return new Promise((resolve: ((value: string) => void)) => {\n        const buffers: Buffer[] = [];\n\n        response.on('data', (data: Buffer) => {\n            buffers.push(data);\n        });\n\n        response.on('end', () => {\n            resolve(Buffer.concat(buffers).toString());\n        });\n    });\n}\n\n/**\n * Returns function, that validates that the status code and content-type from response are the given values\n * @param statusCode\n * @param contentType\n */\nfunction validateStatusCodeAndContentType(statusCode: number, contentType: MediaType): (Res: IncomingMessage) => Promise<void> {\n    return (res: IncomingMessage) => {\n        return new Promise(resolve => {\n            if (res.statusCode !== statusCode) {\n                throw new Error(`${res.statusCode} ${res.statusMessage}`);\n            }\n\n            if (res.headers[constants.HTTP2_HEADER_CONTENT_TYPE] !== contentType) {\n                throw new Error('Wrong content-type ' + res.headers[constants.HTTP2_HEADER_CONTENT_TYPE]);\n            }\n\n            resolve();\n        });\n    };\n}\n\n// DEPRECATED\nexport class ResponseChecker {\n    private readonly contentType;\n    private checkCors = true;\n\n    constructor(contentType: string) {\n        this.contentType = contentType;\n    }\n\n    static forJson(): ResponseChecker {\n        return new ResponseChecker(MediaType.APPLICATION_JSON);\n    }\n\n    static forCSV(): ResponseChecker {\n        return new ResponseChecker(MediaType.TEXT_CSV);\n    }\n\n    static forGeojson(): ResponseChecker {\n        return new ResponseChecker(MediaType.APPLICATION_GEOJSON);\n    }\n\n    static forJpeg(): ResponseChecker {\n        return new ResponseChecker(MediaType.IMAGE_JPEG);\n    }\n\n    check(): CheckerFunction {\n        return this.responseChecker(() => {\n            // no need to do anything\n        });\n    }\n\n    checkJson<T>(fn: (json: T, body: string, res: IncomingMessage) => void): CheckerFunction {\n        return this.responseChecker((body: string, res: IncomingMessage) => {\n            fn(JSON.parse(body), body, res);\n        });\n    }\n\n    responseChecker(fn: (body: string, res: IncomingMessage) => void): CheckerFunction {\n        return async (res: IncomingMessage): Promise<void> => {\n            if (!res.statusCode) {\n                throw new Error('statusCode missing');\n            }\n\n            if (res.statusCode < 200 || res.statusCode > 299) {\n                throw new Error(`${res.statusCode} ${res.statusMessage}`);\n            }\n\n            if (this.checkCors && !res.headers[constants.HTTP2_HEADER_ACCESS_CONTROL_ALLOW_ORIGIN]) {\n                throw new Error('CORS missing');\n            }\n\n            if (res.headers[constants.HTTP2_HEADER_CONTENT_TYPE] !== this.contentType) {\n                throw new Error('Wrong content-type ' + res.headers[constants.HTTP2_HEADER_CONTENT_TYPE]);\n            }\n\n            const body = await getResponseBody(res);\n\n            fn(body, res);\n        };\n    }\n}\n\nexport class ContentChecker {\n    static checkJson<T>(fn: (json: T, body: string, res: IncomingMessage) => void): CheckerFunction {\n        return async (res: IncomingMessage): Promise<void> => {\n            const body = await getResponseBody(res);\n\n            fn(JSON.parse(body), body, res);\n        };\n    }\n\n    static checkResponse(fn: (body: string, res: IncomingMessage) => void): CheckerFunction {\n        return async (res: IncomingMessage): Promise<void> => {\n            const body = await getResponseBody(res);\n\n            fn(body, res);\n        };\n    }\n}\n\nexport class ContentTypeChecker {\n    static checkContentType(contentType: MediaType) {\n        return (res: IncomingMessage) => {\n            if (!res.statusCode) {\n                throw new Error('statusCode missing');\n            }\n\n            if (res.statusCode < 200 || res.statusCode > 299) {\n                throw new Error(`${res.statusCode} ${res.statusMessage}`);\n            }\n\n            if (!res.headers[constants.HTTP2_HEADER_ACCESS_CONTROL_ALLOW_ORIGIN]) {\n                throw new Error('CORS missing');\n            }\n\n            if (res.headers[constants.HTTP2_HEADER_CONTENT_TYPE] !== contentType) {\n                throw new Error('Wrong content-type ' + res.headers[constants.HTTP2_HEADER_CONTENT_TYPE]);\n            }\n        };\n    }\n}\n\nexport class GeoJsonChecker {\n    static validFeatureCollection(fn?: (json: FeatureCollection) => void): CheckerFunction {\n        return ResponseChecker.forGeojson().checkJson((json: FeatureCollection) => {\n            Asserter.assertEquals(json.type, 'FeatureCollection');\n            Asserter.assertTrue(isValidGeoJson(json));\n\n            if (fn) {\n                fn(json);\n            }\n        });\n    }\n}\n\nexport class HeaderChecker {\n    static checkHeaderExists(headerName: string): CheckerFunction {\n        return (res: IncomingMessage) => {\n            if (!res.headers[headerName]) {\n                throw new Error('Missing header: ' + headerName);\n            }\n        };\n    }\n\n    static checkHeaderMissing(headerName: string): CheckerFunction {\n        return (res: IncomingMessage) => {\n            if (res.headers[headerName]) {\n                throw new Error('Header should not exist: ' + headerName);\n            }\n        };\n    }\n}\n"]}