Deep dive into AWS Lambda using Spring Boot 3 and GraalVM

Overview

Serverless computing has become increasingly popular in recent years, and AWS Lambda is one of the most widely used serverless platforms. Spring Boot, on the other hand, is a popular framework for building microservices and web applications in Java. By combining these two technologies, developers can create powerful and scalable serverless applications.

In this comprehensive guide, we'll dive into the details of building an AWS Lambda function with Spring Boot 3 and GraalVM Native Image. We'll explore how to use GraalVM Native Image to create a custom runtime for our Lambda function, which can result in faster startup times and lower memory usage. Additionally, we'll cover how to use the AWS Serverless Application Model (SAM) to deploy our Lambda function to AWS, how to test our function locally, and how to automate the deployment process with GitHub Actions.

By the end of this guide, you'll have a solid understanding of how to build and deploy efficient and scalable serverless applications with Spring Boot and AWS Lambda.

This blog post will guide you through the steps of building a production-ready native AWS Lambda function that receives batch messages from an SQS and writes them into a DynamoDB table. It will explain all the details along the way.

To DIY you will need
  • Basic understanding of Spring and Spring Boot
  • Java 17
  • Docker
  • AWS SAM (Installation guide)
  • GraalVM (for local testing Tutorial how to install it locally) optional (you may need it to build a native image locally)
  • AWS Account optional (you may need it if you want to deploy lambda to AWS)
  • GitHub Repository optional (you may need it if you want to automate deployments to AWS using GitHub Actions workflows)

GraalVM is a universal VM developed by Oracle JDK that is designed to accelerate the execution of applications written in Java and other JVM languages, supporting JVM Runtime Mode, Java on Truffle, and Native Image, with the latter being the focus of this blog post.

The GraalVM native image is a component of GraalVM that ahead-of-time (AOT) compiles Java code into a standalone executable, which can be executed without a JVM. This allows for faster startup times and a lower memory footprint compared to traditional JVM-based applications. Native-Image is particularly useful for serverless and cloud-native applications, where lower startup times and lower memory usage can help reduce resource utilization, improve scalability, and reduce overall costs.

Note: The artifact produced by the GraalVM is platform-dependent, meaning you won't be able to run it on a platform with a different architecture or OS.

There are some minor inconveniences of using GraalVM and assembling standalone executables:

  • Longer build time (depending on machine and application size, it can take anywhere from ~2 minutes to ~10 minutes)
  • GraalVM is not directly aware of the dynamic elements of your code and must be hinted about reflection, dynamic proxies, any resources, and serialization. (The Spring Boot plugin will try to do most of the job for you, but sometimes you have to provide hints on your own.)
  • GraalVM is a relatively new technology, and there may be limited tooling and resources available to help developers debug and troubleshoot issues.

If you're not scared of these inconveniences, let's move forward and try to understand what Spring Ahead-of-Time Processing is and how it simplifies our interaction with native-image assembly.

Most of us know that the magic hidden behind the scenes of the Spring Framework heavily relies on reflection and Dynamic Proxies. However, this can be a complete nightmare when it comes to building a native executable binary using GraalVM. Fortunately, the Spring Native project (which became part of Spring Boot) pre-processes all dynamically accessed and created classes using Ahead-of-Time Processing.

To explain what AOT Processing does, let's take a look at a class from the lambda that we'll be creating:

 1@Configuration
 2public class UseCaseConfig {
 3
 4    @Bean
 5    public ProcessNoteRequest processNoteRequest(
 6            ReadSqsMessageBody readSqsMessageBody,
 7            SaveNoteRequest saveNoteRequest
 8    ) {
 9        return new ProcessNoteRequest(readSqsMessageBody, saveNoteRequest);
10    }
11}
java

This is a simple configuration class that defines a single bean. In a JVM runtime, Spring would create a proxy object for the UseCaseConfig class and attempt to retrieve a bean instance by calling the processNoteRequest method. This method would return either the same instance or a new one based on the bean's scope. When a bean is created, Spring uses reflection to perform dependency injection, call the init method and perform other initialization tasks.

For each method annotated with the @Bean annotation, Spring creates a Bean Definition, which provides instructions on how the bean should be assembled. The configuration class itself also requires a Bean Definition.

