Skip to content

Commit 4c506fb

Browse files
committed
Initial commit
1 parent 0a590ce commit 4c506fb

10 files changed

+18378
-38
lines changed

.prettierrc

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
{
2+
"trailingComma": "es5",
3+
"tabWidth": 2,
4+
"semi": true,
5+
"singleQuote": true
6+
}

README.md

+12-10
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,16 @@
1-
# Welcome to your CDK TypeScript project!
1+
# EventBridge No Match Rule
22

3-
This is a blank project for TypeScript development with CDK.
3+
AWS EventBridge does not natively provide a way to catch events which do not match any rules defined on an event bus. This AWS CDK project defines a custom construct, `NoMatchRule`, which captures un-matched events and forwards them to an SQS queue of your choosing.
44

5-
The `cdk.json` file tells the CDK Toolkit how to execute your app.
5+
This works by creating a rule which matches on **all** events and forwards the event to a Lambda function. The Lambda function introspects the event bus by fetching all rules and testing the event against each rule's pattern. If no rules match the event is sent to the SQS queue.
66

7-
## Useful commands
7+
Note you must only instantiate `NoMatchRule` at most **once** for a given event bus.
88

9-
* `npm run build` compile typescript to js
10-
* `npm run watch` watch for changes and compile
11-
* `npm run test` perform the jest unit tests
12-
* `cdk deploy` deploy this stack to your default AWS account/region
13-
* `cdk diff` compare deployed stack with current state
14-
* `cdk synth` emits the synthesized CloudFormation template
9+
## Commands
10+
11+
- `npm run build` compile typescript to js
12+
- `npm run watch` watch for changes and compile
13+
- `npm run test` perform the jest unit tests
14+
- `cdk deploy` deploy this stack to your default AWS account/region
15+
- `cdk diff` compare deployed stack with current state
16+
- `cdk synth` emits the synthesized CloudFormation template

bin/eventbridge-no-match-rule.ts

+2-5
Original file line numberDiff line numberDiff line change
@@ -1,21 +1,18 @@
11
#!/usr/bin/env node
22
import 'source-map-support/register';
33
import * as cdk from '@aws-cdk/core';
4-
import { EventbridgeNoMatchRuleStack } from '../lib/eventbridge-no-match-rule-stack';
4+
import { NoMatchRuleStack } from '../lib/no-match-rule-stack';
55

