Provisioner 命令:“shell”
示例:
node.vm.provision "shell" do |s|
s.inline = <<-SHELL
# 更改时区
cp /usr/share/zoneinfo/Asia/Shanghai /etc/localtime
timedatectl set-timezone Asia/Shanghai
SHELL
通过 Vagrant Shell 配置程序可以在虚拟机中上传和执行脚本。
Shell 为不熟悉 Chef 或 Puppet 等完整配置管理系统的用户提供了强大的选择。
1. 选项
shell 配置程序支持两个选项:
- inline(字符串)- 在行内指定远程机器要执行的 shell 命令。更多信息参考第二小节 行内脚本。
- path(字符串)- 要上传并执行的脚本的路径。可以是跟项目的 Vagrantfile 相关的脚本,也可以是远程脚本(例如 gist)。
剩下的可用选项是可选的:
- args(字符串或数组) - 作为单个字符串执行时传递给shell脚本的参数。必须将这些参数写入,就好像它们是直接在命令行中键入的一样,所以必须根据需要转义字符,引号等。也可以使用数组传递参数,此时,Vagrant 会自动处理引用。
- env(hash 散列) - 作为环境变量传递给脚本的键值对的列表。Vagrant 将处理环境变量值的引用,但键保持不变。
- binary(boolean) - 自动用 Unix 行结束符替换 Windows 行结束符。如果设置为 false,则 Vagrant 不会这样做。默认情况下是“false”。如果 shell 配置程序通过 WinRM 进行通信,则默认为“true”。
- privileged(boolean) - 指定是否以特权用户身份执行 shell 脚本(sudo)。默认情况下这是“true”。Windows 虚拟机使用计划任务作为真正的管理员运行,而不受 WinRM 限制。
- upload_path(字符串) - shell 脚本将被上传到的远程路径。该脚本通过 SSH 用户上载到 SCP 上,因此该位置必须对该用户可写。默认情况下是“/tmp/vagrant-shell”。
- keep_color(boolean) - 根据输出是来自标准输出 stdout 还是标准错误 stderr,Vagrant 自动以绿色和红色输出。如果设为 true,Vagrant 不会设置颜色,以允许脚本中的原生色彩输出。
- name (字符串) - 此值将显示在输出中,以便在存在多个 shell 提供程序时,用户更容易识别。
- powershell_args(字符串) - 如果在 Windows 上使用 PowerShell 进行配置管理,则可以通过这个参数传递给 PowerShell 额外参数。
- powershell_elevated_interactive(boolean) - 在 Windows 上以交互模式运行提升的脚本。默认情况下这是“false”。需要同时开启 privileged 参数,这个参数才会有效。一定要启用 Windows 的自动登录,因为用户必须登录才能使用交互模式。
- md5(字符串) - 用于验证远程下载的 shell 文件的 MD5 校验和。
- sha1(字符串) - 用于验证远程下载的 shell 文件的 SHA1 校验和。
- sensitive(boolean) - 将 env 选项中使用的哈希值标记为敏感,并将其隐藏在输出中。默认情况下是“false”。
2. 行内脚本(inline script)
对于单行脚本,可以直接在 Vagrantfile 中指定:
Vagrant.configure("2") do |config|
config.vm.provision "shell",
inline: "echo Hello, World"
end
当虚拟机内的配置程序运行时,会打印 Hello, World。
对于多行脚本,需要标明脚本边界:
Vagrant.configure("2") do |config|
node.vm.provision "shell" do |s|
s.inline = <<-SHELL
# 更改时区
cp /usr/share/zoneinfo/Asia/Shanghai /etc/localtime
timedatectl set-timezone Asia/Shanghai
echo 'sync time'
SHELL
end
end
结合一点 Ruby,这使得将多行 shell 脚本直接嵌入到 Vagrantfile 中变得非常简单:
$script = <<-SCRIPT
echo I am provisioning...
date > /etc/vagrant_provisioned_at
SCRIPT
Vagrant.configure("2") do |config|
config.vm.provision "shell", inline: $script
end
上面的内容将脚本分配给全局变量 $script
。这个全局变量包含一个字符串,然后作为内联脚本传递给 Vagrant 配置。
3. 外部脚本
shell 配置程序可以通过路径使用宿主机的脚本。Vagrant 会上传这个脚本到虚拟机并执行。例如:
Vagrant.configure("2") do |config|
config.vm.provision "shell", path: "script.sh"
end
上面例子中使用的是相对路径,相对于项目的根 Vagrantfile 位置。也可以使用绝对路径,同时还支持 ~
家目录和 ..
父目录。
也可以在 path 参数中通过 URL 来使用远程脚本:
Vagrant.configure("2") do |config|
config.vm.provision "shell", path: "https://example.com/provisioner.sh"
end
要运行虚拟机上已有的脚本,可以使用内联脚本来调用虚拟机上的远程脚本:
Vagrant.configure("2") do |config|
config.vm.provision "shell",
inline: "/bin/sh /path/to/the/script/already/on/the/guest.sh"
end
4. 脚本参数
可以像任何普通的 shell 脚本一样参数化脚本。这些参数可以指定给 shell 配置程序。参数应该指定为字符串,因为它们将在命令行中键入,因此请确保正确地转义任何内容:
Vagrant.configure("2") do |config|
config.vm.provision "shell" do |s|
s.inline = "echo $1"
s.args = "'hello, world!'"
end
end
如果不想担心引用的问题,可以将参数指定为数组:
Vagrant.configure("2") do |config|
config.vm.provision "shell" do |s|
s.inline = "echo $1"
s.args = ["hello, world!"]
end
end