通过执行ansible playbook向多个主机中的/etc/hosts文件写入IP、主机解析配置
# cat /etc/hosts 127.0.0.1 localhost localhost.localdomain localhost4 localhost4.localdomain4 ::1 localhost localhost.localdomain localhost6 localhost6.localdomain6 192.168.51.21 candy1使用ansible lineinfile模块,playbook内容如下:
--- - hosts: '{{ host }}' gather_facts: false tasks: - name: setup /etc/hosts lineinfile: path: /etc/hosts line: "{{ content }}"执行命令如下:
source ../env.sh export i=0 for NODE_NAME in ${NODE_NAMES[@]} do content="${NODE_IPS[$(eval echo \$i)]} ${NODE_NAMES[$(eval echo \$i)]}" ansible-playbook setup-hosts/setup-hosts.yaml -e "host=$CLUSTER_NAME content=$content" i=`expr $i + 1` done预期是一行一行将IP、主机名写入/etc/hosts文件,可结果只写入了IP,没有后面的主机名,原因是$content中包含空格。试了各种方法如\空格转义不行,[[:space:]]也不支持。
playbook如下:
--- - hosts: '{{ host }}' gather_facts: false tasks: - name: setup /etc/hosts lineinfile: path: /etc/hosts line: "{{ ip }} {{ hostname }}"执行结果如下:
# cat /etc/hosts 127.0.0.1 localhost localhost.localdomain localhost4 localhost4.localdomain4 ::1 localhost localhost.localdomain localhost6 localhost6.localdomain6 192.168.51.21 candy1 192.168.51.22 candy2 192.168.51.23 candy3这样已经可以满足大多数管理配置文件的需求了,但遇到内容多且不规律的文本就不适用了
改用blockinfile模块,playbook如下:
- hosts: '{{ host }}' tasks: - name: setup /etc/hosts blockinfile: dest: /etc/hosts block: "{{ lookup('file', 'etc-hosts') }}"将文本内容先写入setup-hosts/files/etc-hosts文件(setup-hosts是playbook所在的目录),执行playbook后,结果如下:
127.0.0.1 localhost localhost.localdomain localhost4 localhost4.localdomain4 ::1 localhost localhost.localdomain localhost6 localhost6.localdomain6 # BEGIN ANSIBLE MANAGED BLOCK 192.168.51.21 candy1 192.168.51.22 candy2 192.168.51.23 candy3 # END ANSIBLE MANAGED BLOCK想起我曾经用过google云平台GCP建虚拟机,其中/etc/hosts文件就是用这个方法来管理的
我推荐使用第二种方法来管理文本内容,但如果遇到对#号敏感的情况改用第一种。欢迎留下建议分享给大家。