UNPKG

16.8 kBJavaScriptView Raw
1"use strict";
2Object.defineProperty(exports, "__esModule", { value: true });
3exports.addToDeadLetterQueueResourcePolicy = exports.addLambdaPermission = exports.singletonEventRole = exports.bindBaseTargetConfig = void 0;
4const iam = require("@aws-cdk/aws-iam");
5const core_1 = require("@aws-cdk/core");
6// keep this import separate from other imports to reduce chance for merge conflicts with v2-main
7// eslint-disable-next-line no-duplicate-imports, import/order
8const core_2 = require("@aws-cdk/core");
9/**
10 * Bind props to base rule target config.
11 * @internal
12 */
13function bindBaseTargetConfig(props) {
14 let { deadLetterQueue, retryAttempts, maxEventAge } = props;
15 return {
16 deadLetterConfig: deadLetterQueue ? { arn: deadLetterQueue === null || deadLetterQueue === void 0 ? void 0 : deadLetterQueue.queueArn } : undefined,
17 retryPolicy: retryAttempts || maxEventAge
18 ? {
19 maximumRetryAttempts: retryAttempts,
20 maximumEventAgeInSeconds: maxEventAge === null || maxEventAge === void 0 ? void 0 : maxEventAge.toSeconds({ integral: true }),
21 }
22 : undefined,
23 };
24}
25exports.bindBaseTargetConfig = bindBaseTargetConfig;
26/**
27 * Obtain the Role for the EventBridge event
28 *
29 * If a role already exists, it will be returned. This ensures that if multiple
30 * events have the same target, they will share a role.
31 * @internal
32 */
33function singletonEventRole(scope, policyStatements) {
34 const id = 'EventsRole';
35 const existing = scope.node.tryFindChild(id);
36 if (existing) {
37 return existing;
38 }
39 const role = new iam.Role(scope, id, {
40 roleName: core_1.PhysicalName.GENERATE_IF_NEEDED,
41 assumedBy: new iam.ServicePrincipal('events.amazonaws.com'),
42 });
43 policyStatements.forEach(role.addToPolicy.bind(role));
44 return role;
45}
46exports.singletonEventRole = singletonEventRole;
47/**
48 * Allows a Lambda function to be called from a rule
49 * @internal
50 */
51function addLambdaPermission(rule, handler) {
52 let scope;
53 let node = handler.permissionsNode;
54 let permissionId = `AllowEventRule${core_1.Names.nodeUniqueId(rule.node)}`;
55 if (rule instanceof core_2.Construct) {
56 // Place the Permission resource in the same stack as Rule rather than the Function
57 // This is to reduce circular dependency when the lambda handler and the rule are across stacks.
58 scope = rule;
59 node = rule.node;
60 permissionId = `AllowEventRule${core_1.Names.nodeUniqueId(handler.node)}`;
61 }
62 if (!node.tryFindChild(permissionId)) {
63 handler.addPermission(permissionId, {
64 scope,
65 action: 'lambda:InvokeFunction',
66 principal: new iam.ServicePrincipal('events.amazonaws.com'),
67 sourceArn: rule.ruleArn,
68 });
69 }
70}
71exports.addLambdaPermission = addLambdaPermission;
72/**
73 * Allow a rule to send events with failed invocation to an Amazon SQS queue.
74 * @internal
75 */
76function addToDeadLetterQueueResourcePolicy(rule, queue) {
77 if (!sameEnvDimension(rule.env.region, queue.env.region)) {
78 throw new Error(`Cannot assign Dead Letter Queue in region ${queue.env.region} to the rule ${core_1.Names.nodeUniqueId(rule.node)} in region ${rule.env.region}. Both the queue and the rule must be in the same region.`);
79 }
80 // Skip Resource Policy creation if the Queue is not in the same account.
81 // There is no way to add a target onto an imported rule, so we can assume we will run the following code only
82 // in the account where the rule is created.
83 if (sameEnvDimension(rule.env.account, queue.env.account)) {
84 const policyStatementId = `AllowEventRule${core_1.Names.nodeUniqueId(rule.node)}`;
85 queue.addToResourcePolicy(new iam.PolicyStatement({
86 sid: policyStatementId,
87 principals: [new iam.ServicePrincipal('events.amazonaws.com')],
88 effect: iam.Effect.ALLOW,
89 actions: ['sqs:SendMessage'],
90 resources: [queue.queueArn],
91 conditions: {
92 ArnEquals: {
93 'aws:SourceArn': rule.ruleArn,
94 },
95 },
96 }));
97 }
98 else {
99 core_1.Annotations.of(rule).addWarning(`Cannot add a resource policy to your dead letter queue associated with rule ${rule.ruleName} because the queue is in a different account. You must add the resource policy manually to the dead letter queue in account ${queue.env.account}.`);
100 }
101}
102exports.addToDeadLetterQueueResourcePolicy = addToDeadLetterQueueResourcePolicy;
103/**
104 * Whether two string probably contain the same environment dimension (region or account)
105 *
106 * Used to compare either accounts or regions, and also returns true if both
107 * are unresolved (in which case both are expted to be "current region" or "current account").
108 * @internal
109 */
110function sameEnvDimension(dim1, dim2) {
111 return [core_1.TokenComparison.SAME, core_1.TokenComparison.BOTH_UNRESOLVED].includes(core_1.Token.compareStrings(dim1, dim2));
112}
113//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"util.js","sourceRoot":"","sources":["util.ts"],"names":[],"mappings":"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","sourcesContent":["import * as events from '@aws-cdk/aws-events';\nimport * as iam from '@aws-cdk/aws-iam';\nimport * as lambda from '@aws-cdk/aws-lambda';\nimport * as sqs from '@aws-cdk/aws-sqs';\nimport { Annotations, ConstructNode, IConstruct, Names, Token, TokenComparison, Duration, PhysicalName } from '@aws-cdk/core';\n\n// keep this import separate from other imports to reduce chance for merge conflicts with v2-main\n// eslint-disable-next-line no-duplicate-imports, import/order\nimport { Construct } from '@aws-cdk/core';\n\n/**\n * The generic properties for an RuleTarget\n */\nexport interface TargetBaseProps {\n  /**\n   * The SQS queue to be used as deadLetterQueue.\n   * Check out the [considerations for using a dead-letter queue](https://docs.aws.amazon.com/eventbridge/latest/userguide/rule-dlq.html#dlq-considerations).\n   *\n   * The events not successfully delivered are automatically retried for a specified period of time,\n   * depending on the retry policy of the target.\n   * If an event is not delivered before all retry attempts are exhausted, it will be sent to the dead letter queue.\n   *\n   * @default - no dead-letter queue\n   */\n  readonly deadLetterQueue?: sqs.IQueue;\n  /**\n   * The maximum age of a request that Lambda sends to a function for\n   * processing.\n   *\n   * Minimum value of 60.\n   * Maximum value of 86400.\n   *\n   * @default Duration.hours(24)\n   */\n  readonly maxEventAge?: Duration;\n\n  /**\n   * The maximum number of times to retry when the function returns an error.\n   *\n   * Minimum value of 0.\n   * Maximum value of 185.\n   *\n   * @default 185\n   */\n  readonly retryAttempts?: number;\n}\n\n/**\n * Bind props to base rule target config.\n * @internal\n */\nexport function bindBaseTargetConfig(props: TargetBaseProps) {\n  let { deadLetterQueue, retryAttempts, maxEventAge } = props;\n\n  return {\n    deadLetterConfig: deadLetterQueue ? { arn: deadLetterQueue?.queueArn } : undefined,\n    retryPolicy: retryAttempts || maxEventAge\n      ? {\n        maximumRetryAttempts: retryAttempts,\n        maximumEventAgeInSeconds: maxEventAge?.toSeconds({ integral: true }),\n      }\n      : undefined,\n  };\n}\n\n\n/**\n * Obtain the Role for the EventBridge event\n *\n * If a role already exists, it will be returned. This ensures that if multiple\n * events have the same target, they will share a role.\n * @internal\n */\nexport function singletonEventRole(scope: IConstruct, policyStatements: iam.PolicyStatement[]): iam.IRole {\n  const id = 'EventsRole';\n  const existing = scope.node.tryFindChild(id) as iam.IRole;\n  if (existing) { return existing; }\n\n  const role = new iam.Role(scope as Construct, id, {\n    roleName: PhysicalName.GENERATE_IF_NEEDED,\n    assumedBy: new iam.ServicePrincipal('events.amazonaws.com'),\n  });\n\n  policyStatements.forEach(role.addToPolicy.bind(role));\n\n  return role;\n}\n\n/**\n * Allows a Lambda function to be called from a rule\n * @internal\n */\nexport function addLambdaPermission(rule: events.IRule, handler: lambda.IFunction): void {\n  let scope: Construct | undefined;\n  let node: ConstructNode = handler.permissionsNode;\n  let permissionId = `AllowEventRule${Names.nodeUniqueId(rule.node)}`;\n  if (rule instanceof Construct) {\n    // Place the Permission resource in the same stack as Rule rather than the Function\n    // This is to reduce circular dependency when the lambda handler and the rule are across stacks.\n    scope = rule;\n    node = rule.node;\n    permissionId = `AllowEventRule${Names.nodeUniqueId(handler.node)}`;\n  }\n  if (!node.tryFindChild(permissionId)) {\n    handler.addPermission(permissionId, {\n      scope,\n      action: 'lambda:InvokeFunction',\n      principal: new iam.ServicePrincipal('events.amazonaws.com'),\n      sourceArn: rule.ruleArn,\n    });\n  }\n}\n\n/**\n * Allow a rule to send events with failed invocation to an Amazon SQS queue.\n * @internal\n */\nexport function addToDeadLetterQueueResourcePolicy(rule: events.IRule, queue: sqs.IQueue) {\n  if (!sameEnvDimension(rule.env.region, queue.env.region)) {\n    throw new Error(`Cannot assign Dead Letter Queue in region ${queue.env.region} to the rule ${Names.nodeUniqueId(rule.node)} in region ${rule.env.region}. Both the queue and the rule must be in the same region.`);\n  }\n\n  // Skip Resource Policy creation if the Queue is not in the same account.\n  // There is no way to add a target onto an imported rule, so we can assume we will run the following code only\n  // in the account where the rule is created.\n  if (sameEnvDimension(rule.env.account, queue.env.account)) {\n    const policyStatementId = `AllowEventRule${Names.nodeUniqueId(rule.node)}`;\n\n    queue.addToResourcePolicy(new iam.PolicyStatement({\n      sid: policyStatementId,\n      principals: [new iam.ServicePrincipal('events.amazonaws.com')],\n      effect: iam.Effect.ALLOW,\n      actions: ['sqs:SendMessage'],\n      resources: [queue.queueArn],\n      conditions: {\n        ArnEquals: {\n          'aws:SourceArn': rule.ruleArn,\n        },\n      },\n    }));\n  } else {\n    Annotations.of(rule).addWarning(`Cannot add a resource policy to your dead letter queue associated with rule ${rule.ruleName} because the queue is in a different account. You must add the resource policy manually to the dead letter queue in account ${queue.env.account}.`);\n  }\n}\n\n\n/**\n * Whether two string probably contain the same environment dimension (region or account)\n *\n * Used to compare either accounts or regions, and also returns true if both\n * are unresolved (in which case both are expted to be \"current region\" or \"current account\").\n * @internal\n */\nfunction sameEnvDimension(dim1: string, dim2: string) {\n  return [TokenComparison.SAME, TokenComparison.BOTH_UNRESOLVED].includes(Token.compareStrings(dim1, dim2));\n}"]}
\No newline at end of file