Lambda @edge image resizing based on the demo from AWS
-
From project root folder, run the following commands:
docker build --tag amazonlinux:nodejs .
Note the . (dot) at the end, it is easily missed and is required to specify the current directory.
The Dockerfile is configured to download Amazon Linux and install Node.js 12.x along with dependencies.
-
Set the bucket and cloudfront distribution names in the appropriate files:
- imago/lambda/origin-response-function/index.js
- imago/cloud-formation-template.yml
The default bucket name is: image-resize-test-${AWS::AccountId}-${AWS::Region}
A cloudfront distribution with a domain beginning with this value will be created.
You must change the bucket name on this line in the origin-response-function in order for this to work:
const BUCKET = 'image-resize-test-<your account number>-us-east-1';
For example, if your AWS Account ID is 123456789012, then BUCKET variable would be updated to ‘image-resize-test-123456789012-us-east-1’. If you already have an S3 bucket then update this variable accordingly and modify your CloudFront distribution Origin settings to reflect the same.
-
Install the sharp and querystring module dependencies and compile the ‘Origin-Response’ function.
docker run --rm --volume ${PWD}/lambda/origin-response-function:/build amazonlinux:nodejs /bin/bash -c "source ~/.bashrc; npm init -f -y; npm install sharp --save --only=prod; npm install querystring --save --only=prod; npm install --only=prod"
-
Install the querystring module dependencies and compile the ‘Viewer-Request’ function.
docker run --rm --volume ${PWD}/lambda/viewer-request-function:/build amazonlinux:nodejs /bin/bash -c "source ~/.bashrc; npm init -f -y; npm install querystring --save --only=prod; npm install --only=prod"
-
Package the ‘Origin-Response’ function.
mkdir -p dist && cd lambda/origin-response-function && zip -FS -q -r ../../dist/origin-response-function-12.zip * -x test/\* && cd ../..
-
Package the ‘Viewer-Request’ function.
mkdir -p dist && cd lambda/viewer-request-function && zip -FS -q -r ../../dist/viewer-request-function-12.zip * -x test/\* && cd ../..
-
Choose a bucket in the us-east-1 region to hold the deployment files and upload the zip files created in above steps. If you don't already have an appropriate bucket in that region you can create one using the AWS console or command line tools. These files will be referenced from the CloudFormation template during the next step.
-
In the
imago/cloud-formation-template.yml
file update the<code-bucket>
placeholders with the bucket name you used in the last step. -
Using cloudformation (either cli or the AWS console) deploy the cloudformation template
imago/cloud-formation-template.yml
This will take some time as the cloudfront distribution creation is an extensive process. You can expect up to 20 minutes for this to complete.
-
Upload an high-resolution image to the bucket you created. (eg. images/image.jpg)
-
In a browser navigate to
https://{cloudfront-domain}/images/image.jpg?w=400
Replace cloudfont-domain with the domain name of the cloudfront distribution created during installation. DON'T use the S3 bucket or you won't see any changes.
You should see a resized image that is just 400px wide.
If you look in the bucket you would see a new file:
/<bucket-name>/images/w/400/m/jpg/image.jpg
If your web browser supports webm you might see this file:
/<bucket-name>/images/w/400/m/webm/image.jpg
- o= or original= Requests the original file with no transformation or compression applied. When used all other parameters are ignored.
- w= or width= Sets the desired width. If used without the height parameter the image will be scaled propotionally to the width specified. Values rounded to the nearest 100.
- h= or height= Sets the desired height. If used without the width parameter the image will be scaled propotionally to the height specified. Values rounded to the nearest 100.
- t= or transform= Only used when both width and height are specified. Possible values:
- "c" or "crop": Crop the image to the specified dimensions. Default
- "f" or "fit": Fit the image within the specified dimension.
- q= or quality= Specifies a compression amount for jpeg and webp images. Possible values:
- "l" or "low": Sets quality at 40
- "m", "med", or "medium": Sets quality at 60 Default
- "h" or "high": Sets quality at 80
If the client passes an Accept header that indicates that webp is an acceptable format (Chrome and Android do) a webp image is created instead of a jpeg when jpeg is requested.
S3 keys are constructed following this pattern
/prefix/transform/widthxheight/quality/filetype/filename.extension
12345/myimage.jpg?w=400 yeilds
/12345/w/400/m/jpg/myimage.jpg
note that the default value of medium quality is used
12345/myimage.jpg?h=400 yeilds
/12345/h/400/m/jpg/myimage.jpg
12345/myimage.jpg?w=100&h=100 yeilds
/12345/c/100x100/m/jpg/myimage.jpg
note that the default values of crop and medium quality are used
12345/myimage.jpg?w=600&h=400&t=f yeilds
/12345/f/600x400/m/jpg/myimage.jpg
12345/myimage.jpg?w=1200&q=h yeilds
/12345/w/1200/h/jpg/myimage.jpg
Original source for these functions from: https://aws.amazon.com/blogs/networking-and-content-delivery/resizing-images-with-amazon-cloudfront-lambdaedge-aws-cdn-blog/
Image resizing done using Sharp: https://sharp.pixelplumbing.com
-
Ensure you have Node 12.x installed
-
Clone the repository
-
Test the viewer request function go to the
imago/lambda/viewer-request-function
foldernpm test
-
Test the origin response function go to the
imago/lambda/origin-response-function
foldernpm test
These tests operate on a local image and do not access S3. You can comment out the mock sections to see it operate on live files.