Securing your static web application in
data:image/s3,"s3://crabby-images/fba79/fba79149567d57c52b329b28b2a1606b5f01e1f1" alt=""
data:image/s3,"s3://crabby-images/a6c96/a6c966fca58095dff71af0f8eaac12a8eb7f0fcb" alt=""
Prehistory & motivation
Prehistory & motivation
- Colab was penetration tested (by SwiftSec)
Prehistory & motivation
- Colab was penetration tested (by SwiftSec)
- Numerous security vulnerabilities popped up (mainly in terms of HTTP response headers)
Prehistory & motivation
- Colab was penetration tested (by SwiftSec)
- Numerous security vulnerabilities popped up (mainly in terms of HTTP response headers)
- We had to fix them! 👨🏻🚒
Prerequisites
Prerequisites
- Your application is hosted on AWS S3.
- The static domain (S3) has a CloudFront distribution in front of it.
AWS 101
AWS 101
-
S3 (Simple Static Storage)
- Bucket
- Holds your static files (your app)
- Accessible via URL:
- <bucket-name>.s3.amazonaws.com
AWS 101
-
S3 (Simple Static Storage)
- Bucket
- Holds your static files (your app)
- Accessible via URL:
- <bucket-name>.s3.amazonaws.com
-
CloudFront (CDN)
- Stands in front of your S3 bucket domain
- Caches responses
AWS 101
-
S3 (Simple Static Storage)
- Bucket
- Holds your static files (your app)
- Accessible via URL:
- <bucket-name>.s3.amazonaws.com
-
CloudFront (CDN)
- Stands in front of your S3 bucket domain
- Caches responses
-
Lambda
- A serverless piece of code
- Can be executed from other AWS services' events.
$ curl -X HEAD -I https://domain.com
HTTP/2 200
content-type: text/html
content-length: 2116
date: Wed, 22 Apr 2020 13:20:37 GMT
last-modified: Tue, 07 Apr 2020 13:47:55 GMT
etag: "f84939d115325fbd2da8208c4d83cfd4"
server: AmazonS3
x-cache: Miss from cloudfront
x-amz-cf-pop: SOF50-C
x-amz-cf-id: dtiJbNp4pn_GGAWOamdwULC0Q
CloudFront edges
data:image/s3,"s3://crabby-images/d877e/d877eda1f78dcafd40a573da88bad86d2e73a589" alt=""
CloudFront edges
data:image/s3,"s3://crabby-images/e14f4/e14f49d78e2faeb714aecbee861eb577bb6776da" alt=""
CloudFront edges
data:image/s3,"s3://crabby-images/c6422/c64225584f581d640c943861d94b4967278f7fa3" alt=""
CloudFront edges
data:image/s3,"s3://crabby-images/14725/147253c994426cc97ca5c0aac00920958404fd42" alt=""
Lambda@Edge
data:image/s3,"s3://crabby-images/904ff/904ff59659b9e801597094d65b8623cd4d627e73" alt=""
Adding security headers via L@E
data:image/s3,"s3://crabby-images/7578c/7578c5eabced73a9b9c300d913802a2cf28b9b22" alt=""
Creating security headers L@E
- Important: Lambda@Edge is currently supported in only one AWS region: us-east-1 (North Virginia)
data:image/s3,"s3://crabby-images/9bf09/9bf096edb9139db3e948185f754a73780cd0d1e9" alt=""
Creating security headers L@E
data:image/s3,"s3://crabby-images/7d71f/7d71f3112b124b9788c731ed7f37121f3a10dd54" alt=""
Creating security headers L@E
data:image/s3,"s3://crabby-images/aa999/aa9999228a7a90a42b7d3f1418072012d800b2ff" alt=""
Implementing the Lambda
def lambda_handler(event, context): # ...
Handler signature
Implementing the Lambda
def lambda_handler(event, context): # ...
{
"Records": [
{
"cf": {
"request": {
"clientIp": "203.0.113.178",
"headers": {},
"method": "GET",
"querystring": "",
"uri": "/"
},
"response": {
"headers": {},
"status": "200",
"statusDescription": "OK"
}
}
}
]
}
Event structure
Implementing the Lambda
def lambda_handler(event, context): response = event['Records'][0]['cf']['response'] headers = response['headers'] return response
Fetching the response
Implementing the Lambda
def lambda_handler(event, context): response = event['Records'][0]['cf']['response'] headers = response['headers'] headers['x-frame-options'] = [ { 'key': 'X-Frame-Options', 'value': 'DENY' } ] # ... return response
Adding a security header
Deploying the Lambda 🚢
data:image/s3,"s3://crabby-images/a2ca4/a2ca4b85b194748a80565462ad0833ba3a770247" alt=""
data:image/s3,"s3://crabby-images/f74da/f74da3085e2bf89e1b9d20ac9ff58ebf1372b502" alt=""
Deploying the Lambda 🚢
data:image/s3,"s3://crabby-images/d877e/d877eda1f78dcafd40a573da88bad86d2e73a589" alt=""
Deploying the Lambda 🚢
$ curl -X HEAD -I https://domain.com
HTTP/2 200
content-type: text/html
content-length: 2116
date: Wed, 22 Apr 2020 13:20:37 GMT
last-modified: Tue, 07 Apr 2020 13:47:55 GMT
etag: "f84939d115325fbd2da8208c4d83cfd4"
server: AmazonS3
x-cache: Miss from cloudfront
x-amz-cf-pop: SOF50-C
x-amz-cf-id: dtiJbNp4pn_GGAWOamdwULC0Q
x-frame-options: DENY
$ curl -X HEAD -I https://domain.com
HTTP/2 200
content-type: text/html
content-length: 2116
date: Wed, 22 Apr 2020 13:20:37 GMT
last-modified: Tue, 07 Apr 2020 13:47:55 GMT
etag: "f84939d115325fbd2da8208c4d83cfd4"
server: AmazonS3
x-cache: Miss from cloudfront
x-amz-cf-pop: SOF50-C
x-amz-cf-id: dtiJbNp4pn_GGAWOamdwULC0Q
x-frame-options: DENY
data:image/s3,"s3://crabby-images/3e79b/3e79bd62ffe38c74613b3ea5022cd64f20b53508" alt=""
$ curl -X HEAD -I https://domain.com
HTTP/2 200
content-type: text/html
content-length: 2116
date: Wed, 22 Apr 2020 13:20:37 GMT
last-modified: Tue, 07 Apr 2020 13:47:55 GMT
etag: "f84939d115325fbd2da8208c4d83cfd4"
server: AmazonS3
x-cache: Miss from cloudfront
x-amz-cf-pop: SOF50-C
x-amz-cf-id: dtiJbNp4pn_GGAWOamdwULC0Q
x-frame-options: DENY
Resources 📚
🕵🏻♂️ Analyze your app for security vulnerabilities: https://observatory.mozilla.org/
Securing your static web application in
Securing your static application in AWS
By Ventsislav Tashev
Securing your static application in AWS
How to secure your static application in AWS S3 with Lambda@Edge
- 918