Terraform – Provisioner [远端或本地执行器]

官方文档:https://developer.hashicorp.com/terraform/language/resources/provisioners/syntax

Provisioner 的基本概念

Provisioner 是 Terraform 用来在资源创建或销毁时执行额外操作的一种机制。它可以在资源创建成功后(或销毁前)执行脚本或命令,用于:

  • 初始化系统:例如安装软件、配置系统参数或上传文件;
  • 后续处理:例如在销毁资源前进行数据备份或清理操作。

file

用于将本地文件复制到目标机器

local-exec

当terraform创建了EC2之后把vm的ip写入本地机的txt文件内

remote-exec

当terraform创建了EC2之后在vm当中安装软件,必须nginx等等

使用案例

使用terraform获取最新版本的AMI,然后创建一台获取EC2。接着使用ssh登入EC2安装nginx,最后再把EC2的public ip写入txt文档内。

# 配置 AWS 提供商,指定区域为亚太地区 (ap-southeast-5)
provider "aws" {
  region = "ap-southeast-5"
}

# 获取最新版本的 Amazon Linux 2023 AMI
# 使用 data 数据源查询最新版本的 Amazon Linux AMI,其中:
# - most_recent = true 表示获取最新的镜像
# - owners 指定镜像拥有者为 Amazon
# - 第一个 filter 根据架构筛选为 x86_64
# - 第二个 filter 根据镜像名称筛选符合 "al2023-ami-2023*" 模式的镜像
data "aws_ami" "latest_amazon_linux" {
  most_recent = true
  owners      = ["amazon"]

  filter {
    name   = "architecture"
    values = ["x86_64"]
  }
  filter {
    name   = "name"  // 找出最新版本的 Amazon Linux 2023 AMI
    values = ["al2023-ami-2023*"]
  }
}

# 创建一个安全组,允许 HTTP (80)、HTTPS (443) 和 SSH (22) 访问
resource "aws_security_group" "web_sg" {
  name        = "web-sg"
  description = "https , http, ssh"
  vpc_id      = "vpc-0a85beb97e0734dfa"  # 请替换为实际的 VPC ID

  # 允许 HTTP 访问(80端口)
  ingress {
    description = "HTTP"
    from_port   = 80
    to_port     = 80
    protocol    = "tcp"
    cidr_blocks = ["0.0.0.0/0"]
  }

  # 允许 HTTPS 访问(443端口)
  ingress {
    description = "HTTPS"
    from_port   = 443
    to_port     = 443
    protocol    = "tcp"
    cidr_blocks = ["0.0.0.0/0"]
  }

  # 允许 SSH 访问(22端口)
  ingress {
    description = "SSH"
    from_port   = 22
    to_port     = 22
    protocol    = "tcp"
    cidr_blocks = ["0.0.0.0/0"]
  }

  # 配置所有流量的出口规则
  egress {
    from_port   = 0
    to_port     = 0
    protocol    = "-1"
    cidr_blocks = ["0.0.0.0/0"]
  }
}

# 创建一个 EC2 实例,并使用上面查询到的最新 Amazon Linux AMI
resource "aws_instance" "ec2" {
  ami           = data.aws_ami.latest_amazon_linux.id
  instance_type = "t3.micro"
  key_name      = "test"  # 指定 SSH 密钥名称,请确保已在 AWS 中创建
  subnet_id     = "subnet-08a62e0e1f76c78d9"  # 请替换为实际的子网 ID
  security_groups = [aws_security_group.web_sg.id]

  # 配置 SSH 连接参数,用于后续的远程执行操作
  connection {
    type        = "ssh"
    user        = "ec2-user"
    private_key = file("${path.module}/test.pem")
    host        = self.public_ip
  }

  # 远程执行器,在实例中安装并启动 nginx
  provisioner "remote-exec" {
    inline = [
      "sudo yum install -y nginx",
      "sudo systemctl start nginx"
    ]
  }

  # 本地执行器,将 EC2 实例的公网 IP 写入 public_ip.txt 文件
  provisioner "local-exec" {
    command = "echo ${self.public_ip} >> public_ip.txt"
  }
}

输出结果