66
const app = new cdk.App();
7-
new EventbridgeNoMatchRuleStack(app, 'EventbridgeNoMatchRuleStack', {
7+
new NoMatchRuleStack(app, 'NoMatchRuleStack', {
88
/* If you don't specify 'env', this stack will be environment-agnostic.
99
* Account/Region-dependent features and context lookups will not work,
1010
* but a single synthesized template can be deployed anywhere. */
11-
1211
/* Uncomment the next line to specialize this stack for the AWS Account
1312
* and Region that are implied by the current CLI configuration. */
1413
// env: { account: process.env.CDK_DEFAULT_ACCOUNT, region: process.env.CDK_DEFAULT_REGION },
15-
1614
/* Uncomment the next line if you know exactly what Account and Region you
1715
* want to deploy the stack to. */
1816
// env: { account: '123456789012', region: 'us-east-1' },
19-
2017
/* For more information, see https://docs.aws.amazon.com/cdk/latest/guide/environments.html */
2118
});

lib/eventbridge-no-match-rule-stack.ts

-9
This file was deleted.

lib/no-match-rule-stack.ts

+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
import * as cdk from '@aws-cdk/core';
2+
import * as events from '@aws-cdk/aws-events';
3+
import * as eventsTargets from '@aws-cdk/aws-events-targets';
4+
import * as sqs from '@aws-cdk/aws-sqs';
5+
6+
import { NoMatchRule } from './no-match-rule';
7+
8+
export class NoMatchRuleStack extends cdk.Stack {
9+
constructor(scope: cdk.Construct, id: string, props?: cdk.StackProps) {
10+
super(scope, id, props);
11+
12+
const eventBus = new events.EventBus(this, 'EventBus');
13+
14+
const queue1 = new sqs.Queue(this, 'Queue1');
15+
new events.Rule(this, 'Rule1', {
16+
eventBus: eventBus,
17+
eventPattern: { source: ['foo'] },
18+
targets: [new eventsTargets.SqsQueue(queue1)],
19+
});
20+
21+
const queue2 = new sqs.Queue(this, 'Queue2');
22+
new events.Rule(this, 'Rule2', {
23+
eventBus: eventBus,
24+
eventPattern: { source: ['bar'] },
25+
targets: [new eventsTargets.SqsQueue(queue2)],
26+
});
27+
28+
const noMatchQueue = new sqs.Queue(this, 'NoMatchQueue');
29+
30+
new NoMatchRule(this, 'NoMatchRule', {
31+
eventBus: eventBus,
32+
queue: noMatchQueue,
33+
});
34+
}
35+
}

lib/no-match-rule.func.ts

+50
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
import {
2+
CloudWatchEventsClient,
3+
ListRulesCommand,
4+
TestEventPatternCommand,
5+
} from '@aws-sdk/client-cloudwatch-events';
6+
import { SQSClient, SendMessageCommand } from '@aws-sdk/client-sqs';
7+
8+
const { EVENT_BUS_ARN, SOURCE_RULE_NAME, QUEUE_URL } = process.env;
9+
10+
const eventsClient = new CloudWatchEventsClient({});
11+
const sqsClient = new SQSClient({});
12+
13+
export const handler = async (event: any) => {
14+
let isFirstIteration = true;
15+
let nextToken: string | undefined;
16+
while (isFirstIteration || nextToken) {
17+
isFirstIteration = false;
18+
const response = await eventsClient.send(
19+
new ListRulesCommand({
20+
EventBusName: EVENT_BUS_ARN!,
21+
NextToken: nextToken,
22+
Limit: 100,
23+
})
24+
);
25+
nextToken = response.NextToken;
26+
for (const rule of response.Rules || []) {
27+
// We ignore the rule targeting this Lambda function
28+
if (rule.Name === SOURCE_RULE_NAME!) continue;
29+
// Check if rule matches event
30+
const response = await eventsClient.send(
31+
new TestEventPatternCommand({
32+
Event: JSON.stringify(event),
33+
EventPattern: rule.EventPattern,
34+
})
35+
);
36+
if (response.Result) {
37+
console.log(`Rule ${rule.Arn} matched the event`);
38+
return;
39+
}
40+
}
41+
}
42+
// If we make it here, no rules match the event
43+
console.log(`No rules matched event. Forwarding event to queue.`);
44+
await sqsClient.send(
45+
new SendMessageCommand({
46+
QueueUrl: QUEUE_URL,
47+
MessageBody: JSON.stringify(event),
48+
})
49+
);
50+
};

lib/no-match-rule.ts

+45
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
import * as cdk from '@aws-cdk/core';
2+
import * as events from '@aws-cdk/aws-events';
3+
import * as eventsTargets from '@aws-cdk/aws-events-targets';
4+
import * as lambdaNodejs from '@aws-cdk/aws-lambda-nodejs';
5+
import * as lambda from '@aws-cdk/aws-lambda';
6+
import * as iam from '@aws-cdk/aws-iam';
7+
import * as sqs from '@aws-cdk/aws-sqs';
8+
9+
interface NoMatchRuleProps {
10+
eventBus: events.IEventBus;
11+
queue: sqs.IQueue;
12+
}
13+
14+
export class NoMatchRule extends cdk.Construct {
15+
constructor(scope: cdk.Construct, id: string, props: NoMatchRuleProps) {
16+
super(scope, id);
17+
18+
const func = new lambdaNodejs.NodejsFunction(this, 'func', {
19+
runtime: lambda.Runtime.NODEJS_14_X,
20+
memorySize: 256,
21+
environment: {
22+
EVENT_BUS_ARN: props.eventBus.eventBusArn,
23+
QUEUE_URL: props.queue.queueUrl,
24+
},
25+
});
26+
func.addToRolePolicy(
27+
new iam.PolicyStatement({
28+
effect: iam.Effect.ALLOW,
29+
actions: ['events:ListRules', 'events:TestEventPattern'],
30+
resources: ['*'],
31+
})
32+
);
33+
props.queue.grantSendMessages(func);
34+
35+
const ruleName = `${id}CatchAllRule`;
36+
new events.Rule(this, 'Rule', {
37+
ruleName: ruleName,
38+
eventBus: props.eventBus,
39+
eventPattern: { source: [JSON.stringify({ prefix: '' })] },
40+
targets: [new eventsTargets.LambdaFunction(func)],
41+
});
42+
43+
func.addEnvironment('SOURCE_RULE_NAME', ruleName);
44+
}
45+
}

0 commit comments

Comments
 (0)