To sum up, the configuration above will provide three configuration points during the runtime (❗ keep in mind that all these objects are created during application startup):

  1. A BeanDefinition for the UseCaseConfig class.
  2. A BeanDefinition for the ProcessNoteRequest class.
  3. A proxy on top of the processNoteRequest method that will instantiate this bean.

Now adding the org.springframework.boot plugin from version 3.0.0 and above, add additional steps to our default compilation tasks, which include AOT processing.

After a successful build of the projects the above class will be transformed in the following Java Configuration class(❗ these classes are generated during the build time and later are used by GraalVM to build the final native artifact):

 1public class UseCaseConfig__BeanDefinitions {
 2    public UseCaseConfig__BeanDefinitions() {
 3    }
 4
 5    public static BeanDefinition getUseCaseConfigBeanDefinition() {
 6        Class<?> beanType = UseCaseConfig.class;
 7        RootBeanDefinition beanDefinition = new RootBeanDefinition(beanType);
 8        ConfigurationClassUtils.initializeConfigurationClass(UseCaseConfig.class);
 9        beanDefinition.setInstanceSupplier(UseCaseConfig..SpringCGLIB..0::new);
10        return beanDefinition;
11    }
12
13    private static BeanInstanceSupplier<ProcessNoteRequest> getProcessNoteRequestInstanceSupplier() {
14        return BeanInstanceSupplier.forFactoryMethod(UseCaseConfig.class, "processNoteRequest", new Class[]{ReadSqsMessageBody.class, SaveNoteRequest.class}).withGenerator((registeredBean, args) -> {
15            return ((UseCaseConfig)registeredBean.getBeanFactory().getBean(UseCaseConfig.class)).processNoteRequest((ReadSqsMessageBody)args.get(0), (SaveNoteRequest)args.get(1));
16        });
17    }
18
19    public static BeanDefinition getProcessNoteRequestBeanDefinition() {
20        Class<?> beanType = ProcessNoteRequest.class;
21        RootBeanDefinition beanDefinition = new RootBeanDefinition(beanType);
22        beanDefinition.setInstanceSupplier(getProcessNoteRequestInstanceSupplier());
23        return beanDefinition;
24    }
25}
java

As a result, we can see that we have the exact 3 things we've mentioned before:

  1. A BeanDefinition for the UseCaseConfig class: getUseCaseConfigBeanDefinition.
  2. A BeanDefinition for the ProcessNoteRequest class: getProcessNoteRequestBeanDefinition.
  3. A supplier that instantiates ProcessNoteRequest: getProcessNoteRequestInstanceSupplier.
In a nutshell

Spring Ahead-of-Time Processing takes all the components that will be created and instantiated during runtime by Spring's magic and transforms them into plain Java configurations.

With this understanding, it's clear why Spring Ahead-of-Time Processing is so important for an application built with GraalVM. Without it, all the annotation-based configurations would need to be manually migrated to Java configurations by developers.

The AWS SAM (Serverless Application Model) is a framework for building serverless applications within AWS, and it is essentially a subset of CloudFormation. Using SAM, you can define and configure all the necessary resources for the lambda, including:

  • Resource Configuration: Detailed configuration of resources, such as timeout, memory, ephemeral storage, concurrency, and more.
  • Triggers: Configuring triggers for the lambda, such as HTTP endpoints, SQS messages, SNS notifications, S3 events, etc.
  • Permissions: Configuring AWS permissions for the lambda by referencing existing IAM policies, IAM roles, and connectors.
  • Runtime: Specifying the runtime in which your code will run (e.g. Java 11, Python, Node.js, custom runtime, etc.).
  • Deployment: Handling deployment automatically by SAM, thereby eliminating deployment overhead. You just need to update your code and configuration, and trigger the deployment.

Note: SAM offers a nice feature that allows you to emulate the AWS Lambda environment locally for debugging and testing purposes using the sam local invoke command.

It's worth noting that SAM is based on AWS Lambda and is designed to simplify the process of building and deploying serverless applications. By using SAM, you can save time and reduce complexity, allowing you to focus on developing and refining your application's core features.

In a nutshell

SAM is composed of two components. The first is a Command Line Interface (CLI) that allows you to build, create, deploy, and delete Serverless applications on AWS. The second component is a YAML template that defines all the configurations your application consists of.

For example, in the template below, an AWS function is declared that is triggered by an SQS event:

 1AWSTemplateFormatVersion: "2010-09-09"
 2Transform: AWS::Serverless-2016-10-31
 3Description: >
 4    sam-app
 5
 6Globals:
 7  Function:
 8    Timeout: 20
 9    Runtime: java11
