Logo

dev-resources.site

for different kinds of informations.

Understand Terraform in Just 5 Minutes

Published at
12/26/2024
Categories
webdev
terraform
cicd
aws
Author
Alexander Uspenskiy
Categories
4 categories in total
webdev
open
terraform
open
cicd
open
aws
open
Understand Terraform in Just 5 Minutes

Let’s create a script to set up a ready-to-use architecture. This will include two EC2 instances running Nginx in a private subnet, an Application Load Balancer, a NAT Gateway for egress traffic, and an Internet Gateway in the public subnet.

Architecture Diagram

Terraform Demo Project Architecture

Prerequisites

You need to install AWS CLI and Terraform to your local box.

Source code

Get main.tf from https://github.com/alexander-uspenskiy/terraform-demo

Step by Step instruction

Terraform uses a declarative approach, meaning you simply define the desired final state of your infrastructure, and Terraform determines how to transition from the current state to the desired state for you. In most cases, this works seamlessly, but in complex scenarios, additional instructions, such as depends_on, may need to be provided.

Base infrastructure

Let's begin by creating a new VPC (recommended) in the us-east-1 region. If you prefer a different region, feel free to adjust accordingly. Assign the new VPC a CIDR block of 10.0.0.0/16 and ensure it does not conflict with your default VPC or any existing VPCs. Next, add two public subnets (one for each Availability Zone, as required for the Application Load Balancer) and a private subnet for two web-servers in accordance by security best practices.

locals {
  common_tags = {
    created_by = "terraform"
  }
}

resource "aws_vpc" "terraform-vpc" {
  cidr_block = "10.0.0.0/16"

  tags = merge(local.common_tags, {
    Name = "terraform-vpc"
  })
}

resource "aws_subnet" "public-subnet-terraform-1" {
  vpc_id            = aws_vpc.terraform-vpc.id
  cidr_block        = "10.0.0.0/24"
  availability_zone = "us-east-1a"

  tags = merge(local.common_tags, {
    Name = "public-subnet-terraform-1"
  })
}

resource "aws_subnet" "public-subnet-terraform-2" {
  vpc_id            = aws_vpc.terraform-vpc.id
  cidr_block        = "10.0.3.0/24"
  availability_zone = "us-east-1b"

  tags = merge(local.common_tags, {
    Name = "public-subnet-terraform-2"
  })
}

resource "aws_subnet" "private-subnet-terraform" {
  vpc_id            = aws_vpc.terraform-vpc.id
  cidr_block        = "10.0.2.0/24"
  availability_zone = "us-east-1a"

  tags = merge(local.common_tags, {
    Name = "private-subnet-terraform"
  })
}

Next, create an internet gateway to enable access to the ALB from anywhere, and configure the route tables for both the private and public subnets in the solution. After that, set up a NAT service to allow outbound internet access from the EC2 web servers (necessary for uploading Nginx software and updates). AWS requires an Elastic IP (EIP) for the NAT service. Be mindful that EIP incurs costs, so make sure to release it once testing is complete.

resource "aws_internet_gateway" "terraform-igw" {
  vpc_id = aws_vpc.terraform-vpc.id

  tags = merge(local.common_tags, {
    Name = "terraform-igw"
  })
}

resource "aws_route_table" "terraform-rtb" {
  vpc_id = aws_vpc.terraform-vpc.id

  route {
    cidr_block = "0.0.0.0/0"
    gateway_id = aws_internet_gateway.terraform-igw.id
  }

  tags = merge(local.common_tags, {
    Name = "terraform-rtb"
  })
}

resource "aws_route_table_association" "public-1" {
  subnet_id      = aws_subnet.public-subnet-terraform-1.id
  route_table_id = aws_route_table.terraform-rtb.id
}

resource "aws_route_table_association" "public-2" {
  subnet_id      = aws_subnet.public-subnet-terraform-2.id
  route_table_id = aws_route_table.terraform-rtb.id
}
resource "aws_eip" "terraform-nat-eip" {
  domain = "vpc"
}

resource "aws_route_table" "terraform-private-rtb" {
  vpc_id = aws_vpc.terraform-vpc.id

  route {
    cidr_block     = "0.0.0.0/0"
    nat_gateway_id = aws_nat_gateway.terraform-nat-gateway.id
  }

  tags = merge(local.common_tags, {
    Name = "terraform-private-rtb"
  })
}

