@liflig/cdk 2.24.0 → 2.25.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/alarms/database-alarms.js +29 -30
- package/lib/alarms/index.d.ts +6 -3
- package/lib/alarms/index.js +4 -10
- package/lib/alarms/service-alarms.js +38 -38
- package/lib/alarms/ses-alarms.js +16 -20
- package/lib/alarms/slack-alarm.js +18 -17
- package/lib/bastion-host.js +12 -15
- package/lib/bin/cdk-create-snapshots.js +3 -5
- package/lib/bin/fetch-pipeline-variables.js +4 -6
- package/lib/build-artifacts/github-actions-role.js +9 -14
- package/lib/build-artifacts/index.js +25 -26
- package/lib/cdk-pipelines/cloud-assembly-lookup-handler.js +13 -17
- package/lib/cdk-pipelines/index.d.ts +4 -2
- package/lib/cdk-pipelines/index.js +4 -10
- package/lib/cdk-pipelines/liflig-cdk-pipeline.js +29 -24
- package/lib/cdk-pipelines/slack-notification.js +12 -14
- package/lib/cdk-pipelines/variables.js +12 -18
- package/lib/cloudtrail-slack-integration/cloudtrail-slack-integration.js +16 -17
- package/lib/cloudtrail-slack-integration/index.d.ts +2 -1
- package/lib/cloudtrail-slack-integration/index.js +2 -6
- package/lib/configure-parameters/configure-parameters.js +12 -11
- package/lib/configure-parameters/index.d.ts +2 -1
- package/lib/configure-parameters/index.js +2 -6
- package/lib/cross-region-ssm-parameter.js +4 -8
- package/lib/ecs/cluster.js +11 -13
- package/lib/ecs/fargate-service.js +25 -25
- package/lib/ecs/index.d.ts +6 -3
- package/lib/ecs/index.js +4 -10
- package/lib/ecs/listener-rule.js +13 -10
- package/lib/feature-flags.js +5 -10
- package/lib/hosted-zone-with-param.js +15 -16
- package/lib/index.js +22 -56
- package/lib/kinesis/index.d.ts +2 -1
- package/lib/kinesis/index.js +2 -6
- package/lib/kinesis/kinesis-to-datadog-stream.js +11 -15
- package/lib/load-balancer/index.d.ts +2 -1
- package/lib/load-balancer/index.js +2 -6
- package/lib/load-balancer/load-balancer.js +12 -13
- package/lib/platform/index.d.ts +2 -1
- package/lib/platform/index.js +2 -7
- package/lib/platform/platform.js +9 -10
- package/lib/rds/database.js +15 -15
- package/lib/rds/index.d.ts +2 -1
- package/lib/rds/index.js +2 -6
- package/lib/ses/configurationsetdeliveryoptions/index.js +5 -9
- package/lib/ses/configurationsetsnsdestination/handler.js +8 -12
- package/lib/ses/configurationsetsnsdestination/index.js +22 -19
- package/lib/ses/index.d.ts +2 -1
- package/lib/ses/index.js +5 -12
- package/lib/ses/sesdomain/handler.js +11 -16
- package/lib/ses/sesdomain/index.js +24 -19
- package/lib/ses/sesverifyemail/handler.js +6 -10
- package/lib/ses/sesverifyemail/index.js +19 -14
- package/lib/snapshots.js +13 -19
- package/lib/ssm-parameter-backed-resource.js +15 -15
- package/lib/ssm-parameter-reader.js +5 -10
- package/lib/tags.js +4 -7
- package/lib/utils.js +3 -6
- package/lib/webapp/index.d.ts +2 -1
- package/lib/webapp/index.js +4 -10
- package/lib/webapp/monitor.js +32 -33
- package/lib/webapp/security-headers.js +7 -11
- package/lib/webapp/webapp.js +20 -22
- package/lib/webapp-deploy-via-role.js +7 -11
- package/package.json +20 -20
|
@@ -1,17 +1,19 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
1
|
+
import * as constructs from "constructs";
|
|
2
|
+
import * as iam from "aws-cdk-lib/aws-iam";
|
|
3
|
+
import * as lambda from "aws-cdk-lib/aws-lambda";
|
|
4
|
+
import * as cdk from "aws-cdk-lib";
|
|
5
|
+
import * as cr from "aws-cdk-lib/custom-resources";
|
|
6
|
+
import { RetentionDays } from "aws-cdk-lib/aws-logs";
|
|
7
|
+
import * as snsSubscriptions from "aws-cdk-lib/aws-sns-subscriptions";
|
|
8
|
+
import { NodejsFunction } from "aws-cdk-lib/aws-lambda-nodejs";
|
|
9
|
+
import * as path from "path";
|
|
10
|
+
import { fileURLToPath } from "node:url";
|
|
11
|
+
import { createRequire } from "node:module";
|
|
12
|
+
const require = createRequire(import.meta.url);
|
|
13
|
+
const __filename = fileURLToPath(import.meta.url);
|
|
14
|
+
const __dirname = path.dirname(__filename);
|
|
15
|
+
export class ConfigurationSetSnsDestination extends constructs.Construct {
|
|
13
16
|
constructor(scope, id, props) {
|
|
14
|
-
var _a;
|
|
15
17
|
super(scope, id);
|
|
16
18
|
new cdk.CustomResource(this, "Resource", {
|
|
17
19
|
serviceToken: ConfigurationSetSnsDestinationProvider.getOrCreate(this).serviceToken,
|
|
@@ -27,14 +29,13 @@ class ConfigurationSetSnsDestination extends constructs.Construct {
|
|
|
27
29
|
code: new lambda.InlineCode(`exports.handler = ${sesEventLoggerHandler.toString()};`),
|
|
28
30
|
handler: "index.handler",
|
|
29
31
|
runtime: lambda.Runtime.NODEJS_18_X,
|
|
30
|
-
logRetention:
|
|
32
|
+
logRetention: RetentionDays.THREE_MONTHS,
|
|
31
33
|
});
|
|
32
|
-
if (
|
|
34
|
+
if (props.logEvents ?? true) {
|
|
33
35
|
props.snsTopic.addSubscription(new snsSubscriptions.LambdaSubscription(sesEventLoggerFunction));
|
|
34
36
|
}
|
|
35
37
|
}
|
|
36
38
|
}
|
|
37
|
-
exports.ConfigurationSetSnsDestination = ConfigurationSetSnsDestination;
|
|
38
39
|
const sesEventLoggerHandler = (event) => {
|
|
39
40
|
event.Records.forEach((record) => console.log(`SES event message from SNS: ${record.Sns.Message}`));
|
|
40
41
|
};
|
|
@@ -48,11 +49,13 @@ class ConfigurationSetSnsDestinationProvider extends constructs.Construct {
|
|
|
48
49
|
return (stack.node.tryFindChild(id) ||
|
|
49
50
|
new ConfigurationSetSnsDestinationProvider(stack, id));
|
|
50
51
|
}
|
|
52
|
+
provider;
|
|
53
|
+
serviceToken;
|
|
51
54
|
constructor(scope, id) {
|
|
52
55
|
super(scope, id);
|
|
53
56
|
this.provider = new cr.Provider(this, "Provider", {
|
|
54
|
-
onEventHandler: new
|
|
55
|
-
entry: require.resolve(
|
|
57
|
+
onEventHandler: new NodejsFunction(this, "Function", {
|
|
58
|
+
entry: require.resolve(`${__dirname}/handler`),
|
|
56
59
|
runtime: lambda.Runtime.NODEJS_18_X,
|
|
57
60
|
timeout: cdk.Duration.minutes(5),
|
|
58
61
|
awsSdkConnectionReuse: false,
|
|
@@ -72,4 +75,4 @@ class ConfigurationSetSnsDestinationProvider extends constructs.Construct {
|
|
|
72
75
|
this.serviceToken = this.provider.serviceToken;
|
|
73
76
|
}
|
|
74
77
|
}
|
|
75
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
78
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/ses/configurationsetsnsdestination/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport * as sns from \"aws-cdk-lib/aws-sns\"\nimport * as cdk from \"aws-cdk-lib\"\nimport * as cr from \"aws-cdk-lib/custom-resources\"\nimport { RetentionDays } from \"aws-cdk-lib/aws-logs\"\nimport * as snsSubscriptions from \"aws-cdk-lib/aws-sns-subscriptions\"\nimport { SNSHandler } from \"aws-lambda\"\nimport { NodejsFunction } from \"aws-cdk-lib/aws-lambda-nodejs\"\nimport * as path from \"path\"\nimport { fileURLToPath } from \"node:url\"\nimport { createRequire } from \"node:module\"\n\nconst require = createRequire(import.meta.url)\nconst __filename = fileURLToPath(import.meta.url)\nconst __dirname = path.dirname(__filename)\n\nexport type ConfigurationSetSnsDestinationEventType =\n  | \"SEND\"\n  | \"REJECT\"\n  | \"BOUNCE\"\n  | \"COMPLAINT\"\n  | \"DELIVERY\"\n  | \"OPEN\"\n  | \"CLICK\"\n  | \"RENDERING_FAILURE\"\n  | \"DELIVERY_DELAY\"\n  | \"SUBSCRIPTION\"\n\nexport interface ConfigurationSetSnsDestinationProps {\n  /**\n   * Whether SES events will be logged to CloudWatch\n   * @default true\n   */\n  logEvents?: boolean\n  /**\n   * The SES configuration set name\n   */\n  configurationSetName: string\n  /**\n   * The SES configuration set event destination name\n   */\n  eventDestinationName: string\n  /**\n   * SNS topic to send bounces to\n   */\n  snsTopic: sns.ITopic\n  /**\n   * Event types to match\n   */\n  matchingEventTypes: ConfigurationSetSnsDestinationEventType[]\n}\n\nexport class ConfigurationSetSnsDestination extends constructs.Construct {\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: ConfigurationSetSnsDestinationProps,\n  ) {\n    super(scope, id)\n\n    new cdk.CustomResource(this, \"Resource\", {\n      serviceToken:\n        ConfigurationSetSnsDestinationProvider.getOrCreate(this).serviceToken,\n      properties: {\n        ConfigurationSetName: props.configurationSetName,\n        EventDestinationName: props.eventDestinationName,\n        SnsTopicArn: props.snsTopic.topicArn,\n        MatchingEventTypes: props.matchingEventTypes,\n        Serial: 1,\n      },\n    })\n\n    const sesEventLoggerFunction = new lambda.Function(this, \"EventsHandler\", {\n      code: new lambda.InlineCode(\n        `exports.handler = ${sesEventLoggerHandler.toString()};`,\n      ),\n      handler: \"index.handler\",\n      runtime: lambda.Runtime.NODEJS_18_X,\n      logRetention: RetentionDays.THREE_MONTHS,\n    })\n\n    if (props.logEvents ?? true) {\n      props.snsTopic.addSubscription(\n        new snsSubscriptions.LambdaSubscription(sesEventLoggerFunction),\n      )\n    }\n  }\n}\n\nconst sesEventLoggerHandler: SNSHandler = (event) => {\n  event.Records.forEach((record) =>\n    console.log(`SES event message from SNS: ${record.Sns.Message}`),\n  )\n}\n\nclass ConfigurationSetSnsDestinationProvider extends constructs.Construct {\n  /**\n   * Returns the singleton provider.\n   */\n  public static getOrCreate(scope: constructs.Construct) {\n    const stack = cdk.Stack.of(scope)\n    const id = \"liflig-cdk.configuration-set-sns-destination\"\n    return (\n      (stack.node.tryFindChild(id) as ConfigurationSetSnsDestinationProvider) ||\n      new ConfigurationSetSnsDestinationProvider(stack, id)\n    )\n  }\n\n  private readonly provider: cr.Provider\n  public readonly serviceToken: string\n\n  constructor(scope: constructs.Construct, id: string) {\n    super(scope, id)\n\n    this.provider = new cr.Provider(this, \"Provider\", {\n      onEventHandler: new NodejsFunction(this, \"Function\", {\n        entry: require.resolve(`${__dirname}/handler`),\n        runtime: lambda.Runtime.NODEJS_18_X,\n        timeout: cdk.Duration.minutes(5),\n        awsSdkConnectionReuse: false,\n        initialPolicy: [\n          new iam.PolicyStatement({\n            actions: [\n              \"ses:CreateConfigurationSetEventDestination\",\n              \"ses:UpdateConfigurationSetEventDestination\",\n              \"ses:DeleteConfigurationSetEventDestination\",\n              \"ses:GetConfigurationSetEventDestinations\",\n            ],\n            resources: [\"*\"],\n          }),\n        ],\n      }),\n    })\n\n    this.serviceToken = this.provider.serviceToken\n  }\n}\n"]}
|
package/lib/ses/index.d.ts
CHANGED
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
export { SesDomain } from "./sesdomain";
|
|
2
2
|
export { SesVerifyEmail } from "./sesverifyemail";
|
|
3
|
-
export {
|
|
3
|
+
export { ConfigurationSetSnsDestination } from "./configurationsetsnsdestination";
|
|
4
4
|
export { ConfigurationSetDeliveryOptions } from "./configurationsetdeliveryoptions";
|
|
5
|
+
export type { ConfigurationSetSnsDestinationProps, ConfigurationSetSnsDestinationEventType, } from "./configurationsetsnsdestination";
|
package/lib/ses/index.js
CHANGED
|
@@ -1,12 +1,5 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
var sesverifyemail_1 = require("./sesverifyemail");
|
|
7
|
-
Object.defineProperty(exports, "SesVerifyEmail", { enumerable: true, get: function () { return sesverifyemail_1.SesVerifyEmail; } });
|
|
8
|
-
var configurationsetsnsdestination_1 = require("./configurationsetsnsdestination");
|
|
9
|
-
Object.defineProperty(exports, "ConfigurationSetSnsDestination", { enumerable: true, get: function () { return configurationsetsnsdestination_1.ConfigurationSetSnsDestination; } });
|
|
10
|
-
var configurationsetdeliveryoptions_1 = require("./configurationsetdeliveryoptions");
|
|
11
|
-
Object.defineProperty(exports, "ConfigurationSetDeliveryOptions", { enumerable: true, get: function () { return configurationsetdeliveryoptions_1.ConfigurationSetDeliveryOptions; } });
|
|
12
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc2VzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLHlDQUF1QztBQUE5QixzR0FBQSxTQUFTLE9BQUE7QUFDbEIsbURBQWlEO0FBQXhDLGdIQUFBLGNBQWMsT0FBQTtBQUN2QixtRkFJeUM7QUFEdkMsZ0pBQUEsOEJBQThCLE9BQUE7QUFFaEMscUZBQW1GO0FBQTFFLGtKQUFBLCtCQUErQixPQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgU2VzRG9tYWluIH0gZnJvbSBcIi4vc2VzZG9tYWluXCJcbmV4cG9ydCB7IFNlc1ZlcmlmeUVtYWlsIH0gZnJvbSBcIi4vc2VzdmVyaWZ5ZW1haWxcIlxuZXhwb3J0IHtcbiAgQ29uZmlndXJhdGlvblNldFNuc0Rlc3RpbmF0aW9uUHJvcHMsXG4gIENvbmZpZ3VyYXRpb25TZXRTbnNEZXN0aW5hdGlvbkV2ZW50VHlwZSxcbiAgQ29uZmlndXJhdGlvblNldFNuc0Rlc3RpbmF0aW9uLFxufSBmcm9tIFwiLi9jb25maWd1cmF0aW9uc2V0c25zZGVzdGluYXRpb25cIlxuZXhwb3J0IHsgQ29uZmlndXJhdGlvblNldERlbGl2ZXJ5T3B0aW9ucyB9IGZyb20gXCIuL2NvbmZpZ3VyYXRpb25zZXRkZWxpdmVyeW9wdGlvbnNcIlxuIl19
|
|
1
|
+
export { SesDomain } from "./sesdomain";
|
|
2
|
+
export { SesVerifyEmail } from "./sesverifyemail";
|
|
3
|
+
export { ConfigurationSetSnsDestination } from "./configurationsetsnsdestination";
|
|
4
|
+
export { ConfigurationSetDeliveryOptions } from "./configurationsetdeliveryoptions";
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc2VzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxhQUFhLENBQUE7QUFDdkMsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLGtCQUFrQixDQUFBO0FBQ2pELE9BQU8sRUFBRSw4QkFBOEIsRUFBRSxNQUFNLGtDQUFrQyxDQUFBO0FBQ2pGLE9BQU8sRUFBRSwrQkFBK0IsRUFBRSxNQUFNLG1DQUFtQyxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgU2VzRG9tYWluIH0gZnJvbSBcIi4vc2VzZG9tYWluXCJcbmV4cG9ydCB7IFNlc1ZlcmlmeUVtYWlsIH0gZnJvbSBcIi4vc2VzdmVyaWZ5ZW1haWxcIlxuZXhwb3J0IHsgQ29uZmlndXJhdGlvblNldFNuc0Rlc3RpbmF0aW9uIH0gZnJvbSBcIi4vY29uZmlndXJhdGlvbnNldHNuc2Rlc3RpbmF0aW9uXCJcbmV4cG9ydCB7IENvbmZpZ3VyYXRpb25TZXREZWxpdmVyeU9wdGlvbnMgfSBmcm9tIFwiLi9jb25maWd1cmF0aW9uc2V0ZGVsaXZlcnlvcHRpb25zXCJcbmV4cG9ydCB0eXBlIHtcbiAgQ29uZmlndXJhdGlvblNldFNuc0Rlc3RpbmF0aW9uUHJvcHMsXG4gIENvbmZpZ3VyYXRpb25TZXRTbnNEZXN0aW5hdGlvbkV2ZW50VHlwZSxcbn0gZnJvbSBcIi4vY29uZmlndXJhdGlvbnNldHNuc2Rlc3RpbmF0aW9uXCJcbiJdfQ==
|
|
@@ -1,12 +1,8 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
const
|
|
5
|
-
const
|
|
6
|
-
const handler = async (event) => {
|
|
7
|
-
var _a;
|
|
8
|
-
const sesClient = new client_ses_1.SESClient();
|
|
9
|
-
const sesv2Client = new client_sesv2_1.SESv2Client();
|
|
1
|
+
import { SESClient, VerifyDomainIdentityCommand, VerifyDomainDkimCommand, DeleteIdentityCommand, } from "@aws-sdk/client-ses";
|
|
2
|
+
import { SESv2Client, PutEmailIdentityConfigurationSetAttributesCommand, } from "@aws-sdk/client-sesv2";
|
|
3
|
+
export const handler = async (event) => {
|
|
4
|
+
const sesClient = new SESClient();
|
|
5
|
+
const sesv2Client = new SESv2Client();
|
|
10
6
|
const ttl = "1800";
|
|
11
7
|
const domainName = event.ResourceProperties["DomainName"];
|
|
12
8
|
const includeVerificationRecord = event.ResourceProperties["IncludeVerificationRecord"] == "true";
|
|
@@ -36,7 +32,7 @@ const handler = async (event) => {
|
|
|
36
32
|
}
|
|
37
33
|
switch (event.RequestType) {
|
|
38
34
|
case "Delete":
|
|
39
|
-
const deleteIdentityResp = await sesClient.send(new
|
|
35
|
+
const deleteIdentityResp = await sesClient.send(new DeleteIdentityCommand({
|
|
40
36
|
Identity: domainName,
|
|
41
37
|
}));
|
|
42
38
|
console.log(`ses.deleteIdentity: ${JSON.stringify(deleteIdentityResp)}`);
|
|
@@ -46,7 +42,7 @@ const handler = async (event) => {
|
|
|
46
42
|
case "Create":
|
|
47
43
|
case "Update":
|
|
48
44
|
// Idempotent.
|
|
49
|
-
const verifyDomainIdentityResp = await sesClient.send(new
|
|
45
|
+
const verifyDomainIdentityResp = await sesClient.send(new VerifyDomainIdentityCommand({
|
|
50
46
|
Domain: domainName,
|
|
51
47
|
}));
|
|
52
48
|
console.log(`ses.verifyDomainIdentity: ${JSON.stringify(verifyDomainIdentityResp)}`);
|
|
@@ -55,16 +51,16 @@ const handler = async (event) => {
|
|
|
55
51
|
throw new Error("Verification token not returned");
|
|
56
52
|
}
|
|
57
53
|
// Idempotent.
|
|
58
|
-
const verifyDomainDkimResp = await sesClient.send(new
|
|
54
|
+
const verifyDomainDkimResp = await sesClient.send(new VerifyDomainDkimCommand({
|
|
59
55
|
Domain: domainName,
|
|
60
56
|
}));
|
|
61
57
|
console.log(`ses.verifyDomainDkim: ${JSON.stringify(verifyDomainDkimResp)}`);
|
|
62
|
-
const dkimTokens =
|
|
58
|
+
const dkimTokens = verifyDomainDkimResp.DkimTokens ?? [];
|
|
63
59
|
if (!dkimTokens) {
|
|
64
60
|
throw new Error("DKIM tokens not returned");
|
|
65
61
|
}
|
|
66
62
|
// Idempotent.
|
|
67
|
-
const putEmailIdentityConfigResp = await sesv2Client.send(new
|
|
63
|
+
const putEmailIdentityConfigResp = await sesv2Client.send(new PutEmailIdentityConfigurationSetAttributesCommand({
|
|
68
64
|
EmailIdentity: domainName,
|
|
69
65
|
ConfigurationSetName: defaultConfigurationSetName,
|
|
70
66
|
}));
|
|
@@ -78,5 +74,4 @@ const handler = async (event) => {
|
|
|
78
74
|
};
|
|
79
75
|
}
|
|
80
76
|
};
|
|
81
|
-
|
|
82
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handler.js","sourceRoot":"","sources":["../../../src/ses/sesdomain/handler.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  SESClient,\n  VerifyDomainIdentityCommand,\n  VerifyDomainDkimCommand,\n  DeleteIdentityCommand,\n} from \"@aws-sdk/client-ses\"\n\nimport {\n  SESv2Client,\n  PutEmailIdentityConfigurationSetAttributesCommand,\n} from \"@aws-sdk/client-sesv2\"\n\ntype OnEventHandler = (event: {\n  PhysicalResourceId?: string\n  RequestType: \"Create\" | \"Update\" | \"Delete\"\n  ResourceProperties: Record<string, string>\n}) => Promise<{\n  PhysicalResourceId?: string\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  Data?: Record<string, any>\n}>\n\n// Type to mach CloudFormation AWS::Route53::RecordSetGroup RecordSet\ninterface RecordSetProperty {\n  Name: string\n  Type: string\n  ResourceRecords: string[]\n  TTL: string\n}\n\nexport const handler: OnEventHandler = async (event) => {\n  const sesClient = new SESClient()\n  const sesv2Client = new SESv2Client()\n\n  const ttl = \"1800\"\n\n  const domainName = event.ResourceProperties[\"DomainName\"]\n  const includeVerificationRecord =\n    event.ResourceProperties[\"IncludeVerificationRecord\"] == \"true\"\n\n  const defaultConfigurationSetName =\n    event.ResourceProperties[\"DefaultConfigurationSetName\"]\n\n  if (!includeVerificationRecord) {\n    console.log(\"Excluding verification TXT record\")\n  }\n\n  function createRoute53RecordSets(\n    verificationToken: string,\n    dkimTokens: string[],\n  ) {\n    const records: RecordSetProperty[] = []\n\n    if (includeVerificationRecord) {\n      records.push({\n        Name: `_amazonses.${domainName}.`,\n        Type: \"TXT\",\n        ResourceRecords: [`\"${verificationToken}\"`],\n        TTL: ttl,\n      })\n    }\n\n    for (const token of dkimTokens) {\n      records.push({\n        Name: `${token}._domainkey.${domainName}.`,\n        Type: \"CNAME\",\n        ResourceRecords: [`${token}.dkim.amazonses.com.`],\n        TTL: ttl,\n      })\n    }\n    return records\n  }\n\n  switch (event.RequestType) {\n    case \"Delete\":\n      const deleteIdentityResp = await sesClient.send(\n        new DeleteIdentityCommand({\n          Identity: domainName,\n        }),\n      )\n      console.log(`ses.deleteIdentity: ${JSON.stringify(deleteIdentityResp)}`)\n\n      return {\n        PhysicalResourceId: event.PhysicalResourceId,\n      }\n\n    case \"Create\":\n    case \"Update\":\n      // Idempotent.\n      const verifyDomainIdentityResp = await sesClient.send(\n        new VerifyDomainIdentityCommand({\n          Domain: domainName,\n        }),\n      )\n      console.log(\n        `ses.verifyDomainIdentity: ${JSON.stringify(verifyDomainIdentityResp)}`,\n      )\n      const verificationToken = verifyDomainIdentityResp.VerificationToken\n      if (!verificationToken) {\n        throw new Error(\"Verification token not returned\")\n      }\n\n      // Idempotent.\n      const verifyDomainDkimResp = await sesClient.send(\n        new VerifyDomainDkimCommand({\n          Domain: domainName,\n        }),\n      )\n      console.log(\n        `ses.verifyDomainDkim: ${JSON.stringify(verifyDomainDkimResp)}`,\n      )\n      const dkimTokens = verifyDomainDkimResp.DkimTokens ?? []\n      if (!dkimTokens) {\n        throw new Error(\"DKIM tokens not returned\")\n      }\n\n      // Idempotent.\n      const putEmailIdentityConfigResp = await sesv2Client.send(\n        new PutEmailIdentityConfigurationSetAttributesCommand({\n          EmailIdentity: domainName,\n          ConfigurationSetName: defaultConfigurationSetName,\n        }),\n      )\n      console.log(\n        `sesv2.putEmailIdentityConfigurationSetAttributes ${JSON.stringify(\n          putEmailIdentityConfigResp,\n        )}`,\n      )\n\n      return {\n        PhysicalResourceId: `SesDomain${domainName}`,\n        Data: {\n          Route53RecordSets: createRoute53RecordSets(\n            verificationToken,\n            dkimTokens,\n          ),\n          VerificationToken: verificationToken,\n        },\n      }\n  }\n}\n"]}
|
|
77
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handler.js","sourceRoot":"","sources":["../../../src/ses/sesdomain/handler.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  SESClient,\n  VerifyDomainIdentityCommand,\n  VerifyDomainDkimCommand,\n  DeleteIdentityCommand,\n} from \"@aws-sdk/client-ses\"\n\nimport {\n  SESv2Client,\n  PutEmailIdentityConfigurationSetAttributesCommand,\n} from \"@aws-sdk/client-sesv2\"\n\ntype OnEventHandler = (event: {\n  PhysicalResourceId?: string\n  RequestType: \"Create\" | \"Update\" | \"Delete\"\n  ResourceProperties: Record<string, string>\n}) => Promise<{\n  PhysicalResourceId?: string\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  Data?: Record<string, any>\n}>\n\n// Type to mach CloudFormation AWS::Route53::RecordSetGroup RecordSet\ninterface RecordSetProperty {\n  Name: string\n  Type: string\n  ResourceRecords: string[]\n  TTL: string\n}\n\nexport const handler: OnEventHandler = async (event) => {\n  const sesClient = new SESClient()\n  const sesv2Client = new SESv2Client()\n\n  const ttl = \"1800\"\n\n  const domainName = event.ResourceProperties[\"DomainName\"]\n  const includeVerificationRecord =\n    event.ResourceProperties[\"IncludeVerificationRecord\"] == \"true\"\n\n  const defaultConfigurationSetName =\n    event.ResourceProperties[\"DefaultConfigurationSetName\"]\n\n  if (!includeVerificationRecord) {\n    console.log(\"Excluding verification TXT record\")\n  }\n\n  function createRoute53RecordSets(\n    verificationToken: string,\n    dkimTokens: string[],\n  ) {\n    const records: RecordSetProperty[] = []\n\n    if (includeVerificationRecord) {\n      records.push({\n        Name: `_amazonses.${domainName}.`,\n        Type: \"TXT\",\n        ResourceRecords: [`\"${verificationToken}\"`],\n        TTL: ttl,\n      })\n    }\n\n    for (const token of dkimTokens) {\n      records.push({\n        Name: `${token}._domainkey.${domainName}.`,\n        Type: \"CNAME\",\n        ResourceRecords: [`${token}.dkim.amazonses.com.`],\n        TTL: ttl,\n      })\n    }\n    return records\n  }\n\n  switch (event.RequestType) {\n    case \"Delete\":\n      const deleteIdentityResp = await sesClient.send(\n        new DeleteIdentityCommand({\n          Identity: domainName,\n        }),\n      )\n      console.log(`ses.deleteIdentity: ${JSON.stringify(deleteIdentityResp)}`)\n\n      return {\n        PhysicalResourceId: event.PhysicalResourceId,\n      }\n\n    case \"Create\":\n    case \"Update\":\n      // Idempotent.\n      const verifyDomainIdentityResp = await sesClient.send(\n        new VerifyDomainIdentityCommand({\n          Domain: domainName,\n        }),\n      )\n      console.log(\n        `ses.verifyDomainIdentity: ${JSON.stringify(verifyDomainIdentityResp)}`,\n      )\n      const verificationToken = verifyDomainIdentityResp.VerificationToken\n      if (!verificationToken) {\n        throw new Error(\"Verification token not returned\")\n      }\n\n      // Idempotent.\n      const verifyDomainDkimResp = await sesClient.send(\n        new VerifyDomainDkimCommand({\n          Domain: domainName,\n        }),\n      )\n      console.log(\n        `ses.verifyDomainDkim: ${JSON.stringify(verifyDomainDkimResp)}`,\n      )\n      const dkimTokens = verifyDomainDkimResp.DkimTokens ?? []\n      if (!dkimTokens) {\n        throw new Error(\"DKIM tokens not returned\")\n      }\n\n      // Idempotent.\n      const putEmailIdentityConfigResp = await sesv2Client.send(\n        new PutEmailIdentityConfigurationSetAttributesCommand({\n          EmailIdentity: domainName,\n          ConfigurationSetName: defaultConfigurationSetName,\n        }),\n      )\n      console.log(\n        `sesv2.putEmailIdentityConfigurationSetAttributes ${JSON.stringify(\n          putEmailIdentityConfigResp,\n        )}`,\n      )\n\n      return {\n        PhysicalResourceId: `SesDomain${domainName}`,\n        Data: {\n          Route53RecordSets: createRoute53RecordSets(\n            verificationToken,\n            dkimTokens,\n          ),\n          VerificationToken: verificationToken,\n        },\n      }\n  }\n}\n"]}
|
|
@@ -1,36 +1,40 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
1
|
+
import * as constructs from "constructs";
|
|
2
|
+
import * as iam from "aws-cdk-lib/aws-iam";
|
|
3
|
+
import * as lambda from "aws-cdk-lib/aws-lambda";
|
|
4
|
+
import * as r53 from "aws-cdk-lib/aws-route53";
|
|
5
|
+
import * as cdk from "aws-cdk-lib";
|
|
6
|
+
import * as cr from "aws-cdk-lib/custom-resources";
|
|
7
|
+
import { NodejsFunction } from "aws-cdk-lib/aws-lambda-nodejs";
|
|
8
|
+
import * as path from "path";
|
|
9
|
+
import { fileURLToPath } from "node:url";
|
|
10
|
+
import { createRequire } from "node:module";
|
|
11
|
+
const require = createRequire(import.meta.url);
|
|
12
|
+
const __filename = fileURLToPath(import.meta.url);
|
|
13
|
+
const __dirname = path.dirname(__filename);
|
|
14
|
+
export class SesDomain extends constructs.Construct {
|
|
15
|
+
route53RecordSets;
|
|
16
|
+
verificationToken;
|
|
12
17
|
constructor(scope, id, props) {
|
|
13
|
-
var _a, _b, _c, _d;
|
|
14
18
|
super(scope, id);
|
|
15
19
|
const resource = new cdk.CustomResource(this, "Resource", {
|
|
16
20
|
serviceToken: SesDomainProvider.getOrCreate(this).serviceToken,
|
|
17
21
|
properties: {
|
|
18
22
|
DomainName: props.domainName,
|
|
19
|
-
IncludeVerificationRecord: (
|
|
23
|
+
IncludeVerificationRecord: (props.includeVerificationRecord ?? true).toString(),
|
|
20
24
|
DefaultConfigurationSetName: props.defaultConfigurationSetName,
|
|
21
25
|
// Bump this if changing logic in the lambda that should be
|
|
22
26
|
// re-evaluated.
|
|
23
27
|
Serial: 1,
|
|
24
28
|
},
|
|
25
29
|
});
|
|
26
|
-
const staticRecordSets = (
|
|
30
|
+
const staticRecordSets = (props.spfRecord?.include ?? true)
|
|
27
31
|
? [
|
|
28
32
|
{
|
|
29
33
|
name: props.domainName,
|
|
30
34
|
type: r53.RecordType.TXT,
|
|
31
35
|
ttl: "60",
|
|
32
36
|
resourceRecords: [
|
|
33
|
-
JSON.stringify(
|
|
37
|
+
JSON.stringify(props.spfRecord?.value || "v=spf1 include:amazonses.com ~all"),
|
|
34
38
|
],
|
|
35
39
|
},
|
|
36
40
|
]
|
|
@@ -51,7 +55,6 @@ class SesDomain extends constructs.Construct {
|
|
|
51
55
|
}
|
|
52
56
|
}
|
|
53
57
|
}
|
|
54
|
-
exports.SesDomain = SesDomain;
|
|
55
58
|
class SesDomainProvider extends constructs.Construct {
|
|
56
59
|
/**
|
|
57
60
|
* Returns the singleton provider.
|
|
@@ -62,11 +65,13 @@ class SesDomainProvider extends constructs.Construct {
|
|
|
62
65
|
return (stack.node.tryFindChild(id) ||
|
|
63
66
|
new SesDomainProvider(stack, id));
|
|
64
67
|
}
|
|
68
|
+
provider;
|
|
69
|
+
serviceToken;
|
|
65
70
|
constructor(scope, id) {
|
|
66
71
|
super(scope, id);
|
|
67
72
|
this.provider = new cr.Provider(this, "Provider", {
|
|
68
|
-
onEventHandler: new
|
|
69
|
-
entry: require.resolve(
|
|
73
|
+
onEventHandler: new NodejsFunction(this, "Function", {
|
|
74
|
+
entry: require.resolve(`${__dirname}/handler`),
|
|
70
75
|
runtime: lambda.Runtime.NODEJS_18_X,
|
|
71
76
|
timeout: cdk.Duration.minutes(5),
|
|
72
77
|
awsSdkConnectionReuse: false,
|
|
@@ -91,4 +96,4 @@ class SesDomainProvider extends constructs.Construct {
|
|
|
91
96
|
this.serviceToken = this.provider.serviceToken;
|
|
92
97
|
}
|
|
93
98
|
}
|
|
94
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/ses/sesdomain/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport * as r53 from \"aws-cdk-lib/aws-route53\"\nimport * as cdk from \"aws-cdk-lib\"\nimport * as cr from \"aws-cdk-lib/custom-resources\"\nimport { NodejsFunction } from \"aws-cdk-lib/aws-lambda-nodejs\"\n\ninterface Props {\n  /**\n   * The domain name to register in SES.\n   */\n  domainName: string\n  /**\n   * Hosted Zone to attach DNS records. If not given it must\n   * be performed manually.\n   */\n  hostedZone?: r53.IHostedZone\n  /**\n   * Include or exclude verification TXT record.\n   *\n   * CNAME records for DKIM tokens will still be created.\n   *\n   * Route 53 will not allow multiple TXT records with the same name.\n   * This option allows to \"opt-out\" of the records and leaving\n   * the caller responsible of handling it.\n   *\n   * @default true\n   */\n  includeVerificationRecord?: boolean\n  /**\n   * Default configuration set for emails sent from this domain.\n   */\n  defaultConfigurationSetName?: string\n  /**\n   * Configuration for an SPF record.\n   *\n   * @default - an SPF record with a default value is created.\n   */\n  spfRecord?: {\n    /**\n     * Whether to create the record or not.\n     *\n     * @default true\n     */\n    include?: boolean\n    /**\n     * The value of the SPF record.\n     *\n     * NOTE: The value will be enclosed in double quotes for you.\n     *\n     * @default \"v=spf1 include:amazonses.com ~all\"\n     */\n    value?: string\n  }\n}\n\nexport class SesDomain extends constructs.Construct {\n  public route53RecordSets: cdk.IResolvable\n  public verificationToken: string\n\n  constructor(scope: constructs.Construct, id: string, props: Props) {\n    super(scope, id)\n\n    const resource = new cdk.CustomResource(this, \"Resource\", {\n      serviceToken: SesDomainProvider.getOrCreate(this).serviceToken,\n      properties: {\n        DomainName: props.domainName,\n        IncludeVerificationRecord: (\n          props.includeVerificationRecord ?? true\n        ).toString(),\n        DefaultConfigurationSetName: props.defaultConfigurationSetName,\n        // Bump this if changing logic in the lambda that should be\n        // re-evaluated.\n        Serial: 1,\n      },\n    })\n\n    const staticRecordSets: r53.CfnRecordSetGroup.RecordSetProperty[] =\n      (props.spfRecord?.include ?? true)\n        ? [\n            {\n              name: props.domainName,\n              type: r53.RecordType.TXT,\n              ttl: \"60\",\n              resourceRecords: [\n                JSON.stringify(\n                  props.spfRecord?.value || \"v=spf1 include:amazonses.com ~all\",\n                ),\n              ],\n            },\n          ]\n        : []\n\n    this.route53RecordSets = resource.getAtt(\"Route53RecordSets\")\n    this.verificationToken = resource.getAttString(\"VerificationToken\")\n\n    if (props.hostedZone) {\n      new r53.CfnRecordSetGroup(this, \"RecordSetGroup\", {\n        hostedZoneId: props.hostedZone.hostedZoneId,\n        recordSets: this.route53RecordSets,\n      })\n      if (staticRecordSets.length) {\n        new r53.CfnRecordSetGroup(this, \"StaticRecordSetGroup\", {\n          hostedZoneId: props.hostedZone.hostedZoneId,\n          recordSets: staticRecordSets,\n        })\n      }\n    }\n  }\n}\n\nclass SesDomainProvider extends constructs.Construct {\n  /**\n   * Returns the singleton provider.\n   */\n  public static getOrCreate(scope: constructs.Construct) {\n    const stack = cdk.Stack.of(scope)\n    const id = \"liflig-cdk.ses-domain.provider\"\n    return (\n      (stack.node.tryFindChild(id) as SesDomainProvider) ||\n      new SesDomainProvider(stack, id)\n    )\n  }\n\n  private readonly provider: cr.Provider\n  public readonly serviceToken: string\n\n  constructor(scope: constructs.Construct, id: string) {\n    super(scope, id)\n\n    this.provider = new cr.Provider(this, \"Provider\", {\n      onEventHandler: new NodejsFunction(this, \"Function\", {\n        entry: require.resolve(\"./handler\"),\n        runtime: lambda.Runtime.NODEJS_18_X,\n        timeout: cdk.Duration.minutes(5),\n        awsSdkConnectionReuse: false,\n        initialPolicy: [\n          new iam.PolicyStatement({\n            actions: [\n              \"ses:DeleteIdentity\",\n              \"ses:GetIdentityDkimAttributes\",\n              \"ses:GetIdentityMailFromDomainAttributes\",\n              \"ses:GetIdentityVerificationAttributes\",\n              \"ses:SetIdentityDkimEnabled\",\n              \"ses:SetIdentityMailFromDomain\",\n              \"ses:VerifyDomainDkim\",\n              \"ses:VerifyDomainIdentity\",\n              \"ses:PutEmailIdentityConfigurationSetAttributes\",\n            ],\n            resources: [\"*\"],\n          }),\n        ],\n      }),\n    })\n\n    this.serviceToken = this.provider.serviceToken\n  }\n}\n"]}
|
|
99
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/ses/sesdomain/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport * as r53 from \"aws-cdk-lib/aws-route53\"\nimport * as cdk from \"aws-cdk-lib\"\nimport * as cr from \"aws-cdk-lib/custom-resources\"\nimport { NodejsFunction } from \"aws-cdk-lib/aws-lambda-nodejs\"\nimport * as path from \"path\"\nimport { fileURLToPath } from \"node:url\"\nimport { createRequire } from \"node:module\"\n\nconst require = createRequire(import.meta.url)\nconst __filename = fileURLToPath(import.meta.url)\nconst __dirname = path.dirname(__filename)\n\ninterface Props {\n  /**\n   * The domain name to register in SES.\n   */\n  domainName: string\n  /**\n   * Hosted Zone to attach DNS records. If not given it must\n   * be performed manually.\n   */\n  hostedZone?: r53.IHostedZone\n  /**\n   * Include or exclude verification TXT record.\n   *\n   * CNAME records for DKIM tokens will still be created.\n   *\n   * Route 53 will not allow multiple TXT records with the same name.\n   * This option allows to \"opt-out\" of the records and leaving\n   * the caller responsible of handling it.\n   *\n   * @default true\n   */\n  includeVerificationRecord?: boolean\n  /**\n   * Default configuration set for emails sent from this domain.\n   */\n  defaultConfigurationSetName?: string\n  /**\n   * Configuration for an SPF record.\n   *\n   * @default - an SPF record with a default value is created.\n   */\n  spfRecord?: {\n    /**\n     * Whether to create the record or not.\n     *\n     * @default true\n     */\n    include?: boolean\n    /**\n     * The value of the SPF record.\n     *\n     * NOTE: The value will be enclosed in double quotes for you.\n     *\n     * @default \"v=spf1 include:amazonses.com ~all\"\n     */\n    value?: string\n  }\n}\n\nexport class SesDomain extends constructs.Construct {\n  public route53RecordSets: cdk.IResolvable\n  public verificationToken: string\n\n  constructor(scope: constructs.Construct, id: string, props: Props) {\n    super(scope, id)\n\n    const resource = new cdk.CustomResource(this, \"Resource\", {\n      serviceToken: SesDomainProvider.getOrCreate(this).serviceToken,\n      properties: {\n        DomainName: props.domainName,\n        IncludeVerificationRecord: (\n          props.includeVerificationRecord ?? true\n        ).toString(),\n        DefaultConfigurationSetName: props.defaultConfigurationSetName,\n        // Bump this if changing logic in the lambda that should be\n        // re-evaluated.\n        Serial: 1,\n      },\n    })\n\n    const staticRecordSets: r53.CfnRecordSetGroup.RecordSetProperty[] =\n      (props.spfRecord?.include ?? true)\n        ? [\n            {\n              name: props.domainName,\n              type: r53.RecordType.TXT,\n              ttl: \"60\",\n              resourceRecords: [\n                JSON.stringify(\n                  props.spfRecord?.value || \"v=spf1 include:amazonses.com ~all\",\n                ),\n              ],\n            },\n          ]\n        : []\n\n    this.route53RecordSets = resource.getAtt(\"Route53RecordSets\")\n    this.verificationToken = resource.getAttString(\"VerificationToken\")\n\n    if (props.hostedZone) {\n      new r53.CfnRecordSetGroup(this, \"RecordSetGroup\", {\n        hostedZoneId: props.hostedZone.hostedZoneId,\n        recordSets: this.route53RecordSets,\n      })\n      if (staticRecordSets.length) {\n        new r53.CfnRecordSetGroup(this, \"StaticRecordSetGroup\", {\n          hostedZoneId: props.hostedZone.hostedZoneId,\n          recordSets: staticRecordSets,\n        })\n      }\n    }\n  }\n}\n\nclass SesDomainProvider extends constructs.Construct {\n  /**\n   * Returns the singleton provider.\n   */\n  public static getOrCreate(scope: constructs.Construct) {\n    const stack = cdk.Stack.of(scope)\n    const id = \"liflig-cdk.ses-domain.provider\"\n    return (\n      (stack.node.tryFindChild(id) as SesDomainProvider) ||\n      new SesDomainProvider(stack, id)\n    )\n  }\n\n  private readonly provider: cr.Provider\n  public readonly serviceToken: string\n\n  constructor(scope: constructs.Construct, id: string) {\n    super(scope, id)\n\n    this.provider = new cr.Provider(this, \"Provider\", {\n      onEventHandler: new NodejsFunction(this, \"Function\", {\n        entry: require.resolve(`${__dirname}/handler`),\n        runtime: lambda.Runtime.NODEJS_18_X,\n        timeout: cdk.Duration.minutes(5),\n        awsSdkConnectionReuse: false,\n        initialPolicy: [\n          new iam.PolicyStatement({\n            actions: [\n              \"ses:DeleteIdentity\",\n              \"ses:GetIdentityDkimAttributes\",\n              \"ses:GetIdentityMailFromDomainAttributes\",\n              \"ses:GetIdentityVerificationAttributes\",\n              \"ses:SetIdentityDkimEnabled\",\n              \"ses:SetIdentityMailFromDomain\",\n              \"ses:VerifyDomainDkim\",\n              \"ses:VerifyDomainIdentity\",\n              \"ses:PutEmailIdentityConfigurationSetAttributes\",\n            ],\n            resources: [\"*\"],\n          }),\n        ],\n      }),\n    })\n\n    this.serviceToken = this.provider.serviceToken\n  }\n}\n"]}
|
|
@@ -1,13 +1,10 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
const client_ses_1 = require("@aws-sdk/client-ses");
|
|
5
|
-
const handler = async (event) => {
|
|
6
|
-
const sesClient = new client_ses_1.SESClient();
|
|
1
|
+
import { DeleteIdentityCommand, SESClient, VerifyEmailIdentityCommand, } from "@aws-sdk/client-ses";
|
|
2
|
+
export const handler = async (event) => {
|
|
3
|
+
const sesClient = new SESClient();
|
|
7
4
|
const emailAddress = event.ResourceProperties["EmailAddress"];
|
|
8
5
|
switch (event.RequestType) {
|
|
9
6
|
case "Delete":
|
|
10
|
-
await sesClient.send(new
|
|
7
|
+
await sesClient.send(new DeleteIdentityCommand({
|
|
11
8
|
Identity: emailAddress,
|
|
12
9
|
}));
|
|
13
10
|
return {
|
|
@@ -15,7 +12,7 @@ const handler = async (event) => {
|
|
|
15
12
|
};
|
|
16
13
|
case "Create":
|
|
17
14
|
case "Update":
|
|
18
|
-
await sesClient.send(new
|
|
15
|
+
await sesClient.send(new VerifyEmailIdentityCommand({
|
|
19
16
|
EmailAddress: emailAddress,
|
|
20
17
|
}));
|
|
21
18
|
return {
|
|
@@ -23,5 +20,4 @@ const handler = async (event) => {
|
|
|
23
20
|
};
|
|
24
21
|
}
|
|
25
22
|
};
|
|
26
|
-
|
|
27
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGFuZGxlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9zZXMvc2VzdmVyaWZ5ZW1haWwvaGFuZGxlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSxvREFJNEI7QUFVckIsTUFBTSxPQUFPLEdBQW1CLEtBQUssRUFBRSxLQUFLLEVBQUUsRUFBRTtJQUNyRCxNQUFNLFNBQVMsR0FBRyxJQUFJLHNCQUFTLEVBQUUsQ0FBQTtJQUNqQyxNQUFNLFlBQVksR0FBRyxLQUFLLENBQUMsa0JBQWtCLENBQUMsY0FBYyxDQUFDLENBQUE7SUFFN0QsUUFBUSxLQUFLLENBQUMsV0FBVyxFQUFFLENBQUM7UUFDMUIsS0FBSyxRQUFRO1lBQ1gsTUFBTSxTQUFTLENBQUMsSUFBSSxDQUNsQixJQUFJLGtDQUFxQixDQUFDO2dCQUN4QixRQUFRLEVBQUUsWUFBWTthQUN2QixDQUFDLENBQ0gsQ0FBQTtZQUVELE9BQU87Z0JBQ0wsa0JBQWtCLEVBQUUsS0FBSyxDQUFDLGtCQUFrQjthQUM3QyxDQUFBO1FBRUgsS0FBSyxRQUFRLENBQUM7UUFDZCxLQUFLLFFBQVE7WUFDWCxNQUFNLFNBQVMsQ0FBQyxJQUFJLENBQ2xCLElBQUksdUNBQTBCLENBQUM7Z0JBQzdCLFlBQVksRUFBRSxZQUFZO2FBQzNCLENBQUMsQ0FDSCxDQUFBO1lBRUQsT0FBTztnQkFDTCxrQkFBa0IsRUFBRSxrQkFBa0IsWUFBWSxFQUFFO2FBQ3JELENBQUE7SUFDTCxDQUFDO0FBQ0gsQ0FBQyxDQUFBO0FBNUJZLFFBQUEsT0FBTyxXQTRCbkIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBEZWxldGVJZGVudGl0eUNvbW1hbmQsXG4gIFNFU0NsaWVudCxcbiAgVmVyaWZ5RW1haWxJZGVudGl0eUNvbW1hbmQsXG59IGZyb20gXCJAYXdzLXNkay9jbGllbnQtc2VzXCJcblxudHlwZSBPbkV2ZW50SGFuZGxlciA9IChldmVudDoge1xuICBQaHlzaWNhbFJlc291cmNlSWQ/OiBzdHJpbmdcbiAgUmVxdWVzdFR5cGU6IFwiQ3JlYXRlXCIgfCBcIlVwZGF0ZVwiIHwgXCJEZWxldGVcIlxuICBSZXNvdXJjZVByb3BlcnRpZXM6IFJlY29yZDxzdHJpbmcsIHN0cmluZz5cbn0pID0+IFByb21pc2U8e1xuICBQaHlzaWNhbFJlc291cmNlSWQ/OiBzdHJpbmdcbn0+XG5cbmV4cG9ydCBjb25zdCBoYW5kbGVyOiBPbkV2ZW50SGFuZGxlciA9IGFzeW5jIChldmVudCkgPT4ge1xuICBjb25zdCBzZXNDbGllbnQgPSBuZXcgU0VTQ2xpZW50KClcbiAgY29uc3QgZW1haWxBZGRyZXNzID0gZXZlbnQuUmVzb3VyY2VQcm9wZXJ0aWVzW1wiRW1haWxBZGRyZXNzXCJdXG5cbiAgc3dpdGNoIChldmVudC5SZXF1ZXN0VHlwZSkge1xuICAgIGNhc2UgXCJEZWxldGVcIjpcbiAgICAgIGF3YWl0IHNlc0NsaWVudC5zZW5kKFxuICAgICAgICBuZXcgRGVsZXRlSWRlbnRpdHlDb21tYW5kKHtcbiAgICAgICAgICBJZGVudGl0eTogZW1haWxBZGRyZXNzLFxuICAgICAgICB9KSxcbiAgICAgIClcblxuICAgICAgcmV0dXJuIHtcbiAgICAgICAgUGh5c2ljYWxSZXNvdXJjZUlkOiBldmVudC5QaHlzaWNhbFJlc291cmNlSWQsXG4gICAgICB9XG5cbiAgICBjYXNlIFwiQ3JlYXRlXCI6XG4gICAgY2FzZSBcIlVwZGF0ZVwiOlxuICAgICAgYXdhaXQgc2VzQ2xpZW50LnNlbmQoXG4gICAgICAgIG5ldyBWZXJpZnlFbWFpbElkZW50aXR5Q29tbWFuZCh7XG4gICAgICAgICAgRW1haWxBZGRyZXNzOiBlbWFpbEFkZHJlc3MsXG4gICAgICAgIH0pLFxuICAgICAgKVxuXG4gICAgICByZXR1cm4ge1xuICAgICAgICBQaHlzaWNhbFJlc291cmNlSWQ6IGBTZXNWZXJpZnlFbWFpbDoke2VtYWlsQWRkcmVzc31gLFxuICAgICAgfVxuICB9XG59XG4iXX0=
|
|
23
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGFuZGxlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9zZXMvc2VzdmVyaWZ5ZW1haWwvaGFuZGxlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQ0wscUJBQXFCLEVBQ3JCLFNBQVMsRUFDVCwwQkFBMEIsR0FDM0IsTUFBTSxxQkFBcUIsQ0FBQTtBQVU1QixNQUFNLENBQUMsTUFBTSxPQUFPLEdBQW1CLEtBQUssRUFBRSxLQUFLLEVBQUUsRUFBRTtJQUNyRCxNQUFNLFNBQVMsR0FBRyxJQUFJLFNBQVMsRUFBRSxDQUFBO0lBQ2pDLE1BQU0sWUFBWSxHQUFHLEtBQUssQ0FBQyxrQkFBa0IsQ0FBQyxjQUFjLENBQUMsQ0FBQTtJQUU3RCxRQUFRLEtBQUssQ0FBQyxXQUFXLEVBQUUsQ0FBQztRQUMxQixLQUFLLFFBQVE7WUFDWCxNQUFNLFNBQVMsQ0FBQyxJQUFJLENBQ2xCLElBQUkscUJBQXFCLENBQUM7Z0JBQ3hCLFFBQVEsRUFBRSxZQUFZO2FBQ3ZCLENBQUMsQ0FDSCxDQUFBO1lBRUQsT0FBTztnQkFDTCxrQkFBa0IsRUFBRSxLQUFLLENBQUMsa0JBQWtCO2FBQzdDLENBQUE7UUFFSCxLQUFLLFFBQVEsQ0FBQztRQUNkLEtBQUssUUFBUTtZQUNYLE1BQU0sU0FBUyxDQUFDLElBQUksQ0FDbEIsSUFBSSwwQkFBMEIsQ0FBQztnQkFDN0IsWUFBWSxFQUFFLFlBQVk7YUFDM0IsQ0FBQyxDQUNILENBQUE7WUFFRCxPQUFPO2dCQUNMLGtCQUFrQixFQUFFLGtCQUFrQixZQUFZLEVBQUU7YUFDckQsQ0FBQTtJQUNMLENBQUM7QUFDSCxDQUFDLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBEZWxldGVJZGVudGl0eUNvbW1hbmQsXG4gIFNFU0NsaWVudCxcbiAgVmVyaWZ5RW1haWxJZGVudGl0eUNvbW1hbmQsXG59IGZyb20gXCJAYXdzLXNkay9jbGllbnQtc2VzXCJcblxudHlwZSBPbkV2ZW50SGFuZGxlciA9IChldmVudDoge1xuICBQaHlzaWNhbFJlc291cmNlSWQ/OiBzdHJpbmdcbiAgUmVxdWVzdFR5cGU6IFwiQ3JlYXRlXCIgfCBcIlVwZGF0ZVwiIHwgXCJEZWxldGVcIlxuICBSZXNvdXJjZVByb3BlcnRpZXM6IFJlY29yZDxzdHJpbmcsIHN0cmluZz5cbn0pID0+IFByb21pc2U8e1xuICBQaHlzaWNhbFJlc291cmNlSWQ/OiBzdHJpbmdcbn0+XG5cbmV4cG9ydCBjb25zdCBoYW5kbGVyOiBPbkV2ZW50SGFuZGxlciA9IGFzeW5jIChldmVudCkgPT4ge1xuICBjb25zdCBzZXNDbGllbnQgPSBuZXcgU0VTQ2xpZW50KClcbiAgY29uc3QgZW1haWxBZGRyZXNzID0gZXZlbnQuUmVzb3VyY2VQcm9wZXJ0aWVzW1wiRW1haWxBZGRyZXNzXCJdXG5cbiAgc3dpdGNoIChldmVudC5SZXF1ZXN0VHlwZSkge1xuICAgIGNhc2UgXCJEZWxldGVcIjpcbiAgICAgIGF3YWl0IHNlc0NsaWVudC5zZW5kKFxuICAgICAgICBuZXcgRGVsZXRlSWRlbnRpdHlDb21tYW5kKHtcbiAgICAgICAgICBJZGVudGl0eTogZW1haWxBZGRyZXNzLFxuICAgICAgICB9KSxcbiAgICAgIClcblxuICAgICAgcmV0dXJuIHtcbiAgICAgICAgUGh5c2ljYWxSZXNvdXJjZUlkOiBldmVudC5QaHlzaWNhbFJlc291cmNlSWQsXG4gICAgICB9XG5cbiAgICBjYXNlIFwiQ3JlYXRlXCI6XG4gICAgY2FzZSBcIlVwZGF0ZVwiOlxuICAgICAgYXdhaXQgc2VzQ2xpZW50LnNlbmQoXG4gICAgICAgIG5ldyBWZXJpZnlFbWFpbElkZW50aXR5Q29tbWFuZCh7XG4gICAgICAgICAgRW1haWxBZGRyZXNzOiBlbWFpbEFkZHJlc3MsXG4gICAgICAgIH0pLFxuICAgICAgKVxuXG4gICAgICByZXR1cm4ge1xuICAgICAgICBQaHlzaWNhbFJlc291cmNlSWQ6IGBTZXNWZXJpZnlFbWFpbDoke2VtYWlsQWRkcmVzc31gLFxuICAgICAgfVxuICB9XG59XG4iXX0=
|
|
@@ -1,13 +1,17 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
1
|
+
import * as constructs from "constructs";
|
|
2
|
+
import * as iam from "aws-cdk-lib/aws-iam";
|
|
3
|
+
import * as lambda from "aws-cdk-lib/aws-lambda";
|
|
4
|
+
import * as cdk from "aws-cdk-lib";
|
|
5
|
+
import * as cr from "aws-cdk-lib/custom-resources";
|
|
6
|
+
import { NodejsFunction } from "aws-cdk-lib/aws-lambda-nodejs";
|
|
7
|
+
import * as path from "path";
|
|
8
|
+
import { fileURLToPath } from "node:url";
|
|
9
|
+
import { createRequire } from "node:module";
|
|
10
|
+
const require = createRequire(import.meta.url);
|
|
11
|
+
const __filename = fileURLToPath(import.meta.url);
|
|
12
|
+
const __dirname = path.dirname(__filename);
|
|
13
|
+
export class SesVerifyEmail extends constructs.Construct {
|
|
14
|
+
route53RecordSets;
|
|
11
15
|
constructor(scope, id, props) {
|
|
12
16
|
super(scope, id);
|
|
13
17
|
new cdk.CustomResource(this, "Resource", {
|
|
@@ -18,7 +22,6 @@ class SesVerifyEmail extends constructs.Construct {
|
|
|
18
22
|
});
|
|
19
23
|
}
|
|
20
24
|
}
|
|
21
|
-
exports.SesVerifyEmail = SesVerifyEmail;
|
|
22
25
|
class SesVerifyEmailProvider extends constructs.Construct {
|
|
23
26
|
/**
|
|
24
27
|
* Returns the singleton provider.
|
|
@@ -29,11 +32,13 @@ class SesVerifyEmailProvider extends constructs.Construct {
|
|
|
29
32
|
return (stack.node.tryFindChild(id) ||
|
|
30
33
|
new SesVerifyEmailProvider(stack, id));
|
|
31
34
|
}
|
|
35
|
+
provider;
|
|
36
|
+
serviceToken;
|
|
32
37
|
constructor(scope, id) {
|
|
33
38
|
super(scope, id);
|
|
34
39
|
this.provider = new cr.Provider(this, "Provider", {
|
|
35
|
-
onEventHandler: new
|
|
36
|
-
entry: require.resolve(
|
|
40
|
+
onEventHandler: new NodejsFunction(this, "Function", {
|
|
41
|
+
entry: require.resolve(`${__dirname}/handler`),
|
|
37
42
|
runtime: lambda.Runtime.NODEJS_18_X,
|
|
38
43
|
timeout: cdk.Duration.minutes(5),
|
|
39
44
|
awsSdkConnectionReuse: false,
|
|
@@ -48,4 +53,4 @@ class SesVerifyEmailProvider extends constructs.Construct {
|
|
|
48
53
|
this.serviceToken = this.provider.serviceToken;
|
|
49
54
|
}
|
|
50
55
|
}
|
|
51
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
56
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvc2VzL3Nlc3ZlcmlmeWVtYWlsL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBSyxVQUFVLE1BQU0sWUFBWSxDQUFBO0FBQ3hDLE9BQU8sS0FBSyxHQUFHLE1BQU0scUJBQXFCLENBQUE7QUFDMUMsT0FBTyxLQUFLLE1BQU0sTUFBTSx3QkFBd0IsQ0FBQTtBQUNoRCxPQUFPLEtBQUssR0FBRyxNQUFNLGFBQWEsQ0FBQTtBQUNsQyxPQUFPLEtBQUssRUFBRSxNQUFNLDhCQUE4QixDQUFBO0FBQ2xELE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSwrQkFBK0IsQ0FBQTtBQUM5RCxPQUFPLEtBQUssSUFBSSxNQUFNLE1BQU0sQ0FBQTtBQUM1QixPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sVUFBVSxDQUFBO0FBQ3hDLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxhQUFhLENBQUE7QUFFM0MsTUFBTSxPQUFPLEdBQUcsYUFBYSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUE7QUFDOUMsTUFBTSxVQUFVLEdBQUcsYUFBYSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUE7QUFDakQsTUFBTSxTQUFTLEdBQUcsSUFBSSxDQUFDLE9BQU8sQ0FBQyxVQUFVLENBQUMsQ0FBQTtBQVMxQyxNQUFNLE9BQU8sY0FBZSxTQUFRLFVBQVUsQ0FBQyxTQUFTO0lBQy9DLGlCQUFpQixDQUFpQjtJQUV6QyxZQUFZLEtBQTJCLEVBQUUsRUFBVSxFQUFFLEtBQVk7UUFDL0QsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQTtRQUVoQixJQUFJLEdBQUcsQ0FBQyxjQUFjLENBQUMsSUFBSSxFQUFFLFVBQVUsRUFBRTtZQUN2QyxZQUFZLEVBQUUsc0JBQXNCLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxDQUFDLFlBQVk7WUFDbkUsVUFBVSxFQUFFO2dCQUNWLFlBQVksRUFBRSxLQUFLLENBQUMsWUFBWTthQUNqQztTQUNGLENBQUMsQ0FBQTtJQUNKLENBQUM7Q0FDRjtBQUVELE1BQU0sc0JBQXVCLFNBQVEsVUFBVSxDQUFDLFNBQVM7SUFDdkQ7O09BRUc7SUFDSSxNQUFNLENBQUMsV0FBVyxDQUFDLEtBQTJCO1FBQ25ELE1BQU0sS0FBSyxHQUFHLEdBQUcsQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDLEtBQUssQ0FBQyxDQUFBO1FBQ2pDLE1BQU0sRUFBRSxHQUFHLHNDQUFzQyxDQUFBO1FBQ2pELE9BQU8sQ0FDSixLQUFLLENBQUMsSUFBSSxDQUFDLFlBQVksQ0FBQyxFQUFFLENBQTRCO1lBQ3ZELElBQUksc0JBQXNCLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUN0QyxDQUFBO0lBQ0gsQ0FBQztJQUVnQixRQUFRLENBQWE7SUFDdEIsWUFBWSxDQUFRO0lBRXBDLFlBQVksS0FBMkIsRUFBRSxFQUFVO1FBQ2pELEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUE7UUFFaEIsSUFBSSxDQUFDLFFBQVEsR0FBRyxJQUFJLEVBQUUsQ0FBQyxRQUFRLENBQUMsSUFBSSxFQUFFLFVBQVUsRUFBRTtZQUNoRCxjQUFjLEVBQUUsSUFBSSxjQUFjLENBQUMsSUFBSSxFQUFFLFVBQVUsRUFBRTtnQkFDbkQsS0FBSyxFQUFFLE9BQU8sQ0FBQyxPQUFPLENBQUMsR0FBRyxTQUFTLFVBQVUsQ0FBQztnQkFDOUMsT0FBTyxFQUFFLE1BQU0sQ0FBQyxPQUFPLENBQUMsV0FBVztnQkFDbkMsT0FBTyxFQUFFLEdBQUcsQ0FBQyxRQUFRLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQztnQkFDaEMscUJBQXFCLEVBQUUsS0FBSztnQkFDNUIsYUFBYSxFQUFFO29CQUNiLElBQUksR0FBRyxDQUFDLGVBQWUsQ0FBQzt3QkFDdEIsT0FBTyxFQUFFLENBQUMsb0JBQW9CLEVBQUUseUJBQXlCLENBQUM7d0JBQzFELFNBQVMsRUFBRSxDQUFDLEdBQUcsQ0FBQztxQkFDakIsQ0FBQztpQkFDSDthQUNGLENBQUM7U0FDSCxDQUFDLENBQUE7UUFFRixJQUFJLENBQUMsWUFBWSxHQUFHLElBQUksQ0FBQyxRQUFRLENBQUMsWUFBWSxDQUFBO0lBQ2hELENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIGNvbnN0cnVjdHMgZnJvbSBcImNvbnN0cnVjdHNcIlxuaW1wb3J0ICogYXMgaWFtIGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtaWFtXCJcbmltcG9ydCAqIGFzIGxhbWJkYSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWxhbWJkYVwiXG5pbXBvcnQgKiBhcyBjZGsgZnJvbSBcImF3cy1jZGstbGliXCJcbmltcG9ydCAqIGFzIGNyIGZyb20gXCJhd3MtY2RrLWxpYi9jdXN0b20tcmVzb3VyY2VzXCJcbmltcG9ydCB7IE5vZGVqc0Z1bmN0aW9uIH0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1sYW1iZGEtbm9kZWpzXCJcbmltcG9ydCAqIGFzIHBhdGggZnJvbSBcInBhdGhcIlxuaW1wb3J0IHsgZmlsZVVSTFRvUGF0aCB9IGZyb20gXCJub2RlOnVybFwiXG5pbXBvcnQgeyBjcmVhdGVSZXF1aXJlIH0gZnJvbSBcIm5vZGU6bW9kdWxlXCJcblxuY29uc3QgcmVxdWlyZSA9IGNyZWF0ZVJlcXVpcmUoaW1wb3J0Lm1ldGEudXJsKVxuY29uc3QgX19maWxlbmFtZSA9IGZpbGVVUkxUb1BhdGgoaW1wb3J0Lm1ldGEudXJsKVxuY29uc3QgX19kaXJuYW1lID0gcGF0aC5kaXJuYW1lKF9fZmlsZW5hbWUpXG5cbmludGVyZmFjZSBQcm9wcyB7XG4gIC8qKlxuICAgKiBUaGUgZW1haWwgYWRkcmVzcyB0byBhZGQgYXMgYSB2ZXJpZmllZCBlbWFpbCBpbiBTRVMuXG4gICAqL1xuICBlbWFpbEFkZHJlc3M6IHN0cmluZ1xufVxuXG5leHBvcnQgY2xhc3MgU2VzVmVyaWZ5RW1haWwgZXh0ZW5kcyBjb25zdHJ1Y3RzLkNvbnN0cnVjdCB7XG4gIHB1YmxpYyByb3V0ZTUzUmVjb3JkU2V0czogY2RrLklSZXNvbHZhYmxlXG5cbiAgY29uc3RydWN0b3Ioc2NvcGU6IGNvbnN0cnVjdHMuQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wczogUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQpXG5cbiAgICBuZXcgY2RrLkN1c3RvbVJlc291cmNlKHRoaXMsIFwiUmVzb3VyY2VcIiwge1xuICAgICAgc2VydmljZVRva2VuOiBTZXNWZXJpZnlFbWFpbFByb3ZpZGVyLmdldE9yQ3JlYXRlKHRoaXMpLnNlcnZpY2VUb2tlbixcbiAgICAgIHByb3BlcnRpZXM6IHtcbiAgICAgICAgRW1haWxBZGRyZXNzOiBwcm9wcy5lbWFpbEFkZHJlc3MsXG4gICAgICB9LFxuICAgIH0pXG4gIH1cbn1cblxuY2xhc3MgU2VzVmVyaWZ5RW1haWxQcm92aWRlciBleHRlbmRzIGNvbnN0cnVjdHMuQ29uc3RydWN0IHtcbiAgLyoqXG4gICAqIFJldHVybnMgdGhlIHNpbmdsZXRvbiBwcm92aWRlci5cbiAgICovXG4gIHB1YmxpYyBzdGF0aWMgZ2V0T3JDcmVhdGUoc2NvcGU6IGNvbnN0cnVjdHMuQ29uc3RydWN0KSB7XG4gICAgY29uc3Qgc3RhY2sgPSBjZGsuU3RhY2sub2Yoc2NvcGUpXG4gICAgY29uc3QgaWQgPSBcImxpZmxpZy1jZGsuc2VzLXZlcmlmeS1lbWFpbC5wcm92aWRlclwiXG4gICAgcmV0dXJuIChcbiAgICAgIChzdGFjay5ub2RlLnRyeUZpbmRDaGlsZChpZCkgYXMgU2VzVmVyaWZ5RW1haWxQcm92aWRlcikgfHxcbiAgICAgIG5ldyBTZXNWZXJpZnlFbWFpbFByb3ZpZGVyKHN0YWNrLCBpZClcbiAgICApXG4gIH1cblxuICBwcml2YXRlIHJlYWRvbmx5IHByb3ZpZGVyOiBjci5Qcm92aWRlclxuICBwdWJsaWMgcmVhZG9ubHkgc2VydmljZVRva2VuOiBzdHJpbmdcblxuICBjb25zdHJ1Y3RvcihzY29wZTogY29uc3RydWN0cy5Db25zdHJ1Y3QsIGlkOiBzdHJpbmcpIHtcbiAgICBzdXBlcihzY29wZSwgaWQpXG5cbiAgICB0aGlzLnByb3ZpZGVyID0gbmV3IGNyLlByb3ZpZGVyKHRoaXMsIFwiUHJvdmlkZXJcIiwge1xuICAgICAgb25FdmVudEhhbmRsZXI6IG5ldyBOb2RlanNGdW5jdGlvbih0aGlzLCBcIkZ1bmN0aW9uXCIsIHtcbiAgICAgICAgZW50cnk6IHJlcXVpcmUucmVzb2x2ZShgJHtfX2Rpcm5hbWV9L2hhbmRsZXJgKSxcbiAgICAgICAgcnVudGltZTogbGFtYmRhLlJ1bnRpbWUuTk9ERUpTXzE4X1gsXG4gICAgICAgIHRpbWVvdXQ6IGNkay5EdXJhdGlvbi5taW51dGVzKDUpLFxuICAgICAgICBhd3NTZGtDb25uZWN0aW9uUmV1c2U6IGZhbHNlLFxuICAgICAgICBpbml0aWFsUG9saWN5OiBbXG4gICAgICAgICAgbmV3IGlhbS5Qb2xpY3lTdGF0ZW1lbnQoe1xuICAgICAgICAgICAgYWN0aW9uczogW1wic2VzOkRlbGV0ZUlkZW50aXR5XCIsIFwic2VzOlZlcmlmeUVtYWlsSWRlbnRpdHlcIl0sXG4gICAgICAgICAgICByZXNvdXJjZXM6IFtcIipcIl0sXG4gICAgICAgICAgfSksXG4gICAgICAgIF0sXG4gICAgICB9KSxcbiAgICB9KVxuXG4gICAgdGhpcy5zZXJ2aWNlVG9rZW4gPSB0aGlzLnByb3ZpZGVyLnNlcnZpY2VUb2tlblxuICB9XG59XG4iXX0=
|