Skip to main content

Monitoring

SST integrates with all the major observability providers to help you monitor your apps in production.

Once your app has been deployed to production, there are a few different options on how to monitor your Lambda functions. Let's look at them here.


Datadog

Datadog offers an End-to-end Serverless Monitoring solution that works with Lambda functions. It consists of the following three distinct libraries:

  1. datadog-cdk-constructs-v2 — a CDK construct that sets up Lambda layers and environment variables. It also provides a Lambda handler override that keeps the Datadog API key in an AWS Secrets Manager Secret. This is highly recommended, although technically optional, as you could add the Lambda layers, environment variables, and the Datadog API key separately through SST if you prefer.
  2. dd-trace — Datadog's JavaScript APM tracer. This is necessary if you wish to add traces to your Lambda functions.
  3. datadog-lambda-js — a Lambda-specific version of dd-trace. This is necessary if you wish to add traces or custom metrics to your Lambda functions.

Start by adding these to your project:

npm install --save-dev datadog-cdk-constructs-v2 dd-trace datadog-lambda-js
note

These are added as dev dependencies. The datadog-cdk-constructs-v2 package is used by SST, not by your Lambda code. The dd-trace and datadog-lambda-js packages are not be bundled with your Lambda functions, but rather be loaded from the Lambda layer provided by Datadog's CDK construct.

When you setup an AWS account to be monitored by Datadog, the process uses a CloudFormation stack to set up an API key for forwarding logs, metrics, traces, and other observability data to your Datadog account. The CloudFormation stack sets up this API key in an AWS Secrets Manager Secret, and it can be re-used by the Datadog serverless monitoring CDK construct.

To monitor all the functions in an app, add the following to your sst.config.ts:

sst.config.ts
import { SSTConfig } from "sst";
import { Stack } from "sst/constructs";
import { Datadog } from "datadog-cdk-constructs-v2";
import { Effect, PolicyStatement } from "aws-cdk-lib/aws-iam";

export default {
config() {
// Config
},
async stacks(app) {
// Replace with the API Key Secret ARN from your
// Datadog Integration CloudFormation stack
const datadogApiKeySecretArn =
"arn:aws:secretsmanager:us-east-1:123456789012:secret:DdApiKeySecret-abcdef123456-789ABC";

// Don't enable locally
const enableDatadog = !app.local;

if (enableDatadog) {
// Allow functions to access secret
app.addDefaultFunctionPermissions([
new PolicyStatement({
effect: Effect.ALLOW,
resources: [datadogApiKeySecretArn],
actions: ["secretsmanager:GetSecretValue"],
}),
]);
// Exclude from the function bundle
// since they'll be loaded from the Layer
app.setDefaultFunctionProps({
nodejs: {
esbuild: {
external: ["dd-trace", "datadog-lambda-js"],
},
},
});
}

// Add your stacks
app.stack(/* ... */);

if (enableDatadog) {
// Attach the Datadog contruct to each stack
app.node.children.forEach((stack) => {
if (stack instanceof Stack) {
const datadog = new Datadog(stack, "datadog", {
// Get the latest version from
// https://github.com/Datadog/datadog-lambda-js/releases
nodeLayerVersion: 87,
// Get the latest version from
// https://github.com/Datadog/datadog-lambda-extension/releases
extensionLayerVersion: 40,
site: "datadoghq.com",
apiKeySecretArn: datadogApiKeySecretArn,
env: app.stage,
service: app.name,
// Just a recommendation, feel free to change the version per your CI/CD
version:
process.env.SEED_BUILD_SERVICE_SHA ||
process.env.GITHUB_SHA ||
undefined,
});

datadog.addLambdaFunctions(stack.getAllFunctions());
}
});
}
},
} satisfies SSTConfig;

Then in each Lambda function, should wrap it with the handler provided by datadog-lambda-js:

packages/functions/src/lambda.ts
import { datadog } from "datadog-lambda-js";

export const handler = datadog(async () => {
return {
statusCode: 200,
body: "Hello World!",
};
});

For more details, check out the Datadog docs.


Baselime

Baselime offers a Serverless Observability and Error-Tracking Platform for your AWS Lambda functions and Amazon ECS services.

