Skip to content

This is a practical exercise for the DevOps certification, the objective is to automate the deployment of infrastructure using Terraform with GitHub Actions.

Notifications You must be signed in to change notification settings

marcosgacosta/automating_infrastructure

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

31 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Intro

This project is part of MundosE's DevOps certification, and the task asked is to use GitHub Actions to deploy an Apache on an EC2 using Terraform.

The goal of this Integrative Project is to integrate the following technologies into a project:

  1. GitHub Actions
  2. Apache
  3. Amazon Web Services
    • EC2
    • S3
  4. Terraform

The following diagram shows what we are trying to accomplish: Automating AWS Infrastructure using GitHub Actions and Terraform_.

This is project is for didactic purposes ony so it is important to recognize that many improvements could be made to this implementation but are beyond the scope.

First Step: Create the Terraform Files

The following diagram shows the expected output of our Terraform implementation (main.tf):

Output of main.tf.

Moreover, it is required to create and implement a route table and a security group to allow Internet access to our server limited to the services we need.

So, we need a Terraform File that deploys:

  1. EC2 Instance (webserver_apache)
  2. VPC (custom_vpc)
  3. Subnet (public_subnet)
  4. Internet Gateway (internet_gateway)
  5. Route Table (route_table)
  6. Security Group (apache_security)

EC2 Instance (webserver_-_apache)

Next, let us create the EC2 Instance with the assigned name, using the AMI Amazon Linux 2023, and create it inside our subnet, our security group and commanding it to run the script that will run Apache Webserver.

resource "aws_instance" "webserver_apache" {
  ami           = "ami-0230bd60aa48260c6" # AMI Amazon Linux 2023
  instance_type = "t2.micro"
  associate_public_ip_address = true
  vpc_security_group_ids      = [aws_security_group.apache_security.id]
  subnet_id                   = aws_subnet.public_subnet.id
  user_data                   = "${file("create_apache.sh")}"
}

VPC (custom_vpc)

The following block creates a VPC using the range 192.168.0.0/16:

resource "aws_vpc" "custom_vpc" {
  cidr_block = "192.168.0.0/16"

  tags = {
    Name = "apache_vpc"
  }
}

Subnet (public_subnet)

The following block creates public_subnet inside custom_vpc, using the range 192.168.10.0/24

resource "aws_subnet" "public_subnet" {
  vpc_id            = aws_vpc.custom_vpc.id
  cidr_block        = "192.168.10.0/24"
  availability_zone = "us-east-1a"

}

Internet Gateway (internet_gateway)

The following block creates the Internet Gateway (internet_gateway) so our VPC, hence our Apache Server, can access the Internet:

resource "aws_internet_gateway" "internet_gateway" {
  vpc_id = aws_vpc.custom_vpc.id

}

Route Table (route_table)

The following block creates a route table and attaches it to custom_vpc, and it also creates the default route to the internet_gateway.

resource "aws_route_table" "route_table" {
  vpc_id = aws_vpc.custom_vpc.id

  route {
    cidr_block = "0.0.0.0/0"
    gateway_id = aws_internet_gateway.internet_gateway.id
  }

We need to associate our new Route Table to our Public Subnet:

resource "aws_route_table_association" "a" {
  subnet_id      = aws_subnet.public_subnet.id
  route_table_id = aws_route_table.route_table.id
}

Security Group (apache_security)

Now we need to create a Security Group for our Apache Webserver that allows Web and SSH traffic.

resource "aws_security_group" "apache_security" {
  name        = "apache_security"
  description = "Allow Web and SSH traffic."
  vpc_id      = aws_vpc.main.id

  ingress {
    description      = "Allows SSH"
    from_port        = 22
    to_port          = 22
    protocol         = "tcp"
    cidr_blocks      = ["0.0.0.0/0"]
  }
 ingress {
    description      = "Allows HTTP"
    from_port        = 80
    to_port          = 80
    protocol         = "tcp"
    cidr_blocks      = ["0.0.0.0/0"]
  }
  egress {
    from_port        = 0
    to_port          = 0
    protocol         = "-1"
    cidr_blocks      = ["0.0.0.0/0"]
    ipv6_cidr_blocks = ["::/0"]
  }
}

Region and Provider

To allow our Terraform to work, we need to use all this code and specify our region and our provider:

provider "aws" {
  region = "us-east-1"
}

Output EC2 Instance Public IP

We want to check if the instance is working, so to do that we are going to ask Terraform to provide us the public IP of the Instance:

output "WebServer_Apache_IP" {
  value = aws_instance.webserver_apache.public_ip
}

Results

All this is what we can see in main.tf. If we run it. It gives us a fully functioning EC2 Instance running Apache.

If we run terraform apply --auto-approve we have the following output:

terraform apply Output.

If we write that IP in our web browser we can see our Apache Server fully functioning:

terraform apply Output.

Terraform Backend

To ensure that our Terraform Statefile is secure and have consistency, we are going to set up a remote backend. For this we need to create an S3 Bucket. We create the bucket using the GUI and we call it mundose-pin-test1337.

S3 Bucket created.

There is a possibility to lock the statefile using DynamoDB if we are part of the team. Since it is not our case today, we will not implement it but it is strongly recommended. So we create backend.tf with the following code.:

terraform {
  backend "s3"{
    bucket = "mundose-pin-test1337"
    key    = "terraform/terraform.tfstate"
    region = "us-east-1"
  }
}

Now, everytime we update our infrastructure the Terraform statefile will be stored in the S3 bucket.

GitHub Actions

The following step is to setup GitHub Actions so we can automate the deployment of our infrastructure. To do this we use the Template provided by GitHub Actions with some modifications.

Secret Keys

To be able to execute commands in Terraform and in AWS we need to add the following line when we run Terraform init or Apply.

 env:
        AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }}
        AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }}

To be able to install Terraform on the GitHub Actions enviromnent we also need to set up the secret key TF_API_TOKEN.

So these are the three secret keys we need to configure in our repository:

  1. AWS_ACCESS_KEY_ID
  2. WS_SECRET_ACCESS_KEY
  3. TF_API_TOKEN

After setting up those variables, we can run our Action and we can see the result:

Actions output.

Actions output.

Inside the step Terraform Apply we can find our Webserver IP Address and if we paste it on our Web Browser we have the following:

Webserver running after GitHub Action.

Terraform Destroy

Since we are practicing, we want to be able to remove our infrastructure. For this we create the following terraform_destroy.yml where we set it up to run manually when we want (workflow_dispatch):

name: Terraform Destroy
on:
 workflow_dispatch:

jobs:
 terraform-destroy:
   runs-on: ubuntu-latest
   env:
      AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }}
      AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }}
      AWS_REGION: 'us-east-1'
   steps:
     - name: Terraform destroy
       id: destroy
       run: terraform destroy -auto-approved

Conclusion

So now we have automated our infrastructure. We can make changes in our Terraform files, commit them and GitHub Actions will automatically update our infrastructure to match our new file.

About

This is a practical exercise for the DevOps certification, the objective is to automate the deployment of infrastructure using Terraform with GitHub Actions.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published