10    MemorySize: 256
11    Architectures:
12      - x86_64
13
14Resources:
15  MyFunction:
16    Type: AWS::Serverless::Function
17
18    Properties:
19      CodeUri: .
20      Handler: com.gvart.AppHandler.java
21      Events:
22        SQSEvent:
23          Type: SQS
24          Properties:
25            Queue: my.queue.arn
yaml

AWS Lambda currently supports seven different runtimes, which are:

  1. Java
  2. Python
  3. Node.js
  4. .NET
  5. Go
  6. Ruby
  7. Custom Runtime

Note: The Custom Runtime allows you to implement an AWS Lambda runtime in any programming language by following a simple specification.

The GraalVM native-image produces a binary executable file that is not supported by default runtimes. However, it is possible to run it using custom runtime. The custom runtime is already implemented as part of spring-cloud-function-adapter-aws, but understanding how it works can be helpful. Therefore, we will explore it further.

To create a custom runtime, you simply need to adhere to a straightforward specification that outlines how the runtime should interact with AWS Lambda. By doing so, you gain the ability to construct your runtime using your preferred programming language, which grants you a more flexible approach to constructing serverless applications.

  1. The uploaded artifact should include a script called bootstrap, which serves as the entry point for our AWS Lambda function. This script will be executed to create the runtime.

    Bootstrap script example:

    1#!/bin/sh
    2cd $LAMBDA_TASK_ROOT
    3./application
    
    shell

To visualize the flow, we can use this flow diagram (see explanation below)

AWS Custom runtime Flow Diagram
AWS Custom runtime Flow Diagram

  1. When the application starts up, it needs to perform several initialization steps to ensure it is ready to handle incoming requests efficiently:
    1. Load its configuration from environment variables: _HANDLER, LAMBDA_TASK_ROOT, AWS_LAMBDA_RUNTIME_API. More information on how to do this can be found in the documentation.
    2. Initialize all heavy resources such as JDBC connections, SDK clients, etc. which can be reused across multiple invocations. This helps to avoid the overhead of repeatedly creating and destroying these resources with each request.
    3. Create a global error handler that calls the Lambda's API and immediately exits (This is an essential step to ensure that any unhandled errors are reported to the Lambda's API, allowing the appropriate measures to be taken).
  2. Once the initialization tasks are completed, the custom AWS Lambda runtime enters a loop to process incoming events. The following steps are executed during each iteration of the loop:
    1. The runtime tries to retrieve an incoming event from the Next Invocation API. If there is no response, the runtime exits.
    2. The function handler is invoked. If the invocation fails, the runtime calls the Invocation error API to report the failure.
    3. The runtime returns a response by calling the Invocation response API.
    4. After completing the invocation, the runtime cleans up all the resources used in the current iteration. This is important to prevent any resource leaks and ensure that the runtime is ready to handle the next incoming event.

To find the complete code check out GitHub repository, in this section we will cover only the main parts.

For this implementation, we will focus on a common use case where AWS Lambda is triggered by a message from Amazon Simple Queue Service (SQS), performs some business logic on the message, and then persists the modified message in Amazon DynamoDB.

Project HLD
Project HLD

To prepare the runtime for the AWS Lambda, we will need to add the following plugins to the project:

1plugins {
2    java
3    id("org.springframework.boot") version "$springBootVersion"
4    id("org.graalvm.buildtools.native") version "$nativeBuildToolsVersion"
5}
kotlin

org.graalvm.buildtools.native plugin is needed to create native executable.

And dependencies:

1dependencies {
2    implementation(platform("org.springframework.cloud:spring-cloud-dependencies:$springCloudVersion"))
3    implementation(platform("io.awspring.cloud:spring-cloud-aws-dependencies:$springCloudAwsVersion"))
4
5    implementation("io.awspring.cloud:spring-cloud-aws-starter-dynamodb")
6    implementation("org.springframework.cloud:spring-cloud-starter-function-web")
7    implementation("org.springframework.cloud:spring-cloud-function-adapter-aws")
8    testImplementation("org.springframework.boot:spring-boot-starter-test")
9}
kotlin
  1. spring-cloud-starter-function-web provides the runtime for the function we'll be creating.
  2. spring-cloud-starter-function-adapter-aws adjusts existing runtime and allows to use custom-runtime specification described earlier.

