S3 Version without SSL Certificate #
This tutorial is just a quick solution for testing purposes. It deploys a static websites consisting of html, css and js files. It provides no SSL/TSL encryption and the content of the S3 Bucket is public.
data:image/s3,"s3://crabby-images/251b5/251b59a0a88085c60c8b459d2366d472f6ee0fa0" alt=""
Disable and acknowledge the “Block all public access” option:
data:image/s3,"s3://crabby-images/a3d4e/a3d4ed197ab45b9b56e4fb92315fb49ab1cd4233" alt=""
Go to the permissions section of your S3 Bucket and past the following configuration,
replace s3.jklug.work
with the name of your Bucket:
{
"Version": "2012-10-17",
"Statement": [
{
"Sid": "PublicReadGetObject",
"Effect": "Allow",
"Principal": "*",
"Action": [
"s3:GetObject"
],
"Resource": [
"arn:aws:s3:::s3.jklug.work/*"
]
}
]
}
data:image/s3,"s3://crabby-images/7af58/7af584a9764515e481b07a2ca731842245ef0259" alt=""
Go to Objects Section of your S3 Bucket and upload an index.html file like this one:
<!DOCTYPE html>
<html>
<head>
<title>jklug.work</title>
</head>
<body>
<h1>S3 Static Website</h1>
</body>
</html>
Open the permissions section of your S3 Bucket, enable static website hosting and define an index document, in the example it’s “index.html”:
data:image/s3,"s3://crabby-images/4c6fe/4c6fe977739ecfaecd80c0bc4e198487b17ab70e" alt=""
data:image/s3,"s3://crabby-images/b2984/b298454d1889e1ea14c004a5f1c865d2b0251024" alt=""
data:image/s3,"s3://crabby-images/12327/123271060feaf6cfa0248e39d9016fd1c85c34c3" alt=""
Save the changes, copy the bucket website endpoint and open it in a browser. Note that the website is not encrypted:
data:image/s3,"s3://crabby-images/fdc82/fdc829cedff130d8077261aee54dde73415f6c27" alt=""
data:image/s3,"s3://crabby-images/5457c/5457cf3cd0b67dd52805f1a4d4ebb2e54f6dcdc5" alt=""
Cloudfront Version with SSL/TSL Certificate #
This tutorial uses a S3 Bucket to host a static website but it’s content is only accessable through a Cloudfront Distribution. The Cloudfront Distribution also provides the necessary encryption the provide secure data transfair.
Create S3 Bucket #
Create a new S3 Bucket with standard settings. The “Block all public access” option should be enabled by default:
data:image/s3,"s3://crabby-images/6e3f7/6e3f7b0dcf918bf96373301c3fa4302ae88dc865" alt=""
Go to Objects Section of your S3 Bucket and upload an index.html file like this one:
<!DOCTYPE html>
<html>
<head>
<title>jklug.work</title>
</head>
<body>
<h1>S3 Cloudfront</h1>
</body>
</html>
Create Cloudfront Distribution #
Select your S3 Bucket, enable the “Origin access control settings” and click the “Create control setting” buttion:
data:image/s3,"s3://crabby-images/a7e4a/a7e4a3f4aaf0b1c777818028f9a63e8706701b70" alt=""
Create a control setting and make sure it’s selected in the main view:
data:image/s3,"s3://crabby-images/7c1cb/7c1cbbbfe2d50d074122f96f873580416e81035e" alt=""
data:image/s3,"s3://crabby-images/fb6d8/fb6d86f5a09ef00b30a4aa6be60ef660fbaf77be" alt=""
Enable the “Redirect HTTP to HTTPS” option:
data:image/s3,"s3://crabby-images/193b8/193b8ff1872a19079256279f81c40a0b9bfcafa7" alt=""
Select a price class / region, select your SSL certificate - note: for CloudFront the SSL Certificate must be in the us-east-1 region - and define a default root object, in the example the root object is “index.html”:
data:image/s3,"s3://crabby-images/00d15/00d15f9569e4b311ccdc7ed98789ca3d1c1637ea" alt=""
Create the CloudFront Distribution, open the objects section of your distribution, select the origin name and click “edit”:
data:image/s3,"s3://crabby-images/87987/8798724fb0887f0c0045c23539ca8326055740c4" alt=""
Copy the origin access control bucket policy - it should look like in the example below - and past it as permission to your bucket policy:
data:image/s3,"s3://crabby-images/98fb1/98fb1dc79567d17aa4bfff3a9bd009b6f9ed668a" alt=""
{
"Version": "2008-10-17",
"Id": "PolicyForCloudFrontPrivateContent",
"Statement": [
{
"Sid": "AllowCloudFrontServicePrincipal",
"Effect": "Allow",
"Principal": {
"Service": "cloudfront.amazonaws.com"
},
"Action": "s3:GetObject",
"Resource": "arn:aws:s3:::s3-cloudfront.jklug.work/*",
"Condition": {
"StringEquals": {
"AWS:SourceArn": "arn:aws:cloudfront::073526172187:distribution/E26589F32XA4RO"
}
}
}
]
}
Add Access Policy to S3 Bucket #
Open the permissions section of your S3 Bucket and paste the policy:
data:image/s3,"s3://crabby-images/56903/56903787c9e35bd081fab19955e5999b83af7fa5" alt=""
Route 53 DNS entry #
Open Route 53 and create an DNS entry to routes the traffic to your CloudFront Distribution:
data:image/s3,"s3://crabby-images/87060/87060db114b04650d1910eba893bbc906912a56a" alt=""
data:image/s3,"s3://crabby-images/10742/107423852e98fa7fbaf4d4de47d0b07ee981bada" alt=""
data:image/s3,"s3://crabby-images/dc23a/dc23a31b739eaa15dc4dc460a31f447a2d80c1f8" alt=""
data:image/s3,"s3://crabby-images/7200b/7200bf191c90b9e2b6a771752db179926b98e7ac" alt=""
Open your URL in a Browser, note that it’s secure now:
Cloudfront: Geographic Restrictions #
Option it can be useful to allow traffic to your website only from specified countries. Open the Geographic Restrictions Panel of your CloudFront Distribution:
data:image/s3,"s3://crabby-images/761ec/761ec6d49732aa734b057fa2915950f51623a501" alt=""
Choose allow list and select the desired countries:
data:image/s3,"s3://crabby-images/41e9f/41e9f7c37292d28633cd3785258e4fdbf6b7b580" alt=""
Test the Whitelist Settings with Geo Browse:
https://geotargetly.com/geo-brow
data:image/s3,"s3://crabby-images/2210e/2210e0bd4d15659a2c7337cc789e37360d2e68bc" alt=""
data:image/s3,"s3://crabby-images/33c4c/33c4cc1cf3b25a8fb22c289d78e3b1c8516eaff0" alt=""
Cloudfront Function: Redirect to index.html #
For some web applications it’s necessary to create a function that adds index.html to the end of every URL. This is common with static site generators. Open the CloudFront / Function Panel and create a new function:
data:image/s3,"s3://crabby-images/cbbf9/cbbf933d0ec9f5510a16f0f8b792e9490a1222cf" alt=""
Copy and paste the following function and save the changes:
function handler(event) {
var request = event.request;
var uri = request.uri;
// Check whether the URI is missing a file name.
if (uri.endsWith('/')) {
request.uri += 'index.html';
}
// Check whether the URI is missing a file extension.
else if (!uri.includes('.')) {
request.uri += '/index.html';
}
return request;
}
data:image/s3,"s3://crabby-images/d48b8/d48b8247d57ae0856c53050c1e954526852de6c3" alt=""
Go to the Publish section of your functiona and publish it:
data:image/s3,"s3://crabby-images/ab675/ab67584567098fe863cbedb744869bdcc5a3e194" alt=""
After the function is published add an association / your CloudFront Distribution:
data:image/s3,"s3://crabby-images/55e26/55e26fc2d7832f73bbd9dfb9accef9036bda7d34" alt=""
data:image/s3,"s3://crabby-images/f8c46/f8c461e837a5be231c9d5008dc038fc1f6f4ca0f" alt=""
CloudFront Cache Invalidation #
The remove all files from the cache, for example after the websites was updated, proceed as follows:
- Select the Distribution
- Open the “Invalidations” Tab
- Click “Create invalidation”
- Add object paths:
/*
This invalidates all files in the distribution.