Suppose you want to create a thumbnail for each image file that is uploaded to a bucket. You can create a Lambda function (CreateThumbnail
) that Amazon S3 can invoke when objects are created. Then, the Lambda function can read the image object from the source bucket and create a thumbnail image target bucket.
Upon completing this tutorial, you will have the following Amazon S3, Lambda, and IAM resources in your account:
Lambda Resources
- A Lambda function.
- An access policy associated with your Lambda function that grants Amazon S3 permission to invoke the Lambda function.
IAM Resources
- An execution role that grants permissions that your Lambda function needs through the permissions policy associated with this role.
Amazon S3 Resources
- A source bucket with a notification configuration that invokes the Lambda function.
- A target bucket where the function saves resized images.
This tutorial assumes that you have some knowledge of basic Lambda operations and the Lambda console. If you haven't already, follow the instructions in Getting Started to create your first Lambda function.
To follow the procedures in this guide, you will need a command line terminal or shell to run commands. Commands are shown in listings preceded by a prompt symbol ($) and the name of the current directory, when appropriate:
~/lambda-project$ this is a command
this is output
For long commands, an escape character (\
) is used to split a command over multiple lines.
On Linux and macOS, use your preferred shell and package manager. On Windows 10, you can install the Windows Subsystem for Linux to get a Windows-integrated version of Ubuntu and Bash.
Create the execution role that gives your function permission to access AWS resources.
To create an execution role
-
Open the roles page in the IAM console.
-
Choose Create role.
-
Create a role with the following properties.
- Trusted entity – AWS Lambda.
- Permissions – AWSLambdaExecute.
- Role name – lambda-s3-role.
The AWSLambdaExecute policy has the permissions that the function needs to manage objects in Amazon S3 and write logs to CloudWatch Logs.
Follow the steps to create buckets and upload an object.
-
Open the Amazon S3 console.
-
Create two buckets. The target bucket name must be source followed by resized, where source is the name of the bucket you want to use for the source. For example,
mybucket
andmybucketresized
. -
In the source bucket, upload a .jpg object,
HappyFace.jpg
.When you invoke the Lambda function manually before you connect to Amazon S3, you pass sample event data to the function that specifies the source bucket and
HappyFace.jpg
as the newly created object so you need to create this sample object first.
The following example code receives an Amazon S3 event input and processes the message that it contains. It resizes an image in the source bucket and saves the output to the target bucket.
Note
For sample code in other languages, see Sample Amazon Simple Storage Service Function Code.
Example index.js
// dependencies
var async = require('async');
var AWS = require('aws-sdk');
var gm = require('gm')
.subClass({ imageMagick: true }); // Enable ImageMagick integration.
var util = require('util');
// constants
var MAX_WIDTH = 100;
var MAX_HEIGHT = 100;
// get reference to S3 client
var s3 = new AWS.S3();
exports.handler = function(event, context, callback) {
// Read options from the event.
console.log("Reading options from event:\n", util.inspect(event, {depth: 5}));
var srcBucket = event.Records[0].s3.bucket.name;
// Object key may have spaces or unicode non-ASCII characters.
var srcKey =
decodeURIComponent(event.Records[0].s3.object.key.replace(/\+/g, " "));
var dstBucket = srcBucket + "resized";
var dstKey = "resized-" + srcKey;
// Sanity check: validate that source and destination are different buckets.
if (srcBucket == dstBucket) {
callback("Source and destination buckets are the same.");
return;
}
// Infer the image type.
var typeMatch = srcKey.match(/\.([^.]*)$/);
if (!typeMatch) {
callback("Could not determine the image type.");
return;
}
var imageType = typeMatch[1];
if (imageType != "jpg" && imageType != "png") {
callback('Unsupported image type: ${imageType}');
return;
}
// Download the image from S3, transform, and upload to a different S3 bucket.
async.waterfall([
function download(next) {
// Download the image from S3 into a buffer.
s3.getObject({
Bucket: srcBucket,
Key: srcKey
},
next);
},
function transform(response, next) {
gm(response.Body).size(function(err, size) {
// Infer the scaling factor to avoid stretching the image unnaturally.
var scalingFactor = Math.min(
MAX_WIDTH / size.width,
MAX_HEIGHT / size.height
);
var width = scalingFactor * size.width;
var height = scalingFactor * size.height;
// Transform the image buffer in memory.
this.resize(width, height)
.toBuffer(imageType, function(err, buffer) {
if (err) {
next(err);
} else {
next(null, response.ContentType, buffer);
}
});
});
},
function upload(contentType, data, next) {
// Stream the transformed image to a different S3 bucket.
s3.putObject({
Bucket: dstBucket,
Key: dstKey,
Body: data,
ContentType: contentType
},
next);
}
], function (err) {
if (err) {
console.error(
'Unable to resize ' + srcBucket + '/' + srcKey +
' and upload to ' + dstBucket + '/' + dstKey +
' due to an error: ' + err
);
} else {
console.log(
'Successfully resized ' + srcBucket + '/' + srcKey +
' and uploaded to ' + dstBucket + '/' + dstKey
);
}
callback(null, "message");
}
);
};
The deployment package is a .zip file containing your Lambda function code and dependencies.
To create a deployment package
-
Create a folder (
examplefolder
), and then create a subfolder (node_modules
). -
Install the Node.js platform. For more information, see the Node.js website.
-
Install dependencies. The code examples use the following libraries:
- AWS SDK for JavaScript in Node.js
- gm, GraphicsMagick for node.js
- Async utility module
The AWS Lambda runtime already has the AWS SDK for JavaScript in Node.js, so you only need to install the other libraries. Open a command prompt, navigate to the
examplefolder
, and install the libraries using thenpm
command, which is part of Node.js.npm install async gm
-
Save the sample code to a file named index.js.
-
Review the preceding code and note the following:
- The function knows the source bucket name and the key name of the object from the event data it receives as parameters. If the object is a .jpg, the code creates a thumbnail and saves it to the target bucket.
- The code assumes that the destination bucket exists and its name is a concatenation of the source bucket name followed by the string
resized
. For example, if the source bucket identified in the event data isexamplebucket
, the code assumes you have anexamplebucketresized
destination bucket. - For the thumbnail it creates, the code derives its key name as the concatenation of the string
resized-
followed by the source object key name. For example, if the source object key issample.jpg
, the code creates a thumbnail object that has the keyresized-sample.jpg
.
-
Save the file as
index.js
inexamplefolder
. After you complete this step, you will have the following folder structure:index.js /node_modules/gm /node_modules/async
-
Zip the index.js file and the node_modules folder as
CreateThumbnail.zip
.
To create the function
-
Create a Lambda function with the
create-function
command.$ aws lambda create-function --function-name CreateThumbnail \ --zip-file fileb://function.zip --handler index.handler --runtime nodejs8.10 \ --role role-arn \ --timeout 10 --memory-size 1024
The preceding command specifies a 10-second timeout value as the function configuration. Depending on the size of objects you upload, you might need to increase the timeout value using the following AWS CLI command.
$ aws lambda update-function-configuration --function-name CreateThumbnail --timeout 30
In this step, you invoke the Lambda function manually using sample Amazon S3 event data.
To test the Lambda function
-
Save the following Amazon S3 sample event data in a file and save it as
inputFile.txt
. You need to update the JSON by providing your sourcebucket name and a .jpg object key.{ "Records":[ { "eventVersion":"2.0", "eventSource":"aws:s3", "awsRegion":"us-west-2", "eventTime":"1970-01-01T00:00:00.000Z", "eventName":"ObjectCreated:Put", "userIdentity":{ "principalId":"AIDAJDPLRKLG7UEXAMPLE" }, "requestParameters":{ "sourceIPAddress":"127.0.0.1" }, "responseElements":{ "x-amz-request-id":"C3D13FE58DE4C810", "x-amz-id-2":"FMyUVURIY8/IgAtTv8xRjskZQpcIZ9KG4V5Wp6S7S/JRWeUWerMUE5JgHvANOjpD" }, "s3":{ "s3SchemaVersion":"1.0", "configurationId":"testConfigRule", "bucket":{ "name":"sourcebucket", "ownerIdentity":{ "principalId":"A3NL1KOZZKExample" }, "arn":"arn:aws:s3:::sourcebucket" }, "object":{ "key":"HappyFace.jpg", "size":1024, "eTag":"d41d8cd98f00b204e9800998ecf8427e", "versionId":"096fKKXTRTtl3on89fVO.nfljtsv6qko" } } } ] }
-
Run the following Lambda CLI
invoke
command to invoke the function. Note that the command requests asynchronous execution. You can optionally invoke it synchronously by specifyingRequestResponse
as theinvocation-type
parameter value.$ aws lambda invoke --function-name CreateThumbnail --invocation-type Event \ --payload file://inputfile.txt outputfile.txt
-
Verify that the thumbnail was created in the target bucket.
In this step, you add the remaining configuration so that Amazon S3 can publish object-created events to AWS Lambda and invoke your Lambda function. You do the following in this step:
- Add permissions to the Lambda function access policy to allow Amazon S3 to invoke the function.
- Add notification configuration to your source bucket. In the notification configuration, you provide the following:
- Event type for which you want Amazon S3 to publish events. For this tutorial, you specify the
s3:ObjectCreated:*
event type so that Amazon S3 publishes events when objects are created. - Lambda function to invoke.
- Event type for which you want Amazon S3 to publish events. For this tutorial, you specify the
To add permissions to the function policy
-
Run the following Lambda CLI
add-permission
command to grant Amazon S3 service principal (s3.amazonaws.com
) permissions to perform thelambda:InvokeFunction
action. Note that permission is granted to Amazon S3 to invoke the function only if the following conditions are met:- An object-created event is detected on a specific bucket.
- The bucket is owned by a specific AWS account. If a bucket owner deletes a bucket, some other AWS account can create a bucket with the same name. This condition ensures that only a specific AWS account can invoke your Lambda function.
$ aws lambda add-permission --function-name CreateThumbnail --principal s3.amazonaws.com \ --statement-id some-unique-id --action "lambda:InvokeFunction" \ --source-arn arn:aws:s3:::sourcebucket \ --source-account bucket-owner-account-id
-
Verify the function's access policy by running the AWS CLI
get-policy
command.$ aws lambda get-policy --function-name function-name
Add notification configuration on the source bucket to request Amazon S3 to publish object-created events to Lambda.
To configure notifications
-
Open the Amazon S3 console.
-
Choose the source bucket.
-
Choose Properties.
-
Under Events, configure a notification with the following settings.
- Name – lambda-trigger.
- Events – ObjectCreate (All).
- Send to – Lambda function.
- Lambda – CreateThumbnail.
For more information on event configuration, see Enabling Event Notifications in the Amazon Simple Storage Service Console User Guide.
Now you can test the setup as follows:
-
Upload .jpg or .png objects to the source bucket using the Amazon S3 console.
-
Verify that the thumbnail was created in the target bucket using the
CreateThumbnail
function. -
View logs in the CloudWatch console.