遇到的问题
通过执行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文件就是用这个方法来管理的
总结
我推荐使用第二种方法来管理文本内容,但如果遇到对#号敏感的情况改用第一种。欢迎留下建议分享给大家。