いま、新規サービスを開発するにあたり、Terraformを利用しています。
Terraformは便利ですが、なかなかハマりどころも多いですね。
Cloud KMSのKeyを扱う際に"Instance cannot be destroyed"エラーが出て時間を溶かしたので、対応した内容を記録します。
環境
- GCP
- Terraform : Ver.1.0.1
- 利用したmodule : terraform-google-modules/kms/google Ver. 1.2
手動で作成したCloud KMS KeyをTerraformに反映させる
- Terraform経由でCloud KMSのkeyringと、key("key_terraform")を作成しました。
- 手動でKey("key_manual")をGCPのConsoleから手動で作成しました。
手動で作成した鍵をTerraformに反映させようとしました。
"Instance cannot be destroyed"が発生
Terraformに反映させようと以下のtfファイルでterraform plan
したところ以下のエラーが発生しました。
module "kms" {
source = "terraform-google-modules/kms/google"
version = "~> 1.2"
project_id = "sample_project"
location = "global"
keyring = "sample_key_ring"
keys = ["key_terraform", "key_manual"]
set_owners_for = []
key_rotation_period = "157680000s"
}
$ terraform plan
....
│ Error: Instance cannot be destroyed
│
│ on .terraform/modules/kms/main.tf line 27:
│ 27: resource "google_kms_crypto_key" "key" {
│
│ Resource module.kms.google_kms_crypto_key.key[0] has lifecycle.prevent_destroy set, but the plan calls for this resource to be destroyed. To avoid this error and continue with the plan, either disable
│ lifecycle.prevent_destroy or reduce the scope of the plan using the -target flag.
│ Error: Instance cannot be destroyed
│
│ on .terraform/modules/kms/main.tf line 27:
│ 27: resource "google_kms_crypto_key" "key" {
│
│ Resource module.kms.google_kms_crypto_key.key has lifecycle.prevent_destroy set, but the plan calls for this resource to be destroyed. To avoid this error and continue with the plan, either disable
│ lifecycle.prevent_destroy or reduce the scope of the plan using the -target flag.
このエラーが出ていると、他のリソースのapplyが反映されないので、解決する必要があります。
prevent_destroyをtrueにする
Terraformはリソースの再作成を行うのに対し、
エラーメッセージの通り、Cloud KMSの鍵は削除できないためです。
moduleのドキュメントを見ると、prevent_destroyというパラメータがあるので付与します。
module "kms" {
source = "terraform-google-modules/kms/google"
version = "~> 1.2"
project_id = "sample_project"
location = "global"
keyring = "sample_key_ring"
keys = ["key_terraform", "key_manual"]
set_owners_for = []
key_rotation_period = "157680000s" # 追加
}
$ terraform plan
....
│ Error: Instance cannot be destroyed
│
│ on .terraform/modules/kms/main.tf line 27:
│ 27: resource "google_kms_crypto_key" "key" {
│
│ Resource module.kms.google_kms_crypto_key.key has lifecycle.prevent_destroy set, but the plan calls for this resource to be destroyed. To avoid this error and continue with the plan, either disable
│ lifecycle.prevent_destroy or reduce the scope of the plan using the -target flag.
エラーは一つ減りましたが、同じエラーが発生しました。
手動で作成した鍵"key_manual"の再作成をTerraformが試みており、
名前は同じでもTerraform設定と実際の鍵が異なるものと認識されているようです。
key_rotation_period(鍵のローテーション期間)をtfファイルと揃えてもだめでした。
GCPのConsole上はterraformで作成したもの"key_terraform"と差分がありません。
tfstateファイルを編集する
調べていくうちに以下のstackoverflowの投稿がありました。
こちらにあるように、あとはGCS上に配置したstateファイルを編集するしかなさそうです。
ダウンロードしたstateファイルの中で、"key_terraform"と"key_manual"の設定値を比較します。
terraform経由で作成したkeyと比較すると、やはり内部的にはいくつか差分がありました。
- indexがない
- timeoutsの形式が異なる
- dependenciesがない
バックアップをとりつつ、"key_terraform"にあわせるように編集しました。
そして、改めて、terraform plan
を実行します。
$ terraform plan
....
│ No changes. Your infrastructure matches the configuration.
ようやく、差分が出なくなり、反映されました!
stackoverflowのコメントにあるように、Cloud KMSの鍵は破棄して利用できなくさせることはできても、
完全に削除することができないので、再作成するterraformの設計にはそぐわないようです。
GCPでは、CloudSQLなども同じ名前のインスタンスをすぐ再作成できないなどもあるので、その場合も削除しないように回避したり、stateファイルを直接編集したりする必要がありそうです。