vmware.vmware_rest.vcenter_vm_guest_customization 模块 – 在虚拟机上应用自定义规范

注意

此模块是 vmware.vmware_rest 集合 (版本 4.3.0) 的一部分。

如果您使用的是 ansible 包,则可能已安装此集合。它不包含在 ansible-core 中。要检查它是否已安装,请运行 ansible-galaxy collection list

要安装它,请使用: ansible-galaxy collection install vmware.vmware_rest。您需要其他要求才能使用此模块,有关详细信息,请参阅 要求

要在 playbook 中使用它,请指定: vmware.vmware_rest.vcenter_vm_guest_customization

vmware.vmware_rest 0.1.0 中的新增功能

概要

  • {@param.name vm} 中的虚拟机上应用自定义规范。虚拟机启动时,实际的自定义会在客户机内部进行。如果虚拟机存在挂起的自定义设置,并且设置了新的自定义设置,则现有自定义设置将被新的设置覆盖。

要求

执行此模块的主机需要以下要求。

  • vSphere 7.0.3 或更高版本

  • python >= 3.6

  • aiohttp

参数

参数

注释

configuration_spec

字典 / 必需

要在自定义期间应用于客户机的设置。此参数是必需的。

有效属性为

  • windows_config (dict): Windows 客户机操作系统的客户机自定义规范

如果未设置,则必须设置 ConfigurationSpec.linux-config 或 ConfigurationSpec.cloud-config。否则,将抛出相应的错误。([‘set’])

  • 可接受的键

  • reboot (string): 此选项指定自定义后应对客户机执行的操作。

此字段的可接受值

  • NO_REBOOT

  • REBOOT

  • SHUTDOWN

  • sysprep (object): Windows 客户机操作系统的自定义设置,例如用户详细信息、管理员详细信息等。必须指定 WindowsConfiguration.sysprep 或 WindowsConfiguration.sysprep-xml 中的一个。

如果未设置,则不会将 sysprep 设置应用于 Windows 客户机操作系统。

如果未设置,则不会将 sysprep 设置应用于 Windows 客户机操作系统。

  • linux_config (dict): Linux 客户机操作系统的客户机自定义规范

如果未设置,则必须设置 ConfigurationSpec.windows-config 或 ConfigurationSpec.cloud-config。否则,将抛出相应的错误。([‘set’])

  • 可接受的键

  • hostname (object): (Windows) 虚拟机的计算机名。计算机名可以包含字母 (A-Z)、数字 (0-9) 和连字符 (-),但不能包含空格或句点 (.)。名称不能完全由数字组成。计算机名的长度限制为 15 个字符。如果计算机名超过 15 个字符,则将被截断为 15 个字符。请查看 HostnameGenerator 以了解各种选项。

  • domain (string): 完全限定域名。

  • time_zone (string): 区分大小写的时区,例如 Europe/Sofia。有效的时区值基于 Linux 使用的 tz (时区) 数据库。这些值是“区域/位置”形式的字符串 (string),其中区域是大陆或海洋名称,位置是城市、岛屿或其他区域名称。

有关 Linux 中不同版本的受支持时区的列表,请参阅 https://kb.vmware.com/kb/2145518

如果未设置,则不会修改客户机操作系统中的时区。

  • script_text (string): 在 Linux 客户机自定义之前和之后运行的脚本。

脚本的最大大小为 65536 字节。只要脚本 (shell、perl、python…) 在标题中具有正确的“#!”,它就受支持。当运行脚本时,调用者不应假定任何环境变量。

自定义引擎使用命令行调用脚本:1) 在自定义之前使用参数“precustomization”,2) 在自定义之后使用参数“postcustomization”。脚本应解析此参数并实现相应块中预自定义或后自定义任务代码的详细信息。

一个 Linux shell 脚本示例

#!/bin/sh

if [ x$1 == x”precustomization” ]; then

echo “执行预自定义任务”

#预自定义操作代码…

