🚀 Big News: Socket Acquires Coana to Bring Reachability Analysis to Every Appsec Team.Learn more
Socket
Book a DemoInstallSign in
Socket

lambda-error-sns-sender

Package Overview
Dependencies
Maintainers
1
Versions
57
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

lambda-error-sns-sender - npm Package Compare versions

Comparing version

to
0.0.53

lambda-error-sns-sender.png

3

.projenrc.ts

@@ -132,5 +132,4 @@ import { awscdk } from 'projen';

],
}),
console.log();
});
project.synth();

@@ -5,7 +5,16 @@ # Lambda Error SNS Sender

How does it work:
- Lambda is subscribed to the SNS topic where you receive your alarms. There is a message body subscription filter that limits errors based on the Lambda error metric. If you defied your metic in some other way, not the default one, you will have to change that filter.
- Lambda analyzes the message, finds a log of the failed Lambda, and queries the log for the period that the metic was configured, plus some additional safety time, so we do not miss that error message. It extracts just a number of errors that can fit in one SNS message.
-Lambda sends errors to the same SNS you use for alerts. So apart from the common Alarm message, you will receive an additional one with error messages.
How does it work?
1. **Lambda is subscribed to the SNS topic where you receive your alarms.** There is a message body subscription filter that limits errors based on the Lambda error metric. You must change the filter if you defied your metric in some other way, not the default one.
1. **Lambda analyzes the message**, finds a log of the failed Lambda, and queries the log group for the period that the metric was configured, plus some additional safety time, so we do not miss that error message. It extracts just a number of errors that can fit in one SNS message.
1. **Lambda sends errors to the same SNS** that you use for alerts. So, apart from the Alarm message for changing the error state, you will receive an additional one with detailed error messages.
![lambda-error-sns-sender](lambda-error-sns-sender.png)
- **[CDK construct](https://constructs.dev/packages/lambda-error-sns-sender)**
If you are building your system with CDK (or [SST](https://sst.dev/)). Available for TypeScript, Java, C#, Python, and Go.
- **[CloudFormation](https://lambda-error-sns-sender.s3.eu-west-1.amazonaws.com/lambda-error-sns-sender.yaml)**
For existing solutions, so you do not have to modify them. You deploy and point to the existing SNS used for CloudWatch alarms.
**If you are interested how to use it and how was build see the [blog post](https://www.serverlesslife.com/live/Improving_CloudWatch_Alarms_for_Lambda_Errors.html).**
# API Reference <a name="API Reference" id="api-reference"></a>

@@ -12,0 +21,0 @@

@@ -107,3 +107,2 @@ import { Buffer } from 'node:buffer';

//const joinedLogs = `LAMBDA ${functionName} ERRORS:\n\n${logs.join('\n')}`;
const joinedLogs = logs.join('\n');

@@ -110,0 +109,0 @@

@@ -92,9 +92,2 @@ "use strict";

exports.CloudFormationExportStack = CloudFormationExportStack;
const app = new aws_cdk_lib_1.App();
new CloudFormationExportStack(app, 'lambda-error-sns-sender-cf', {
synthesizer: new aws_cdk_lib_1.DefaultStackSynthesizer({
generateBootstrapVersionRule: false,
}),
});
app.synth();
function setNodeNames(constructs, parentName) {

@@ -117,2 +110,2 @@ let i = 0;

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudFormationExport.js","sourceRoot":"","sources":["../src/cloudFormationExport.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  App,\n  Stack,\n  StackProps,\n  CfnParameter,\n  DefaultStackSynthesizer,\n} from 'aws-cdk-lib';\nimport * as cdk from 'aws-cdk-lib';\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 { Construct, IConstruct } from 'constructs';\nimport { LambdaErrorSnsSender } from './index';\n\nexport class CloudFormationExportStack extends Stack {\n  constructor(scope: Construct, id: string, props: StackProps = {}) {\n    super(scope, id, props);\n\n    const noOfParameters = 5;\n    const snsTopics: sns.Topic[] = [];\n    const conditionsDict: Record<string, cdk.CfnCondition> = {};\n\n    for (let i = 1; i <= noOfParameters; i++) {\n      const snsTopicArnParam = new CfnParameter(this, `snsTopicArn${i}`, {\n        type: 'String',\n        description: 'SNS Topic ARN receive alarms and send Lambda errors to.',\n      });\n\n      const snsTopic = sns.Topic.fromTopicArn(\n        this,\n        `snsTopic${i}`,\n        snsTopicArnParam.valueAsString,\n      );\n      snsTopics.push(snsTopic as sns.Topic);\n\n      const condition = new cdk.CfnCondition(this, `conditionSnsTopic${i}`, {\n        expression: cdk.Fn.conditionNot(\n          cdk.Fn.conditionEquals('', snsTopic.topicArn),\n        ),\n      });\n\n      conditionsDict[snsTopicArnParam.valueAsString] = condition;\n    }\n\n    const maxNumberOfLogsParam = new CfnParameter(this, `maxNumberOfLogs`, {\n      type: 'Number',\n      description: 'Max number of logs to send to SNS Topic.',\n    });\n\n    const lambdaErrorSnsSender = new LambdaErrorSnsSender(\n      this,\n      'lambdaErrorSnsSender',\n      {\n        snsTopics: snsTopics,\n        maxNumberOfLogs: maxNumberOfLogsParam.valueAsNumber,\n      },\n    );\n\n    const subscriptions = lambdaErrorSnsSender.node\n      .findAll()\n      .filter((s) => s instanceof sns.CfnSubscription) as sns.CfnSubscription[];\n\n    for (const subscription of subscriptions) {\n      if (conditionsDict[subscription.topicArn]) {\n        subscription.cfnOptions.condition =\n          conditionsDict[subscription.topicArn];\n      }\n    }\n\n    const snsErrorFunc = lambdaErrorSnsSender.node.findChild(\n      'lambdaSnsError',\n    ) as lambda.Function;\n\n    for (const lambdaPermission of snsErrorFunc.permissionsNode.children) {\n      if (lambdaPermission instanceof lambda.CfnPermission) {\n        if (\n          lambdaPermission.sourceArn &&\n          conditionsDict[lambdaPermission.sourceArn]\n        ) {\n          lambdaPermission.cfnOptions.condition =\n            conditionsDict[lambdaPermission.sourceArn];\n        }\n      }\n    }\n\n    const lambdaPolicy = snsErrorFunc.node\n      .findAll()\n      .find((c) => c instanceof iam.CfnPolicy) as iam.CfnPolicy;\n\n    const newPolicyStatements = [];\n    for (const statement of (lambdaPolicy.policyDocument?.statements ??\n      []) as iam.PolicyStatement[]) {\n      let addCondition = false;\n      let condition: cdk.CfnCondition | undefined;\n\n      for (const resource of statement.resources) {\n        if (conditionsDict[resource]) {\n          addCondition = true;\n          condition = conditionsDict[resource];\n          break;\n        }\n      }\n\n      if (addCondition && condition) {\n        const newStatement = {\n          'Fn::If': [\n            condition.logicalId,\n            statement.toJSON(),\n            ,\n            { Ref: 'AWS::NoValue' },\n          ],\n        };\n\n        newPolicyStatements.push(newStatement);\n      } else {\n        newPolicyStatements.push(statement.toJSON());\n      }\n    }\n\n    lambdaPolicy.policyDocument = {\n      Statement: newPolicyStatements,\n    };\n\n    setNodeNames(this.node.children);\n  }\n}\n\nconst app = new App();\n\nnew CloudFormationExportStack(app, 'lambda-error-sns-sender-cf', {\n  synthesizer: new DefaultStackSynthesizer({\n    generateBootstrapVersionRule: false,\n  }),\n});\n\napp.synth();\n\nfunction setNodeNames(constructs: IConstruct[], parentName?: string) {\n  let i = 0;\n  for (const construct of constructs) {\n    let newParentName = parentName;\n\n    if (construct.node.defaultChild instanceof cdk.CfnElement) {\n      newParentName = parentName\n        ? `${parentName}${construct.node.id}`\n        : construct.node.id;\n      construct.node.defaultChild.overrideLogicalId(newParentName);\n    }\n\n    if (construct instanceof lambda.CfnPermission) {\n      construct.overrideLogicalId(`${newParentName}Permission${i}`);\n      i++;\n    }\n\n    setNodeNames(construct.node.children, newParentName);\n  }\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudFormationExport.js","sourceRoot":"","sources":["../src/cloudFormationExport.ts"],"names":[],"mappings":"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","sourcesContent":["import { Stack, StackProps, CfnParameter } from 'aws-cdk-lib';\nimport * as cdk from 'aws-cdk-lib';\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 { Construct, IConstruct } from 'constructs';\nimport { LambdaErrorSnsSender } from './index';\n\nexport class CloudFormationExportStack extends Stack {\n  constructor(scope: Construct, id: string, props: StackProps = {}) {\n    super(scope, id, props);\n\n    const noOfParameters = 5;\n    const snsTopics: sns.Topic[] = [];\n    const conditionsDict: Record<string, cdk.CfnCondition> = {};\n\n    for (let i = 1; i <= noOfParameters; i++) {\n      const snsTopicArnParam = new CfnParameter(this, `snsTopicArn${i}`, {\n        type: 'String',\n        description: 'SNS Topic ARN receive alarms and send Lambda errors to.',\n      });\n\n      const snsTopic = sns.Topic.fromTopicArn(\n        this,\n        `snsTopic${i}`,\n        snsTopicArnParam.valueAsString,\n      );\n      snsTopics.push(snsTopic as sns.Topic);\n\n      const condition = new cdk.CfnCondition(this, `conditionSnsTopic${i}`, {\n        expression: cdk.Fn.conditionNot(\n          cdk.Fn.conditionEquals('', snsTopic.topicArn),\n        ),\n      });\n\n      conditionsDict[snsTopicArnParam.valueAsString] = condition;\n    }\n\n    const maxNumberOfLogsParam = new CfnParameter(this, `maxNumberOfLogs`, {\n      type: 'Number',\n      description: 'Max number of logs to send to SNS Topic.',\n    });\n\n    const lambdaErrorSnsSender = new LambdaErrorSnsSender(\n      this,\n      'lambdaErrorSnsSender',\n      {\n        snsTopics: snsTopics,\n        maxNumberOfLogs: maxNumberOfLogsParam.valueAsNumber,\n      },\n    );\n\n    const subscriptions = lambdaErrorSnsSender.node\n      .findAll()\n      .filter((s) => s instanceof sns.CfnSubscription) as sns.CfnSubscription[];\n\n    for (const subscription of subscriptions) {\n      if (conditionsDict[subscription.topicArn]) {\n        subscription.cfnOptions.condition =\n          conditionsDict[subscription.topicArn];\n      }\n    }\n\n    const snsErrorFunc = lambdaErrorSnsSender.node.findChild(\n      'lambdaSnsError',\n    ) as lambda.Function;\n\n    for (const lambdaPermission of snsErrorFunc.permissionsNode.children) {\n      if (lambdaPermission instanceof lambda.CfnPermission) {\n        if (\n          lambdaPermission.sourceArn &&\n          conditionsDict[lambdaPermission.sourceArn]\n        ) {\n          lambdaPermission.cfnOptions.condition =\n            conditionsDict[lambdaPermission.sourceArn];\n        }\n      }\n    }\n\n    const lambdaPolicy = snsErrorFunc.node\n      .findAll()\n      .find((c) => c instanceof iam.CfnPolicy) as iam.CfnPolicy;\n\n    const newPolicyStatements = [];\n    for (const statement of (lambdaPolicy.policyDocument?.statements ??\n      []) as iam.PolicyStatement[]) {\n      let addCondition = false;\n      let condition: cdk.CfnCondition | undefined;\n\n      for (const resource of statement.resources) {\n        if (conditionsDict[resource]) {\n          addCondition = true;\n          condition = conditionsDict[resource];\n          break;\n        }\n      }\n\n      if (addCondition && condition) {\n        const newStatement = {\n          'Fn::If': [\n            condition.logicalId,\n            statement.toJSON(),\n            ,\n            { Ref: 'AWS::NoValue' },\n          ],\n        };\n\n        newPolicyStatements.push(newStatement);\n      } else {\n        newPolicyStatements.push(statement.toJSON());\n      }\n    }\n\n    lambdaPolicy.policyDocument = {\n      Statement: newPolicyStatements,\n    };\n\n    setNodeNames(this.node.children);\n  }\n}\n\nfunction setNodeNames(constructs: IConstruct[], parentName?: string) {\n  let i = 0;\n  for (const construct of constructs) {\n    let newParentName = parentName;\n\n    if (construct.node.defaultChild instanceof cdk.CfnElement) {\n      newParentName = parentName\n        ? `${parentName}${construct.node.id}`\n        : construct.node.id;\n      construct.node.defaultChild.overrideLogicalId(newParentName);\n    }\n\n    if (construct instanceof lambda.CfnPermission) {\n      construct.overrideLogicalId(`${newParentName}Permission${i}`);\n      i++;\n    }\n\n    setNodeNames(construct.node.children, newParentName);\n  }\n}\n"]}

@@ -59,3 +59,3 @@ "use strict";

_a = JSII_RTTI_SYMBOL_1;
LambdaErrorSnsSender[_a] = { fqn: "lambda-error-sns-sender.LambdaErrorSnsSender", version: "0.0.52" };
LambdaErrorSnsSender[_a] = { fqn: "lambda-error-sns-sender.LambdaErrorSnsSender", version: "0.0.53" };
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7QUFBQSw2QkFBNkI7QUFDN0IsbUNBQW1DO0FBQ25DLDJDQUEyQztBQUMzQyxpREFBaUQ7QUFDakQsMkNBQTJDO0FBRTNDLGtFQUFrRTtBQUNsRSwyQ0FBdUM7QUFVdkMsTUFBYSxvQkFBcUIsU0FBUSxzQkFBUztJQUNqRCxZQUFZLEtBQWdCLEVBQUUsRUFBVSxFQUFFLEtBQWdDO1FBQ3hFLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUM7UUFFakIsSUFBSSxDQUFDLEtBQUssQ0FBQyxTQUFTLElBQUksS0FBSyxFQUFFLFNBQVMsQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFO1lBQ3JELE1BQU0sSUFBSSxLQUFLLENBQUMsd0JBQXdCLENBQUMsQ0FBQztTQUMzQztRQUVELE1BQU0sWUFBWSxHQUFHLElBQUksTUFBTSxDQUFDLFFBQVEsQ0FBQyxJQUFJLEVBQUUsZ0JBQWdCLEVBQUU7WUFDL0QsT0FBTyxFQUFFLGVBQWU7WUFDeEIsSUFBSSxFQUFFLE1BQU0sQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUN6QixJQUFJLENBQUMsSUFBSSxDQUFDLFNBQVMsRUFBRSxpQ0FBaUMsQ0FBQyxDQUN4RDtZQUNELE9BQU8sRUFBRSxNQUFNLENBQUMsT0FBTyxDQUFDLFdBQVc7WUFDbkMsV0FBVyxFQUFFO2dCQUNYLGtCQUFrQixFQUFFLEtBQUssRUFBRSxlQUFlLEVBQUUsUUFBUSxFQUFFLElBQUksS0FBSzthQUNoRTtZQUNELE9BQU8sRUFBRSxHQUFHLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUM7U0FDakMsQ0FBQyxDQUFDO1FBRUgsaUNBQWlDO1FBQ2pDLFlBQVksQ0FBQyxlQUFlLENBQzFCLElBQUksR0FBRyxDQUFDLGVBQWUsQ0FBQztZQUN0QixPQUFPLEVBQUUsQ0FBQyxzQkFBc0IsQ0FBQztZQUNqQyxTQUFTLEVBQUUsQ0FBQyxHQUFHLENBQUMsRUFBRSxzREFBc0Q7U0FDekUsQ0FBQyxDQUNILENBQUM7UUFDRixLQUFLLE1BQU0sUUFBUSxJQUFJLEtBQUssRUFBRSxTQUFTLEVBQUU7WUFDdkMsTUFBTSxhQUFhLEdBQUc7Z0JBQ3BCLFVBQVUsRUFBRSxHQUFHLENBQUMsY0FBYyxDQUFDLE1BQU0sQ0FDbkMsR0FBRyxDQUFDLGtCQUFrQixDQUFDLFlBQVksQ0FBQztvQkFDbEMsU0FBUyxFQUFFLENBQUMsUUFBUSxDQUFDO2lCQUN0QixDQUFDLENBQ0g7Z0JBQ0QsU0FBUyxFQUFFLEdBQUcsQ0FBQyxjQUFjLENBQUMsTUFBTSxDQUNsQyxHQUFHLENBQUMsa0JBQWtCLENBQUMsWUFBWSxDQUFDO29CQUNsQyxTQUFTLEVBQUUsQ0FBQyxZQUFZLENBQUM7aUJBQzFCLENBQUMsQ0FDSDtnQkFDRCxhQUFhLEVBQUUsR0FBRyxDQUFDLGNBQWMsQ0FBQyxNQUFNLENBQ3RDLEdBQUcsQ0FBQyxrQkFBa0IsQ0FBQyxZQUFZLENBQUM7b0JBQ2xDLFNBQVMsRUFBRSxDQUFDLFdBQVcsQ0FBQztpQkFDekIsQ0FBQyxDQUNIO2dCQUNELFNBQVMsRUFBRSxHQUFHLENBQUMsY0FBYyxDQUFDLE1BQU0sQ0FDbEMsR0FBRyxDQUFDLGtCQUFrQixDQUFDLFlBQVksQ0FBQztvQkFDbEMsU0FBUyxFQUFFLENBQUMsS0FBSyxDQUFDO2lCQUNuQixDQUFDLENBQ0g7YUFDRixDQUFDO1lBRUYsUUFBUSxDQUFDLGVBQWUsQ0FDdEIsSUFBSSxZQUFZLENBQUMsa0JBQWtCLENBQUMsWUFBWSxFQUFFO2dCQUNoRCwyQkFBMkIsRUFBRTtvQkFDM0IsT0FBTyxFQUFFLEdBQUcsQ0FBQyxjQUFjLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxNQUFNLElBQUksYUFBYSxDQUFDO2lCQUNsRTthQUNGLENBQUMsQ0FDSCxDQUFDO1lBRUYsUUFBUSxDQUFDLFlBQVksQ0FBQyxZQUFZLENBQUMsQ0FBQztTQUNyQztJQUNILENBQUM7O0FBN0RILG9EQThEQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIHBhdGggZnJvbSAncGF0aCc7XG5pbXBvcnQgKiBhcyBjZGsgZnJvbSAnYXdzLWNkay1saWInO1xuaW1wb3J0ICogYXMgaWFtIGZyb20gJ2F3cy1jZGstbGliL2F3cy1pYW0nO1xuaW1wb3J0ICogYXMgbGFtYmRhIGZyb20gJ2F3cy1jZGstbGliL2F3cy1sYW1iZGEnO1xuaW1wb3J0ICogYXMgc25zIGZyb20gJ2F3cy1jZGstbGliL2F3cy1zbnMnO1xuaW1wb3J0IHsgRmlsdGVyT3JQb2xpY3kgfSBmcm9tICdhd3MtY2RrLWxpYi9hd3Mtc25zJztcbmltcG9ydCAqIGFzIHN1YnNjcmlwdGlvbiBmcm9tICdhd3MtY2RrLWxpYi9hd3Mtc25zLXN1YnNjcmlwdGlvbnMnO1xuaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSAnY29uc3RydWN0cyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgTGFtYmRhRXJyb3JTbnNTZW5kZXJQcm9wcyBleHRlbmRzIGNkay5TdGFja1Byb3BzIHtcbiAgcmVhZG9ubHkgc25zVG9waWNzOiBzbnMuVG9waWNbXTtcbiAgcmVhZG9ubHkgbWF4TnVtYmVyT2ZMb2dzPzogbnVtYmVyO1xuICByZWFkb25seSBmaWx0ZXI/OiB7XG4gICAgW2F0dHJpYnV0ZTogc3RyaW5nXTogRmlsdGVyT3JQb2xpY3k7XG4gIH07XG59XG5cbmV4cG9ydCBjbGFzcyBMYW1iZGFFcnJvclNuc1NlbmRlciBleHRlbmRzIENvbnN0cnVjdCB7XG4gIGNvbnN0cnVjdG9yKHNjb3BlOiBDb25zdHJ1Y3QsIGlkOiBzdHJpbmcsIHByb3BzOiBMYW1iZGFFcnJvclNuc1NlbmRlclByb3BzKSB7XG4gICAgc3VwZXIoc2NvcGUsIGlkKTtcblxuICAgIGlmICghcHJvcHMuc25zVG9waWNzIHx8IHByb3BzPy5zbnNUb3BpY3MubGVuZ3RoID09PSAwKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoJ05vIFNOUyBUb3BpY3MgcHJvdmlkZWQnKTtcbiAgICB9XG5cbiAgICBjb25zdCBzbnNFcnJvckZ1bmMgPSBuZXcgbGFtYmRhLkZ1bmN0aW9uKHRoaXMsICdsYW1iZGFTbnNFcnJvcicsIHtcbiAgICAgIGhhbmRsZXI6ICdpbmRleC5oYW5kbGVyJyxcbiAgICAgIGNvZGU6IGxhbWJkYS5Db2RlLmZyb21Bc3NldChcbiAgICAgICAgcGF0aC5qb2luKF9fZGlybmFtZSwgJy4uL2xpYi9mdW5jdGlvbnMvbGFtYmRhU25zRXJyb3InKSxcbiAgICAgICksXG4gICAgICBydW50aW1lOiBsYW1iZGEuUnVudGltZS5OT0RFSlNfMThfWCxcbiAgICAgIGVudmlyb25tZW50OiB7XG4gICAgICAgIE1BWF9OVU1CRVJfT0ZfTE9HUzogcHJvcHM/Lm1heE51bWJlck9mTG9ncz8udG9TdHJpbmcoKSA/PyAnMTAwJyxcbiAgICAgIH0sXG4gICAgICB0aW1lb3V0OiBjZGsuRHVyYXRpb24ubWludXRlcygxKSxcbiAgICB9KTtcblxuICAgIC8vZ3JhbmQgYWNjZXNzIHRvIGNsb3Vkd2F0Y2ggbG9nc1xuICAgIHNuc0Vycm9yRnVuYy5hZGRUb1JvbGVQb2xpY3koXG4gICAgICBuZXcgaWFtLlBvbGljeVN0YXRlbWVudCh7XG4gICAgICAgIGFjdGlvbnM6IFsnbG9nczpGaWx0ZXJMb2dFdmVudHMnXSxcbiAgICAgICAgcmVzb3VyY2VzOiBbJyonXSwgLy8gSSBkbyBub3Qga25vdyB3aGljaCByZXNvdXJjZXMgSSB3aWxsIG5lZWQgdG8gYWNjZXNzXG4gICAgICB9KSxcbiAgICApO1xuICAgIGZvciAoY29uc3Qgc25zVG9waWMgb2YgcHJvcHM/LnNuc1RvcGljcykge1xuICAgICAgY29uc3QgZGVmYXVsdEZpbHRlciA9IHtcbiAgICAgICAgTWV0cmljTmFtZTogc25zLkZpbHRlck9yUG9saWN5LmZpbHRlcihcbiAgICAgICAgICBzbnMuU3Vic2NyaXB0aW9uRmlsdGVyLnN0cmluZ0ZpbHRlcih7XG4gICAgICAgICAgICBhbGxvd2xpc3Q6IFsnRXJyb3JzJ10sXG4gICAgICAgICAgfSksXG4gICAgICAgICksXG4gICAgICAgIE5hbWVzcGFjZTogc25zLkZpbHRlck9yUG9saWN5LmZpbHRlcihcbiAgICAgICAgICBzbnMuU3Vic2NyaXB0aW9uRmlsdGVyLnN0cmluZ0ZpbHRlcih7XG4gICAgICAgICAgICBhbGxvd2xpc3Q6IFsnQVdTL0xhbWJkYSddLFxuICAgICAgICAgIH0pLFxuICAgICAgICApLFxuICAgICAgICBTdGF0aXN0aWNUeXBlOiBzbnMuRmlsdGVyT3JQb2xpY3kuZmlsdGVyKFxuICAgICAgICAgIHNucy5TdWJzY3JpcHRpb25GaWx0ZXIuc3RyaW5nRmlsdGVyKHtcbiAgICAgICAgICAgIGFsbG93bGlzdDogWydTdGF0aXN0aWMnXSxcbiAgICAgICAgICB9KSxcbiAgICAgICAgKSxcbiAgICAgICAgU3RhdGlzdGljOiBzbnMuRmlsdGVyT3JQb2xpY3kuZmlsdGVyKFxuICAgICAgICAgIHNucy5TdWJzY3JpcHRpb25GaWx0ZXIuc3RyaW5nRmlsdGVyKHtcbiAgICAgICAgICAgIGFsbG93bGlzdDogWydTVU0nXSxcbiAgICAgICAgICB9KSxcbiAgICAgICAgKSxcbiAgICAgIH07XG5cbiAgICAgIHNuc1RvcGljLmFkZFN1YnNjcmlwdGlvbihcbiAgICAgICAgbmV3IHN1YnNjcmlwdGlvbi5MYW1iZGFTdWJzY3JpcHRpb24oc25zRXJyb3JGdW5jLCB7XG4gICAgICAgICAgZmlsdGVyUG9saWN5V2l0aE1lc3NhZ2VCb2R5OiB7XG4gICAgICAgICAgICBUcmlnZ2VyOiBzbnMuRmlsdGVyT3JQb2xpY3kucG9saWN5KHByb3BzLmZpbHRlciA/PyBkZWZhdWx0RmlsdGVyKSxcbiAgICAgICAgICB9LFxuICAgICAgICB9KSxcbiAgICAgICk7XG5cbiAgICAgIHNuc1RvcGljLmdyYW50UHVibGlzaChzbnNFcnJvckZ1bmMpO1xuICAgIH1cbiAgfVxufVxuIl19

@@ -44,7 +44,7 @@ {

"@types/node": "^16",
"@typescript-eslint/eslint-plugin": "^5",
"@typescript-eslint/parser": "^5",
"@typescript-eslint/eslint-plugin": "^6",
"@typescript-eslint/parser": "^6",
"aws-cdk-lib": "2.70.0",
"constructs": "10.0.5",
"esbuild": "^0.19.2",
"esbuild": "^0.19.4",
"esbuild-runner": "^2.2.2",

@@ -54,16 +54,16 @@ "eslint": "^8",

"eslint-import-resolver-node": "^0.3.9",
"eslint-import-resolver-typescript": "^3.6.0",
"eslint-plugin-import": "^2.28.0",
"eslint-import-resolver-typescript": "^3.6.1",
"eslint-plugin-import": "^2.28.1",
"eslint-plugin-prettier": "^5.0.0",
"jsii": "^5.1.9",
"jsii-diff": "^1.87.0",
"jsii-docgen": "^9.1.2",
"jsii-pacmak": "^1.87.0",
"jsii-diff": "^1.89.0",
"jsii-docgen": "^9.2.2",
"jsii-pacmak": "^1.89.0",
"jsii-rosetta": "^5.1.9",
"npm-check-updates": "^16",
"prettier": "^3.0.1",
"projen": "^0.72.11",
"prettier": "^3.0.3",
"projen": "^0.73.38",
"standard-version": "^9",
"ts-node": "^10.9.1",
"typescript": "^5.1.6"
"typescript": "^5.2.2"
},

@@ -89,3 +89,3 @@ "peerDependencies": {

"license": "Apache-2.0",
"version": "0.0.52",
"version": "0.0.53",
"types": "lib/index.d.ts",

@@ -92,0 +92,0 @@ "stability": "stable",

@@ -5,5 +5,15 @@ # Lambda Error SNS Sender

How does it work:
- Lambda is subscribed to the SNS topic where you receive your alarms. There is a message body subscription filter that limits errors based on the Lambda error metric. If you defied your metic in some other way, not the default one, you will have to change that filter.
- Lambda analyzes the message, finds a log of the failed Lambda, and queries the log for the period that the metic was configured, plus some additional safety time, so we do not miss that error message. It extracts just a number of errors that can fit in one SNS message.
-Lambda sends errors to the same SNS you use for alerts. So apart from the common Alarm message, you will receive an additional one with error messages.
How does it work?
1. **Lambda is subscribed to the SNS topic where you receive your alarms.** There is a message body subscription filter that limits errors based on the Lambda error metric. You must change the filter if you defied your metric in some other way, not the default one.
1. **Lambda analyzes the message**, finds a log of the failed Lambda, and queries the log group for the period that the metric was configured, plus some additional safety time, so we do not miss that error message. It extracts just a number of errors that can fit in one SNS message.
1. **Lambda sends errors to the same SNS** that you use for alerts. So, apart from the Alarm message for changing the error state, you will receive an additional one with detailed error messages.
![lambda-error-sns-sender](lambda-error-sns-sender.png)
- **[CDK construct](https://constructs.dev/packages/lambda-error-sns-sender)**
If you are building your system with CDK (or [SST](https://sst.dev/)). Available for TypeScript, Java, C#, Python, and Go.
- **[CloudFormation](https://lambda-error-sns-sender.s3.eu-west-1.amazonaws.com/lambda-error-sns-sender.yaml)**
For existing solutions, so you do not have to modify them. You deploy and point to the existing SNS used for CloudWatch alarms.
**If you are interested how to use it and how was build see the [blog post](https://www.serverlesslife.com/live/Improving_CloudWatch_Alarms_for_Lambda_Errors.html).**

@@ -140,3 +140,2 @@ import * as fs from 'fs';

Body: body,
//ACL: 'public-read',
};

@@ -143,0 +142,0 @@

Sorry, the diff of this file is not supported yet

Sorry, the diff of this file is not supported yet

Sorry, the diff of this file is not supported yet