Building a Custom EC2 Module with an Amazon Machine Image Using Terraform
data:image/s3,"s3://crabby-images/32303/3230324d2f9532226f40184718cd0c3b78cb36ce" alt=""
In Terraform, modules are blocks containing multiple resources being used together. Modules allow you to feed information and other “child” modules into a “parent” module, though “parent” modules cannot be fed into “child” modules. In this exercise, we will make a custom module so that a standardized EC2 instance with an Amazon Linux 2 AMI can be easily reproduced and adjusted in the future.
Using modules in Terraform is great, but one should be cautious of over-using modules because it can be challenging to understand and maintain an unorganized or redundant family of modules.
For this exercise, I needed:
- My Github
- Terraform
- AWS CLI
- IDE of choice (I used Atom)
STEP 1: Fork & Clone Repo
First I forked a “create ec2” repository from my Level Up In Tech course’s Github account (Image 1).
data:image/s3,"s3://crabby-images/197f3/197f36bbd03c9377b883d41652d34bce92a03f85" alt=""
Then in Terminal I confirmed that I was in my home directory, and I used the command git clone https://github.com/kczarzasty/terraformec2.git to clone the forked repo (Image 2).
data:image/s3,"s3://crabby-images/037df/037dfa7ff91820af5e2a7de622e0175479256436" alt=""
Then I confirmed that I had a new working directory named “terraformec2” by changing into that directory (Image 3).
data:image/s3,"s3://crabby-images/e9f6d/e9f6d14c691fdf15136f0f900ce500a2326679f1" alt=""
STEP 2: Module Setup
Now that the repository was set, I needed to set up my files for the module. Within the “terraformec2” directory, I made a new directory called “instance,” and therein I created my main.tf, outputs.tf, providers.tf, and variables.tf files (Image 4).
data:image/s3,"s3://crabby-images/6b464/6b464dd5339b8f2e5e8bb5d90e418c71e1d9d28b" alt=""
We then needed to transfer the resource and provider section of ec2.tf to our new providers.tf of our instance module, and so I did so in Atom as seen in Image 5 & 6.
data:image/s3,"s3://crabby-images/7e5f9/7e5f91f3d7a8b3df43c8dfcbdc012e9466e6aaeb" alt=""
data:image/s3,"s3://crabby-images/336ef/336efe033d5dd7f2baaec2edcc59610203899a37" alt=""
I then copied the instance resource from ec2.tf and pasted it in main.tf to be the provider (Image 7).
data:image/s3,"s3://crabby-images/0830a/0830ac2a1f5f01ddac88929612364404e63c06d1" alt=""
I then cleared the information in ec2.tf, and replaced it with , as it will be our module file intaking code from the other files.
data:image/s3,"s3://crabby-images/f81c1/f81c15c5a66aeb68c64795090f264c2f0fd80947" alt=""
I then changed directories back one by using the command cd ../. Once in the terraformec2 directory, I ran the following respectively to ensure my EC2 Instance was being created: terraform init, terraform fmt, terraform validate, terraform plan, terraform apply. Our successful result after typing “yes” to confirm terraform apply is shown in Image 9.
data:image/s3,"s3://crabby-images/2be93/2be93bff415a2c739afcdc24e7a56d56bfd50a92" alt=""
I confirmed in the my AWS Console that the EC2 instance had been created in region us-west-2 (Image 10).
data:image/s3,"s3://crabby-images/152eb/152eb8a124419dd740715684f459a21d2577fae9" alt=""
We no longer needed this instance (it was just a test), so I then ran the command terraform destroy and typed “yes” to confirm the action (Image 11).
data:image/s3,"s3://crabby-images/2c231/2c231276e2f7b8223856564102bd6073c6e5f193" alt=""
At this point we had successfully set up the module, but as it stands at this point in the article, it is just a simple EC2 module offering no variability.
STEP 3: Variables
So now added variables to the module so that we can adjust the EC2 Instance in the future as needed.
First I took the three original resources & made appropriate variables for each in variables.tf (Image 12).
data:image/s3,"s3://crabby-images/78139/78139694b9e5860f5d3688a1b9933f3ea1c14a7d" alt=""
I then referenced these variables in main.tf, (Image 13) where main.tf formly had the fixed values set.
data:image/s3,"s3://crabby-images/4f727/4f72715b0b9425bc4432c89e803b28207b96dc56" alt=""
I then added a variable to variables.tf so that in the future we can specify how many instances we want to create (Image 14).
data:image/s3,"s3://crabby-images/98ba9/98ba97d7e610e01b3269fd9be5bba14aa5966a88" alt=""
Then in ec2.tf I added a line for Instance Count and made it equal to 2 (Image 15).
data:image/s3,"s3://crabby-images/9607f/9607f09f9c7eb942e23445ea339d7890b7437fbe" alt=""
Then I made an adjustment. I was concerned that, when launching multiple EC2 instances, it would be difficult to differentiate between instances. Therefore in main.tf I wrote Line 6 so that the Instance Count would be indicated in the name. Note I also wrote +1 because the instance count starts at zero (Image 16).
data:image/s3,"s3://crabby-images/a563c/a563c1cdea44d71d15abf13d36878a110054a390" alt=""
STEP 4: Outputs
Now that our module was increasingly customizable through the use of variables, I wrote my outputs.tf file which govern what information is printed in Terminal after successful terraform apply.
I elected to clarify Public IPs and EC2 Tags by writing in outputs.tf as seen in Image 17.
data:image/s3,"s3://crabby-images/b2bf0/b2bf00cacde9779c827b8a380ed35a565fb0ef0b" alt=""
I then wrote the desired outputs in a new rootoutputs.tf in /terraformec2 to go along with the ec2.tf module (Image 18).
data:image/s3,"s3://crabby-images/005e4/005e42438517823c14836a84c8cfb6ca275a9f78" alt=""
I now wanted to pause and ensure the module was working. When I ran terraform validate, I discovered that my count index was invalid (Image 19).
data:image/s3,"s3://crabby-images/0e2da/0e2da9c9e0c286929ebe78b5451fb3da95748ae1" alt=""
I then resolved this issue by adding Line 4 to main.tf as seen in Image 20.
data:image/s3,"s3://crabby-images/d7d8d/d7d8d9bcf044a1a203ae9e824663c8f469fa6a8e" alt=""
I then re-ran terraform validate, and this time everything was valid (Image 21).
data:image/s3,"s3://crabby-images/c5150/c51506fe254f4f130691b1fc76b931ebc66056c3" alt=""
I then ran terraform apply and confirmed “yes.” Image 22 shows that the resources were created and our outputs from rootoutputs.tf were printed.
data:image/s3,"s3://crabby-images/12648/12648197f988ba55c02af0ad0047d7ae03f24c85" alt=""
I then ran terraform destroy again, so as to avoid running these resources, accruing costs, etc.
STEP 5: Github
Now we’ve made our custom module. This would be a great tool to reuse in the future, so we will push it to Github.
To do so I first made a .gitignore file in the root directory so as to ignore files that I didn’t want pushed (Image 23).
data:image/s3,"s3://crabby-images/99fb7/99fb70cbeee370ccd801f3a25f9f33f9e83ae0de" alt=""
To ensure all files were set, I ran the command git status (Image 24).
data:image/s3,"s3://crabby-images/840a7/840a743ee9aa7845c4f13426bfe999706f1a71ec" alt=""
To stage my files I ran the following command as seen in Image 25 to include all files that weren’t tracked or staged were ready.
git add ec2.tf .DS_Store .gitignore .terraform.lock.hcl instance/ rootoutputs.tf
data:image/s3,"s3://crabby-images/efadf/efadf0c71ae40211740c595281aeb9e82c4df253" alt=""
I then ran the command git commit -m “initial commit” to commit my files (Image 26).
data:image/s3,"s3://crabby-images/d8661/d8661f13c7466bd2d2eb1e1dcaa1aec7e867f083" alt=""
I then used the command git push origin to push the files to my Github (Image 27).
data:image/s3,"s3://crabby-images/4f308/4f308c7a31a09b467561fc14ce2697525c388a22" alt=""
I then confirmed in my Github that the custom EC2 module and all relevant files were there, and indeed they were (Image 28).
data:image/s3,"s3://crabby-images/0493e/0493e4778349d368b3d430fe439895d83d3d9115" alt=""
Thanks for reading.
Credit: Level Up In Tech as was credited in the body of the article.