elif [ x$1 == x”postcustomization” ]; then

echo “执行后自定义任务”

#后自定义操作代码…

fi

如果未设置,则不会执行任何脚本。

  • cloud_config (dict): 包含云配置的客户机自定义规范。

如果未设置,则必须设置 ConfigurationSpec.windows-config 或 ConfigurationSpec.linux-config。否则,将抛出相应的错误。([‘set’])

  • 可接受的键

  • type (字符串): 此选项指定云配置的不同类型。

此字段的可接受值

  • CLOUDINIT

  • cloudinit (对象): cloud-init 配置

此字段是可选的,只有当 CloudConfiguration.type 的值为 CLOUDINIT 时才相关。

global_DNS_settings

字典 / 必需

全局 DNS 设置构成不特定于特定虚拟网络适配器的 DNS 设置。此参数是必需的。

有效属性为

  • dns_suffix_list (列表): 虚拟网络适配器的名称解析后缀列表。此列表适用于 Windows 和 Linux 客机自定义。对于 Linux,此设置是全局的,而在 Windows 中,此设置是在每个适配器基础上列出的。

如果未设置,则不设置 DNS 后缀。([‘set’])

  • dns_servers (列表): 具有静态 IP 地址的虚拟网络适配器的 DNS 服务器列表。如果此列表为空,则预期客户机操作系统使用 DHCP 服务器获取其 DNS 服务器设置。这些设置配置虚拟机以使用指定的 DNS 服务器。这些 DNS 服务器设置按优先级顺序列出。

如果未设置,则不设置 DNS 服务器。([‘set’])

interfaces

列表 / 元素=字典 / 必需

特定于特定虚拟网络适配器的 IP 设置。AdapterMapping 结构将网络适配器的 MAC 地址映射到其 IPSettings。如果没有任何网络适配器,则可以为空,否则应与为虚拟机配置的网络适配器数量匹配。此参数是必需的。

有效属性为

  • mac_address (字符串): 正在自定义的网络适配器的 MAC 地址。

在 vSphere 7.0 系列中,必须按 pciSlotNumber 的升序指定 MAC 地址,否则将报告 MAC 地址不匹配错误。更多详细信息,请参见 https://kb.vmware.com/s/article/87648

如果未设置,则自定义过程将 CustomizationSpec.interfaces 中的 *i_p_settings* 列表中的设置映射到虚拟机的网络适配器,按 PCI 插槽顺序排列。PCI 总线上的第一个虚拟网络适配器分配 interfaces[0].IPSettings,第二个适配器分配 interfaces[1].IPSettings,依此类推。([‘set’])

  • adapter (字典): 关联虚拟网络适配器的 IP 设置。([‘set’])

此键与 [‘set’] 一起是必需的。

  • 可接受的键

  • ipv4 (对象): 指定为此虚拟网络适配器配置 IPv4 地址、子网掩码和网关信息。

如果未设置,则不设置 IPv4 地址。

  • ipv6 (对象): 指定为此虚拟网络适配器配置 IPv6 地址、子网掩码和网关信息。

如果未设置,则不设置 IPv6 地址。

  • windows (对象): 为此特定的虚拟网络适配器配置的 Windows 设置。这只对 Windows 客户机操作系统有效。

如果未设置,则不设置任何特定的 Windows 设置。

session_timeout

浮点数

在 vmware.vmware_rest 2.1.0 中添加

客户端会话的超时设置。

整个操作(包括连接建立、请求发送和响应)的最大秒数。

默认值为 300 秒。

state

字符串

选项

  • "set" ← (默认)

vcenter_hostname

字符串 / 必需

vSphere vCenter 的主机名或 IP 地址

如果任务中未指定此值,则将改用环境变量 VMWARE_HOST 的值。

vcenter_password

字符串 / 必需

vSphere vCenter 密码

如果任务中未指定此值,则将改用环境变量 VMWARE_PASSWORD 的值。

vcenter_rest_log_file