resource "aws_route_table_association" "private-association" {
  subnet_id      = aws_subnet.private-subnet-terraform.id
  route_table_id = aws_route_table.terraform-private-rtb.id
}

resource "aws_nat_gateway" "terraform-nat-gateway" {
  allocation_id = aws_eip.terraform-nat-eip.id
  subnet_id     = aws_subnet.public-subnet-terraform-1.id

  tags = merge(local.common_tags, {
    Name = "terraform-nat-gateway"
  })
}

Continue with security group creation. We need two different groups for ALB and web servers for better security maintenance.

resource "aws_security_group" "secgrp-alb" {
  description = "Security group for ALB"
  name        = "secgrp-alb"
  vpc_id      = aws_vpc.terraform-vpc.id

  # Allow inbound HTTP traffic from the internet
  ingress {
    from_port   = 80
    to_port     = 80
    protocol    = "tcp"
    cidr_blocks = ["0.0.0.0/0"]
  }

  # Allow all outbound traffic
  egress {
    from_port   = 0
    to_port     = 0
    protocol    = "-1"
    cidr_blocks = ["0.0.0.0/0"]
  }

  tags = merge(local.common_tags, {
    Name = "secgrp-alb"
  })
}

resource "aws_security_group" "secgrp-web-server" {
  description = "Security group for web servers in private subnets"
  name        = "secgrp-web-server"
  vpc_id      = aws_vpc.terraform-vpc.id

  # Allow inbound HTTP traffic from the ALB security group
  ingress {
    from_port       = 80
    to_port         = 80
    protocol        = "tcp"
    security_groups = [aws_security_group.secgrp-alb.id]
  }

  # Allow all outbound traffic
  egress {
    from_port   = 0
    to_port     = 0
    protocol    = "-1"
    cidr_blocks = ["0.0.0.0/0"]
  }

  tags = merge(local.common_tags, {
    Name = "secgrp-web-server"
  })
}

Then add ALB with listener and Target Group (for two web servers).

resource "aws_lb" "terraform-alb" {
  name               = "terraform-alb"
  internal           = false
  load_balancer_type = "application"
  security_groups    = [aws_security_group.secgrp-alb.id]
  subnets            = [
    aws_subnet.public-subnet-terraform-1.id,
    aws_subnet.public-subnet-terraform-2.id
  ]

  enable_deletion_protection = false

  tags = merge(local.common_tags, {
    Name = "terraform-alb"
  })
}


resource "aws_lb_target_group" "terraform-alb-tg" {
  name     = "terraform-alb-tg"
  port     = 80
  protocol = "HTTP"
  vpc_id   = aws_vpc.terraform-vpc.id

  health_check {
    path                = "/"
    interval            = 30
    timeout             = 5
    healthy_threshold   = 5
    unhealthy_threshold = 2
    matcher             = "200"
  }

  tags = merge(local.common_tags, {
    Name = "terraform-alb-tg"
  })
}

resource "aws_lb_listener" "app-lb-listener" {
  load_balancer_arn = aws_lb.terraform-alb.arn
  port              = 80
  protocol          = "HTTP"

  default_action {
    type             = "forward"
    target_group_arn = aws_lb_target_group.terraform-alb-tg.arn
  }
}

Now it’s time to add web servers and associate them with the ALB Target Group. For this setup, I’m using the standard AWS Linux 2023 image, but you can choose a different OS based on your requirements. To stay within the free tier, use the t2.micro instance type. The user_data script is designed to install Nginx on the EC2 instances and set up a default web page with a simple test output. Note that this script will execute only once during instance creation and will not run again if you restart or stop and start the instance.

resource "aws_instance" "web-server-1" {
  ami                         = "ami-01816d07b1128cd2d"
  associate_public_ip_address = false
  instance_type               = "t2.micro"
  subnet_id                   = aws_subnet.private-subnet-terraform.id
  vpc_security_group_ids      = [aws_security_group.secgrp-web-server.id]

  root_block_device {
    delete_on_termination = true
    volume_size           = 10
    volume_type           = "gp3"
  }

  user_data = <<-EOF
             #!/bin/bash
              sudo yum update -y
              sudo yum install -y nginx
              sudo systemctl start nginx
              sudo systemctl enable nginx
              echo "<html><body><h1>Hello from teraform web server 1!</h1><p>Public IP: $PUBLIC_IP</p><p>Private IP: $PRIVATE_IP</p></body></html>" | sudo tee /usr/share/nginx/html/index.html
              EOF

  tags = merge(local.common_tags, {
    Name = "terraform-web-server-1"
  })

  lifecycle {
    create_before_destroy = true
  }

  depends_on = [aws_lb.terraform-alb, aws_lb_target_group.terraform-alb-tg]
}