aws_security_group.web_sg: Creating...
aws_security_group.web_sg: Creation complete after 1s [id=sg-0da0b1d044deefafb]
aws_instance.ec2: Creating...
aws_instance.ec2: Still creating... [10s elapsed]
aws_instance.ec2: Provisioning with 'remote-exec'...
aws_instance.ec2 (remote-exec): Connecting to remote host via SSH...
aws_instance.ec2 (remote-exec):   Host: 43.216.199.232
aws_instance.ec2 (remote-exec):   User: ec2-user
aws_instance.ec2 (remote-exec):   Password: false
aws_instance.ec2 (remote-exec):   Private key: true
aws_instance.ec2 (remote-exec):   Certificate: false
aws_instance.ec2 (remote-exec):   SSH Agent: true
aws_instance.ec2 (remote-exec):   Checking Host Key: false
aws_instance.ec2 (remote-exec):   Target Platform: unix
aws_instance.ec2 (remote-exec): Connecting to remote host via SSH...
aws_instance.ec2 (remote-exec):   Host: 43.216.199.232
aws_instance.ec2 (remote-exec):   User: ec2-user
aws_instance.ec2 (remote-exec):   Password: false
aws_instance.ec2 (remote-exec):   Private key: true
aws_instance.ec2 (remote-exec):   Certificate: false
aws_instance.ec2 (remote-exec):   SSH Agent: true
aws_instance.ec2 (remote-exec):   Checking Host Key: false
aws_instance.ec2 (remote-exec):   Target Platform: unix
aws_instance.ec2 (remote-exec): Connecting to remote host via SSH...
aws_instance.ec2 (remote-exec):   Host: 43.216.199.232
aws_instance.ec2 (remote-exec):   User: ec2-user
aws_instance.ec2 (remote-exec):   Password: false
aws_instance.ec2 (remote-exec):   Private key: true
aws_instance.ec2 (remote-exec):   Certificate: false
aws_instance.ec2 (remote-exec):   SSH Agent: true
aws_instance.ec2 (remote-exec):   Checking Host Key: false
aws_instance.ec2 (remote-exec):   Target Platform: unix
aws_instance.ec2 (remote-exec): Connected!
aws_instance.ec2 (remote-exec): Amazon  ---  B/s |   0  B     --:-- ETA
aws_instance.ec2 (remote-exec): Amazon  ---  B/s |   0  B     --:-- ETA
aws_instance.ec2 (remote-exec): Amazon  ---  B/s |   0  B     --:-- ETA
aws_instance.ec2: Still creating... [20s elapsed]
aws_instance.ec2 (remote-exec): Amazon   25 MB/s |  33 MB     00:01
aws_instance.ec2: Still creating... [30s elapsed]
aws_instance.ec2 (remote-exec): Amazon  ---  B/s |   0  B     --:-- ETA
aws_instance.ec2 (remote-exec): Amazon  ---  B/s |   0  B     --:-- ETA
aws_instance.ec2 (remote-exec): Amazon  ---  B/s |   0  B     --:-- ETA
aws_instance.ec2 (remote-exec): Amazon   17 kB/s |  14 kB     00:00
aws_instance.ec2 (remote-exec): Last metadata expiration check: 0:00:01 ago on Thu Mar 20 15:39:19 2025.
aws_instance.ec2: Still creating... [40s elapsed]
aws_instance.ec2 (remote-exec): Dependencies resolved.
aws_instance.ec2 (remote-exec): ========================================
aws_instance.ec2 (remote-exec):  Package    Arch   Version
aws_instance.ec2 (remote-exec):                       Repository   Size
aws_instance.ec2 (remote-exec): ========================================
aws_instance.ec2 (remote-exec): Installing:
aws_instance.ec2 (remote-exec):  nginx      x86_64 1:1.26.3-1.amzn2023.0.1
aws_instance.ec2 (remote-exec):                       amazonlinux  33 k
aws_instance.ec2 (remote-exec): Installing dependencies:
aws_instance.ec2 (remote-exec):  generic-logos-httpd
aws_instance.ec2 (remote-exec):             noarch 18.0.0-12.amzn2023.0.3
aws_instance.ec2 (remote-exec):                       amazonlinux  19 k
aws_instance.ec2 (remote-exec):  gperftools-libs
aws_instance.ec2 (remote-exec):             x86_64 2.9.1-1.amzn2023.0.3
aws_instance.ec2 (remote-exec):                       amazonlinux 308 k
aws_instance.ec2 (remote-exec):  libunwind  x86_64 1.4.0-5.amzn2023.0.2
aws_instance.ec2 (remote-exec):                       amazonlinux  66 k
aws_instance.ec2 (remote-exec):  nginx-core x86_64 1:1.26.3-1.amzn2023.0.1
aws_instance.ec2 (remote-exec):                       amazonlinux 670 k
aws_instance.ec2 (remote-exec):  nginx-filesystem
aws_instance.ec2 (remote-exec):             noarch 1:1.26.3-1.amzn2023.0.1
aws_instance.ec2 (remote-exec):                       amazonlinux 9.6 k
aws_instance.ec2 (remote-exec):  nginx-mimetypes
aws_instance.ec2 (remote-exec):             noarch 2.1.49-3.amzn2023.0.3
aws_instance.ec2 (remote-exec):                       amazonlinux  21 k

