Deploying a Single Page Application (SPA) on AWS: A Beginner's Guide. Part 1. Theory
Understanding the Basics
Preface
In the realm of IT solutions, the age-old question of where to deploy your containers invariably leads to the ubiquitous answer, "It depends." The choice hinges on various factors such as the scale, skill set, time constraints, and budget considerations.
This article is dedicated to the specific scenario of migrating on-premises single-page applications (SPAs) to the cloud. One notable feature of this series is its commitment to technical neutrality. You have the freedom to construct your Frontend static assets with any framework of your preference or even opt for raw HTML. Likewise, you can employ any programming language to craft your Backend, provided it exposes business logic through a REST API.
Prerequisites
To embark on this tutorial, ensure you have the following essentials at your disposal:
AWS CLI - AWS Command Line Interface
Frontend - static web assets for your SPA - encompassing index.html, js, css, and other relevant files
Backend - a REST API server, encapsulated within a container
That's all you need to get started.
Source code repository
javatask/aws-spa-beginners-guide: Code for the blog series on porting single page application to AWS (github.com)
Cloud Applications Should Be FROSST by Gregor Hohpe
In his book Gregor Hohpe Cloud Strategy: A Decision-based Approach to Successful Cloud Migration described the principle of making your application "cloud ready. The goal of this series is to follow it:
FROSST means Cloud Ready: Frugal, Relocatable, Observable, Seamlessly updatable, internally Secured, and failure Tolerant.
Let's deep dive into these principles:
•Frugal - refers to the conservative and efficient use of CPU and memory resources, avoiding unnecessary wastefulness, while also considering aspects like deployment footprint and start-up time, without resorting to premature optimization or employing unnecessarily complex or cumbersome solutions.
•Relocatable - refers to the ability to move an application from one data center or server to another without requiring a complex migration process, allowing for scaling, quick redeployment in case of failure, updates, or use in discounted cloud instances, facilitated by methods such as externalizing state and avoiding reliance on single physical instances.
•Observable - refers to the ability to analyze and understand the state and performance of the system without needing to plug in a debugger, through means such as exposing endpoints that report readiness and liveness, monitoring SLA compliance, and utilizing logs and tools like distributed tracing, thereby enabling scenarios like relocatability, failure prediction, and troubleshooting in distributed systems.
•Seamlessly updatable - the ability to deploy updates without downtime or disruption, often during regular business hours, through the use of practices like backward compatibility, feature toggles, simultaneous existence in different versions, and aided by characteristics like frugality, relocatability, observability, and external tools and automation such as green/blue deploys.
•internally Secured - refers to the practice of incorporating security measures within the application itself, rather than solely relying on external perimeter security, adopting a "trust no one" approach that may include authenticating API gateways and end-users, testing against common security risks, and implementing embedded authorization or authentication mechanisms, to ensure robust protection against unauthorized access and vulnerabilities, irrespective of the environment in which the application is deployed.
•failure Tolerant - refers to the ability to continue functioning in a degraded mode without complete failure, even when components or dependent services encounter problems, by accepting and managing failures internally without propagating them to the end user, and having mechanisms to recover automatically, a characteristic supported by frugality, relocatability, and observability, and requiring a conscious design decision.
Use Case
The SPA embodies a classic three-layered architecture (Figure 1). For practical application, we will employ a combination of HTML files for Frontend and opt for the Spring Boot Tutorial Rest API app as our Backend. You can access all the code increments in my Git repository - javatask/aws-spa-beginners-guide: Code for the blog series on porting single page application to AWS (github.com).
Figure 1. Single-page application (Link to Wikipedia)
The application migration process unfolds in the following steps:
Host your static assets using CloudFront and S3 services. The primary objective at this juncture is to have your Frontend up and running on a publicly accessible https URL.
Proceed to obtain an independent URL for your Backend and perform testing.
Subsequently, link your Backend URL to the
/api
path on the primary Frontend URL to facilitate AJAX calls.The final step involves the addition of Sign-In functionality and the fortification of your Backend URL with the JWT Authorization header. This ensures that the Backend API rejects all requests devoid of a valid JWT.
Refer to Figure 2 and Figure 3 for a comprehensive depiction of the entire architecture and the interplay of its components.
Figure 2. Generic Architecture for Deploying Our SPA on AWS
Figure 3. Sequence Diagram Illustrating Interactions Among the Moving Parts
Description of the Sequence diagram:
The user initiates a webpage request, which CloudFront fulfills by fetching static files from the S3 bucket.
When the user generates an API request, CloudFront forwards it to the API Gateway.
The API Gateway, in turn, validates the request's authenticity with Cognito.
If the authentication process proves successful, the Backend proceeds to handle the request, potentially fetching or storing data from the Database. The processed data is subsequently relayed to the user via the API Gateway and CloudFront.
In the event of authentication failure, the user receives an error message.
In Summation
This section delineates various approaches for migrating a Single Page Application to the Cloud. It also outlines the prerequisites necessary to embark on the tutorial and delves into the use case, along with five possible architectural variations for hosting your Backend.
Stay tuned for the next instalment - "Securing Static Hosting."