resource "aws_instance" "web-server-2" {
  ami                         = "ami-01816d07b1128cd2d"
  associate_public_ip_address = false
  instance_type               = "t2.micro"
  subnet_id                   = aws_subnet.private-subnet-terraform.id
  vpc_security_group_ids      = [aws_security_group.secgrp-web-server.id]

  root_block_device {
    delete_on_termination = true
    volume_size           = 10
    volume_type           = "gp3"
  }

user_data = <<-EOF
             #!/bin/bash
              sudo yum update -y
              sudo yum install -y nginx
              sudo systemctl start nginx
              sudo systemctl enable nginx
              echo "<html><body><h1>Hello from teraform web server 2!</h1><p>Public IP: $PUBLIC_IP</p><p>Private IP: $PRIVATE_IP</p></body></html>" | sudo tee /usr/share/nginx/html/index.html
              EOF

  tags = merge(local.common_tags, {
    Name = "terraform-web-server-2"
  })

  lifecycle {
    create_before_destroy = true
  }

  depends_on = [aws_lb.terraform-alb, aws_lb_target_group.terraform-alb-tg]
}

resource "aws_lb_target_group_attachment" "web_server_1_attachment" {
  target_group_arn = aws_lb_target_group.terraform-alb-tg.arn
  target_id        = aws_instance.web-server-1.id
  port             = 80

  depends_on = [aws_instance.web-server-1]
}

resource "aws_lb_target_group_attachment" "web_server_2_attachment" {
  target_group_arn = aws_lb_target_group.terraform-alb-tg.arn
  target_id        = aws_instance.web-server-2.id
  port             = 80

  depends_on = [aws_instance.web-server-2]
}

The final step is to add output to the Terraform script to monitor the status of the web servers and retrieve the public URL of the Application Load Balancer (ALB).

output "web_server_1_private_ip" {
  description = "The private IP address of web server 1"
  value       = aws_instance.web-server-1.private_ip
}

output "web_server_2_private_ip" {
  description = "The private IP address of web server 2"
  value       = aws_instance.web-server-2.private_ip
}

output "alb_dns_name" {
  description = "The DNS name of the ALB"
  value       = aws_lb.terraform-alb.dns_name
}

Local Software Installation

AWS CLI

  1. Install the AWS CLI using Homebrew:

    brew install awscli
    
  2. Verify the installation:

    aws --version
    
  3. Configure the AWS CLI:

    aws configure
    

Terraform

  1. Install Terraform using Homebrew:

    brew tap hashicorp/tap
    brew install hashicorp/tap/terraform
    
  2. Verify the installation:

    terraform -version
    

Terraform Usage

  1. Initialize the Terraform configuration:

    terraform init
    
  2. Create an execution plan:

    terraform plan
    
  3. Apply the Terraform configuration:

    terraform apply
    
  4. Destroy the Terraform-managed infrastructure:

    terraform destroy
    

How to execute

  1. Create a Free Tier Account in AWS (https://aws.amazon.com/free/)

  2. Setup CLI

  3. Install Terraform

  4. Apply configuration

  5. See output (for example alb_dns_name = "terraform-alb-1525092884.us-east-1.elb.amazonaws.com")

  6. Browse to your alb_dns_name url (don't forget to use http://)

  7. Refresh the page several time to make sure ALB routes you to web-server-1 or web-server-2

  8. Do terraform destroy to release resources and stay on free tier credit.

How to Enhance Architecture

  1. To enhance the resilience of the solution, add another private subnet in a different availability zone and move some of the web servers there.
  2. Integrate a Terraform step into your CI/CD pipeline.
  3. Use a tool like Ansible to deploy configuration to web server instances, improving supportability and security.
  4. Set up multiple NAT Gateways in different public subnets to ensure high availability.

Resources

Happy Coding!

Featured ones: