ブロック@Terraform¶
はじめに¶
本サイトにつきまして、以下をご認識のほど宜しくお願いいたします。
01. resource
ブロック¶
resource
ブロックとは¶
ルートモジュール/チャイルドモジュールにて、クラウドプロバイダーのAPIに対してリクエストを送信し、クラウドインフラを作成する。
resource
タイプ¶
操作されるリソースの種類のこと。
リソースとTerraformのresource
タイプはおおよそ一致している。
resource
ブロックの実装方法¶
▼ AWSの場合¶
*実装例*
# ---------------------------------------------
# AWS ALB
# ---------------------------------------------
resource "aws_lb" "this" {
name = "prd-foo-alb"
load_balancer_type = "application"
security_groups = ["*****"]
subnets = ["subnet-*****","subnet-*****"]
}
02. data
ブロック¶
data
ブロックとは¶
クラウドプロバイダーのAPIに対してリクエストを送信し、クラウドインフラに関するデータを取得する。
data
ブロックの実装方法¶
▼ AWSの場合¶
*実装例*
例として、ECSタスク定義名を指定して、AWSから
# ---------------------------------------------
# Data ECS task definition
# ---------------------------------------------
data "aws_ecs_task_definition" "this" {
task_definition = "prd-foo-ecs-task-definition"
}
*実装例*
AMIを検索した上で、AWSから特定のAMIを取得する。
# ---------------------------------------------
# Data AMI
# ---------------------------------------------
data "aws_ami" "bastion" {
most_recent = true
owners = ["amazon"]
filter {
name = "architecture"
values = ["x86_64"]
}
filter {
name = "root-device-type"
values = ["ebs"]
}
filter {
name = "name"
values = ["amzn-ami-hvm-*"]
}
filter {
name = "virtualization-type"
values = ["hvm"]
}
filter {
name = "block-device-mapping.volume-type"
values = ["gp2"]
}
}
03. output
ブロック¶
output
ブロックとは¶
可読性の観点から、resource
ブロック一括で出力するのではなく、resource
ブロックの特定のattribute
値を出力するようにした方が良い。
循環参照エラー (Error: Cycle
)¶
output
ブロックで値を出力するモジュールは、コール元に依存することになる。
コール元もoutput
ブロックを出力するモジュールを使用する場合、相互依存するために循環参照エラーが発生する。
循環参照エラーは、ローカルモジュール / 自前リモートモジュール内で公式リモートモジュールを使用する場合に起こる。
ユースケースの種類¶
▼ ルートモジュール内で使用する場合¶
ルートモジュールが持つ値を、data
ブロックのterraform_remote_state
リソースを使用して、異なるバックエンドに出力する。
この場合、ルートモジュールがコール元に依存することになる。
repository/
├── main.tf # ルートモジュール
├── outputs.tf
...
*実装例*
ルートモジュール内でoutput
ブロックを使用したとする。
# @ルートモジュール
# resourceブロックを、outputブロックとして出力する。
output "bastion_ec2_instance_id" {
value = aws_instance.bastion.id
}
# ローカルモジュールのoutputブロックを、さらにoutputブロックとして出力する。
output "alb_zone_id" {
value = module.alb.alb_zone_id
}
任意の場所で、data
ブロックのterraform_remote_state
からoutput
ブロックを使用できる。
# ---------------------------------------------
# Data
# ---------------------------------------------
data "terraform_remote_state" "alb" {
backend = "s3"
config = {
bucket = "bucket"
key = "alb.tfstate"
}
}
data "terraform_remote_state" "ec2" {
backend = "s3"
config = {
bucket = "bucket"
key = "ec2.tfstate"
}
}
# ---------------------------------------------
# Resource
# ---------------------------------------------
resource "foo" "this" {
foo_id = data.terraform_remote_state.alb.outputs.alb_zone_id
}
resource "bar" "this" {
bar_id = data.terraform_remote_state.ec2.outputs.bastion_ec2_instance_id
}
▼ ローカルモジュール内で使用する場合¶
ローカルモジュール内のresource
ブロックが持つ値を、ルートモジュールに出力する。
この場合、ローカルモジュールがコール元に依存することになる。
可読性の観点から、resource
ブロック一括で出力するのではなく、resource
ブロックの特定のattribute
値を出力するようにした方が良い。
repository/
├── modules/ # ローカルモジュール
│ ├── foo-module/
│ │ ├── main.tf
│ │ ├── outputs.tf
│ │ ├── variables.tf
...
*実装例*
ローカルモジュール内でoutput
ブロックを使用したとする。
# @ローカルモジュール
# ---------------------------------------------
# Output ALB
# ---------------------------------------------
output "alb_zone_id" {
value = aws_lb.this.zone_id
}
output "elb_service_account_arn" {
value = data.aws_elb_service_account.this.arn
}
ルートモジュールのmodule
ブロックでローカルモジュールをコールし、output
ブロックを渡せる。
# @ルートモジュール
# ルートモジュールでローカルモジュールをコールする。
module "alb" {
source = "../modules/alb"
}
resource "foo" "this" {
foo_id = module.alb.alb_zone_id
}
▼ リモートモジュール内で使用する場合¶
リモートモジュールのresource
ブロックや、リモートモジュール内ローカルモジュールが持つ値を、コール先のルートモジュールに出力する。
この場合、リモートモジュールがコール元に依存することになる。
repository/
├── main.tf # リモートモジュール
├── outputs.tf
...
# リモートモジュール内のローカルモジュールを出力する。
# ---------------------------------------------
# AWS ALB
# ---------------------------------------------
output "alb_zone_id" {
value = module.alb.alb_zone_id
}
# ---------------------------------------------
# AWS EC2
# ---------------------------------------------
output "bastion_ec2_instance_id" {
value = aws_instance.bastion.id
}
ルートモジュールのmodule
ブロックでリモートモジュールをコールし、output
ブロックを渡せる。
# @ルートモジュール
# ルートモジュールでリモートモジュールをコールする。
module "alb" {
source = "git::https://github.com/hiroki-hasegawa/terraform-alb-modules.git"
}
resource "foo" "this" {
foo_id = module.alb.alb_zone_id
}
output
ブロックのデータ型¶
▼ count
引数を使用した場合の注意点¶
resource
ブロックの作成にcount
引数を使用した場合、そのresource
ブロックはlist型として扱われる。
そのため、output
ブロックではキー名を指定して出力できる。
補足として、for_each
引数で作成したresource
ブロックはアスタリスクでインデックス名を指定できないため、注意。
*実装例*
例として、VPCのサブネットを示す。
ここでは、パブリックサブネット、applicationサブネット、datastoreサブネットをcount
引数で作成したとする。
# ---------------------------------------------
# AWS パブリックサブネット
# ---------------------------------------------
resource "aws_subnet" "public" {
count = 2
...
}
# ---------------------------------------------
# AWS プライベートサブネット
# ---------------------------------------------
resource "aws_subnet" "private_app" {
count = 2
...
}
resource "aws_subnet" "private_datastore" {
count = 2
...
}
▼ list型output
ブロック¶
インデックスキーをアスタリスクを指定した場合、list型になる。
# ---------------------------------------------
# AWS VPC
# ---------------------------------------------
output "public_subnet_ids" {
value = aws_subnet.public[*].id # IDのリスト型
}
output "private_app_subnet_ids" {
value = aws_subnet.private_app[*].id # IDのリスト型
}
output "private_datastore_subnet_ids" {
value = aws_subnet.private_datastore[*].id # IDのリスト型
}
▼ スカラー型output
ブロック¶
インデックスキー (0
番目) を指定した場合、スカラー型になる。
# ---------------------------------------------
# AWS VPC
# ---------------------------------------------
output "public_subnet_ids" {
value = aws_subnet.public[0].id # IDの文字列
}
output "private_app_subnet_ids" {
value = aws_subnet.private_app[0].id # IDの文字列
}
output "private_datastore_subnet_ids" {
value = aws_subnet.private_datastore[0].id # IDの文字列
}
04. local
ブロック¶
local
ブロックとは¶
通常変数であり、定義されたローカル/リモートモジュール内にのみスコープを持つ。
ルートモジュールとローカル/リモートモジュールが異なるリポジトリで管理されている場合に有効である。
これらが同じリポジトリにある場合は、環境変数を使用した方が可読性が高くなる。
locals {
foo = "FOO"
}
resource "aws_instance" "example" {
foo = local.foo
}
▼ よくある¶
locals {
service = "foo"
prefix = "${local.service}-${var.env}"
tags = {
Service = local.service
Env = var.env
# 管理するリポジトリ
ManagedBy = "https://github.com/hiroki-hasegawa/foo-terraform.git"
}
}
05. variable
ブロック¶
variable
ブロックとは¶
.tfvars
ファイル、module
ブロック、resource
ブロック、で使用する変数に関して、データ型やデフォルト値を定義する。
オプション¶
▼ データ型¶
単一値、list型、map型を定義できる。
*実装例*
AZ、サブネットのCIDRブロック、RDSのパラメーターグループ値などはmap型として保持しておくと良い。
また、IPアドレスのセット、ユーザーエージェントなどはlist型として保持しておくと良い。
# ---------------------------------------------
# Variables ECS
# ---------------------------------------------
variable "ecs_container_laravel_port_http" {
type = number
}
variable "ecs_container_nginx_port_http" {
type = number
}
# ---------------------------------------------
# Variables RDS
# ---------------------------------------------
variable "rds_auto_minor_version_upgrade" {
type = bool
}
variable "rds_instance_class" {
type = string
}
variable "rds_parameter_group_values" {
type = map(string)
}
# ---------------------------------------------
# Variables VPC
# ---------------------------------------------
variable "vpc_availability_zones" {
type = map(string)
}
variable "vpc_cidr" {
type = string
}
variable "vpc_endpoint_port_https" {
type = number
}
variable "vpc_subnet_private_datastore_cidrs" {
type = map(string)
}
variable "vpc_subnet_private_app_cidrs" {
type = map(string)
}
variable "vpc_subnet_public_cidrs" {
type = map(string)
}
# ---------------------------------------------
# Variables WAF
# ---------------------------------------------
variable "waf_allowed_global_ip_addresses" {
type = list(string)
}
variable "waf_blocked_user_agents" {
type = list(string)
}
▼ デフォルト値¶
変数のデフォルト値を定義できる。
有効な値を設定してしまうと可読性が悪くなる。
そのため、無効値 (例:boolean型であればfalse
、string型であれば空文字、list型であれば空配列など) を設定する。
*実装例*
module
ブロックやresource
ブロック内で、count
引数を使用して条件分岐を定義した場合に、そのフラグ値となるboolean値をデフォルト値として定義すると良い。
variable "enable_provision" {
description = "enable provision"
type = bool
default = false
}
06. メタ引数¶
メタ引数とは¶
全てのresource
ブロックで使用できるオプションのこと。
depends_on
引数¶
▼ depends_on
引数とは¶
resource
ブロック間の依存関係を明示的に定義する。
Terraformでは、基本的にresource
ブロック間の依存関係が暗黙的に定義されている。
しかし、複数のresource
ブロックが関わると、resource
ブロックを適切な順番で作成できない場合があるため、そういった時に使用する。
▼ ALB target group vs. ALB、ECS¶
例として、ALB target groupを示す。
ALB Target groupとALBのresource
ブロックを適切な順番で作成できないため、ECSの作成時にエラーが起こる。
ALBの後にALB target groupを作成する必要がある。
*実装例*
# ---------------------------------------------
# AWS ALB target group
# ---------------------------------------------
resource "aws_lb_target_group" "this" {
name = "prd-foo-alb-tg"
port = 80
protocol = "HTTP"
vpc_id = "vpc-*****"
deregistration_delay = "60"
target_type = "ip"
slow_start = "60"
health_check {
interval = 30
path = "/healthcheck"
protocol = "HTTP"
timeout = 5
unhealthy_threshold = 2
matcher = 200
}
depends_on = [aws_lb.this]
}
▼ Internet Gateway vs. EC2、Elastic IP、AWS NAT Gateway¶
例として、AWS NAT Gatewayを示す。
AWS NAT Gateway、Internet Gateway、のresource
ブロックを適切な順番で作成できない。
そのため、Internet Gatewayの作成後に、AWS NAT Gatewayを作成するように定義する必要がある。
# ---------------------------------------------
# AWS EC2
# ---------------------------------------------
resource "aws_instance" "bastion" {
ami = "*****"
instance_type = "t2.micro"
vpc_security_group_ids = ["*****"]
subnet_id = "*****"
key_name = "prd-foo-bastion"
associate_public_ip_address = true
disable_api_termination = true
tags = {
Name = "prd-foo-bastion"
}
depends_on = [var.internet_gateway]
}
# ---------------------------------------------
# AWS Elastic IP
# ---------------------------------------------
resource "aws_eip" "nat_gateway" {
for_each = var.vpc_availability_zones
vpc = true
tags = {
Name = format(
"prd-foo-ngw-%s-eip",
each.value
)
}
depends_on = [aws_internet_gateway.this]
}
# ---------------------------------------------
# AWS AWS NAT Gateway
# ---------------------------------------------
resource "aws_nat_gateway" "this" {
for_each = var.vpc_availability_zones
subnet_id = aws_subnet.public[each.key].id
allocation_id = aws_eip.nat_gateway[each.key].id
tags = {
Name = format(
"prd-foo-%s-ngw",
each.value
)
}
depends_on = [aws_internet_gateway.this]
}
▼ AWS S3バケットポリシー vs. パブリックアクセスブロックポリシー¶
例として、AWS S3を示す。
バケットポリシーとパブリックアクセスブロックポリシーを同時に作成できないため、作成のタイミングが重ならないようにする必要がある。
# ---------------------------------------------
# AWS S3
# ---------------------------------------------
# foo bucket
resource "aws_s3_bucket" "foo" {
bucket = "prd-foo-bucket"
acl = "private"
}
# Public access block
resource "aws_s3_bucket_public_access_block" "foo" {
bucket = aws_s3_bucket.foo.id
block_public_acls = true
block_public_policy = true
ignore_public_acls = true
restrict_public_buckets = true
}
# Bucket policy attachment
resource "aws_s3_bucket_policy" "foo" {
bucket = aws_s3_bucket.foo.id
policy = templatefile(
"${path.module}/policies/foo_bucket_policy.tpl",
{
foo_s3_bucket_arn = aws_s3_bucket.foo.arn
s3_cloudfront_origin_access_identity_iam_arn = var.s3_cloudfront_origin_access_identity_iam_arn
}
)
depends_on = [aws_s3_bucket_public_access_block.foo]
}
count
引数¶
▼ count
引数とは¶
指定した数だけ、resource
ブロックの作成を繰り返す。
count.index
オプションでインデックス数を展開する。
*実装例*
# ---------------------------------------------
# Resource EC2
# ---------------------------------------------
resource "aws_instance" "server" {
count = 4
ami = "ami-a1b2c3d4"
instance_type = "t2.micro"
tags = {
Name = "ec2-${count.index}"
}
}
▼ 作成の有無の条件分岐¶
特定の実行環境でリソースの作成の有無を切り替えたい場合、.terraform.tfvars
ファイルからフラグ値を渡し、これがあるかないかをcount
引数で判定し、条件分岐を実現する。
フラグ値を渡さない場合は、デフォルト値を渡すようにする。
# 特定の実行環境の.terraform.tfvarsファイル
enable_provision = true
# ---------------------------------------------
# Variables EC2
# ---------------------------------------------
variable "enable_provision" {
description = "enable provision"
type = bool
default = false
}
# ---------------------------------------------
# AWS EC2
# ---------------------------------------------
resource "aws_instance" "server" {
count = var.enable_provision ? 1 : 0
ami = "ami-a1b2c3d4"
instance_type = "t2.micro"
tags = {
Name = "ec2-${count.index}"
}
}
注意点として、count
関数を使用すると、他のブロック (例:resource
ブロック、output
ブロック) で設定値にアクセスする時にインデックス番号0
を指定する必要がある。
resource "foo" "server" {
foo = aws_instance.server[0].ami
}
▼ 実行環境別の作成¶
# 特定の実行環境の.terraform.tfvarsファイル
env = dev
# ---------------------------------------------
# Variables EC2
# ---------------------------------------------
variable "env" {
description = "system environment"
type = string
}
# ---------------------------------------------
# AWS EC2
# ---------------------------------------------
resource "aws_instance" "server" {
# テスト環境とステージング環境以外でプロビジョニングする
count = (
var.environment != "tes"
|| var.environment != "stg"
) ? 1 : 0
ami = "ami-a1b2c3d4"
instance_type = "t2.micro"
tags = {
Name = "ec2-${count.index}"
}
}
▼ count
関数で作成されなかったresource
ブロックを検知¶
count
関数で作成したブロックを他のブロックで使用する場合、それを検知できるようにする必要がある。
count
関数で作成されたリソースが存在するかどうかは、length
関数で検知できる。
resource "aws_kms_key" "foo" {
count = var.region == "ap-northeast-1" ? 1 : 0
policy = data.aws_iam_policy_document.foo.json
# ここでは、マルチリージョンが必須とする。
multi_region = true
tags = {
Name = foo
}
}
resource "aws_kms_alias" "foo" {
# count関数によるaws_kms_key.footリソースがなければ、本リソースも作成しない
# count = var.region == "ap-northeast-1" ? 1 : 0 でもよい。
count = length(aws_kms_key.foo)
name = "alias/foo"
target_key_id = aws_kms_key.foo[0].key_id
}
resource "aws_kms_replica_key" "foo" {
# count関数によるaws_kms_key.fooリソースがなければ、本リソースも作成しない
# count = var.region == "ap-northeast-1" ? 1 : 0 でもよい。
count = length(aws_kms_key.k8s_secret)
provider = aws.ap-northeast-3
primary_key_arn = aws_kms_key.foo[0].arn
policy = data.aws_iam_policy_document.foo.json
tags = {
Name = foo
}
}
▼ count
関数で作成されなかったoutput
ブロックはnull
¶
count
関数で作成されたリソースに対してのみoutput
ブロックで値を出力し、もしリソースがなければnull
や空文字 (""
) を出力するようにする。
補足として、count
関数の結果の検知には、length
関数を使用する。
output "foo_kms_key_arn" {
value = length(aws_kms_key.foo) > 0 ? aws_kms_key.foo[0].arn : null
}
for_each
引数¶
▼ for_each
引数とは¶
事前にfor_each
引数に格納したmap型のkey
の数だけ、resource
ブロックを繰り返し実行する。
繰り返し処理を実行する時に、count
引数とは違い、要素名を指定して出力できる。
*実装例*
例として、subnetを繰り返し作成する。
# ---------------------------------------------
# Variables Global
# ---------------------------------------------
vpc_availability_zones = { a = "a", c = "c" }
vpc_cidr = "*.*.*.*/23"
vpc_subnet_private_datastore_cidrs = { a = "*.*.*.*/27", c = "*.*.*.*/27" }
vpc_subnet_private_app_cidrs = { a = "*.*.*.*/25", c = "*.*.*.*/25" }
vpc_subnet_public_cidrs = { a = "*.*.*.*/27", c = "*.*.*.*/27" }
# ---------------------------------------------
# Resrouce VPC
# ---------------------------------------------
resource "aws_subnet" "public" {
for_each = var.vpc_availability_zones
vpc_id = aws_vpc.this.id
cidr_block = var.vpc_subnet_public_cidrs[each.key]
availability_zone = "${var.region}${each.value}"
map_public_ip_on_launch = true
tags = {
Name = format(
"prd-foo-pub-%s-subnet",
each.value
)
}
}
▼ 冗長化されたAZにおける設定¶
冗長化されたAZで共通のルートテーブルを作成する場合、そこで、for_each
引数を使用すると、少ない実装で作成できる。
for_each
引数で作成されたresource
ブロックはterraform apply
コマンド実行中にmap構造として扱われ、resource
ブロック名の下層にキー名でresource
ブロックが並ぶ構造になっている。
これを参照するために、『<resourceタイプ>.<resourceブロック名>[each.key].<attribute>
』とする。
*実装例*
パブリックサブネット、プライベートサブネット、プライベートサブネットに紐付くAWS NAT Gatewayの設定が冗長化されたAZで共通の場合、for_each
引数で作成する。
# ---------------------------------------------
# Variables Global
# ---------------------------------------------
vpc_availability_zones = { a = "a", c = "c" }
# ---------------------------------------------
# Resrouce Internet Gateway
# ---------------------------------------------
resource "aws_internet_gateway" "this" {
vpc_id = aws_vpc.this.id
tags = {
Name = "prd-foo-igw"
}
}
# ---------------------------------------------
# Resrouce ルートテーブル (パブリック)
# ---------------------------------------------
resource "aws_route_table" "public" {
vpc_id = aws_vpc.this.id
route {
cidr_block = "0.0.0.0/0"
gateway_id = aws_internet_gateway.this.id
}
tags = {
Name = "prd-foo-pub-rtb"
}
}
# ---------------------------------------------
# Resrouce ルートテーブル (プライベート)
# ---------------------------------------------
resource "aws_route_table" "private_app" {
for_each = var.vpc_availability_zones
vpc_id = aws_vpc.this.id
route {
cidr_block = "0.0.0.0/0"
nat_gateway_id = aws_nat_gateway.this[each.key].id
}
tags = {
Name = format(
"prd-foo-pvt-%s-app-rtb",
each.value
)
}
}
# ---------------------------------------------
# Resrouce AWS NAT Gateway
# ---------------------------------------------
resource "aws_nat_gateway" "this" {
for_each = var.vpc_availability_zones
subnet_id = aws_subnet.public[each.key].id
allocation_id = aws_eip.nat_gateway[each.key].id
tags = {
Name = format(
"prd-foo-%s-ngw",
each.value
)
}
depends_on = [aws_internet_gateway.this]
}
▼ 単一値でoutput
ブロック¶
resource
ブロックの作成にfor_each
引数を使用した場合、そのresource
ブロックはmap型として扱われる。
そのため、キー名を指定して出力できる。
# ---------------------------------------------
# Variables Global
# ---------------------------------------------
vpc_availability_zones = { a = "a", c = "c" }
# ---------------------------------------------
# AWS VPC
# ---------------------------------------------
output "public_a_subnet_id" {
value = aws_subnet.public[var.vpc_availability_zones.a].id
}
output "public_c_subnet_id" {
value = aws_subnet.public[var.vpc_availability_zones.c].id
}
▼ map型でoutput
ブロック¶
*実装例*
# ---------------------------------------------
# Variables Global
# ---------------------------------------------
vpc_availability_zones = { a = "a", c = "c" }
# ---------------------------------------------
# AWS VPC
# ---------------------------------------------
output "public_subnet_ids" {
value = {
a = aws_subnet.public[var.vpc_availability_zones.a].id,
c = aws_subnet.public[var.vpc_availability_zones.c].id
}
}
output "private_app_subnet_ids" {
value = {
a = aws_subnet.private_app[var.vpc_availability_zones.a].id,
c = aws_subnet.private_app[var.vpc_availability_zones.c].id
}
}
output "private_datastore_subnet_ids" {
value = {
a = aws_subnet.private_datastore[var.vpc_availability_zones.a].id,
c = aws_subnet.private_datastore[var.vpc_availability_zones.c].id
}
}
# ---------------------------------------------
# ALB
# ---------------------------------------------
resource "aws_lb" "this" {
name = "prd-foo-alb"
subnets = values(private_app_subnet_ids)
security_groups = [var.alb_security_group_id]
internal = false
idle_timeout = 120
enable_deletion_protection = true
access_logs {
enabled = true
bucket = var.alb_s3_bucket_id
}
}
dynamic
引数¶
▼ dynamic
引数とは¶
指定したブロックを繰り返し作成する。
▼ map型の場合¶
*実装例*
map型のキー名と値の両方を設定値として使用する。
例として、RDSパラメーターグループのparameter
ブロックを、map型通常変数を使用して繰り返し作成する。
# ---------------------------------------------
# Variables RDS
# ---------------------------------------------
rds_parameter_group_values = {
time_zone = "asia/tokyo"
character_set_client = "utf8mb4"
character_set_connection = "utf8mb4"
character_set_database = "utf8mb4"
character_set_results = "utf8mb4"
character_set_server = "utf8mb4"
server_audit_events = "connect,query,query_dcl,query_ddl,query_dml,table"
server_audit_logging = 1
server_audit_logs_upload = 1
general_log = 1
slow_query_log = 1
long_query_time = 3
}
# ---------------------------------------------
# AWS RDS Cluster Parameter Group
# ---------------------------------------------
resource "aws_rds_cluster_parameter_group" "this" {
name = "prd-foo-cluster-pg"
description = "The cluster parameter group for prd-foo-rds"
family = "aurora-mysql5.7"
dynamic "parameter" {
for_each = var.rds_parameter_group_values
content {
# parameterブロックのnameオプションとvalueオプションに出力する。
name = parameter.key
value = parameter.value
}
}
}
*実装例*
map型の値を設定値として使用する。
security_group_ingress_ec2_ssh = {
cidr_blocks = "*.*.*.*"
description = "SSH access from foo ip address"
from_port = 22
to_port = 22
protocol = "TCP"
}
# ---------------------------------------------
# AWS Security Group
# ---------------------------------------------
resource "aws_security_group" "ec2" {
...
dynamic ingress {
for_each = var.security_group_ingress_ec2_ssh
content {
cidr_blocks = [ ingress.value["cidr_blocks"] ]
description = ingress.value["description"]
from_port = ingress.value["from_port"]
to_port = ingress.value["to_port"]
protocol = ingress.value["protocol"]
}
}
...
}
▼ list型の場合¶
*実装例*
例として、WAFの正規表現パターンセットのregular_expression
ブロックを、list型通常変数を使用して繰り返し作成する。
# ---------------------------------------------
# Variables WAF
# ---------------------------------------------
waf_blocked_user_agents = [
"FooCrawler",
"BarSpider",
"BazBot",
]
# ---------------------------------------------
# WAF Regex Pattern Sets
# ---------------------------------------------
resource "aws_wafv2_regex_pattern_set" "cloudfront" {
name = "blocked-user-agents"
description = "Blocked user agents"
scope = "CLOUDFRONT"
dynamic "regular_expression" {
for_each = var.waf_blocked_user_agents
content {
# regex_stringブロックのregex_stringオプションに出力する。
regex_string = regular_expression.value
}
}
}
lifecycle
引数¶
▼ lifecycle
引数とは¶
resource
ブロックの作成、更新、そして削除のプロセスをカスタマイズする。
▼ create_before_destroy¶
resource
ブロックを新しく作成した後に削除するように、変更できる。
通常時、Terraformの処理順序として、resource
ブロックの削除後に作成が行われる。
しかし、他のリソースと依存関係が存在する場合、先に削除が行われることによって、他のリソースに影響が出てしまう。
これに対処するために、先に新しいresource
ブロックを作成し、紐付けし直してから、削除する必要がある。
*実装例*
例として、ACMのSSL証明書を示す。
ACMのSSL証明書は、ALBやCloudFrontに紐付いており、新しい証明書に紐付け直した後に、既存のものを削除する必要がある。
# ---------------------------------------------
# For foo domain
# ---------------------------------------------
resource "aws_acm_certificate" "foo" {
...
# 新しいSSL証明書を作成した後に削除する。
lifecycle {
create_before_destroy = true
}
}
*実装例*
例として、RDSのクラスターパラメーターグループとサブネットグループを示す。
クラスターパラメーターグループとサブネットグループは、DBクラスターに紐付いており、新しいクラスターパラメーターグループに紐付け直した後に、既存のものを削除する必要がある。
# ---------------------------------------------
# AWS RDS Cluster Parameter Group
# ---------------------------------------------
resource "aws_rds_cluster_parameter_group" "this" {
...
lifecycle {
create_before_destroy = true
}
}
# ---------------------------------------------
# AWS RDS Subnet Group
# ---------------------------------------------
resource "aws_db_subnet_group" "this" {
...
lifecycle {
create_before_destroy = true
}
}
*実装例*
例として、Redisのパラメーターグループとサブネットグループを示す。
ラメータグループとサブネットグループは、RDSに紐付いており、新しいパラメーターグループとサブネットグループに紐付け直した後に、既存のものを削除する必要がある。
# ---------------------------------------------
# AWS Redis Parameter Group
# ---------------------------------------------
resource "aws_elasticache_parameter_group" "redis" {
...
lifecycle {
create_before_destroy = true
}
}
# ---------------------------------------------
# AWS Redis Subnet Group
# ---------------------------------------------
resource "aws_elasticache_subnet_group" "redis" {
...
lifecycle {
create_before_destroy = true
}
}
▼ ignore_changes¶
実インフラのみで発生したresource
ブロックの作成・更新・削除を無視し、tfstate
ファイルに反映しないようにする。
これにより、ignore_changes
引数を定義したタイミング以降、実インフラとtfstate
ファイルに差分があっても、tfstate
ファイルの値が更新されなくなる。
ただし、リソースによってはignore_changes
引数を使えないものがある (例:SSMパラメーターストア) 。
*実装例*
例として、ECSを示す。
ECSでは、AWS AutoScalingによってECSタスク数が増加する。
そのため、これらを無視する必要がある。
# ---------------------------------------------
# AWS ECS Service
# ---------------------------------------------
resource "aws_ecs_service" "this" {
...
lifecycle {
ignore_changes = [
# AWS AutoScalingによるECSタスク数の増減を無視。
desired_count,
]
}
}
*実装例*
例として、Redisを示す。
Redisでは、AWS AutoScalingによってプライマリー数とレプリカ数が増減する。
そのため、これらを無視する必要がある。
# ---------------------------------------------
# AWS Redis Cluster
# ---------------------------------------------
resource "aws_elasticache_replication_group" "redis" {
...
lifecycle {
ignore_changes = [
# プライマリー数とレプリカ数の増減を無視します。
number_cache_clusters
]
}
}
*実装例*
使用例はすくないが、補足としてresource
ブロック全体を無視する場合はall
値を設定する。
resource "aws_foo" "foo" {
...
lifecycle {
ignore_changes = all
}
}
regexall
引数¶
▼ regexall
引数とは¶
正規表現ルールに基づいて、文字列の中から文字を抽出する。
これを応用して、特定の文字列を含む場合に条件を分岐させるようにできる。
security_group_ingress_ec2_ssh = {
prd = {
cidr_blocks = "*.*.*.*"
description = "SSH access from foo ip address"
from_port = 22
to_port = 22
protocol = "TCP"
}
stg = {
cidr_blocks = "*.*.*.*"
description = "SSH access from foo ip address"
from_port = 22
to_port = 22
protocol = "TCP"
}
}
resource "aws_security_group" "ec2" {
...
dynamic ingress {
# 環境が複数あるとする (prd-1、prd-2、stg-1、stg-2) 。
# 環境名がprdという文字を含むキーがあった場合、全てprdキーの方を使用する。
for_each = length(regexall("prd", var.environment)) > 0 ? var.security_group_ingress_ec2_ssh.prd : var.security_group_ingress_ec2_ssh.stg
content {
cidr_blocks = [ ingress.value["cidr_blocks"] ]
description = ingress.value["description"]
from_port = ingress.value["from_port"]
to_port = ingress.value["to_port"]
protocol = ingress.value["protocol"]
}
}
...
}
07. tpl形式の切り出しと読み出し¶
templatefile
関数¶
▼ templatefile
関数とは¶
第一引数でポリシーが定義されたファイルを読み出し、第二引数でファイルに変数を渡す。
ファイルの拡張子はtplとするのが良い。
*実装例*
例として、AWS S3を示す。
# ---------------------------------------------
# AWS AWS S3 bucket policy
# ---------------------------------------------
resource "aws_s3_bucket_policy" "alb" {
bucket = aws_s3_bucket.alb_logs.id
policy = templatefile(
"${path.module}/policies/alb_bucket_policy.tpl",
{
aws_elb_service_account_arn = var.aws_elb_service_account_arn
aws_s3_bucket_alb_logs_arn = aws_s3_bucket.alb_logs.arn
}
)
}
バケットポリシーを定義するtpl形式ファイルでは、string型の場合は"${}"
で、integer型の場合は${}
で変数を展開する。
ここで拡張子をjsonにしてしまうと、integer型の出力をjsonの構文エラーとして扱われてしまう。
{
"Version": "2012-10-17",
"Statement":
[
{
"Effect": "Allow",
"Principal": {"AWS": "${aws_elb_service_account_arn}/*"},
"Action": "s3:PutObject",
"Resource": "${aws_s3_bucket_alb_logs_arn}/*",
},
],
}
▼ path式¶
変数 | 値 | 例 |
---|---|---|
path.module |
path 式が実行された.tf ファイルがあるディレクトリのパス。 |
/project/module/foo/ |
path.root |
terraform コマンドの作業ディレクトリのパス |
/var/www/ |
path.root |
module ディレクトリのルートパス |
/project/module/ |
terraform.workplace |
現在使用しているワークスペース名 | prd |
ポリシーの紐付け¶
containerDefinitionsの設定¶
▼ containerDefinitionsとは¶
ECSタスク定義のうち、コンテナを定義する部分のこと。
*実装例*
{
"ipcMode": null,
"executionRoleArn": "<ecsTaskExecutionRoleのARN>",
"containerDefinitions": [],
...,
}
▼ 設定方法¶
integer型を通常変数として渡せるように、拡張子をjsonではなくtplとするのが良い。
image
キーでは、ECRイメージのURLを設定する。
バージョンタグは任意で指定でき、もし指定しない場合は、『latest
』という名前のタグが自動的に割り当てられる。
バージョンタグにハッシュ値が割り当てられている場合、Terraformでは時系列で最新のタグ名を取得する方法がないため、secrets
キーでは、パラメーターストアの値を参照できる。
ログ分割の目印を設定するawslogs-datetime-format
キーでは、タイムスタンプを表す\\[%Y-%m-%d %H:%M:%S\\]
を設定すると良い。
これにより、同じ時間に発生したログを1つのログとしてまとめられるため、スタックトレースが見やすくなる。
*実装例*
[
{
# コンテナ名
"name": "laravel",
# ECRのURL。タグを指定しない場合はlatestが割り当てられる。
"image": "${laravel_ecr_repository_url}",
"essential": "true",
"portMappings": [
# AWS ECSのホストとコンテナのポートマッピング
{"containerPort": 80, "hostPort": 80, "protocol": "tcp"},
],
"secrets": [
{
# アプリケーションの環境変数名
"name": "DB_HOST",
# AWS Systems Managerのパラメーター名
"valueFrom": "/prd-foo/DB_HOST",
},
{"name": "DB_DATABASE", "valueFrom": "/prd-foo/DB_DATABASE"},
{"name": "DB_PASSWORD", "valueFrom": "/prd-foo/DB_PASSWORD"},
{"name": "DB_USERNAME", "valueFrom": "/prd-foo/DB_USERNAME"},
{"name": "REDIS_HOST", "valueFrom": "/prd-foo/REDIS_HOST"},
{"name": "REDIS_PASSWORD", "valueFrom": "/prd-foo/REDIS_PASSWORD"},
{"name": "REDIS_PORT", "valueFrom": "/prd-foo/REDIS_PORT"},
],
"logConfiguration": {
# ログドライバー
"logDriver": "awslogs",
"options": {
# ロググループ名
"awslogs-group": "/prd-foo/laravel/log",
# スタックトレースのグループ化 (同時刻ログのグループ化)
"awslogs-datetime-format": "\\[%Y-%m-%d %H:%M:%S\\]",
# リージョン
"awslogs-region": "ap-northeast-1",
# ログストリーム名の接頭辞
"awslogs-stream-prefix": "/container",
},
},
},
]
08. 関数¶
jsonencode¶
Terraformのデータ型をJSON文字列型に変換する。