forked from pulumi/examples
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Thumbnailer with AWS Lambda on containers (pulumi#850)
- Loading branch information
1 parent
04e1ed7
commit b5c7e8b
Showing
9 changed files
with
248 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
name: video-thumbnailer-lambda | ||
runtime: nodejs | ||
description: A video thumbnail extractor using serverless functions packaged with containers | ||
template: | ||
config: | ||
aws:region: | ||
description: The AWS region to deploy into | ||
default: us-west-2 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,121 @@ | ||
[data:image/s3,"s3://crabby-images/6cda6/6cda6ebc2b40bbe5456b997a85c4bc3ff16add5d" alt="Deploy"](https://app.pulumi.com/new) | ||
|
||
# Video Thumbnailer Using AWS Lambda | ||
|
||
A video thumbnail extractor using serverless functions. The video processing function is packaged as a Docker container. | ||
|
||
Navigate to [Running Container Images in AWS Lambda](https://www.pulumi.com/blog/aws-lambda-container-support/) for a full walkthrough. | ||
|
||
## Prerequisites | ||
|
||
To run this example, make sure [Docker](https://docs.docker.com/engine/installation/) is installed and running. | ||
|
||
## Running the App | ||
|
||
1. Create a new stack: | ||
|
||
``` | ||
pulumi stack init dev | ||
``` | ||
1. Configure Pulumi to use an AWS region of your choice, for example: | ||
``` | ||
pulumi config set aws:region us-west-2 | ||
``` | ||
1. Restore NPM modules via `npm install` or `yarn install`. | ||
1. Preview and deploy the app via `pulumi up`. The preview will take some time, as it builds a Docker container. A total of 16 resources are created. | ||
``` | ||
$ pulumi up | ||
Previewing update (dev) | ||
... | ||
Do you want to perform this update? yes | ||
Updating (dev) | ||
Type Name Status | ||
+ pulumi:pulumi:Stack video-thumbnailer-lambda-dev created | ||
+ ├─ awsx:ecr:Repository sampleapp created | ||
+ │ ├─ aws:ecr:Repository sampleapp created | ||
+ │ └─ aws:ecr:LifecyclePolicy sampleapp created | ||
+ ├─ aws:s3:Bucket bucket created | ||
+ │ ├─ aws:s3:BucketEventSubscription onNewThumbnail created | ||
+ │ │ └─ aws:lambda:Permission onNewThumbnail created | ||
+ │ ├─ aws:s3:BucketEventSubscription onNewVideo created | ||
+ │ │ └─ aws:lambda:Permission onNewVideo created | ||
+ │ └─ aws:s3:BucketNotification onNewVideo created | ||
+ ├─ aws:iam:Role onNewThumbnail created | ||
+ ├─ aws:iam:Role thumbnailerRole created | ||
+ ├─ aws:lambda:Function onNewThumbnail created | ||
+ ├─ aws:iam:RolePolicyAttachment onNewThumbnail-32be53a2 created | ||
+ ├─ aws:iam:RolePolicyAttachment lambdaFullAccess created | ||
+ └─ aws:lambda:Function thumbnailer created | ||
Outputs: | ||
bucketName: "bucket-7c6b55a" | ||
Resources: | ||
+ 16 created | ||
Duration: 1m41s | ||
``` | ||
1. View the stack outputs: | ||
``` | ||
$ pulumi stack output | ||
Current stack outputs (1): | ||
OUTPUT VALUE | ||
bucketName bucket-7c6b55a | ||
``` | ||
1. Upload a video, embedding the timestamp in the filename: | ||
``` | ||
$ aws s3 cp ./sample/cat.mp4 s3://$(pulumi stack output bucketName)/cat_00-01.mp4 | ||
upload: sample/cat.mp4 to s3://***/cat_00-01.mp4 | ||
``` | ||
1. View the logs from both Lambda functions: | ||
``` | ||
$ pulumi logs -f | ||
Collecting logs for stack dev since 2020-12-02T08:58:43.000+01:00. | ||
2020-12-02T09:58:39.747+01:00[ thumbnailer-dbb2a35] START RequestId: 3ec2886e-e739-4764-be3b-a8e5a48a4986 Version: $LATEST | ||
2020-12-02T09:58:39.750+01:00[ thumbnailer-dbb2a35] 2020-12-02T08:58:39.748Z 3ec2886e-e739-4764-be3b-a8e5a48a4986 INFO Video handler called | ||
2020-12-02T09:58:39.750+01:00[ thumbnailer-dbb2a35] 2020-12-02T08:58:39.750Z 3ec2886e-e739-4764-be3b-a8e5a48a4986 INFO aws s3 cp s3://bucket-33b87c2/cat_00-01.mp4 /tmp/cat_00-01.mp4 | ||
download: s3://bucket-33b87c2/cat_00-01.mp4 to ../../tmp/cat_00-01.mp4ed 256.0 KiB/666.5 KiB (1.2 MiB/s) with 1 file(s) remaining | ||
2020-12-02T09:58:53.068+01:00[ thumbnailer-dbb2a35] 2020-12-02T08:58:53.068Z 3ec2886e-e739-4764-be3b-a8e5a48a4986 INFO ffmpeg -v error -i /tmp/cat_00-01.mp4 -ss 00:01 -vframes 1 -f image2 -an -y /tmp/cat.jpg | ||
2020-12-02T09:59:01.701+01:00[ thumbnailer-dbb2a35] 2020-12-02T08:59:01.701Z 3ec2886e-e739-4764-be3b-a8e5a48a4986 INFO aws s3 cp /tmp/cat.jpg s3://bucket-33b87c2/cat.jpg | ||
upload: ../../tmp/cat.jpg to s3://bucket-33b87c2/cat.jpg pleted 86.6 KiB/86.6 KiB (315.8 KiB/s) with 1 file(s) remaining | ||
2020-12-02T09:59:11.628+01:00[ thumbnailer-dbb2a35] 2020-12-02T08:59:11.627Z 3ec2886e-e739-4764-be3b-a8e5a48a4986 INFO *** New thumbnail: file cat_00-01.mp4 was saved at 2020-12-02T08:58:33.845Z. | ||
2020-12-02T09:59:11.668+01:00[ thumbnailer-dbb2a35] END RequestId: 3ec2886e-e739-4764-be3b-a8e5a48a4986 | ||
2020-12-02T09:59:11.668+01:00[ thumbnailer-dbb2a35] REPORT RequestId: 3ec2886e-e739-4764-be3b-a8e5a48a4986 Duration: 31920.84 ms Billed Duration: 32733 ms Memory Size: 128 MB Max Memory Used: 128 MB Init Duration: 811.55 ms | ||
2020-12-02T09:59:11.777+01:00[ onNewThumbnail-2f969e0] START RequestId: 07c13039-eccb-4e38-a3cf-c7fa11982b84 Version: $LATEST | ||
2020-12-02T09:59:11.788+01:00[ onNewThumbnail-2f969e0] 2020-12-02T08:59:11.782Z 07c13039-eccb-4e38-a3cf-c7fa11982b84 INFO onNewThumbnail called | ||
2020-12-02T09:59:11.788+01:00[ onNewThumbnail-2f969e0] 2020-12-02T08:59:11.788Z 07c13039-eccb-4e38-a3cf-c7fa11982b84 INFO *** New thumbnail: file cat.jpg was saved at 2020-12-02T08:59:06.333Z. | ||
2020-12-02T09:59:11.809+01:00[ onNewThumbnail-2f969e0] END RequestId: 07c13039-eccb-4e38-a3cf-c7fa11982b84 | ||
2020-12-02T09:59:11.809+01:00[ onNewThumbnail-2f969e0] REPORT RequestId: 07c13039-eccb-4e38-a3cf-c7fa11982b84 Duration: 31.96 ms Billed Duration: 32 ms Memory Size: 128 MB Max Memory Used: 65 MB Init Duration: 171.22 ms | ||
``` | ||
1. Download the key frame: | ||
``` | ||
$ aws s3 cp s3://$(pulumi stack output bucketName)/cat.jpg . | ||
download: s3://***/cat.jpg to ./cat.jpg | ||
``` | ||
## Clean up | ||
To clean up the resources, you will first need to clear the contents of the bucket. | ||
```bash | ||
aws s3 rm s3://$(pulumi stack output bucketName) --recursive | ||
``` | ||
|
||
Then, run `pulumi destroy` and answer the confirmation question at the prompt. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,24 @@ | ||
FROM amazon/aws-lambda-nodejs:12 | ||
ARG FUNCTION_DIR="/var/task" | ||
|
||
# Install tar and xz | ||
RUN yum install tar xz unzip -y | ||
|
||
# Install awscli | ||
RUN curl "https://awscli.amazonaws.com/awscli-exe-linux-x86_64.zip" -o "awscliv2.zip" -s | ||
RUN unzip -q awscliv2.zip | ||
RUN ./aws/install | ||
|
||
# Install ffmpeg | ||
RUN curl https://johnvansickle.com/ffmpeg/releases/ffmpeg-release-amd64-static.tar.xz -o ffmpeg.tar.xz -s | ||
RUN tar -xf ffmpeg.tar.xz | ||
RUN mv ffmpeg-4.3.1-amd64-static/ffmpeg /usr/bin | ||
|
||
# Create function directory | ||
RUN mkdir -p ${FUNCTION_DIR} | ||
|
||
# Copy handler function and package.json | ||
COPY index.js ${FUNCTION_DIR} | ||
|
||
# Set the CMD to your handler (could also be done as a parameter override outside of the Dockerfile) | ||
CMD [ "index.handler" ] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,31 @@ | ||
'use strict'; | ||
const { execSync } = require('child_process'); | ||
|
||
function run(command) { | ||
console.log(command); | ||
const result = execSync(command, {stdio: 'inherit'}); | ||
if (result != null) { | ||
console.log(result.toString()); | ||
} | ||
} | ||
|
||
exports.handler = async (event) => { | ||
console.log("Video handler called"); | ||
|
||
if (!event.Records) { | ||
return; | ||
} | ||
|
||
for (const record of event.Records) { | ||
const fileName = record.s3.object.key; | ||
const bucketName = record.s3.bucket.name; | ||
const thumbnailFile = fileName.substring(0, fileName.indexOf("_")) + ".jpg"; | ||
const framePos = fileName.substring(fileName.indexOf("_")+1, fileName.indexOf(".")).replace("-", ":"); | ||
|
||
run(`aws s3 cp s3://${bucketName}/${fileName} /tmp/${fileName}`); | ||
run(`ffmpeg -v error -i /tmp/${fileName} -ss ${framePos} -vframes 1 -f image2 -an -y /tmp/${thumbnailFile}`); | ||
run(`aws s3 cp /tmp/${thumbnailFile} s3://${bucketName}/${thumbnailFile}`); | ||
|
||
console.log(`*** New thumbnail: file ${fileName} was saved at ${record.eventTime}.`); | ||
} | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,49 @@ | ||
// Copyright 2016-2020, Pulumi Corporation. All rights reserved. | ||
|
||
import * as aws from "@pulumi/aws"; | ||
import * as awsx from "@pulumi/awsx"; | ||
|
||
// A bucket to store videos and thumbnails. | ||
const bucket = new aws.s3.Bucket("bucket"); | ||
|
||
const image = awsx.ecr.buildAndPushImage("sampleapp", { | ||
context: "./docker-ffmpeg-thumb", | ||
}); | ||
const role = new aws.iam.Role("thumbnailerRole", { | ||
assumeRolePolicy: aws.iam.assumeRolePolicyForPrincipal({ Service: "lambda.amazonaws.com" }), | ||
}); | ||
new aws.iam.RolePolicyAttachment("lambdaFullAccess", { | ||
role: role.name, | ||
policyArn: aws.iam.ManagedPolicies.AWSLambdaFullAccess | ||
}); | ||
|
||
const thumbnailer = new aws.lambda.Function("thumbnailer", { | ||
packageType: "Image", | ||
imageUri: image.imageValue, | ||
role: role.arn, | ||
timeout: 900, | ||
}); | ||
|
||
// When a new video is uploaded, run the FFMPEG task on the video file. | ||
// Use the time index specified in the filename (e.g. cat_00-01.mp4 uses timestamp 00:01) | ||
bucket.onObjectCreated("onNewVideo", thumbnailer, { filterSuffix: ".mp4" }); | ||
|
||
// Export the bucket name. | ||
export const bucketName = bucket.id; | ||
|
||
// When a new thumbnail is created, log a message. | ||
bucket.onObjectCreated("onNewThumbnail", new aws.lambda.CallbackFunction<aws.s3.BucketEvent, void>("onNewThumbnail", { | ||
callback: async bucketArgs => { | ||
console.log("onNewThumbnail called"); | ||
if (!bucketArgs.Records) { | ||
return; | ||
} | ||
|
||
for (const record of bucketArgs.Records) { | ||
console.log(`*** New thumbnail: file ${record.s3.object.key} was saved at ${record.eventTime}.`); | ||
} | ||
}, | ||
policies: [ | ||
aws.iam.ManagedPolicies.AWSLambdaFullAccess, // Provides wide access to "serverless" services (Dynamo, S3, etc.) | ||
], | ||
}), { filterSuffix: ".jpg" }); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
{ | ||
"name": "video-thumbnailer-lambda", | ||
"version": "0.1.0", | ||
"main": "index.js", | ||
"dependencies": { | ||
"@pulumi/pulumi": "^2.0.0", | ||
"@pulumi/aws": "^3.17.0", | ||
"@pulumi/awsx": "^0.20.0" | ||
} | ||
} |
Binary file not shown.