To define a lambda handler (which serves as the entry point to the application), a bean of type Consumer<T>, Function<T,R>, or Supplier<T> needs to be registered in the Spring context.

If we have multiple beans that implement any of these functional interfaces, the spring.cloud.function.definition property in the application.properties or application.yml file must be explicitly configured.

For the given use case where a Lambda function consumes an event from SQS and saves it in DynamoDB, the implementation should use the Consumer<SqsEvent> interface. Here, SqsEvent represents an SQS object that can contain from 1 to 10 messages.

 1@Component
 2public class LambdaFunction implements Consumer<SqsEvent> {
 3
 4    private static final Logger log = LoggerFactory.getLogger(LambdaFunction.class);
 5
 6    private final ProcessNoteRequest processNoteRequest;
 7
 8    public LambdaFunction(ProcessNoteRequest processNoteRequest) {
 9        this.processNoteRequest = processNoteRequest;
10    }
11
12    @Override
13    public void accept(SqsEvent sqsEvent) {
14        log.info("Received event: {}", sqsEvent);
15
16        sqsEvent.records()
17                .stream()
18                .map(SqsMessage::body)
19                .forEach(processNoteRequest::process);
20    }
21}
java

The ProcessNoteRequest implementation reads the message body and saves it in DynamoDB.

 1public class ProcessNoteRequest {
 2
 3    private final ReadSqsMessageBody readSqsMessageBody;
 4    private final SaveNoteRequest saveNoteRequest;
 5
 6    public ProcessNoteRequest(ReadSqsMessageBody readSqsMessageBody, SaveNoteRequest saveNoteRequest) {
 7        this.readSqsMessageBody = readSqsMessageBody;
 8        this.saveNoteRequest = saveNoteRequest;
 9    }
10
11    public void process(String messageBody) {
12        var result = readSqsMessageBody.read(messageBody);
13        saveNoteRequest.save(result);
14    }
15}
java

The ReadSqsMessageBody implementation uses Jackson ObjectMapper to read String message body and convert it to a DTO object.

 1@Component
 2public class DefaultReadSqsMessageBody implements ReadSqsMessageBody {
 3
 4    private final Logger log = LoggerFactory.getLogger(DefaultReadSqsMessageBody.class);
 5
 6    private final ObjectMapper mapper;
 7
 8    public DefaultReadSqsMessageBody(ObjectMapper mapper) {
 9        this.mapper = mapper;
10    }
11
12    @Override
13    public PersistNoteRequest read(String body) {
14        try {
15            return mapper.readValue(body, PersistNoteRequest.class);
16        } catch (JsonProcessingException exception) {
17            log.error("Failed to read value: {}", body, exception);
18            throw new RuntimeException("Failed to process json.");
19        }
20    }
21}
java

The SaveNoteRequest implementation applied some changes on provided DTO using RequestTransformer instance, and finally object is saved in a DynamodbDB table.

 1@Component
 2public class DefaultSaveNoteRequest implements SaveNoteRequest {
 3
 4    private final Logger log = LoggerFactory.getLogger(DefaultSaveNoteRequest.class);
 5
 6    private final DynamoDbTemplate template;
 7    private final RequestTransformer<PersistNoteRequest, Note> transformer;
 8
 9    public DefaultSaveNoteRequest(DynamoDbTemplate template, RequestTransformer<PersistNoteRequest, Note> transformer) {
10        this.template = template;
11        this.transformer = transformer;
12    }
13
14    @Override
15    public void save(PersistNoteRequest request) {
16        log.info("Persisting {}", request);
17
18        var dbEntry = transformer.transform(request);
19        template.save(dbEntry);
20    }
21}
java

The classes mentioned above are quite simple and do not require any special code for the native-image. It's a typical Spring Boot application that we create regularly. This code is almost ready to be deployed to AWS Lambda runtime, but before that, we need to provide some reflection hints to GraalVM. This is because the following operations use reflection:

  • ObjectMapper is used internally by Spring Cloud Function to create SqsEvent and SqsMessage objects.
  • ObjectMapper#readValue uses reflection to access our DTO object PersistNoteRequest
  • The internals of DynamoDbTemplate#save also use reflection to persist our message Note.

To provide these hints, we can put all the details in a /META-INF/native-image/reflect-config.json file. Alternatively, we can use an annotation provided by Spring Boot, which is @RegisterReflectionForBinding, to provide a type-safe way of doing this. This is the last step we need to take, and the following code accomplishes it:

