/latency

Uses Terraform to explore latency / cost tradeoffs with minimalist examples using various AWS architectures including EC2, Lambda functions, S3, DynamoDB and Fargate.

Primary LanguageHCL

Latency

The purpose of this repo is to compare the performance of various architectures for a simple web application with a backend that keeps track of the number of times it has been called. It should be fairly straightforward to adapt to your specific needs.

UPDATE: New section on queues.

Each architecture can be built by running

terraform apply

in the root directory. The application URL will be output once the process is complete. To switch between the various architectures, change the line

  source   = "./1-fastapi-ec2"

in main.tf to point to the corresponding directory.

  • FastAPI + EC2 (1-fastapi-ec2). "Serverful" architecture. Spins up an EC2 instance and installs a Python FastAPI server which serves the web page, provides an end-point to increment & return an in-memory counter. The server is exposed on port 8000.

  • Lambda Python + S3 (2-lambda-python-s3). Serverless architecture. API Gateway serves the web page from S3 bucket and provides route to a Python Lambda function which stores the counter as an object in a S3 bucket. Note that AWS reuses Lambda functions as much as possible, meaning that moving as much initialization code outside of the handler function itself improves the latency significantly.

  • Lambda Python + DynamoDB (3-lambda-python-dynamodb). Serverless architecture. API Gateway serves the web page from S3 bucket and provides route to a Python Lambda function which stores the counter as an item in a DynamoDB table.

  • Lambda JavaScript + DynamoDB (4-lambda-js-dynamodb). Same as above, but with a JavaScript Lambda function.

  • Lambda C++ + DynamoDB (5-lambda-c++-dynamodb). Same as above, but with a C++ Lambda function. To build the Lambda function yourself, adapt these instructions https://github.com/awslabs/aws-lambda-cpp/tree/master/examples/dynamodb.

  • Lambda Rust + DynamoDB (6-lambda-rust-dynamodb). Same as above, but with a Rust Lambda function. I included this example because I am a big fan of Rust (at least in theory). The compiler is very unforgiving but, if you do manage to make it build, you know it will be thread-safe and robust. The code is much easier to read than it is to write, and including modules ("crates") is very straightforward. And, of course, it is blisteringly fast. To build the Lambda function yourself, adapt these instructions https://github.com/awslabs/aws-lambda-rust-runtime.

  • Fargate ECS (7-fargate-ecs). Serverless architecture. Creates a Fargate ECS service and Load Balancer on the default VPC (for simplicity) that runs the FastAPI server as a task inside a container. One advantage of using Fargate is that it can be configured to use spot instances which are up to 70% cheaper than on-demand instances. As spot instances can be terminated at any time (although, in practice, this is only about 5% of the time), the app persists its state on Elastic File Storage (EFS). You can build the Docker image yourself by running docker build . in the root directory of the repository.

  • Kubernetes (8-k8s). The main alternatives for creating a Kubernetes cluster on AWS are kOps and AWS EKS. An advantage of using Kubernetes is that you can build a complex application from individually scalable and re-deployable microservices that is cloud vendor agnostic. With kOps it is a lot easier to create a cluster than with EKS using Terraform. It is also cheaper (the control node costs a minimum of $12.06 a month, whereas the EKS cluster comes in at $73 not including the worker nodes). It allows finer grained control, but it is arguably harder to configure if you are used to using the AWS console. To build the cluster and install the application, run ./8-8ks/apply.sh <your Route53 managed domain> in a bash shell - be patient, the whole process can take up to 20 minutes. (To tear it down again, run ./8-8ks/destroy.sh <your Route53 managed domain>). You will need to have installed kOps, kubectl and bash.

To test the performance over the internet run

python test\latency.py <URL>/hits 1000

providing the application URL and the number of times to increment the counter in succession. Alternatively, you can invoke the test from a Lambda function. Build the Lambda function by running terraform apply in the test directory and launch the test with

python test\lambda-latency.py <URL>/hits 1000

Results

The cost estimates are based on current AWS pricing for the region eu-west-2 and don't take into account network usage. I have included EC2, Lambda, S3, DynamoDB, EFS and API Gateway costs.

Architecture Average latency (ms) Monthly cost ($) Cost per million calls ($)
FastAPI + EC2 13 10.57 N/A
Lambda Python + S3 120 N/A 7.16
Lambda Python + DynamoDB 60 N/A 3.24
Lambda JavaScript + DynamoDB 105 N/A 3.39
Lambda C++ + DynamoDB 23 N/A 3.09
Lambda Rust + DynamoDB 26 N/A 3.09
Fargate Spot ECS 16 3.11* N/A
Kubernetes 13 22.63* N/A

* I haven't included the Elastic Load Balancer here as this is something you would probably want anyway. Even for a minimalist example, it is not possible to set up and access a Fargate cluster without one. kOps also creates one by default. It will set you back at least $19.32 a month.

Queues

Now let's suppose we have a function calc that takes a parameter x and takes x seconds to return x. We might want to use a queue to run many calculations in parallel.

  • Lambda Python + SQS (9-lambda-python-sqs). Serverless architecture. API Gateway posts a message to an SQS queue which triggers a Python Lambda function that runs the calculation and then increments a counter in a DynamoDB table. To test it, python 9-lambda-sqs/sqs.py 1000 2 will run 1000 "calculations" that take a random time to complete between 0 and 2 seconds.

  • EKS + RabbitMQ (10-eks-rabbitmq). This time, we use AWS EKS to create a cluster and deploy RabbitMQ, Prometheus (for custom metrics), a Horizontal Pod Autoscaler (HPA) that spawns our calc Deployment according to the length of the RabbitMQ queue and Karpenter, which scales up nodes in the cluster. To test it, python 10-eks-rabbitmq/rabbitmq.py 10000 2 will run 10000 "calculations" that take a random time to complete between 0 and 2 seconds. Because the Kubernetes provider depends on the EKS cluster, you have to terragrunt run-all apply. You may need to run it a second time in order to get the URL of the RabbitMQ queue as the Load Balancer creates this asynchronously. Please also be sure to delete the PVC Volumes* in the AWS console once you have finished with them. [If the rabbitmq service gets stuck deleting, run kubectl patch svc rabbitmq -p '{"metadata":{"finalizers":[]}}' --type=merge.]

Of course, a real use case is needed to compare the two architectures, and one may be better for some cases than others. Nevertheless, I found that the SQS solution was much easier to deploy although it was unsurprisingly far less configurable than the RabbitMQ approach. It was also less performant. Most of the compexity of setting up the Kubernetes deployment was concerned with Prometheus and this would typically already be configured in an existing cluster.

* You can delete all the PVC Volumes with the following command:

aws ec2 describe-volumes --query "Volumes[?starts_with(Tags[?Key=='Name'].Value | [0], 'latency-dynamic-pvc-')].VolumeId" --output text | tr '\t' '\n' | xargs -I {} aws ec2 delete-volume --volume-id {}