Configuring bare-metal Packet servers with cloud-init

Introduction

Oftentimes there will be cases where you will want to automate the provisioning and configuration of your Packer bare-metal infrastructure. There are a plethora of tools out there, however, cloud-init is an industry-standard that is used to initialize and configure instances with user-data.

What is Terraform?

Terraform is an Infrastructure-as-code tool that allows users to build, change, and version your infrastructure safely and efficiently. It uses a high-level syntax to declaratively provision and manage infrastructure, allowing the ability to break down the configuration into smaller chunks for better organization, re-use, and maintainability. Information on installing and running Terraform can be found here. By passing the user_data parameter into a Terraform.yaml file, you can use automation to configure your Packet instance at boot time. More on that below.

Using Terraform to configure Packet devices with cloud-init

If Terraform is your preferred infrastructure provisioning method then you can find the Packet Terraform Provider at the Github Repo here.

For any infrastructure provider, when using Terraform as a provisioning tool you will always need to specify the provider block as seen here:

provider "packet" {
  auth_token = "${var.auth_token}"
}

Here’s an example module that utilizes user-data to configure a Packet instance at boot time:

# Create a server
resource "packet_device" "my-dream-server-1" {
    hostname         = "tf.coreos2"
    plan             = "t1.small.x86"
    facilities       = ["ewr1"]
    operating_system = "coreos_stable"
    billing_cycle    = "hourly"
    project_id       = local.project_id
    user_data = "I2Nsb3VkLWNvbmZpZwpwYWNrYWdlczoKICAtIGlmdG9wCiAgLSBubW9uCg=="
    project_ssh_keys_ids = ["95"]
    tags          = {
        Name        = "Application Server 2"
        Environment = "development"
    }
}

With this module, you have a resource that is designating packet_device as the type of resource you want to provision, and using variables such as project_id and user_data to handle the provisioning. When you provide the string for user_data, you are designating a startup script that the bare-metal server will run on boot-up.

Using cloud-init to configure Packet devices

You can provision new servers via the API to fetch user data for your Packet device via the cloud-init service. This allows you to automate various server configuration tasks by fetching user data directives upon server deployment. Your provided tasks will be executed when your server boots for the first time. There are two ways of doing this – shell scripts or cloud-init directives. We’re going to talk about cloud-init directives.

Cloud-Init directives are executed when your server boots for the first time, but the syntax is slightly different. Your scenario must start with #cloud-config line, otherwise user data directives will be rejected. For further reference, I recommend checking the cloud-init official documentation: https://cloudinit.readthedocs.io/en/latest/index.html

A simple example of a cloud-init script that would be passed is:

#cloud-config

packages:

 - httpd

 - mariadb-server

runcmd:

 - systemctl start httpd

 - sudo systemctl enable httpd

 - [ sh, -c, "chmod 755 /var/tmp" ]

In order to pass this data scenario to the Packet API, it must first be converted into base64 format. On a Linux system you would do the following for your test.yaml file:

# base64 test.yaml

I2Nsb3VkLWNvbmZpZwpwYWNrYWdlczogCiAtIGh0dHBkIAogLSBtYXJpYWRiLXNlcnZlcgpydW5j

bWQ6CiAtIHN5c3RlbWN0bCBzdGFydCBodHRwZAogLSBzdWRvIHN5c3RlbWN0bCBlbmFibGUgaHR0

cGQKIC0gWyBzaCwgLWMsICJjaG1vZCA3NTUgL3Zhci90bXAiIF0K

This output text then has to be fetched via Packet API user_data parameter when ordering a new server.

Putting it all together

To see this in action, specify the resources provider so that you can designate Packet as the platform you’re provisioning to. Your final script should look like this:

# Set provider

provider "packet" {
  auth_token = "${var.auth_token}"
}

# Create a server
resource "packet_device" "my-dream-server-1" {
    hostname         = "tf.coreos2"
    plan             = "t1.small.x86"
    facilities       = ["ewr1"]
    operating_system = "coreos_stable"
    billing_cycle    = "hourly"
    project_id       = local.project_id
    user_data = "I2Nsb3VkLWNvbmZpZwpwYWNrYWdlczoKICAtIGlmdG9wCiAgLSBubW9uCg=="
    project_ssh_keys_ids = ["95"]
    tags          = {
        Name        = "Application Server 2"
        Environment = "development"
    }
}

Finishing up

That’s all there is to it! Terraform is a really great tool for automating infrastructure once you understand the syntax and how it works. I hope you liked this article!

Subscribe and discover the newest
updates, news, and features

We value your inbox and are committed to preventing spam