1@Configuration
2@RegisterReflectionForBinding({
3        PersistNoteRequest.class,
4        SqsMessage.class,
5        SqsEvent.class,
6        Note.class
7})
8public class NativeConfig { }
java

After these steps, our code is ready to be compiled into a native image that can be deployed to AWS Lambda runtime.

The main SAM file is template.yaml, which defines all the resources related to the Lambda function.
Using SAM, we will define the following:

  • Custom runtime via Runtime parameter
  • An AWS Lambda function named EveryNotePersisterFunction
    • An SQS trigger for that function called SQSEvent
    • Build instructions via the BuildMethod parameter (this is necessary to guide SAM on how to assemble our application)
  • An SQS queue named MainSqs
  • A simple DynamoDB table with a PK named NoteTable
 1AWSTemplateFormatVersion: "2010-09-09"
 2Transform: AWS::Serverless-2016-10-31
 3
 4Globals:
 5  Function:
 6    Timeout: 20
 7    Runtime: provided.al2 
 8    MemorySize: 256
 9    Architectures:
10      - x86_64
11    Tracing: Active # https://docs.aws.amazon.com/lambda/latest/dg/lambda-x-ray.html
12
13Resources:
14  EveryNotePersisterFunction:
15    Type: AWS::Serverless::Function 
16    Connectors: # <- This section grants permissions to Read and Write to the DynamoDB Table
17      NoteTableConnector:
18        Properties:
19          Destination:
20            Id: NoteTable
21          Permissions:
22            - Read
23            - Write
24
25    Properties:
26      CodeUri: .
27      Handler: none
28      Events:
29        SQSEvent: 
30          Type: SQS
31          Properties:
32            Queue: !GetAtt MainSqs.Arn
33            BatchSize: 10
34            Enabled: true
35    Metadata:
36      BuildMethod: makefile
37
38  MainSqs:
39    Type: AWS::SQS::Queue
40
41  NoteTable:
42    Type: AWS::Serverless::SimpleTable
43    Properties:
44      TableName: note
45      PrimaryKey:
46        Name: id
47        Type: String
...
yaml

To provide instructions for SAM, we need to create a Makefile that will clean the build directory, compile the native executable file, create the previously mentioned bootstrap script and place everything in the designated ARTIFACTS_DIR location.

 1build-EveryNotePersisterFunction:
 2	rm -rf ./build
 3	./gradlew clean
 4	./gradlew nativeCompile
 5	echo '#!/bin/sh' > ./build/bootstrap
 6	echo 'set -euo pipefail' >> ./build/bootstrap
 7	echo './application' >> ./build/bootstrap
 8	chmod 777 ./build/bootstrap
 9	cp ./build/native/nativeCompile/application $(ARTIFACTS_DIR)
10	cp ./build/bootstrap $(ARTIFACTS_DIR)
makefile

The final step is to create a build environment for the Lambda function. Since the build artifact will run in the Amazon Linux 2 operating system, we need to ensure that the artifact can be built and executed on provided.al2. To do so, all build steps will happen inside a Docker container with the base image public.ecr.aws/amazonlinux/amazonlinux:2

Note: This file was generated by the SAM CLI and has been modified to make it easier to manage versions by exposing the GraalVM, Java, and Gradle versions as arguments.

 1FROM public.ecr.aws/amazonlinux/amazonlinux:2
 2
 3RUN yum -y update \
 4    && yum install -y unzip tar gzip bzip2-devel ed gcc gcc-c++ gcc-gfortran \
 5    less libcurl-devel openssl openssl-devel readline-devel xz-devel \
 6    zlib-devel glibc-static libcxx libcxx-devel llvm-toolset-7 zlib-static \
 7    && rm -rf /var/cache/yum
 8
 9# Graal VM
