Skip to content

Commit

Permalink
feat(lambda-nodejs): use docker instead of npm package for parcel-bun…
Browse files Browse the repository at this point in the history
…dler

* feat(lambda-nodejs): use docker instead of npm package for parcel-bundler

* require version ^1 of parcel-bundler in Dockerfile

Co-Authored-By: Jonathan Goldwasser <[email protected]>

* update README: docker requirement

* comment in Dockerfile, make node version a build ARG

* add test for when docker not installed

* add optional nodeDockerTag to props

* Expose nodeDockerTag in NodejsFunctionProps and pass it to the Builder

* cdk-build pre

* chore(lambda-nodejs): remove dockerd pre directive

* add missing closing parenthesis

* fix linter errors

* fix linter error

* start docker daemon inside the container

* Update buildspec.yaml

* Fix docker location.

* Update buildspec.yaml

Co-authored-by: Jonathan Goldwasser <[email protected]>
Co-authored-by: Elad Ben-Israel <[email protected]>
Co-authored-by: mergify[bot] <37929162+mergify[bot]@users.noreply.github.com>
  • Loading branch information
4 people authored May 1, 2020
1 parent 0e96415 commit 55c4d0b
Show file tree
Hide file tree
Showing 10 changed files with 362 additions and 2,899 deletions.
4 changes: 4 additions & 0 deletions buildspec-pr.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,10 @@ version: 0.2
phases:
install:
commands:
# Start docker daemon inside the container
- nohup /usr/bin/dockerd --host=unix:///var/run/docker.sock --host=tcp://127.0.0.1:2375 --storage-driver=overlay2&
- timeout 15 sh -c "until docker info; do echo .; sleep 1; done"

# Install yarn if it wasn't already present in the image
- yarn --version || npm -g install yarn
build:
Expand Down
4 changes: 4 additions & 0 deletions buildspec.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,10 @@ version: 0.2
phases:
install:
commands:
# Start docker daemon inside the container
- nohup /usr/bin/dockerd --host=unix:///var/run/docker.sock --host=tcp://127.0.0.1:2375 --storage-driver=overlay2&
- timeout 15 sh -c "until docker info; do echo .; sleep 1; done"

# Install yarn if it wasn't already present in the image
- yarn --version || npm -g install yarn
pre_build:
Expand Down
9 changes: 1 addition & 8 deletions packages/@aws-cdk/aws-lambda-nodejs/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -11,14 +11,7 @@

This library provides constructs for Node.js Lambda functions.

To use this module, you will need to add a dependency on `parcel-bundler` in your
`package.json`:

```
yarn add parcel-bundler@^1
# or
npm install parcel-bundler@^1
```
To use this module, you will need to have Docker installed.

### Node.js Function
Define a `NodejsFunction`:
Expand Down
55 changes: 30 additions & 25 deletions packages/@aws-cdk/aws-lambda-nodejs/lib/builder.ts
Original file line number Diff line number Diff line change
Expand Up @@ -41,29 +41,20 @@ export interface BuilderOptions {
* The node version to use as target for Babel
*/
readonly nodeVersion?: string;

/**
* The docker tag of the node base image to use in the parcel-bundler docker image
*
* @see https://hub.docker.com/_/node/?tab=tags
*/
readonly nodeDockerTag: string;
}

/**
* Builder
*/
export class Builder {
private readonly parcelBinPath: string;

constructor(private readonly options: BuilderOptions) {
let parcelPkgPath: string;
try {
parcelPkgPath = require.resolve('parcel-bundler/package.json'); // This will throw if `parcel-bundler` cannot be found
} catch (err) {
throw new Error('It looks like parcel-bundler is not installed. Please install v1.x of parcel-bundler with yarn or npm.');
}
const parcelDir = path.dirname(parcelPkgPath);
const parcelPkg = JSON.parse(fs.readFileSync(parcelPkgPath, 'utf8'));

if (!parcelPkg.version || !/^1\./.test(parcelPkg.version)) { // Peer dependency on parcel v1.x
throw new Error(`This module has a peer dependency on parcel-bundler v1.x. Got v${parcelPkg.version}.`);
}

this.parcelBinPath = path.join(parcelDir, parcelPkg.bin.parcel);
}

public build(): void {
Expand All @@ -78,29 +69,43 @@ export class Builder {
});
}