To get started, sign up for an account. Then connect your AWS account by deploying their CloudFormation stack in your AWS account.

Baselime will start to automatically collect logs, metrics and AWS X-Ray traces from your AWS account.

Then to enable OpenTelemetry tracing for a function, add a baselime:tracing tag and set it to true.

stacks/Foo.js
import * as cdk from "aws-cdk-lib";

cdk.Tags.of(myfunc).add("baselime:tracing", "true");

To enable OpenTelemetry tracing on all your functions, you can use the Stack construct's getAllFunctions method and do the following at the bottom of your stack definition.

stacks/Foo.js
import * as cdk from "aws-cdk-lib";

stack
.getAllFunctions()
.forEach((fn) => cdk.Tags.of(fn).add("baselime:tracing", "true"));

To enable capturing logs from your ECS services, you can use AWS Firelens log driver to send logs to Baselime’s backend.

stacks/Foo.js
import { StackContext, Service } from "sst/constructs";
import { StringParameter } from 'aws-cdk-lib/aws-ssm';
import * as ecs from 'aws-cdk-lib/aws-ecs';

export function API({ stack }: StackContext) {
// Get your API key from the Baselime console and store it in SSM or secrets manager
const key = StringParameter.valueForStringParameter(stack, 'baselime-key');

const service = new Service(stack, 'sst-service', {
path: './',
environment: {
BASELIME_KEY: key
},
cdk: {
container: {
// Set the container logging to send logs to the Baselime dedicated ECS logs endpoint
logging: new ecs.FireLensLogDriver({
options: {
"Name": "http",
"Host": "ecs-logs-ingest.baselime.io",
"Port": "443",
"TLS": "on",
"format": "json",
"retry_limit": "2",
"header": `x-api-key ${key}`,
},
}),
}
}
});
}

Baselime also enables you to create dashboards and alerts using SST. Install the @baselime/cdk package from npm.

npm install --save-dev @baselime/cdk

To create an alert that sends all error and critial logs to a Slack channel:

stacks/Foo.js
import { Alert, filter, Baselime } from "@baselime/cdk";

// Get your API key from the Baselime console and store it in SSM or secrets manager
const key = StringParameter.valueForStringParameter(stack, 'baselime-key');

Baselime.init(stack, {
apiKey: key,
});

new Alert("service-errors", {
parameters: {
query: {
filters: [
filter.inArray("LogLevel", ["ERROR", "CRITICAL"]),
],
},
channels: [{ type: "slack", targets: ["baselime-alerts"] }]
},
});

For more details, check out the Baselime docs.

Sentry

Sentry offers Serverless Error Monitoring for your Lambda functions. Integration is done through a Lambda Layer.

Head over to the Layer that Sentry provides, select your region and copy the layer ARN. Then add the Layer to your stack.

stacks/Foo.js
import { LayerVersion } from "aws-cdk-lib/aws-lambda";

const sentry = LayerVersion.fromLayerVersionArn(
stack,
"SentryLayer",
`arn:aws:lambda:${scope.region}:943013980633:layer:SentryNodeServerlessSDK:34`
);

You can then set it for all the functions in your stack using the addDefaultFunctionLayers and addDefaultFunctionEnv. Note we only want to enable this when the function is deployed, not when using Live Lambda Dev.

stacks/Foo.js
if (!scope.local) {
stack.addDefaultFunctionLayers([layer]);
stack.addDefaultFunctionEnv({
SENTRY_DSN: "<SENTRY_DSN>",
NODE_OPTIONS: "-r @sentry/serverless/dist/awslambda-auto",
});
}

Sentry also offers performance monitoring for serverless. To enable, add the SENTRY_TRACES_SAMPLE_RATE environment variable.

stack.addDefaultFunctionEnv({
SENTRY_DSN: "<SENTRY_DSN>",
SENTRY_TRACES_SAMPLE_RATE: "1.0",
NODE_OPTIONS: "-r @sentry/serverless/dist/awslambda-auto",
});

This can be tuned between the values of 0 and 1. Where 0 means that no performance related information is sent, and 1 means that information for all the invocations are sent. This should be tuned based on the volume of invocations and the amount of transactions available in your Sentry account. A value of 0.5 should work for most projects.

