Creating AWS Lambda functions from R
code can be a powerful way to make our local R
code available in the cloud as an on-demand serverless service. In recent weeks I’ve been working on my package r2lambda to build and deploy AWS Lambda functions from the R
console. In an introductory article about the package a few weeks ago, I covered the basics and showcased the main usage:
# install_packages("remotes")
::install_github("discindo/r2lambda")
remotes
::deploy_lambda(
r2lambdatag = "my-lambda",
runtime_function = "my_fun",
runtime_path = "path/to/script/of/my_fun",
dependencies = c("ggplot2", "dplyr")
)
In my first post on the topic, I noted that this very much a work in progress, and that I hope to actively develop this package by adding features that would make it useful in more realistic scenarios.
Testing R-based AWS Lambda functions locally
One such feature was the ability to test our R
-based Lambda function locally before deploying to the AWS cloud. This is super useful, because, depending on the size of the docker
image it might take a while to push it to the AWS ECR repository. Also, creating an AWS Lambda function from the ECR docker image, requires granting the function a role and permissions policy to execute or access other services. All of these steps create resources/services in your AWS account, so ideally, would only be done when we are certain that docker image we are deploying as a Lambda function works correctly.
This procedure is well documented. After creating the Lambda docker
image, to test it locally, we need to 1) run a container on our local machine and 2) send a request to it with curl
. Essentially, we are ‘invoking’ the function with the same payload the same way we’d do it in the cloud, but locally. This is the best way to know that everything works as intended. Merely testing the R
code separately, without the Lambda docker
context, might not be enough.
I packaged this routine in the function test_lambda
and added this step to the {r2lamdba}
deployment workflow. With these changes, instead of one function deploy_lambda
that would build and deploy the image, we now have thee steps:
build_lambda
– to build and tag thedocker
image locallytest_lambda
– to test the lambda docker container locally (optional but recommended)deploy_lambda
– to push the docker image to the cloud and create the function
Or in code:
Build a docker image for the lambda function
<- "parity"
runtime_function <- system.file("parity.R", package = "r2lambda")
runtime_path <- NULL
dependencies
# Might take a while, its building a docker image
build_lambda(
tag = "parity1",
runtime_function = runtime_function,
runtime_path = runtime_path,
dependencies = dependencies
)
Test the lambda docker image locally
<- list(number = 2)
payload <- "parity1"
tag test_lambda(tag = "parity1", payload)
Deploy to AWS Lambda
# Might take a while, its pushing it to a remote repository
deploy_lambda(tag = "parity1")
Invoke deployed lambda
invoke_lambda(
function_name = "parity1",
invocation_type = "RequestResponse",
payload = list(number = 2),
include_logs = FALSE
)
#> Lambda response payload:
#> {"parity":"even"}
So, although we’ve added a couple of steps to the workflow, I think its for the better, as we can have finer control over building and deploying. For example, sometimes we might want to deploy a Lambda function from an existing docker
image, so de-coupling the build and deploy steps makes a lot of sense.
Would love to hear from you! Let me know if you try the r2lamdba
package or if you know of any similar projects.