Introduction To Serverless Security: Part 3 - Preventing Accidental Deletion
Avoid falling victim to the pitfall of accidentally deleting your critical data when using Serverless. Learn how to enable AWS CloudFormation termination protection.
data:image/s3,"s3://crabby-images/264a2/264a2004ecd92661ad2dbbb53546e79df43b32b8" alt="Introduction To Serverless Security: Part 3 - Preventing Accidental Deletion"
"Did I just delete the database with all my customer data?!" you might say if you failed to enable the measure to prevent accidental deletion. We will explore how to avoid this pitfall in your Serverless environment.
Serverless Make It Really Easy to Deploy Your Environment—and Undeploy Too
The Serverless framework has made it extremely easy to deploy you functions, create databases, provision storage, and more with one deploy command. This is powerful and convenient, but the same goes for tearing down your deployment. (You may want to read the first article in this series, "Introduction To Serverless Security: Part 1 - Dependencies," to get a quick overview on serverless environments.)
Deploying Your Environment
This example deployment file shows how you can configure the resources you want in one configuration.
service: secjuice-example
provider:
name: aws
runtime: nodejs8.10
stage: ${opt:stage, 'dev'}
region: us-east-1
functions:
exampleFunction:
handler: functions/example.handler
events:
- http:
path: secjuice/example
method: get
resources:
Resources:
S3BucketFiles:
Type: AWS::S3::Bucket
Properties:
# must be globally unique across all AWS
BucketName: secjuice-example-files
CustomersTable:
Type: AWS::DynamoDB::Table
Properties:
TableName: secjuice-example-customers
AttributeDefinitions:
- AttributeName: AccountId
AttributeType: S
KeySchema:
- AttributeName: AccountId
KeyType: HASH
ProvisionedThroughput:
ReadCapacityUnits: 1
WriteCapacityUnits: 1
serverless.yml
configuration file.The serverless.yml
configuration creates the following upon deployment to Amazon Web Services (AWS):
- One function called
secjuice-example-dev-exampleFunction
using AWS Lambdas; it automatically appends the "service" and "stage" to the lambda function name
data:image/s3,"s3://crabby-images/213ce/213ceeb9b4fdc0db7c475f6567426f9d60f979c3" alt=""
- One file storage system called
secjuice-example-files
using AWS Simple Storage Service (S3)
data:image/s3,"s3://crabby-images/ce571/ce571b1e4cf3ae8dfaec338b00f304c954bb3a39" alt=""
- One database called
secjuice-example-customers
using AWS DynamoDB.
data:image/s3,"s3://crabby-images/f8f5b/f8f5b7e051a9ac2f9e0a5fddce9d58b0d8d8e3d6" alt=""
To start the deploy, you navigate to the project folder where the serverless.yml
file exists and run the following command:
sls deploy
Wow! Deploying the stack is really simple.
Undeploying (i.e. Removing) Your Enviroment
As simple as it was to deploy, the same applies to removing your environment. This is a double-edge sword. You may want to remove your environment quickly when developing for multiple reasons, but you might not want that same ease with your production environment (and important data).
To start the removal, you navigate to the project folder where the serverless.yml
file exists and run the following command:
sls remove
It is scary how easy it was to delete the entire stack.
Issuing this command deleted the following:
- The DynamoDB database
data:image/s3,"s3://crabby-images/edb32/edb32d33af5835939c76584d96b3b5bd4a131fa5" alt=""
- The S3 bucket
data:image/s3,"s3://crabby-images/1c947/1c94765666993682c25acfc49db9bf7d27065036" alt=""
- The Lambda function
data:image/s3,"s3://crabby-images/e7c11/e7c1145e0ab4abfe06150831818fc0dc9b6e6249" alt=""
How can you protect your data from an accidental (or maliciously intended) deletion?
Strategies to Protect Your Data From Accidental Deletion
Separating Function and Data Stacks
You can separate your functions and data into two stacks. I did accomplished this by creating two sub-folders "data" and "functions", each with its own `serverless.yml` configuration file.
service: secjuice-example-functions
provider:
name: aws
runtime: nodejs8.10
stage: ${opt:stage, 'dev'}
region: us-east-1
functions:
exampleFunction:
handler: example.handler
events:
- http:
path: secjuice/example
method: get
functions/serverless.yml
configuration file.service: secjuice-example-data
provider:
name: aws
runtime: nodejs8.10
stage: ${opt:stage, 'dev'}
region: us-east-1
resources:
Resources:
S3BucketFiles:
Type: AWS::S3::Bucket
Properties:
# must be globally unique across all AWS
BucketName: secjuice-example-files
CustomersTable:
Type: AWS::DynamoDB::Table
Properties:
TableName: secjuice-example-customers
AttributeDefinitions:
- AttributeName: AccountId
AttributeType: S
KeySchema:
- AttributeName: AccountId
KeyType: HASH
ProvisionedThroughput:
ReadCapacityUnits: 1
WriteCapacityUnits: 1
data/serverless.yml
configuration file.To start the deploy, you navigate to the project folder where the original serverless.yml
file existed and run the following commands:
cd functions
sls deploy
cd ../data
sls deploy
cd ..
These two deploys did the following:
- Created two CloudFormation stacks
data:image/s3,"s3://crabby-images/4abd1/4abd10a27c21939b42b9f5c5ac9d94b393489fc5" alt=""
- Created the Lambda function
data:image/s3,"s3://crabby-images/b2772/b277256b7c91e55951da17a2259aa08b2520eb07" alt=""
- Created the DynamoDB table
data:image/s3,"s3://crabby-images/f8f5b/f8f5b7e051a9ac2f9e0a5fddce9d58b0d8d8e3d6" alt=""
- Created the S3 buckets
data:image/s3,"s3://crabby-images/a03d8/a03d8dbf346816415a6b118dd94d2630ff654481" alt=""
Now lets remove on the functions stack.
cd functions
sls remove
cd ..
You will notice only the Lambda function and the S3 bucket associated with the functions stack is removed:
- Only the data CloudFormation stack remains.
data:image/s3,"s3://crabby-images/01fdc/01fdc3e2d471ed5bc35cfb4c25668742742bc144" alt=""
- Only the data stack S3 buckets remain.
data:image/s3,"s3://crabby-images/2b16d/2b16d5e2b778aee68ce3d9fe3d792ccf97144201" alt=""
- The DynamoDB table remains.
data:image/s3,"s3://crabby-images/f8f5b/f8f5b7e051a9ac2f9e0a5fddce9d58b0d8d8e3d6" alt=""
- There is no Lambda function.
data:image/s3,"s3://crabby-images/a1d53/a1d53233af6203d7940d904274f2dcb17d3e295a" alt=""
This approach allows a developer to work on the Lambda functions without worrying about the effect it has to the data.
Enabling Termination Protection
AWS CloudFormation has a nice feature to protect against accidental termination: it is called "termination protection." Termination protection is disabled by default. To enable it:
- Go to the stack
- Click "Stack actions"
- Click "Edit termination protection"
data:image/s3,"s3://crabby-images/7ece2/7ece247e87596d462094168754b83afbfe6e35ff" alt=""
- Click "Enabled"
- Click "Save"
data:image/s3,"s3://crabby-images/470d0/470d00277bae39120b500494f24605b56f2116a8" alt=""
Now when you try to remove the data stack you will get the following error:
Serverless: Getting all objects in S3 bucket...
Serverless: Removing objects in S3 bucket...
Serverless: Removing Stack...
Serverless Error ---------------------------------------
Stack [secjuice-example-data-dev] cannot be deleted while TerminationProtection is enabled
Enabling termination protection via the web console is very simple, but can be time consuming if you have a lot of stacks to manage. I recommend enabling it as part of the Serverless deploy.
At the time of this writing, the Serverless framework version 1.x has no support for termination protection. You need to use a Serverless plugin to add that capability. You can use one of two plugins:
The "serverless-stack-termination-protection" plugin
This plugin enables termination protection during deploy without any additional configuration.
To install it, run the following commands in your project folder:
cd data
npm install --save-dev serverless-stack-termination-protection
Open the serverless.yml
configuration file for the data stack and add the plugin.
service: secjuice-example-data
provider:
name: aws
runtime: nodejs8.10
stage: ${opt:stage, 'dev'}
region: us-east-1
resources:
Resources:
S3BucketFiles:
Type: AWS::S3::Bucket
Properties:
# must be globally unique across all AWS
BucketName: secjuice-example-files
CustomersTable:
Type: AWS::DynamoDB::Table
Properties:
TableName: secjuice-example-customers
AttributeDefinitions:
- AttributeName: AccountId
AttributeType: S
KeySchema:
- AttributeName: AccountId
KeyType: HASH
ProvisionedThroughput:
ReadCapacityUnits: 1
WriteCapacityUnits: 1
plugins:
- serverless-stack-termination-protection
data/serverless.yml
configuration file.You will see the following output when you deploy the data stack:
serverless-stack-termination-protection: Successfully enabled termination protection
Disclosure: I wrote the "serverless-stack-termination-protection" plugin.
The "serverless-termination-protection" plugin
This plugin also enables termination protection during deploy, but offers additional configuration options to deploy to specific stages. For example, you can specify to enable termination protection only for your "prod" stage/environment.
To install it, run the following commands in your project folder:
cd data
npm install --save-dev serverless-termination-protection aws-sdk
Open the serverless.yml
configuration file for the data stack and add the plugin.
service: secjuice-example-data
provider:
name: aws
runtime: nodejs8.10
stage: ${opt:stage, 'dev'}
region: us-east-1
resources:
Resources:
S3BucketFiles:
Type: AWS::S3::Bucket
Properties:
# must be globally unique across all AWS
BucketName: secjuice-example-files
CustomersTable:
Type: AWS::DynamoDB::Table
Properties:
TableName: secjuice-example-customers
AttributeDefinitions:
- AttributeName: AccountId
AttributeType: S
KeySchema:
- AttributeName: AccountId
KeyType: HASH
ProvisionedThroughput:
ReadCapacityUnits: 1
WriteCapacityUnits: 1
plugins:
- serverless-termination-protection
custom:
serverlessTerminationProtection:
stages:
- prod
data/serverless.yml
configuration file.You will see the following output when you deploy the data stack:
Serverless: STP: Adding termination protection to secjuice-example-data-dev..
Serverless: STP: Checking if termination protection should be added..
Serverless: STP: Stages to check: prod
Serverless: STP: Not applying termination protection for dev stage
With either plugin, your data stack is now protected, unless you specify to exclude a stage.
Conclusion
Consider separating your stacks and enabling termination protection on the data stack at a minimum to protect your application from accidental deletion.
Before You Go
Other Articles in this Series
- Introduction To Serverless Security: Part 1 - Dependencies
- Introduction To Serverless Security: Part 2 - Input Validation
Source
The source files are available at https://github.com/miguel-a-calles-mba/secjuice/tree/master/termination-protection-examples for your enjoyment.
A Note from the Author
Join my mailing list to get updates on my writings, my short stories, my upcoming books, and cybersecurity news. Visit https://miguelacallesmba.com/subscribe to join.
Stay secure, Miguel
data:image/s3,"s3://crabby-images/d0fa2/d0fa2b7e4fc619f6838cf5c7855af5121c8ff8e3" alt=""