Run AWS Lambda functions locally in node.
The main focus of this project is to enable you to use your functions locally, either in a dev-server or for testing, without any adjustments needed.
With a variety of alternatives available, why choose this library?
It's lightweight and comes with zero runtime dependencies. Moreover, it runs your lambda functions as isolated as possible, executing each lambda in a separate child process.
Instead of loading the lambda handler directly into the current node process,
RunL
forks a child process in which the handler is executed. To execute the
handler the child process requires the handler code, executes it and passes the
result back to the parent process.
Why so complicated?
Other libraries, which repeatedly re-import lambda code in the same node process, encounter an issue where residual memory from each import is not fully cleared by the garbage collector. This results in memory leaks. Consequently, when running numerous tests locally or doing a lot of requests, these memory leaks can cause the development server to eventually cease functioning.
Unfortunately, this scenario is not merely theoretical; in fact, it was the primary motivation behind the creation of this library.
RunL comes with two execution modes, Ephemeral
and Persistent
. Ephemeral
always forks a new child process, executes the handler within this process and
passes the result to the parent process.
This has the advantage that every request runs totally isolated from every other request. The disadvantage is, that the lambda handler code must always be newly required inside the child process. This can take a few seconds if the handler is several megabytes in size.
const { Lambda } = require('runl');
const lambda = new Lambda({
mode: 'Ephemeral',
lambdaPath: __dirname + '/handler/example-handler.js',
environment: {
BASE_URL: '/'
}
});
lambda.execute({ path: '/index.html' });
The Persistent
mode in contrast creates a long running fork that is reused for
every invocation. The advantage is that the handler code must not be newly
required for every request, which can drastically reduce the request times in
your dev server.
import { APIGatewayProxyResult } from 'aws-lambda';
import { Lambda, LambdaMode } from 'runl';
const lambda = new Lambda({
mode: LambdaMode.Persistent,
lambdaPath: __dirname + '/handler/example-handler.js'
});
const result = lambda.execute<APIGatewayProxyResult>();
// you can manually stop the child process
// otherwise the child process lives
// until the parent process is terminated
lambda.stop();
The Lambda
constructor accepts the following options:
required:
- lambdaPath: absolute path to the lambda handler.
optional:
-
environment: environment variables accessible in the lambda handler.
-
lambdaHandler: the name of lambda handler, defaults to handler.
-
lambdaTimeout: maximum execution time for the lambda, defaults to 30.000ms.
-
autoReload: if true, the lambda handler is automatically updated every time the file associated with lambdaPath is changed. Defaults to false.
-
debugPort: when node is started with the debug flag
(--inspect or NODE_OPTIONS="--inspect")
and a debugPort is specified, the forked process will listen on this port for debug messages.
The execute
method:
- event: The APIGatewayProxyEvent that is passed to the lambda handler.
You can also use the CLI to run a lambda function from a file.
CLI Options | |
---|---|
-t, --timeout | Maximal execution time in ms |
-e, --env | Environment variables, either a JSON object or file path |
-o, --event | The event object passed to the lambda function, either a JSON object or file path |
-h, --handler | The name of lambda handler, defaults to: handler |
# passing environment variables
runl -e '{"LOG_LEVEL": "DEBUG"}' ./index.cjs
# specify event data and milliseconds until lambda timeout
runl -o '{"action": "delete"}' -t 1000 ./index.cjs