See: Description
| Interface | Description |
|---|---|
| BundlingOptions |
(experimental) Bundling options.
|
| ICommandHooks |
(experimental) Command hooks.
|
| ICommandHooks.Jsii$Default |
Internal default implementation for
ICommandHooks. |
| NodejsFunctionProps |
(experimental) Properties for a NodejsFunction.
|
| Class | Description |
|---|---|
| BundlingOptions.Builder |
A builder for
BundlingOptions |
| BundlingOptions.Jsii$Proxy |
An implementation for
BundlingOptions |
| ICommandHooks.Jsii$Proxy |
A proxy class which represents a concrete javascript instance of this type.
|
| NodejsFunction |
(experimental) A Node.js Lambda function bundled using esbuild.
|
| NodejsFunction.Builder |
(experimental) A fluent builder for
NodejsFunction. |
| NodejsFunctionProps.Builder |
A builder for
NodejsFunctionProps |
| NodejsFunctionProps.Jsii$Proxy |
An implementation for
NodejsFunctionProps |
| Enum | Description |
|---|---|
| LogLevel |
(experimental) Log level for esbuild.
|
---
This library provides constructs for Node.js Lambda functions.
The NodejsFunction construct creates a Lambda function with automatic transpiling and bundling
of TypeScript or Javascript code. This results in smaller Lambda packages that contain only the
code and dependencies needed to run the function.
It uses esbuild under the hood.
The NodejsFunction allows you to define your CDK and runtime dependencies in a single
package.json and to collocate your runtime code with your infrastructure code:
. ├── lib │ ├── my-construct.api.ts # Lambda handler for API │ ├── my-construct.auth.ts # Lambda handler for Auth │ └── my-construct.ts # CDK construct with two Lambda functions ├── package-lock.json # single lock file ├── package.json # CDK and runtime dependencies defined in a single package.json └── tsconfig.json
By default, the construct will use the name of the defining file and the construct's
id to look up the entry file. In my-construct.ts above we have:
// Example automatically generated without compilation. See https://github.com/aws/jsii/issues/826 // automatic entry look up Object apiHandler = new NodejsFunction(this, "api"); Object authHandler = new NodejsFunction(this, "auth");
Alternatively, an entry file and handler can be specified:
// Example automatically generated without compilation. See https://github.com/aws/jsii/issues/826
NodejsFunction.Builder.create(this, "MyFunction")
.entry("/path/to/my/file.ts")// accepts .js, .jsx, .ts and .tsx files
.handler("myExportedFunc")
.build();
For monorepos, the reference architecture becomes:
. ├── packages │ ├── cool-package │ │ ├── lib │ │ │ ├── cool-construct.api.ts │ │ │ ├── cool-construct.auth.ts │ │ │ └── cool-construct.ts │ │ ├── package.json # CDK and runtime dependencies for cool-package │ │ └── tsconfig.json │ └── super-package │ ├── lib │ │ ├── super-construct.handler.ts │ │ └── super-construct.ts │ ├── package.json # CDK and runtime dependencies for super-package │ └── tsconfig.json ├── package-lock.json # single lock file ├── package.json # root dependencies └── tsconfig.json
All properties of lambda.Function can be used to customize the underlying lambda.Function.
See also the AWS Lambda construct library.
The NodejsFunction construct automatically reuses existing connections
when working with the AWS SDK for JavaScript. Set the awsSdkConnectionReuse prop to false to disable it.
The NodejsFunction requires a dependencies lock file (yarn.lock, pnpm-lock.yaml or
package-lock.json). When bundling in a Docker container, the path containing this lock file is
used as the source (/asset-input) for the volume mounted in the container.
By default, the construct will try to automatically determine your project lock file.
Alternatively, you can specify the depsLockFilePath prop manually. In this
case you need to ensure that this path includes entry and any module/dependencies
used by your function. Otherwise bundling will fail.
If esbuild is available it will be used to bundle your code in your environment. Otherwise,
bundling will happen in a Lambda compatible Docker container.
For macOS the recommendend approach is to install esbuild as Docker volume performance is really poor.
esbuild can be installed with:
$ npm install --save-dev esbuild@0
OR
$ yarn add --dev esbuild@0
To force bundling in a Docker container even if esbuild is available in your environment,
set bundling.forceDockerBundling to true. This is useful if your function relies on node
modules that should be installed (nodeModules prop, see below) in a Lambda
compatible environment. This is usually the case with modules using native dependencies.
By default, all node modules are bundled except for aws-sdk. This can be configured by specifying
bundling.externalModules:
// Example automatically generated without compilation. See https://github.com/aws/jsii/issues/826
NodejsFunction.Builder.create(this, "my-handler")
.bundling(Map.of(
"externalModules", asList("aws-sdk", "cool-module")))
.build();
By default, all node modules referenced in your Lambda code will be bundled by esbuild.
Use the nodeModules prop under bundling to specify a list of modules that should not be
bundled but instead included in the node_modules folder of the Lambda package. This is useful
when working with native dependencies or when esbuild fails to bundle a module.
// Example automatically generated without compilation. See https://github.com/aws/jsii/issues/826
NodejsFunction.Builder.create(this, "my-handler")
.bundling(Map.of(
"nodeModules", asList("native-module", "other-module")))
.build();
The modules listed in nodeModules must be present in the package.json's dependencies or
installed. The same version will be used for installation. The lock file (yarn.lock,
pnpm-lock.yaml or package-lock.json) will be used along with the right installer (yarn,
pnpm or npm).
When working with nodeModules using native dependencies, you might want to force bundling in a
Docker container even if esbuild is available in your environment. This can be done by setting
bundling.forceDockerBundling to true.
esbuild
The NodejsFunction construct exposes some esbuild options
via properties under bundling:
// Example automatically generated without compilation. See https://github.com/aws/jsii/issues/826
NodejsFunction.Builder.create(this, "my-handler")
.bundling(Map.of(
"minify", true, // minify code, defaults to false
"sourceMap", true, // include source map, defaults to false
"target", "es2020", // target environment for the generated JavaScript code
"loader", Map.of(// Use the 'dataurl' loader for '.png' files
".png", "dataurl"),
"define", Map.of(// Replace strings during build time
"process.env.API_KEY", JSON.stringify("xxx-xxxx-xxx"),
"process.env.PRODUCTION", JSON.stringify(true),
"process.env.NUMBER", JSON.stringify(123)),
"logLevel", LogLevel.getSILENT(), // defaults to LogLevel.WARNING
"keepNames", true, // defaults to false
"tsconfig", "custom-tsconfig.json", // use custom-tsconfig.json instead of default,
"metafile", true, // include meta file, defaults to false
"banner", "/* comments * /", // requires esbuild >= 0.9.0, defaults to none
"footer", "/* comments * /"))
.build();
It is possible to run additional commands by specifying the commandHooks prop:
// Example automatically generated without compilation. See https://github.com/aws/jsii/issues/826
NodejsFunction.Builder.create(this, "my-handler-with-commands")
.bundling(Map.of(
"commandHooks", Map.of(
// Copy a file so that it will be included in the bundled asset
public Array afterBundling(String inputDir, String outputDir) {return asList(String.format("cp %s/my-binary.node %s", inputDir, outputDir))
})))
.build();
The following hooks are available:
beforeBundling: runs before all bundling commandsbeforeInstall: runs before node modules installationafterBundling: runs after all bundling commands
They all receive the directory containing the lock file (inputDir) and the
directory where the bundled asset will be output (outputDir). They must return
an array of commands to run. Commands are chained with &&.
The commands will run in the environment in which bundling occurs: inside the container for Docker bundling or on the host OS for local bundling.
Use bundling.environment to define environments variables when esbuild runs:
// Example automatically generated without compilation. See https://github.com/aws/jsii/issues/826
NodejsFunction.Builder.create(this, "my-handler")
.bundling(Map.of(
"environment", Map.of(
"NODE_ENV", "production")))
.build();
Use bundling.buildArgs to pass build arguments when building the Docker bundling image:
// Example automatically generated without compilation. See https://github.com/aws/jsii/issues/826
NodejsFunction.Builder.create(this, "my-handler")
.bundling(Map.of(
"buildArgs", Map.of(
"HTTPS_PROXY", "https://127.0.0.1:3001")))
.build();
Use bundling.dockerImage to use a custom Docker bundling image:
// Example automatically generated without compilation. See https://github.com/aws/jsii/issues/826
NodejsFunction.Builder.create(this, "my-handler")
.bundling(Map.of(
"dockerImage", cdk.DockerImage.fromBuild("/path/to/Dockerfile")))
.build();
This image should have esbuild installed globally. If you plan to use nodeModules it
should also have npm, yarn or pnpm depending on the lock file you're using.
Use the default image provided by @aws-cdk/aws-lambda-nodejs
as a source of inspiration.
Copyright © 2021. All rights reserved.