const args = [
'build', this.options.entry,
'--out-dir', this.options.outDir,
const dockerBuildArgs = [
'build', '--build-arg', `NODE_TAG=${this.options.nodeDockerTag}`, '-t', 'parcel-bundler', path.join(__dirname, '../parcel-bundler'),
];
const dockerRunArgs = [
'run', '--rm',
'-v', `${path.dirname(path.resolve(this.options.entry))}:/entry`,
'-v', `${path.resolve(this.options.outDir)}:/out`,
...(this.options.cacheDir ? ['-v', `${path.resolve(this.options.cacheDir)}:/cache`] : []),
'parcel-bundler',
];
const parcelArgs = [
'parcel', 'build', `/entry/${path.basename(this.options.entry)}`,
'--out-dir', '/out',
'--out-file', 'index.js',
'--global', this.options.global,
'--target', 'node',
'--bundle-node-modules',
'--log-level', '2',
!this.options.minify && '--no-minify',
!this.options.sourceMaps && '--no-source-maps',
...this.options.cacheDir
? ['--cache-dir', this.options.cacheDir]
: [],
...(this.options.cacheDir ? ['--cache-dir', '/cache'] : []),
].filter(Boolean) as string[];

const parcel = spawnSync(this.parcelBinPath, args);
const build = spawnSync('docker', dockerBuildArgs);
if (build.error) {
throw build.error;
}
if (build.status !== 0) {
throw new Error(`[Status ${build.status}] stdout: ${build.stdout?.toString().trim()}\n\n\nstderr: ${build.stderr?.toString().trim()}`);
}

const parcel = spawnSync('docker', [...dockerRunArgs, ...parcelArgs]);
if (parcel.error) {
throw parcel.error;
}

if (parcel.status !== 0) {
throw new Error(parcel.stdout.toString().trim());
throw new Error(`[Status ${parcel.status}] stdout: ${parcel.stdout?.toString().trim()}\n\n\nstderr: ${parcel.stderr?.toString().trim()}`);
}
} catch (err) {
throw new Error(`Failed to build file at ${this.options.entry}: ${err}`);
Expand Down
11 changes: 11 additions & 0 deletions packages/@aws-cdk/aws-lambda-nodejs/lib/function.ts
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,15 @@ export interface NodejsFunctionProps extends lambda.FunctionOptions {
* @default - `.cache` in the root directory
*/
readonly cacheDir?: string;

/**
* The docker tag of the node base image to use in the parcel-bundler docker image
*
* @see https://hub.docker.com/_/node/?tab=tags
*
* @default - 13.8.0-alpine3.11
*/
readonly nodeDockerTag?: string;
}

/**
Expand All @@ -84,6 +93,7 @@ export class NodejsFunction extends lambda.Function {
? lambda.Runtime.NODEJS_12_X
: lambda.Runtime.NODEJS_10_X;
const runtime = props.runtime || defaultRunTime;
const nodeDockerTag = props.nodeDockerTag || '13.8.0-alpine3.11';

// Build with Parcel
const builder = new Builder({
Expand All @@ -94,6 +104,7 @@ export class NodejsFunction extends lambda.Function {
sourceMaps: props.sourceMaps,
cacheDir: props.cacheDir,
nodeVersion: extractVersion(runtime),
nodeDockerTag,
});
builder.build();

Expand Down
9 changes: 8 additions & 1 deletion packages/@aws-cdk/aws-lambda-nodejs/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,14 @@
"build+test": "npm run build && npm test",
"compat": "cdk-compat"
},
"cdk-build": {
"eslint": {
"ignore-pattern": [
"test/function.test.handler2.js",
"test/integ-handlers/js-handler.js"
]
}
},
"keywords": [
"aws",
"cdk",
Expand Down Expand Up @@ -80,7 +88,6 @@
"cdk-build-tools": "0.0.0",
"cdk-integ-tools": "0.0.0",
"fs-extra": "^8.1.0",
"parcel-bundler": "^1.12.4",
"pkglint": "0.0.0"
},
"dependencies": {
Expand Down
7 changes: 7 additions & 0 deletions packages/@aws-cdk/aws-lambda-nodejs/parcel-bundler/Dockerfile
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
# runs the parcel-bundler npm package to package and install dependencies of nodejs lambda functions
ARG NODE_TAG
FROM node:${NODE_TAG}

RUN yarn global add parcel-bundler@^1

CMD [ "parcel" ]
63 changes: 34 additions & 29 deletions packages/@aws-cdk/aws-lambda-nodejs/test/builder.test.ts
Original file line number Diff line number Diff line change
@@ -1,60 +1,66 @@
import { spawnSync } from 'child_process';
import * as fs from 'fs';
import * as path from 'path';
import { Builder } from '../lib/builder';

let parcelPkgPath: string;
let parcelPkg: Buffer;
beforeAll(() => {
parcelPkgPath = require.resolve('parcel-bundler/package.json');
parcelPkg = fs.readFileSync(parcelPkgPath);
});

afterEach(() => {
fs.writeFileSync(parcelPkgPath, parcelPkg);
});

jest.mock('child_process', () => ({
spawnSync: jest.fn((_cmd: string, args: string[]) => {
if (args[1] === 'error') {
if (args.includes('/entry/error')) {
return { error: 'parcel-error' };
}

if (args[1] === 'status') {
if (args.includes('/entry/status')) {
return { status: 1, stdout: Buffer.from('status-error') };
}

if (args.includes('/entry/no-docker')) {
return { error: 'Error: spawnSync docker ENOENT' };
}

return { error: null, status: 0 };
}),
}));

test('calls parcel with the correct args', () => {
test('calls docker with the correct args', () => {
const builder = new Builder({
entry: 'entry',
global: 'handler',
outDir: 'out-dir',
cacheDir: 'cache-dir',
nodeDockerTag: '13.8.0-alpine3.11',
});
builder.build();

expect(spawnSync).toHaveBeenCalledWith(expect.stringContaining('parcel-bundler'), expect.arrayContaining([
'build', 'entry',
'--out-dir', 'out-dir',
// docker build
expect(spawnSync).toHaveBeenNthCalledWith(1, 'docker', [
'build', '--build-arg', 'NODE_TAG=13.8.0-alpine3.11', '-t', 'parcel-bundler', path.join(__dirname, '../parcel-bundler'),
]);

// docker run
expect(spawnSync).toHaveBeenNthCalledWith(2, 'docker', [
'run', '--rm',
'-v', `${path.join(__dirname, '..')}:/entry`,
'-v', `${path.join(__dirname, '../out-dir')}:/out`,
'-v', `${path.join(__dirname, '../cache-dir')}:/cache`,
'parcel-bundler',
'parcel', 'build', '/entry/entry',
'--out-dir', '/out',
'--out-file', 'index.js',
'--global', 'handler',
'--target', 'node',
'--bundle-node-modules',
'--log-level', '2',
'--no-minify',
'--no-source-maps',
'--cache-dir', 'cache-dir',
]));
'--cache-dir', '/cache',
]);
});

test('throws in case of error', () => {
const builder = new Builder({
entry: 'error',
global: 'handler',
outDir: 'out-dir',
nodeDockerTag: '13.8.0-alpine3.11',
});
expect(() => builder.build()).toThrow('parcel-error');
});
Expand All @@ -64,18 +70,17 @@ test('throws if status is not 0', () => {
entry: 'status',
global: 'handler',
outDir: 'out-dir',
nodeDockerTag: '13.8.0-alpine3.11',
});
expect(() => builder.build()).toThrow('status-error');
});

test('throws when parcel-bundler is not 1.x', () => {
fs.writeFileSync(parcelPkgPath, JSON.stringify({
...JSON.parse(parcelPkg.toString()),
version: '2.3.4',
}));
expect(() => new Builder({
entry: 'entry',
test('throws if docker is not installed', () => {
const builder = new Builder({
entry: 'no-docker',
global: 'handler',
outDir: 'out-dur',
})).toThrow(/This module has a peer dependency on parcel-bundler v1.x. Got v2.3.4./);
outDir: 'out-dir',
nodeDockerTag: '13.8.0-alpine3.11',
});
expect(() => builder.build()).toThrow('Error: spawnSync docker ENOENT');
});
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,7 @@
"Properties": {
"Code": {
"S3Bucket": {
"Ref": "AssetParametersa736580e4382e6d67a7acaedbbd271f559ed918382808ae1431364ae70286543S3BucketD096DC83"
"Ref": "AssetParametersfdefbd5f0d90231e11e34991218a3674ec3ac1a3a99948038164c6d7a8efa88eS3Bucket9E60CFAE"
},
"S3Key": {
"Fn::Join": [
Expand All @@ -49,7 +49,7 @@
"Fn::Split": [
"||",
{
"Ref": "AssetParametersa736580e4382e6d67a7acaedbbd271f559ed918382808ae1431364ae70286543S3VersionKeyA76287A5"
"Ref": "AssetParametersfdefbd5f0d90231e11e34991218a3674ec3ac1a3a99948038164c6d7a8efa88eS3VersionKey74429E96"
}
]
}
Expand All @@ -62,7 +62,7 @@
"Fn::Split": [
"||",
{
"Ref": "AssetParametersa736580e4382e6d67a7acaedbbd271f559ed918382808ae1431364ae70286543S3VersionKeyA76287A5"
"Ref": "AssetParametersfdefbd5f0d90231e11e34991218a3674ec3ac1a3a99948038164c6d7a8efa88eS3VersionKey74429E96"
}
]
}
Expand Down Expand Up @@ -172,17 +172,17 @@
}
},
"Parameters": {
"AssetParametersa736580e4382e6d67a7acaedbbd271f559ed918382808ae1431364ae70286543S3BucketD096DC83": {
"AssetParametersfdefbd5f0d90231e11e34991218a3674ec3ac1a3a99948038164c6d7a8efa88eS3Bucket9E60CFAE": {
"Type": "String",
"Description": "S3 bucket for asset \"a736580e4382e6d67a7acaedbbd271f559ed918382808ae1431364ae70286543\""
"Description": "S3 bucket for asset \"fdefbd5f0d90231e11e34991218a3674ec3ac1a3a99948038164c6d7a8efa88e\""
},
"AssetParametersa736580e4382e6d67a7acaedbbd271f559ed918382808ae1431364ae70286543S3VersionKeyA76287A5": {
"AssetParametersfdefbd5f0d90231e11e34991218a3674ec3ac1a3a99948038164c6d7a8efa88eS3VersionKey74429E96": {
"Type": "String",
"Description": "S3 key for asset version \"a736580e4382e6d67a7acaedbbd271f559ed918382808ae1431364ae70286543\""
"Description": "S3 key for asset version \"fdefbd5f0d90231e11e34991218a3674ec3ac1a3a99948038164c6d7a8efa88e\""
},
"AssetParametersa736580e4382e6d67a7acaedbbd271f559ed918382808ae1431364ae70286543ArtifactHashB7337532": {
"AssetParametersfdefbd5f0d90231e11e34991218a3674ec3ac1a3a99948038164c6d7a8efa88eArtifactHashAA5D4787": {
"Type": "String",
"Description": "Artifact hash for asset \"a736580e4382e6d67a7acaedbbd271f559ed918382808ae1431364ae70286543\""
"Description": "Artifact hash for asset \"fdefbd5f0d90231e11e34991218a3674ec3ac1a3a99948038164c6d7a8efa88e\""
},
"AssetParameters0d445d366e339b4344917ba638a4cb2dcddfd0e063b10fb909340fd1cc51c278S3BucketDBD288E6": {
"Type": "String",
Expand Down
Loading

0 comments on commit 55c4d0b

Please sign in to comment.