Who hasn\u2019t had problems with\u00a0AWS CloudFormation?<\/strong><\/p>\n
Sadly, it is commonly known that AWS CloudFormation is as powerful of a tool as it is frustrating; this short article will highlight the most common pitfalls and the most annoying situations, as well as, obviously,\u00a0ways to avoid them.<\/strong><\/p>\n
We begin our journey by\u00a0describing the service:<\/strong><\/p>\n
\u201cAWS CloudFormation provides you with a common language for describing and provisioning all infrastructure resources in your cloud environment. With CloudFormation you can use a simple text file to model and implement provisioning, in an automated and secure manner, of all resources needed for your applications on all regions and all accounts. Such file will be the only source of your cloud environment.<\/p><\/blockquote>\n
CloudFormation is available at no additional cost; you are only charged the cost of AWS resources needed to run the applications.\u201d<\/p><\/blockquote>\n
One of the main advantages of using AWS CloudFormation is precisely to have\u00a0a template of your own infrastructure, which can be versioned, revised, and easily replicated<\/strong>\u00a0as often as necessary.<\/p>\n
There are problems, however, right from the description of the service.<\/p>\n<\/div>\n<\/div>\n<\/section>\n
\n <\/div>\n\n\n#1 An (in)complete modeling<\/h3>\n
Although the AWS service presentation page explicitly states that AWS CloudFormation offers complete cloud infrastructure modeling, this, in fact, does not happen.<\/p>\n
At least, not always; in fact,\u00a0there are many configurations and\/or resources that cannot be specified in an AWS CloudFormation template.<\/strong><\/p>\n
To date, for example, it is not possible to:<\/p>\n
\n
- Define the settings related to Cognito UserPool for federation with an external IDP<\/li>\n
- Add a route to a routing table that points to a Transit Gateway<\/li>\n
- Implement SSH key provisioning for EC2 (probably for security reasons)<\/li>\n<\/ul>\n
Our advice is<\/strong>\u00a0to make sure that all the necessary services and configurations can be truly modeled with the current version.<\/p>\n
Almost all services are supported, although for some it is necessary to wait a considerable amount of time between the release and official support of AWS CloudFormation.<\/p>\n
There is also a way to\u00a0extend the capabilities of AWS CloudFormation<\/strong>using custom resources. In fact, they allow the deployment of a Lambda function and execute it as a \u201cCustom Resource\u201d at a given time during the creation of the stack.<\/p>\n
Here is the official documentation of this feature.<\/a><\/p>\n<\/div>\n<\/div>\n<\/section>\n
\n <\/div>\n\n\n#2 Waiting time<\/h3>\n
If it is true that once the template is packaged, the execution is faster than the manual one, the template production process is a long process based on trial and error.<\/p>\n
It is not possible to 100% check a template locally,<\/strong>\u00a0it is only possible to be certain of syntactic correctness without first executing it.<\/p>\n
Which means that prior to arriving at a working version, you have to go through numerous attempts, each having long execution times.<\/p>\n
To make life easier we can count on the following features:<\/strong><\/p>\n
The first is the\u00a0ValidateTemplate<\/a>\u00a0API which does a much more thorough check than just a linter.<\/p>\n
Another way is the extensive use of\u00a0changesets<\/a>, which allow you to have a preview of the actions that will result in changes to a template.<\/p>\n
\n <\/div>\n\n\n#3 Unintelligible errors<\/h3>\n
Error messages are often meaningless, misleading, useless, or simply wrong.<\/p>\n
One can only be certain that the stack has failed.\u00a0What has actually failed and why is as useful to know as it is difficult to find out.<\/strong><\/p>\n
The only way to quickly find the reason for failure is to\u00a0analyze the execution log<\/strong>\u00a0to see exactly on which resource an error is found, and when the provisioning for that resource occurred.<\/p>\n<\/div>\n<\/div>\n<\/section>\n
\n <\/div>\n\n