字符串

您可以使用此可选参数来设置日志文件的位置。

此文件将用于记录 HTTP REST 交互。

该文件将存储在运行模块的主机上。

如果任务中未指定此值,则将改用

环境变量 VMWARE_REST_LOG_FILE 的值。

vcenter_username

字符串 / 必需

vSphere vCenter 用户名

如果任务中未指定此值,则将改用环境变量 VMWARE_USER 的值。

vcenter_validate_certs

布尔值

允许在 SSL 证书无效时连接。当证书不受信任时,将其设置为 false

如果任务中未指定此值,则将改用环境变量 VMWARE_VALIDATE_CERTS 的值。

选项

  • false

  • true ← (默认)

vm

字符串 / 必需

需要自定义的虚拟机的唯一标识符。

此参数必须是 vmware.vmware_rest.vcenter_vm_info 返回的资源的 ID。此参数是必需的。

备注

注意

  • 在 vSphere 7.0.3 上测试

示例

##########
#
# VM customization can be difficult to troubleshoot, since each environment is different. Here are some general tips:
#
# 1. Make sure perl is installed on the Linux systems. Make sure cloud-init is installed if using cloud-init
# 2. Custom script execution is disabled by default. To enable it, you can run as root: vmware-toolbox-cmd config set deployPkg enable-custom-scripts  true
# 3. VMware tools must be installed and recognized by vCenter before you can apply customization. See the example below for one approach to this.
# 4. On Linux (RHEL specifically), customization script logs can be found at /var/log/vmware-imc/toolsDeployPkg.log
# 5. Once the VM is started, the pending customization is applied. Even if that fails, the customization is then cleared. Meaning, you need to re-apply
#    the customization spec in order to try again. Simply rebooting the VM will not change anything.
#
##########

# Here is the basic workflow for creating a new VM and then customizing it
- name: Deploy a new VM based on a template
  vmware.vmware_rest.vcenter_vmtemplate_libraryitems:
    name: vm-from-template
    library: "{{ library_id }}"
    template_library_item: "{{ template_id }}"
    placement:
      cluster: "{{ lookup('vmware.vmware_rest.cluster_moid', '/my_dc/host/my_cluster') }}"
    state: deploy
  register: my_new_vm

- name: Power on the VM to register VMware tools
  vmware.vmware_rest.vcenter_vm_power:
    state: start
    vm: "{{ my_new_vm.id }}"

- name: Wait until my VMware tools are recognized
  vmware.vmware_rest.vcenter_vm_tools_info:
    vm: "{{ my_new_vm.id }}"
  register: vm_tools_info
  until:
    - vm_tools_info is not failed
    - vm_tools_info.value.run_state == "RUNNING"
  retries: 60
  delay: 5

- name: Power Off VM
  vmware.vmware_rest.vcenter_vm_power:
    state: stop
    vm: "{{ my_new_vm.id }}"

- name: Customize the VM
  vmware.vmware_rest.vcenter_vm_guest_customization:
    vm: "{{ my_new_vm.id }}"
    global_DNS_settings:
      dns_suffix_list:
        - lan
        - foo.internal
      dns_servers:
        - "8.8.8.8"
    interfaces:
      - adapter:
          ipv4:
            type: DHCP
    configuration_spec:
      linux_config:
        domain: test
        hostname:
          fixed_name: myhost
          type: FIXED

# Here is an example using the Linux script text. The script shebang can be anything (bash, perl, python), so long as the script will actually run
# There is also size and length limitation on the script text, as described in the module documentation.
# Finally, note the script is run twice. Once before all of the other customization and once after.
- name: Customize the VM
  vmware.vmware_rest.vcenter_vm_guest_customization:
    vm: "{{ my_new_vm.id }}"
    global_DNS_settings:
      dns_suffix_list:
        - lan
        - foo.internal
      dns_servers:
        - "8.8.8.8"
    interfaces:
      - adapter:
          ipv4:
            type: DHCP
    configuration_spec:
      linux_config:
        domain: test
        hostname:
          fixed_name: myhost
          type: FIXED
        script_text: |
          #!/bin/sh
          if [ x$1 == x"precustomization" ]; then
            echo "PRE" >> /tmp/vmware_rest_init_script.log
            # add any other pre-customization tasks here
          fi

          if [ x$1 == x"postcustomization" ]; then
            echo "POST" >> /tmp/vmware_rest_init_script.log
            # add any other post-customization tasks here
          fi

