How to Use Linode Object Storage as a Terraform Backend

(or any other S3-compatible object store)

This is mostly a note to my future self since I couldn't find any documentation on using Terraform with Linode Object Storage as a Terraform state backend. With that said, I hope to save other intrepid Linode users some time if they're lucky enough to come across this post. While I've explained the problem in terms of Linode, the solution here applies to any non-S3 object store compatible with the S3 API. Swap in the appropriate endpoint for your provider.

Chances are, if you've ever tried using Terraform's S3 backend with a non-S3 object store, you've seen an error similar to the following:

Error: error using credentials to get account ID: error calling sts:GetCallerIdentity: InvalidClientTokenId: The security token included in the request is invalid. status code: 403, request id: 60a33ac0-e460-4745-89dc-668d72413df3

Linode's Object Storage is S3-compatible, which means anywhere you'd typically use Amazon's S3 object storage – as long as you can swap in a custom endpoint URL – you can use Linode Object Storage instead. Which, by the way, is really cheap. So you should do it. Think of all the non-Bezos pockets you'll be lining.

Change Your Backend

It's time to update your Terraform backend. First of all, you can mostly refer to Terraform's S3 documentation for this, but there are two key changes you'll want to make.

  1. endpoint: This is the hostname that Terraform uses to connect to Linode's Object Store API. Terraform thinks it's talking to S3 but – S3-compatibility to the rescue – we're dropping Linode into place.
  2. skip_credentials_validation: According to Terraform's documentation: (Optional) Skip the credentials validation via the STS API. Presumably this is referring to Amazon STS, but I'll be honest; I don't care. If this isn't set to true, Terraform will ask Amazon if you Linode credentials are valid S3 credentials. You don't want that.

Example Backend Config

terraform {
  backend "s3" {
    bucket = "<YOUR LINODE BUCKET NAME>"
    key    = "/path/to/your/tfstate"
    region = "<LINODE REGION>"                        # e.g. us-east-1
    endpoint = "<LINODE REGION>"    # e.g.
    skip_credentials_validation = true                # just do it

That should get you all set up using Linode Object Storage as your Terraform state backend.

Cheers, AC

See also