aws_instance.ec2 (remote-exec): Transaction Summary
aws_instance.ec2 (remote-exec): ========================================
aws_instance.ec2 (remote-exec): Install  7 Packages

aws_instance.ec2 (remote-exec): Total download size: 1.1 M
aws_instance.ec2 (remote-exec): Installed size: 3.6 M
aws_instance.ec2 (remote-exec): Downloading Packages:
aws_instance.ec2 (remote-exec): Amazon  ---  B/s |   0  B     --:-- ETA
aws_instance.ec2 (remote-exec): (1/7):  ---  B/s |   0  B     --:-- ETA
aws_instance.ec2 (remote-exec): (1/7):   70 kB/s |  19 kB     00:00
aws_instance.ec2 (remote-exec): (2-3/7)  70 kB/s |  19 kB     00:15 ETA
aws_instance.ec2 (remote-exec): (2/7):  234 kB/s |  66 kB     00:00
aws_instance.ec2 (remote-exec): (3-4/7)  83 kB/s |  85 kB     00:12 ETA
aws_instance.ec2 (remote-exec): (3/7):  1.0 MB/s | 308 kB     00:00
aws_instance.ec2 (remote-exec): (4-5/7) 149 kB/s | 419 kB     00:04 ETA
aws_instance.ec2 (remote-exec): (4/7):  951 kB/s |  33 kB     00:00
aws_instance.ec2 (remote-exec): (5-6/7) 150 kB/s | 426 kB     00:04 ETA
aws_instance.ec2 (remote-exec): (5/7):  497 kB/s | 9.6 kB     00:00
aws_instance.ec2 (remote-exec): (6-7/7) 152 kB/s | 435 kB     00:04 ETA
aws_instance.ec2 (remote-exec): (6/7):   11 MB/s | 670 kB     00:00
aws_instance.ec2 (remote-exec): (7/7):  289 kB/s | 1.1 MB     00:00 ETA
aws_instance.ec2 (remote-exec): (7/7):  558 kB/s |  21 kB     00:00
aws_instance.ec2 (remote-exec): ----------------------------------------
aws_instance.ec2 (remote-exec): Total   1.7 MB/s | 1.1 MB     00:00
aws_instance.ec2 (remote-exec): Running transaction check
aws_instance.ec2 (remote-exec): Transaction check succeeded.
aws_instance.ec2 (remote-exec): Running transaction test
aws_instance.ec2 (remote-exec): Transaction test succeeded.
aws_instance.ec2 (remote-exec): Running transaction
aws_instance.ec2 (remote-exec):   Preparing        :  [=          ] 1/1
aws_instance.ec2 (remote-exec):   Preparing        :  [===        ] 1/1
aws_instance.ec2 (remote-exec):   Preparing        :  [====       ] 1/1
aws_instance.ec2 (remote-exec):   Preparing        :  [======     ] 1/1
aws_instance.ec2 (remote-exec):   Preparing        :  [=======    ] 1/1
aws_instance.ec2 (remote-exec):   Preparing        :  [=========  ] 1/1
aws_instance.ec2 (remote-exec):   Preparing        :                1/1
aws_instance.ec2 (remote-exec):   Running scriptlet: nginx-filesy   1/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [      ] 1/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [=     ] 1/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [==    ] 1/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [===   ] 1/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [====  ] 1/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [===== ] 1/7
aws_instance.ec2 (remote-exec):   Installing       : nginx-filesy   1/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [      ] 2/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [====  ] 2/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [===== ] 2/7
aws_instance.ec2 (remote-exec):   Installing       : nginx-mimety   2/7
aws_instance.ec2 (remote-exec):   Installing       : libun [      ] 3/7
aws_instance.ec2 (remote-exec):   Installing       : libun [=     ] 3/7
aws_instance.ec2 (remote-exec):   Installing       : libun [===   ] 3/7
aws_instance.ec2 (remote-exec):   Installing       : libun [====  ] 3/7
aws_instance.ec2 (remote-exec):   Installing       : libun [===== ] 3/7
aws_instance.ec2 (remote-exec):   Installing       : libunwind-1.   3/7
aws_instance.ec2 (remote-exec):   Installing       : gperf [      ] 4/7
aws_instance.ec2 (remote-exec):   Installing       : gperf [=     ] 4/7
aws_instance.ec2 (remote-exec):   Installing       : gperf [==    ] 4/7
aws_instance.ec2 (remote-exec):   Installing       : gperf [===   ] 4/7
aws_instance.ec2 (remote-exec):   Installing       : gperf [====  ] 4/7
aws_instance.ec2 (remote-exec):   Installing       : gperf [===== ] 4/7
aws_instance.ec2 (remote-exec):   Installing       : gperftools-l   4/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [      ] 5/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [=     ] 5/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [==    ] 5/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [===   ] 5/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [====  ] 5/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [===== ] 5/7
aws_instance.ec2 (remote-exec):   Installing       : nginx-core-1   5/7
aws_instance.ec2 (remote-exec):   Installing       : gener [      ] 6/7
aws_instance.ec2 (remote-exec):   Installing       : gener [====  ] 6/7
aws_instance.ec2 (remote-exec):   Installing       : gener [===== ] 6/7
aws_instance.ec2 (remote-exec):   Installing       : generic-logo   6/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [      ] 7/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [==    ] 7/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [===   ] 7/7
aws_instance.ec2 (remote-exec):   Installing       : nginx [===== ] 7/7
aws_instance.ec2 (remote-exec):   Installing       : nginx-1:1.26   7/7
aws_instance.ec2 (remote-exec):   Running scriptlet: nginx-1:1.26   7/7
aws_instance.ec2 (remote-exec):   Verifying        : generic-logo   1/7
aws_instance.ec2 (remote-exec):   Verifying        : gperftools-l   2/7
aws_instance.ec2 (remote-exec):   Verifying        : libunwind-1.   3/7
aws_instance.ec2 (remote-exec):   Verifying        : nginx-1:1.26   4/7
aws_instance.ec2 (remote-exec):   Verifying        : nginx-core-1   5/7
aws_instance.ec2 (remote-exec):   Verifying        : nginx-filesy   6/7
aws_instance.ec2 (remote-exec):   Verifying        : nginx-mimety   7/7

