Table of Contents
AWS services are the LEGO blocks of the IT world. Businesses can snap together building blocks like SQS queues, EC2 instances, and S3 buckets to rapidly create a service that delivers value.
You can treat AWS like a freestyle LEGO construction project. Just pull pieces from a big bin and put them together until you’re satisfied with the toy you’ve built:
Or you could be a little more rigorous, and follow the (468-page!) instruction manual to produce this 7,541-piece monster:
This is essentially the difference between the AWS Management Console and AWS CloudFormation.
In Amazon’s own words:
The Console provides an assortment of simplified and automated workflows and wizards that make it easier to test and build with AWS services.
AWS CloudFormation allows you to model your entire infrastructure and application resources with either a text file or programming languages.
The console is great for exploratory prototyping, but bad for reproducible infrastructure. Once you switch to defining your infrastructure in CloudFormation stacks, you’ll never look back. Here are a few best practices to help you on your way.
You can define your CloudFormation stacks in YAML or JSON. You should use YAML.
For starters, there’s a lot less syntactic cruft. Compare the same stack, first in JSON:
"Resources" : { | |
"S3Bucket" : { | |
"Type" : "AWS::S3::Bucket", | |
"DeletionPolicy": "Retain", | |
"Properties" : { | |
"BucketName" : "my-bucket" | |
} | |
} | |
} |
And now in YAML:
Resources: | |
S3Bucket: | |
Type: AWS::S3::Bucket | |
DeletionPolicy: Retain | |
Properties: | |
BucketName: my-bucket |
If that wasn’t enough, YAML also supports:
#
characterUserData
sectionsImagine you have a very complex CloudFormation stack that creates dozens of resources. Buried in your stack is this buggy code:
BucketPolicy: | |
Type: AWS::S3::BucketPolicy | |
Properties: | |
BucketName: my-bucket | |
PolicyDocument: | |
Statement: | |
- Sid: PermitPublic | |
Action: s3:getObject | |
Effect: Allow | |
Principal: '*' | |
Resource: arn:aws:s3:::my-bucket/public/* |
Spot the bug? Probably not, and so you submit your stack. After several minutes of CloudFormation churning away, it reaches the buggy part of the stack:
It then spends several minutes undoing the changes. Frustrating!
If you had instead used Amazon’s excellent cfn-python-lint tool to validate your stack, you would have been warned immediately:
Defining your infrastructure in a CloudFormation stack means it’s trivial to make copies of it. Imagine a world where every developer can easily and safely test their code changes, without putting production at risk.
This does require some forethought. After all, you can’t have two S3 buckets named the same thing, so stacks with hardcoded names are likely to be hard to duplicate.
To ensure multiple instances of a stack don’t collide with each other, you have a few options:
Parameters: | |
Suffix: | |
Type: String | |
Resources: | |
S3Bucket: | |
Type: AWS::S3::Bucket | |
Properties: | |
BucketName: !Sub my-bucket-${Suffix} |
${AWS::StackName}
and ${AWS::AccountId}
.Note that CloudFormation parameters can be strongly typed. Both simple types, such as String
or Number
, and AWS-specific types, such as AWS::EC2::AvailabilityZone::Name
, are supported. Choosing appropriate types for your parameters helps developers understand what sort of values they are expected to provide.
CloudFormation templates aren’t merely static text files. The last example used the Sub
intrinsic function to replace a placeholder value with a user-provided value. This programmability is what grants CloudFormation much of its power.
The whole list of intrinsic functions is worth reviewing, although we’ll discuss a few particularly powerful ones in more detail in this article.
CloudFormation stacks don’t have to be one size fits all. For example, production workloads may require powerful, and therefore expensive, resources. A developer kicking the tires likely won’t require the same amount of hardware. CloudFormation provides ways to conditionally create different kinds of resources.
The simplest example uses a condition intrinsic function to declare a Condition
that controls whether a resource is created or not.
Imagine a stack that creates a CloudFront distribution. For developer environments, it’s OK to use the auto-generated ddcfg6dsj43sm.cloudfront.net
domain name. For production environments, you’d like to do a bit more work and assign a user-friendly DNS entry like example.com
.
Line 6 declares the IsProd
condition, which tests whether the Env
parameter is set to a certain value using the Equals
intrinsic function. Line 16 makes the creation of the CNAME record happen only when IsProd
is true:
Parameters: | |
Env: | |
Type: String | |
Conditions: | |
IsProd: !Equals [ !Ref Env, "prod" ] | |
Resources: | |
CloudFrontDistribution: | |
Type: AWS::CloudFront::Distribution | |
Properties: | |
... | |
CNAMERecord: | |
Type: AWS::Route53::RecordSet | |
Condition: IsProd | |
Properties: | |
HostedZoneId: Z2F3H7JOBSMNZW | |
Name: example.com | |
ResourceRecords: | |
- !GetAtt CloudFrontDistribution.DomainName | |
TTL: 60 | |
Type: CNAME |
Sometimes you need to create the resources in both cases, but with different values for its attributes. The stack below creates a small EC2 instance for developers, but a large one for production, using the If
intrinsic function:
Parameters: | |
Env: | |
Type: String | |
Conditions: | |
IsProd: !Equals [ !Ref Env, "prod" ] | |
Resources: | |
Instance: | |
Type: AWS::EC2::Instance | |
Properties: | |
ImageId: ami-0d4147bb461c62a2c | |
InstanceType: !If [ IsProd, "m6g.2xlarge", "m6g.medium" ] |
Finally, sometimes you need to include an attribute in some cases, and exclude it altogether in others. For this, there is the special AWS::NoValue
reference. Think of it like null
in a programming language – it can take the place of a primitive value like a string (as in line 6), or of a composite record-type value (as in line 13).
Resources: | |
CloudFrontDistribution: | |
Type: AWS::CloudFront::Distribution | |
Properties: | |
DistributionConfig: | |
Aliases: !If [ IsProd, "example.com", !Ref "AWS::NoValue" ] | |
ViewerCertificate: | |
!If | |
- IsProd | |
- AcmCertificateArn: arn:aws:acm:us-east-1:1234567890:certificate/1141ddff-638c-451e-a4dd-3f2708abbb3d | |
MinimumProtocolVersion: TLSv1 | |
SslSupportMethod: sni-only | |
- !Ref "AWS::NoValue" | |
Origins: | |
- DomainName: backend.example.com | |
Id: default | |
CustomOriginConfig: | |
OriginProtocolPolicy: https-only | |
OriginPath: / | |
Enabled: true | |
DefaultCacheBehavior: | |
... |
Before creating your resources, CloudFormation firsts analyzes them to make a directed, acyclic graph of dependencies. This is how CloudFormation decides which resources to create first, and which can be created in parallel.
For example, a stack that defines a new VPC with a security group, subnet, database instance, and EC2 instance will have this graph:
There are some scenarios where you’ll want to fine tune CloudFormation’s default dependency analysis.
The DependsOn
attribute tells CloudFormation to wait until the depended-on resource has been created before creating the depending-on resource. Imagine that we wanted the database instance to be fully created before launching the EC2 instance. We could achieve that with this stack:
DBInstance: | |
Type: AWS::RDS::DBInstance | |
... | |
EC2Instance: | |
Type: AWS::EC2::Instance | |
DependsOn: DBInstance | |
... |
That would cause the new graph of dependencies to look like this:
By default, an EC2 instance is considered created when it is launched. This is fast, but not always useful. For example, what if you need to install some software and start some services?
The CreationPolicy
attribute provides a solution. It prevents an EC2 instance from being automatically marked as created. Instead, to mark it as created, and thus permit the stack to continue creating dependent resources, you need to signal CloudFormation. The easiest way to do this is with the cfn-signal
script. A common approach is to provide a UserData
script that does your initialization steps and then finishes by calling the cfn-signal
script.
The CreationPolicy
attribute only applies to EC2 instance and auto-scaling groups. Sometimes you’ll need to block progress of a CloudFormation stack pending some external action. In those cases, you can create an AWS::CloudFormation::WaitCondition
resource. The WaitCondition
can also be signalled with cfn-signal
, and other resources can use DependsOn
to wait until it has been signalled.
Once you’ve created your stack, you often need to translate from a logical resource ID like S3Bucket
, which identifies the resource in your stack, to a physical resource ID like stack-s3bucket-13qb2k9okcsc1
, which identifies the actual AWS entity.
There are a few ways of doing this, depending on the context.
One option is to use the aws
command-line tool to interrogate your stack after it has been created. The aws
tool supports the JMESPath query language, which lets you extract the particular values of interest to you:
$ aws cloudformation describe-stack-resources \
--stack-name your-stack \
--logical-resource-id S3Bucket \
--query StackResources[0].PhysicalResourceId \
--output text
stack-s3bucket-13qb2k9okcsc1
You could then pass this value to another stack as a parameter value.
An alternative approach uses the special ImportValue
intrinsic function. ImportValue
retrieves a previously exported Output
value from an earlier stack for use in a subsequent stack.
Imagine that we create a stack which creates an S3 bucket:
Resources: | |
S3Bucket: | |
Type: AWS::S3::Bucket | |
Outputs: | |
BucketName: | |
Value: !Ref S3Bucket | |
Export: | |
Name: TheS3Bucket |
In a second stack, we’d like to create an EC2 instance that knows which S3 bucket it should use. We can configure a bash script that runs on instance start up and writes the bucket name to a file for future access:
Resources: | |
Instance: | |
Type: AWS::EC2::Instance | |
Properties: | |
ImageId: ami-0d4147bb461c62a2c | |
InstanceType: m6g.medium | |
UserData: | |
Fn::Base64: | |
!Sub | |
- | | |
#!/bin/bash | |
echo ${BucketName} > /tmp/bucket.txt | |
- BucketName: !ImportValue TheS3Bucket |
Note that the exported name of an output value must be unique across all of your account’s stacks in a given region.
CloudFormation is a powerful, flexible tool. While there are many other features we haven’t touched on – macros and YAML anchors and aliases, for example – we’ve highlighted the most useful bits in the list of best practices above. Whether you’re starting out or an experienced user, hopefully you have found something useful.