# Here is a simple example using cloud-init
# See also:
#   https://developer.broadcom.com/xapis/vsphere-automation-api/latest/vcenter/data-structures/Guest_CloudinitConfiguration/
#   https://knowledge.broadcom.com/external/article/311895/how-to-customize-virtual-machine-using-c.html
#   https://cloudinit.readthedocs.io/en/latest/reference/examples.html
#   https://cloudinit.readthedocs.io/en/23.4.1/reference/datasources/vmware.html#walkthrough-of-guestinfo-keys-transport
#
#   cloud-init required: metadata as plain-text JSON/YAML, maximum 512KB file size
#   cloud-init optional: userdata as plain-text in raw cloud-init format with no compression / no base64 encoding, maximum 512KB file size
- name: Customize the VM
  vmware.vmware_rest.vcenter_vm_guest_customization:
    vm: "{{ my_new_vm.id }}"
    global_DNS_settings:
      dns_suffix_list: []
      dns_servers:
        - "8.8.8.8"
    interfaces:
      - adapter:
          ipv4:
            type: DHCP
    configuration_spec:
      cloud_config:
        type: CLOUDINIT
        cloudinit:
          metadata: |
            instance-id: cloud-vm-example-1
            local-hostname: cloud-vm
            network:
              config: disabled
          userdata: |
            #cloud-config
            disable_root: 0
            write_files:
              - content: |
                  This is a test
                path: /root/cloud-init-example

# Here is a more complex cloud-init example
- name: Set cloud-init variables for customization specification
  ansible.builtin.set_fact:
    metadata_yaml:
      instance-id: "{{ vm_name }}"
      hostname: "{{ vm_name }}"
      local-hostname: "{{ vm_name }}"
      network:
        version: 2
        ethernets:
          nics:
            match:
              name: e*
            dhcp4: true
            dhcp6: false
      public_ssh_keys:
        - "{{ lookup('ansible.builtin.file', vmware_vm_ssh_public_key_file_path) }}"

    userdata_yaml_text: |
      #cloud-config
      hostname: {{ vm_name }}
      fqdn: {{ vm_name }}.{{ vm_domain }}

      disable_root: false
      ssh_pwauth: false
      ssh_deletekeys: true
      ssh:
        emit_keys_to_console: false
      no_ssh_fingerprints: false
      ssh_authorized_keys:
        - {{ lookup('ansible.builtin.file', vmware_vm_ssh_public_key_file_path) }}

      users:
        - name: root
          ssh_authorized_keys:
            - {{ lookup('ansible.builtin.file', vmware_vm_ssh_public_key_file_path) }}
          lock_passwd: false

      write_files:
        - path: /etc/cloud/cloud-init.disabled
          permissions: "0644"
          content: ""

- name: Apply customization specification to the VM in Powered Off state
  vmware.vmware_rest.vcenter_vm_guest_customization:
    vm: "{{ my_new_vm.id }}"
    configuration_spec:
      cloud_config:
        type: CLOUDINIT
        cloudinit:
          metadata: "{{ metadata_yaml | to_json(ensure_ascii=true) }}"
          userdata: "{{ userdata_yaml_text | trim }}" # remove last newline character
    interfaces: []
    global_DNS_settings: {}

返回值

常见的返回值已在 此处 记录,以下是此模块特有的字段

描述

字典

自定义虚拟机

返回:成功时

示例: {}

作者

  • Ansible 云团队 (@ansible-collections)