The Terraform SSH Module

I've been revisting Terraform lately and have worked through a few of the awkward bits I ran into on my first time around. Near the top of this list was SSH key management.

I'd always encountered SSH keys provisioned along with the compute resources you'd be accessing with them so I just configured them like any other variable. This worked well enough until I added a second compute resource and started debugging error messages about keys that already existed.

I was treating SSH keys as a VM-specific resource, but Amazon and DigitalOcean both approached the matter differently.

It's the Service, Stupid

Amazon and DigitalOcean both associate an SSH keypair with the service first, and then the instance. If you provision box1 with keypair1 and then try to provision box2 with keypair1, you'll get an error that the key already exists, which is totally unsurprising with the benefit of hindsight.

Both of these services treat keypairs as a general security feature of your account. New instances are simply associated with these keypairs during provisioning.

The SSH Module

Enter the SSH module. I'm going to focus on AWS here since that's the page I've got up in the other window, but there are very few moving parts here so you'll be able to apply this elsewhere with minimal fuss.

Here's what a fully-configured resource looks like:

1
2
3
4
resource "aws_key_pair" "default" {
  key_name   = "yourname-personal"
  public_key = "ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAABAQD3F6tyPEFEzV0LX3X8BsXdMsQz1x2cEikKDEY0aIj41qgxMCP/iteneqXSIFZBp5vizPvaoIR3Um9xK7PGoW8giupGn+EPuxIA4cDM4vzOqOkiMPhz5XK0whEjkVzTo4+S0puvDZuwIsdiW9mxhJc7tgBNL0cYlWSYVkz4G/fslNfRPW5mYAM49f4fhtxPb5ok4Q2Lg9dPKVHO/Bgeu5woMc7RY0p1ej6D4CKFE6lymSDJpW0YHX/wqE9+cfEauh7xZcG0q9t2ta6F6fmX0agvpFyZo8aFbXeUBr7osSCJNgvavWbM/06niWrOvYX2xwWdhXmXSrbX8ZbabVohBK41 you@example.com"
}

That quickly becomes this:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
# main.tf
resource "aws_key_pair" "default" {
  key_name   = "${var.key_name}"
  public_key = "${var.public_key}"
}

# variables.tf
variable "name" {
  description = "Friendly name for the public key.
}
variable "public_key" {
  description = "Your public key"
}

# outputs.tf
output "name" {
  description = "Name of the public key"
  value       = "${aws_key_pair.default.name}"
}

output "fingerprint" {
  description = "SSH fingerprint of the key we just added"
  value       = "${aws_key_pair.default.fingerprint}"
}

Create an environment named "aws-ssh" or something descriptive and run init/plan/apply. You'll get some output like this:

1
2
name = yourname-personal
ssh_fingerprint = 00:11:22:33:44:55:66:77:88:99:aa:bb:cc:dd:ee:ff

This is helpful, since the fingerprint is how you'll tell your provider which key to use. If you need to access this output again, you can just run terraform output from your environment to get it.

One More Thing

There are two ways you can provide your key to your SSH module. The first is to simply paste in the contents of id_rsa.pub into your config. The second is a bit more elegant:

1
2
3
4
5
module "ssh-you-personal" {
  source = "../../modules/aws/ssh/"
  name       = "you-personal"
  public_key = "${file("~/.ssh/id_rsa.pub")}"
}

This loads the keyfile from disk first, then passes it to the module for provisioning.

Nice!

<<
>>