10ARG JAVA_VERSION
11ARG GRAAL_VERSION
12ENV GRAAL_FOLDERNAME graalvm-ce-${JAVA_VERSION}-${GRAAL_VERSION}
13ENV GRAAL_FILENAME graalvm-ce-${JAVA_VERSION}-linux-amd64-${GRAAL_VERSION}.tar.gz
14RUN curl -4 -L https://github.com/graalvm/graalvm-ce-builds/releases/download/vm-${GRAAL_VERSION}/${GRAAL_FILENAME} | tar -xvz
15RUN mv $GRAAL_FOLDERNAME /usr/lib/graalvm
16RUN rm -rf $GRAAL_FOLDERNAME
17
18# Gradle
19ARG GRADLE_VERSION
20ENV GRADLE_FOLDERNAME gradle-${GRADLE_VERSION}
21ENV GRALE_FILENAME ${GRADLE_FOLDERNAME}-bin.zip
22RUN curl -L https://services.gradle.org/distributions/${GRALE_FILENAME} > $GRALE_FILENAME
23RUN unzip -o $GRALE_FILENAME
24RUN mv $GRADLE_FOLDERNAME /usr/lib/gradle
25RUN rm $GRALE_FILENAME
26ENV PATH=$PATH:/usr/lib/gradle/bin
27
28# AWS Lambda Builders
29RUN amazon-linux-extras enable python3.8
30RUN yum clean metadata && yum -y install python3.8
31RUN curl -L get-pip.io | python3.8
32RUN pip3 install aws-lambda-builders
33
34VOLUME /project
35WORKDIR /project
36
37RUN /usr/lib/graalvm/bin/gu install native-image
38RUN ln -s /usr/lib/graalvm/bin/native-image /usr/bin/native-image
39
40ENV JAVA_HOME /usr/lib/graalvm
41
42ENTRYPOINT ["sh"]
...
Dockerfile

