terragrate helps doing state migrations in terraform. It is inspired by database migration tools like Flyway.
You'll need to install terraform and set it up. At least terraform state list should work for terragrate to be any useful.
Download the latest release for your OS and put it in your PATH. Running terragrate alone will print the detailed help.
Let's assume I have a resource like this in terraform:
resource "docker_network" "network" {
  name     = "public_network"
}I have provisioned the resource, which is identified as public_network.docker_network.network in the state.
During a refactoring, we extract this resource to a module called network to make it more reusable. Now we will use it a bit differently:
module "public_network" {
  source = "./network"
  name   = "public_network"
}If we run terraform apply, it will try to destroy the old resource and create a new one. Even though it's the exact same entity, the new id is called module.public_network.docker_network.network.
This is fine for a toy example like this, but it makes no sense to reprovision a complete database just because of this!
The typical approach to fix this is by migrating the state with the terraform state commands by hand. This is error-prone and untraceable.
Instead, I propose using a migration reflected in code. Let's define the move_to_module.json migration, and store it together with the infrastructure code:
{
  "name": "move to module",
  "description": "Convert network to module",
  "transformations": [
    {
      "kind": "MV",
      "matcher": "public_network",
      "replacement": "module.public_network"
    }
  ]
}I can use that migration and feed it my current state with this:
terraform state list | terragrate --state - --migration move_to_module.json tfThis will output the list of terraform commands needed to migrate the state properly, in this case:
terraform state mv module.public_network.docker_network.network module.module.public_network.docker_network.networkBy keeping the migration in your repository, you can known which migrations have been applied thus far. An automated script is much less likely to leave your state in an undefined situation.
The following transformations are supported:
- MV: Move items in terraform state.
- RM: /remove items from the terraform state.
- As of today, terragratedoes not offer a way of automatically running migrations in case your state is out of date.
- Right now, terragrateassumes that you callterraformdirectly. It doesn't support something like terragrunt
See LICENSE
See the open issues for a list of proposed features (and known issues).