Skip to content

ljacobsson/lambda-debug

Repository files navigation

DEPRECATED

Please use samp-cli for a much smoother local debugging experience

lambda-debug

Lambda-Debug is a tool that enables you to invoke AWS Lambda functions in the cloud from any event source and intercept the requests with breakpoints locally. There's similar functionality in SST which this is inspired by, but is designed to work with AWS SAM and CloudFormation and, in a later version, CDK.

<blink>** NOTE: This is intended to be used to debug functions in a development environment. DO NOT use in production. **</blink>

Features

  • 🏠 Local debugging: Set breakpoints in your code and step through your functions invocations locally on native events triggered in the cloud.
  • 👍 No code changes: No need to modify your code to enable debugging. Just add some dev dependencies and some configuration.
  • 🔐 Same IAM permissions: Your functions will run with the same IAM permissions as they do in the cloud.
  • Fast iterations: No need to deploy your code to the cloud to test changes. Just save your code and invoke your functions in the cloud.

Demo

Setup

Follow these steps to configure lambda-debug:

Step 1: Install Dependencies

You need to install the lambda-debug and tsc-watch dependencies.

npm install --save-dev lambda-debug tsc-watch nodemon

Step 2: Configure development environment.

Currently there's only config support for vscode, but nothing suggests this can't be used with other code editors.

npx lambda-debug-setup

This will update tsconfig.json, .vscode/launch-config and .vscode/tasks.json. Make sure to review the diff to understand the changes.

Summary of changes

1: Update tsconfig.json

Add the following to your tsconfig.json file along with the rest of your config:

{
  "extends": "@tsconfig/node18/tsconfig.json",
  "compilerOptions": {
    "resolveJsonModule": true,
    "outDir": "dist",
    "sourceMap": true
  },
  "exclude": ["node_modules", "dist/**/*"]
}

2: Update .vscode/launch.json

Add the following configuration to your .vscode/launch.json file:

{
  "version": "0.2.0",
  "configurations": [
    {
      "type": "node",
      "request": "launch",
      "name": "Debug Lambda functions",
      "runtimeExecutable": "${workspaceFolder}/node_modules/.bin/nodemon",
      "skipFiles": ["<node_internals>/**"],
      "program": "${workspaceFolder}/node_modules/lambda-debug/connect.js",
      "restart": true,
      "env": {
        "outDir": "dist"
      },
      "postDebugTask": "lambda-debug-cleanup",
      "console": "integratedTerminal",
      "internalConsoleOptions": "neverOpen"      
    },
  ]
}

3: Update .vscode/tasks.json

Add the following task to your .vscode/tasks.json file:

{
  "version": "2.0.0",
  "tasks": [
    {
      "label": "lambda-debug-cleanup",
      "type": "shell",      
      "command": "node ${workspaceFolder}/node_modules/lambda-debug/cleanup.js"
    }
  ]
}

Step 5: Deploy Your Stack

Deploy your stack using SAM if you haven't already done so. You'll need the SAM CLI for this. Check out the AWS SAM CLI documentation for more information.

sam deploy

Step 6: Set up File Listener

Use tsc-watch to set up a file listener that automatically compiles your TypeScript files as you change them. You can start it with the following command:

npx tsc-watch

Step 7: Debug!

Now you're ready to debug. Set some breakpoints in your code, hit F5 in VS Code to start debugging, and then invoke your functions in the cloud. You'll be able to step through your function invocations locally in VS Code, just as if the code were running on your machine.

Configuration

The env section within your .vscode/launch.json allows you to set environment variables that define how Lambda-Debug operates. Here's a brief explanation of each variable:

  • outDir: This is where your compiled JavaScript files are located after TypeScript compilation. For example, if you have set "outDir" to "dist", Lambda-Debug will look for the compiled JavaScript files in the "dist" folder.

  • configEnv: This specifies the environment to use from your samconfig.toml-file. For example, if you have multiple profiles for different AWS environments (such as default, staging, production), you can set "configEnv" to the one you want to use.

  • includeFunctions: If you wish to debug only specific functions, list them here, separating multiple function names with a comma. For example, "HelloWorldFunction,AnotherFunction". If left empty, all functions will be included.

  • excludeFunctions: Use this if you wish to exclude certain functions from being debugged. List the functions you want to exclude, separating multiple function names with a comma. For example, "GoodbyeWorldFunction,SomeOtherFunction". If left empty, no functions will be excluded.

  • timeout: This is the timeout in seconds that will allocated to your function when debugging. The default is 300 seconds. If you spend longer than this on a breakpoint then you'll be allowd to finish stepping through the code, but any return value will be ignored.

How it Works

Lambda-debug temporarily replaces your function code with a relay proxy function. This relay sets up an MQTT connection with your local machine over AWS IoT, enabling a bidirectional transfer of execution context between the cloud and your local machine.

While in debug mode, the function's configuration is tweaked to better accommodate local debugging needs:

  • Timeout: The function's timeout is increased. This extension is to ensure that your function does not time out while you are stepping through code in the debugger.
  • MemorySize: The memory allocation is reduced to 128mb. Given that the actual compute happens on your local machine when debugging, this smaller memory footprint helps minimize costs without impacting performance.

Cost Implications

Though AWS IoT usage in this scenario doesn't carry additional costs, be aware that while a function is at a breakpoint during debugging, AWS charges approximately $0.0000000021 per millisecond or roughly $0.0076 per accumulated hour.

Restoration Post-Debugging

Once your debugging session is concluded, Lambda-Debug restores your function back to its original state. This includes resetting the function code as well as the function configuration to their original values, ensuring your live environment is left untouched and as intended.

Troubleshooting

>>> I log in to AWS SSO, but still get access denied exception

This is due to a bug in the aws-sdk-v3 and is unrelated to this project. It happens sometimes when the SSO session expires. To get around this, go to your sso start url, clear cookies and log in again

>>> I make a code change and my function invocation doesn't reach my code

When the code is changed, the debugger will reload. This results in a sub-second (depending on your internet connection) disconnection from the MQTT topic. If your function invocation happens during this time, it will not reach your code. To avoid this, wait a few seconds after making a code change before invoking your function.

About

Library to help debug JS/TS AWS Lambda functions locally

Topics

Resources

License

Stars

Watchers

Forks

Packages

No packages published