aws_instance.ec2 (remote-exec): Installed:
aws_instance.ec2 (remote-exec):   generic-logos-httpd-18.0.0-12.amzn2023.0.3.noarch
aws_instance.ec2 (remote-exec):   gperftools-libs-2.9.1-1.amzn2023.0.3.x86_64
aws_instance.ec2 (remote-exec):   libunwind-1.4.0-5.amzn2023.0.2.x86_64
aws_instance.ec2 (remote-exec):   nginx-1:1.26.3-1.amzn2023.0.1.x86_64
aws_instance.ec2 (remote-exec):   nginx-core-1:1.26.3-1.amzn2023.0.1.x86_64
aws_instance.ec2 (remote-exec):   nginx-filesystem-1:1.26.3-1.amzn2023.0.1.noarch
aws_instance.ec2 (remote-exec):   nginx-mimetypes-2.1.49-3.amzn2023.0.3.noarch

aws_instance.ec2 (remote-exec): Complete!
aws_instance.ec2: Provisioning with 'local-exec'...
aws_instance.ec2 (local-exec): Executing: ["/bin/sh" "-c" "echo 43.216.199.232 >> public_ip.txt"]
aws_instance.ec2: Creation complete after 44s [id=i-0fa059b345ed3d9d5]

浏览public ip就能打开网站

当销毁资源之后执行local-exec

需要在local-exec当中加入when = destroy即可

resource "aws_iam_user" "ss" {
  name = "test"

  provisioner "local-exec" {
    when = destroy
    command = "echo this is destroy"
  }
}

执行命令有错误应该怎么办?

执行的命令行有错误的话,在terraform state 是taint的状态,还未完整。IAM用户是创建了,但是卡在中间的还未完成的状态,所以再次apply的话会进行销毁了再尝试重新创建过。

resource "aws_iam_user" "ss" {
  name = "test"
  provisioner "local-exec" {
    command = "sssd"
  }
}

执行命令有错误,但是想继续执行应该怎么办?

想继续执行的话可以加上on_failure = continue , 那么就算命令行有错误都能完成执行terraform

resource "aws_iam_user" "ss" {
  name = "test"
  provisioner "local-exec" {
    command = "sssd"
    on_failure = continue
  }
}

Loading

Facebook评论