Skip to content

A sample implementation of an evolutionary architecture for a serverless application using safe deployments, automatically computing the fitness function at deployment time, with the possibility to rollback back if fitness is not improving.

License

danilop/evolutionary-serverless-architectures-with-safe-deployments

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

21 Commits
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Evolutionary Serverless Architectures with Safe Deployments

A sample implementation of an evolutionary architecture for a serverless application using safe deployments, automatically computing the fitness function at deployment time, with the possibility to rollback if fitness is not improving.

Made with ❤️ by Danilo Poccia.

To build this implementation, I started from the sample code in the AWS SAM repository: https://github.com/awslabs/serverless-application-model/tree/master/examples/2016-10-31/lambda_safe_deployments

I updated the Node.js runtime to version 8.10, so that I could make use of the new async/await syntax.

The AWS SAM template.yaml creates:

To test the deployment, you can use the SAM CLI and the following build/package/deploy commands two times:

  • the first time to create the CloudFormation stack for the application
  • the second time to update the stack, see how safe deployments work and follow them in the CodeDeploy console, and how the fitness of the architecture is measured by the PreTraffic function and updated in a custom CloudWatch metric
sam build

sam package --s3-bucket <YOUR_BUCKET> \
            --output-template-file packaged.yaml

sam deploy --template-file packaged.yaml \
           --stack-name evolutionary-deployment \
           --capabilities CAPABILITY_IAM

You can follow the first implementation of the stack, and the next updates, from the CloudFormation console. The previous commands use the default region set for the AWS CLI.

For the two Lambda functions providing an API, different deployment strategies are implemented:

  • myFirstFunction is using a Linear deployment adding 10% of the invocations to the new version every minute (Linear10PercentEvery1Minute), taking 10 minutes to complete
  • mySecondFunction is using a Canary deployment with 10% of the invocations to the new version for 5 minutes, and then a rollout to 100% (Canary10Percent5Minutes)

The preTrafficHook function is running some tests to check if the deployment must Succeed or Fail and at the same time is computing the value of the fitness function for this deployment:

  • some of the tests are actually atomic fitness functions themselves, testing a single reource in the CloudFormation stack
  • other tests can act as holistic functions, testing that multiple resources (such as functions and databases) are working together in the expected way

To simplify and reuse atomic tests on single resources, the SAM template is passing the StackId to the preTrafficHook function as an environment variable.

Using the StackId, the function is getting the list of the resources in the stack, on which it can iterate with a switch that can apply specific tests depending on the resource type.

Most of the tests involve invocations to AWS services, so to make it more efficient are reduce the overall duration of this function:

  • all tests are implemented as async functions (so that are automatically wrapped as promises)
  • all tests are added to a list (array) that is then executed using Promises.all()

For example, some of the tests that can be implemented on non-functional requirements, such as security and scalability, are:

  • check that encryption at rest is enabled on all S3 buckets
  • check that encryption at rest is enabled on all DynamoDB tables
  • check that public write and/or read is prohibited for all S3 buckets
  • check that S3 buckets accept HTTPS requests only
  • check that auto scaling is enabled for all DynamoDB tables

Those checks contribute to the measurement of the fitness function, so that if you change you architecture (and possibly your application) to be more secure or scalable, you automatically increase the resulting fitness.

Instead of implementing all tests, you can leverage the existing AWS Config managed rules, such as:

  • s3-bucket-logging-enabled
  • s3-bucket-replication-enabled
  • s3-bucket-versioning-enabled
  • s3-bucket-public-write-prohibited
  • s3-bucket-public-read-prohibited
  • s3-bucket-ssl-requests-only
  • s3-bucket-server-side-encryption-enabled
  • dynamodb-autoscaling-enabled
  • dynamodb-throughput-limit-check
  • lambda-function-public-access-prohibited
  • lambda-function-settings-check

A full lists of AWS Config managed rules is available here. To check the compliance to one or more of those rules, I am using the AWS Config getComplianceDetailsByResource API.

About

A sample implementation of an evolutionary architecture for a serverless application using safe deployments, automatically computing the fitness function at deployment time, with the possibility to rollback back if fitness is not improving.

Topics

Resources

License

Stars

Watchers

Forks

Packages

No packages published