You also need to wrap your function handlers.

packages/functions/src/foo.js
import * as Sentry from "@sentry/serverless";

export const handler = Sentry.AWSLambda.wrapHandler(async (event) => {
// ...
});

For more details, check out the Sentry docs. Also, check out our docs on how to upload source maps to Sentry.


Lumigo

Lumigo offers a Serverless Monitoring and Debugging Platform.

To get started, sign up for an account. Then follow their wizard to deploy their stack in your AWS production account.

Then to enable Lambda monitoring for a function, add a lumigo:auto-trace tag and set it to true.

stacks/Foo.js
import * as cdk from "aws-cdk-lib";

cdk.Tags.of(myfunc).add("lumigo:auto-trace", "true");

To monitor all the functions in a stack, you can use the Stack construct's getAllFunctions method and do the following at the bottom of your stack definition.

stacks/Foo.js
import * as cdk from "aws-cdk-lib";

stack
.getAllFunctions()
.forEach((fn) => cdk.Tags.of(fn).add("lumigo:auto-trace", "true"));

For more details, check out the Lumigo docs on auto-tracing.


New Relic

New Relic offers New Relic Serverless for AWS Lambda. To get started, follow the steps in the documentation.

To enable Lambda monitoring, you'll need to add a layer to the functions you want to monitor. To figure out the layer ARN for the latest version, check the available layers per region here.

With the layer ARN, you can use the layer construct in your CDK code. To ensure the Lambda function is instrumented correctly, the function handler must be set to the handler provided by the New Relic layer. Note we only want to enable this when the function is deployed, not in Live Lambda Dev as the layer will prevent the debugger from connecting.

info

If you use Node.js then you'll probably need to use Node 18 if you plan on using ESM modules.

Add the following to your sst.config.ts file.

sst.config.ts
export default {
config() {
// Config
},
async stacks(app) {
// Don't enable locally
const enableNR = !app.local;

if (enableNR) {
app.setDefaultFunctionProps((stack) => {
const newRelicLayer = LayerVersion.fromLayerVersionArn(
stack,
"NewRelicLayer",
// Find your "<ARN>" here: https://layers.newrelic-external.com/
// Make sure you select the correct region and version
"<ARN>"
);

return {
layers: [newRelicLayer.layerVersionArn],
};
});
}

// Add your stacks
app.stack(/* ... */);

if (enableNR) {
await app.finish();

// Loop through each stack in the app
app.node.children.forEach((stack) => {
if (stack instanceof sst.Stack) {
const policy = new PolicyStatement({
actions: ["secretsmanager:GetSecretValue"],
effect: Effect.ALLOW,
resources: [
Fn.importValue(
"NewRelicLicenseKeySecret-NewRelic-LicenseKeySecretARN"
),
],
});

stack.getAllFunctions().forEach((fn) => {
const cfnFunction = fn.node.defaultChild as CfnFunction;
if (cfnFunction.handler) {
fn.addEnvironment(
"NEW_RELIC_LAMBDA_HANDLER",
cfnFunction.handler
);
fn.addEnvironment("NEW_RELIC_ACCOUNT_ID", "YOUR_ACCOUNT_ID");
// If your New Relic account has a parent account, this value should be that account ID. Otherwise, just
// your account id.
fn.addEnvironment(
"NEW_RELIC_TRUSTED_ACCOUNT_KEY",
"YOUR_ACCOUNT_ID_OR_PARENT_ACCOUNT_ID"
);
}

// Give your function access to the secret containing your New Relic license key
// You will set this key using the `newrelic-lambda integrations install` command
// More info: https://docs.newrelic.com/docs/serverless-function-monitoring/aws-lambda-monitoring/enable-lambda-monitoring/account-linking/
fn.attachPermissions([policy]);

// See #3 on the link below for the correct handler name to use based on your runtime
// The handler name below is for NodeJS
// https://github.com/newrelic/newrelic-lambda-layers#manual-instrumentation-using-layers
cfnFunction.handler = "newrelic-lambda-wrapper.handler";
});
}
});
}
},
} satisfies SSTConfig;