Note: Before proceeding with deployment, make sure that:

  1. Docker has at least 4GB RAM.
  2. AWS credentials are properly configured on your local environment.
  3. You have sufficient permissions to create such resources as CloudFormation, S3, SQS, Lambda, and DynamoDB.
  1. Build the Docker image that SAM will use to assemble the native-image executable file by running the ./build-image.sh script
     1 #!/bin/sh
     2 set -e
     3
     4 JAVA_VERSION=java17
     5 GRAAL_VERSION=22.3.0
     6 GRADLE_VERSION=7.6
     7
     8 docker build --build-arg GRADLE_VERSION=$GRADLE_VERSION \
     9 --build-arg JAVA_VERSION=$JAVA_VERSION \
    10 --build-arg GRAAL_VERSION=$GRAAL_VERSION \
    11 -t al2-graalvm:gradle .
    
    shell
  2. Execute sam build --use-container command to build a deployable artifact. It will use the Docker container built in step 1 together with the Makefile created previously.
  3. Now artifact can be deployed (Don't forget to replace <AWS_REGION> placeholder).
    1 sam deploy \
    2 --no-confirm-changeset \
    3 --no-fail-on-empty-changeset \
    4 --resolve-s3 \
    5 --region <AWS_REGION> \
    6 --capabilities CAPABILITY_IAM \
    7 --stack-name every-note-persister
    
    shell

To test created lambda function, I found several options that can be applied separately or combined for different situations.

  1. Using Integration tests with LocalStack and TestContainers (JVM)

    To test our application, we can use old good unit tests or integration tests. However, since our application is not aware of the trigger, such as an SQS message, an API Gateway request, or an SNS notification, we won't be able to create a complete integration test. Fortunately, we can leverage Spring Cloud Function, which provides a utility class that emulates the APIs provided by Amazon for the lambdas with custom-runtime.

    The AWSCustomRuntime can be useful for creating a complete integration test with AWS custom runtime. Additionally, we will need to use LocalStack to run DynamoDB locally and test the complete execution flow.

    Before listing the steps of the test with AWSCustomRuntime, it is important to note a few important things:

    1. We need to add AWSCustomRuntime to the application context.
    2. We need to use a real webEnvironment since it is required by the AWSCustomRuntime.
    3. We need to specify explicitly the function name (via spring.cloud.function.definition property) that will be used as a handler because AWSCustomRuntime defines additional beans that implement functional interfaces.
    4. We need to provide a property with the key _HANDLER. This is necessary for the AWS adapter to realize that the CustomRuntimeEventLoop should be started.

    With all of this in mind, we can create an integration test that will send an event to the CustomRuntime (which will later be pulled by our function), and we expect that within 5 seconds, we will have persisted the event in DynamoDB.

     1@ExtendWith(DynamoDbExtension.class)
     2@SpringBootTest(
     3        classes = {Bootstrap.class, AWSCustomRuntime.class},
     4        webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT,
     5        properties = {
     6                "spring.cloud.function.definition=lambdaFunction",
     7                "_HANDLER=lambdaFunction"
     8        })
     9class LambdaFunctionTest {
    10
    11    // Some setup and dependency injection
    12
    13    @Test
    14    void Test() throws IOException {
    15        //Given
    16        var sqsEvent = Files.readString(event);
    17
    18        //When message is sent to custom-runtime, and it gets consumed by handler
    19        customRuntime.exchange(sqsEvent);
    20
    21        //Then item should be saved in database
    22        await().atMost(Duration.ofSeconds(5)).until(itemSavedInDatabase());
    23
    24    }
    25
    26    private Callable<Boolean> itemSavedInDatabase() {
    27        return () -> {
    28            var items = table.scan().items().stream().toList();
    29            if (items.size() == 1) {
    30                var savedItem = items.get(0);
    31                assertThat(savedItem.getId()).isNotNull();
    32                assertThat(savedItem.getCreatedAt()).isNotNull();
    33                assertThat(savedItem.getNoteBody()).isEqualTo("message body");
    34
    35                return true;
    36            }
    37            //exact one item wasn't saved
    38            return false;
    39        };
    40    }
    41}
    
    ...
    java

    I would recommend using this approach as part of the quality check before deployment, which should be a part of the CI/CD process.

  2. Run locally as a Spring Boot application (JVM)

    Once the application is created, it can be run as a regular Spring Boot application. On startup, the created Lambda handler is exposed as an HTTP endpoint that can be invoked at http://localhost:8080/lambdaFunction. The lambdaFunction is the bean name which is used as the entry point for our application.

    To emulate SQS message receiving, we can use the following curl call:

     1curl --location --request POST 'http://localhost:8080/lambdaFunction' \
     2--header 'Content-Type: application/json' \
     3--data-raw '{
     4    "Records": [
     5        {
     6            "messageId": "19dd0b57-b21e-4ac1-bd88-01bbb068cb78",
     7            "receiptHandle": "MessageReceiptHandle",
     8            "body": "{\"body\":\"hello world\"}",
     9            "attributes": {
    10                "ApproximateReceiveCount": "1",
    11                "SentTimestamp": "1523232000000",
    12                "SenderId": "123456789012",
    13                "ApproximateFirstReceiveTimestamp": "1523232000001"
    14            },
    15            "messageAttributes": {},
    16            "md5OfBody": "7b270e59b47ff90a553787216d55d91d",
    17            "eventSource": "aws:sqs",
    18            "eventSourceARN": "arn:aws:sqs:eu-west-1:156445482446:TestQueue",
    19            "awsRegion": "eu-west-1"
    20        }
    21    ]
    22}'
    
    shell

    I would recommend using this approach when it comes to manual testing and debugging.

  3. Test on AWS (native-image)

    To test a deployed lambda, we can send a message to SQS since that is the trigger for the created lambda function. Afterward, we can verify the DynamoDB table to ensure that the message was saved.

    Send a message to SQS using CLI:

    1aws sqs send-message \
    2--queue-url=https://sqs.eu-west-1.amazonaws.com/<ACCOUNT-ID>/<QUEUE-NAME> \
    3--message-body='{"body":"my message"}'
    
    shell

    Note: Don't forget to replace <ACCOUNT-ID> and <QUEUE-NAME> placeholders with values that correspond to your account and created queue name.

    After that, we should check the Lambda's monitor tab to ensure that the lambda was executed successfully.

    Lambda Monitor screen
    Lambda Monitor screen
    We should also check that there is an entry in DynamoDB with the posted message.
    Dynamodb items
    Dynamodb items

    I would recommend using this approach when we want to perform end-to-end (E2E) tests in a pre-production account or to perform load/performance tests within a real AWS environment, all these steps can be automated and also introduced as part of CI/CD.

We all like to be lazy and delegate our work to someone else, so let's delegate the building, testing, and deployment to a GitHub workflow.

Note: The workflows described below are pretty simple and can be adjusted based on your requirements.

I've created two workflows. The first one is triggered when a pull request is opened, and GitHub Actions will try to build our application, run tests, and finally build a native executable using SAM:

 1on:
 2  pull_request:
 3    branches:
 4      - main
 5jobs:
 6  build_and_test:
 7    name: "Build and Test"
 8    runs-on: ubuntu-latest
 9    steps:
10      - uses: actions/checkout@v3
11
12      - uses: actions/cache@v2
13        with:
14          path: |
15            ~/.gradle/caches
16            ~/.gradle/wrapper            
17          key: ${{ runner.os }}-gradle-${{ hashFiles('**/*.gradle*', '**/gradle-wrapper.properties') }}
18          restore-keys: |
19                        ${{ runner.os }}-gradle-
20      - uses: actions/setup-java@v3
21        with:
22          distribution: 'zulu'
23          java-version: '17'
24
25      - name: "Gradle Build"
26        run: ./gradlew build -x test
27
28      - name: "Gradle Test"
29        run: ./gradlew test
30
31  sam_build:
32    name: "Build using SAM"
33    needs: build_and_test
34    runs-on: ubuntu-latest
35    steps:
36      - uses: actions/checkout@v3
37      - uses: actions/setup-python@v4
38        with:
39          python-version: "3.8"
40      - uses: aws-actions/setup-sam@v2
41
42      - name: "build image if missing"
43        run: |
44          if [[ "$(docker images -q al2-graalvm:gradle 2> /dev/null)" == "" ]]; then
45            echo "Image not found. Building image."
46            ./build-image.sh
47          else
48            echo "Image found. Skipping build."
49          fi          
50      - run: sam build --use-container
...
yaml

The second workflow gets triggered when a merge occurs on the main branch. It will assume an IAM role, build the native-image (this step can be improved by caching the artifact from the previous step to reduce execution time), and finally deploy our serverless application.

Note: Before running this workflow, please ensure that you have completed the following steps:

  1. Create Secrets for GitHub Actions with AWS_REGION and AWS_ROLE which have sufficient permissions to create all resources defined in template.yml.
  2. Configure the OIDC provider to allow GitHub to assume the IAM role without any AWS credentials. Alternatively, you can use AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY
  3. Verify that the role/user you are using for deployment has sufficient permissions. For testing purposes, you can use AdministratorAccess, but it's better to configure a fine-grained IAM policy. You can find more details on how to do this here: https://sst.dev/chapters/customize-the-serverless-iam-policy.html
 1on:
 2  push:
 3    branches:
 4      - main
 5
 6permissions:
 7  id-token: write
 8  contents: read
 9jobs:
10  deploy:
11    runs-on: ubuntu-latest
12    steps:
13      - uses: actions/checkout@v3
14      - uses: actions/setup-python@v4
15        with:
16          python-version: "3.8"
17      - uses: aws-actions/setup-sam@v2
18      - uses: aws-actions/configure-aws-credentials@v1
19        with:
20          role-to-assume: ${{ secrets.AWS_ROLE }}
21          aws-region: ${{ secrets.AWS_REGION }}
22
23      - name: "build image if missing"
24        run: |
25          if [[ "$(docker images -q al2-graalvm:gradle 2> /dev/null)" == "" ]]; then
26            echo "Image not found. Building image."
27            ./build-image.sh
28          else
29            echo "Image found. Skipping build."
30          fi          
31
32      - run: sam build --use-container
33      # Prevent prompts and failure when the stack is unchanged
34      - run: |
35          sam deploy \
36          --no-confirm-changeset \
37          --no-fail-on-empty-changeset \
38          --resolve-s3 \
39          --region ${{ secrets.AWS_REGION }} \
40          --capabilities CAPABILITY_IAM \
41          --stack-name every-note-persister          
...
yaml

In conclusion, building an AWS Lambda using Spring Boot 3 GraalVM native image can provide significant benefits, such as improved performance and reduced cold start times. By leveraging custom runtime, Spring AOT, SAM, and GitHub actions, developers can streamline the process of deploying and testing their serverless applications.

Using code examples, diagrams, and fine-grained configurations can help developers understand the details of building and deploying an AWS Lambda using Spring Boot 3 GraalVM native image. Additionally, referencing relevant resources and documentation can provide valuable insights and best practices for developers looking to build scalable and efficient serverless applications on AWS Lambda.

Overall, the combination of Spring Boot 3 GraalVM native image and AWS Lambda provides a powerful platform for developing and deploying serverless applications. By following the steps outlined in this blog post, developers can quickly and easily create high-performance and scalable serverless applications using modern technologies and tools.

Thank you for taking the time to read this blog post. I hope you found it informative and helpful in your journey of building AWS Lambdas with Spring Boot 3 GraalVM native image. Be sure to stay tuned for future posts, and I look forward to connecting with you again soon.