黑帽联盟

 找回密码
 会员注册
查看: 1441|回复: 0
打印 上一主题 下一主题

[基础服务] ansible笔记(18):变量(五)

[复制链接]
yun 黑帽联盟官方人员 

920

主题

37

听众

1364

积分

超级版主

Rank: 8Rank: 8

  • TA的每日心情
    奋斗
    2019-10-18 11:20
  • 签到天数: 678 天

    [LV.9]以坛为家II

    本帖最后由 yun 于 2019-9-11 17:22 编辑

    ansible是一个系列文章,我们会尽量以通俗易懂的方式总结ansible的相关知识点。
    ansible系列博文直达链接:ansible轻松入门系列
    "ansible系列"中的每篇文章都建立在前文的基础之上,所以,请按照顺序阅读这些文章,否则有可能在阅读中遇到障碍。


    ansible中还有一些内置变量可供我们使用,当然,这些内置变量的变量名是被ansible保留的,我们定义变量时不能使用这些变量名。

    内置变量ansible_version
    先从一个简单的内置变量说起,比如,我们可以通过内置变量ansible_version获取到ansible的版本号,示例命令如下
    1. ansible test70 -m debug -a "msg={{ansible_version}}"
    复制代码
    内置变量hostvars
    除了ansible_version,还有一些非常有用的内置变量。比如内置变量hostvars
    hostvars可以帮助我们在操作当前主机时获取到其他主机中的信息。
    假设,我想要在操作test70主机时获取到test71主机中的facts信息,我该怎么办呢?示例如下
    1. ---
    2. - name: "play 1: Gather facts of test71"
    3.   hosts: test71
    4.   remote_user: root

    5. - name: "play 2: Get facts of test71 when operating on test70"
    6.   hosts: test70
    7.   remote_user: root
    8.   tasks:
    9.   - debug:
    10.       msg: "{{hostvars['test71'].ansible_ens35.ipv4}}"
    复制代码
    上例中有两个play,第一个play针对test71主机执行,但是第一个play中没有显式指定任何task(后文会解释原因),第二个play针对test70主机执行,在第二个play中只有一个task,即使用debug模块,输出了test71主机中的ens35网卡的IP信息,如你所见,我们可以借助hostvars在操作当前主机时输出其他主机中的facts信息,上例中使用hostvars加上清单中的主机名称再加上facts的key,即可获取到对应的facts信息,有了前文的总结作为基础,你一定想到了,上例中的msg的值改为如下写法也是可以的。
    1. "{{hostvars.test71.ansible_ens35.ipv4}}"
    复制代码
    上例中的第一个play中并没有任何的task,为什么还需要第一个play呢?如果你将上例的第一个play删除,只保留第二个play,运行时则会报错,这是因为,虽然第一个play中没有任何task,但是当第一个play执行时,默认会调用"[Gathering Facts]"任务,也就是说,默认会收集test71主机的facts信息,只有被收集过的facts信息才能被后面的play引用到,如果压根没有收集对应主机的facts信息,即使使用hostvars内置变量,也无法获取到对应主机的facts信息,我们来做个试验,我们可以直接把上例的第一个play从playbook中删除,也可以指明让第一个play不收集对应的facts信息,使用"gather_facts"关键字可以控制当前play是否收集对应主机的facts信息,示例如下:
    1. ---
    2. - name: "play 1: Gather facts of test71"
    3.   hosts: test71
    4.   remote_user: root
    5.   gather_facts: no

    6. - name: "play 2: Get facts of test71 when operating on test70"
    7.   hosts: test70
    8.   remote_user: root
    9.   tasks:
    10.   - debug:
    11.       msg: "{{hostvars['test71'].ansible_ens35.ipv4}}"
    复制代码
    如上例所示,第一个play中的"gather_facts: no"表示设置当前play不收集对应主机的信息,运行上例playbook会报错,因为第二个play在操作test70时,无法获取到test71主机中的facts信息,原因是test71的facts信息并未被收集过,所以,调用其他主机的facts信息的前提是对应主机的facts信息已经被收集过。

    其实,除了facts信息,我们还能够利用hostvars内置变量从别的主机中获取到其他类型的一些变量信息,比如,其他主机的注册变量、主机变量、组变量等信息,我们先来看一个获取其他主机的注册变量的小示例,如下:
    1. ---
    2. - hosts: test71
    3.   remote_user: root
    4.   gather_facts: no
    5.   tasks:
    6.   - shell: "echo register_var_in_play1"
    7.     register: shellreturn

    8. - hosts: test70
    9.   remote_user: root
    10.   gather_facts: no
    11.   tasks:
    12.   - debug:
    13.       msg: "{{hostvars.test71.shellreturn.stdout}}"
    复制代码
    如上例所示,通过hostvars内置变量可以直接获取到其他主机中的注册变量,你一定发现了,注册变量并不用像facts信息那样需要事先收集,即可直接通过hostvars跨主机被引用到,同理,如果你在清单中为test71主机配置了主机变量,或者为test71主机所在的组配置了组变量,也是可以通过hostvars直接跨主机引用的,这里就不进行示例了,动手试试吧。

    你可能会问,如果我直接在play中为当前主机定义一个变量,可以在之后的play中操作其他主机时被引用到吗?那么我们来做个实验,示例如下
    1. ---
    2. - hosts: test71
    3.   remote_user: root
    4.   gather_facts: no
    5.   vars:
    6.     testvar: testvar_in_71
    7.   tasks:
    8.   - debug:
    9.       msg: "{{testvar}}"

    10. - hosts: test70
    11.   remote_user: root
    12.   gather_facts: no
    13.   tasks:
    14.   - debug:
    15.       msg: "{{hostvars.test71.testvar}}"
    复制代码
    在上例的第一个play中我们为test71主机定义了一个变量,变量名称为testvar,在第二个play中操作test70主机时,使用hostvars尝试引用test71主机中的变量,如果执行上述playbook则会报错,看来通过vars关键字定义的变量使用上例中的方法是无法被跨主机引用的,聪明如你,一定想到了解决方案,前一篇文章中,我们总结了怎样使用"set_fact"关键字定义变量,通过"set_fact"关键字定义的变量拥有类似"facts"信息的特性(如果不明白可以参考前文),所以,我们可以把"vars"关键字中定义的变量通过"set_fact"关键字去定义,这样这些变量就好像facts信息被收集过一样,能被之后的play引用到了,示例如下
    1. ---
    2. - hosts: test71
    3.   remote_user: root
    4.   gather_facts: no
    5.   tasks:
    6.   - set_fact:
    7.       testvar: "testvar_in_71"
    8.   - debug:
    9.       msg: "{{testvar}}"

    10. - hosts: test70
    11.   remote_user: root
    12.   gather_facts: no
    13.   tasks:
    14.   - debug:
    15.       msg: "{{hostvars.test71.testvar}}"
    复制代码
    上例通过"set_fact"结合"hostvars"的方式,实现了跨play获取其他主机中的变量信息的功能,还是很方便的。


    内置变量inventory_hostname
    通过inventory_hostname变量可以获取到被操作的当前主机的主机名称,这里所说的主机名称并不是linux系统的主机名,而是对应主机在清单中配置的名称,假设我的清单配置如下
    1. [test_group]
    2. 10.1.1.60
    3. test70.zsythink.net ansible_host=10.1.1.70
    4. test71 anisble_host=10.1.1.71
    复制代码
    清单中配置了三个主机,第一个主机以IP的形式配置,第二个主机和第三个主机都以别名的方式配置,他们同属于test_group组。
    那么我们使用内置变量inventory_hostname获取一下各个主机的对应的主机名,看看会返回什么,示例如下
    1. # ansible test_group -m debug -a "msg={{inventory_hostname}}"
    2. test70.zsythink.net | SUCCESS => {
    3.     "changed": false,
    4.     "msg": "test70.zsythink.net"
    5. }
    6. 10.1.1.60 | SUCCESS => {
    7.     "changed": false,
    8.     "msg": "10.1.1.60"
    9. }
    10. test71 | SUCCESS => {
    11.     "changed": false,
    12.     "msg": "test71"
    13. }
    复制代码
    从返回信息可以看出,如果使用IP配置主机,inventory_hostname的值就是IP,如果使用别名,inventory_hostname的值就是别名。


    内置变量inventory_hostname_short
    与内置变量inventory_hostname类似,通过inventory_hostname_short也可以获取当前play操作的主机在清单中对应的名称,但是这个名称更加简短,假设我的清单配置如下
    1. [test_group]
    2. 10.1.1.60
    3. test70.cnblackhat.com ansible_host=10.1.1.70
    4. test71 anisble_host=10.1.1.71
    复制代码
    那么通过内置变量inventory_hostname_short获取到的主机的简短名称如下:
    1. # ansible test_group -m debug -a "msg={{inventory_hostname_short}}"
    2. test70.zsythink.net | SUCCESS => {
    3.     "changed": false,
    4.     "msg": "test70"
    5. }
    6. 10.1.1.60 | SUCCESS => {
    7.     "changed": false,
    8.     "msg": "10"
    9. }
    10. test71 | SUCCESS => {
    11.     "changed": false,
    12.     "msg": "test71"
    13. }
    复制代码
    可以看到,无论是IP还是别名,如果清单的主机名称中包含".",inventory_hostname_short都会取得主机名中第一个"."之前的字符作为主机的简短名称。


    内置变量play_hosts
    通过内置变量play_hosts可以获取到当前play所操作的所有主机的主机名列表,示例playbook如下:
    1. ---
    2. - hosts: test70,test71
    3.   remote_user: root
    4.   gather_facts: no
    5.   tasks:
    6.   - debug:
    7.       msg: "{{play_hosts}}"
    复制代码
    执行上例的playbook,返回信息如下
    1. TASK [debug] *************************
    2. ok: [test70] => {
    3.     "msg": [
    4.         "test71",
    5.         "test70"
    6.     ]
    7. }
    8. ok: [test71] => {
    9.     "msg": [
    10.         "test71",
    11.         "test70"
    12.     ]
    13. }
    复制代码
    可以看到,此play每操作一个主机,都会将当前play操作的所有主机的主机名列表返回。
    没错,inventory_hostname和play_hosts都是返回主机名,只不过,inventory_hostname只返回当前被操作的主机的主机名,而play_hosts则返回当前play中所有被操作主机的主机名列表。

    内置变量groups
    通过groups内置变量可以获取到清单中"所有分组"的"分组信息",什么意思呢?我们先来看一个清单配置,假设我的清单配置如下:
    1. 10.1.1.60
    2. test70.zsythink.net ansible_host=10.1.1.70
    3. test71 anisble_host=10.1.1.71

    4. [testA]
    5. test60 ansible_host=10.1.1.60
    6. test61 ansible_host=10.1.1.61

    7. [testB]
    8. test70 ansible_host=10.1.1.70

    9. [test:children]
    10. testA
    11. testB
    复制代码
    上述清单中,显式的指定了三个组,testA组、testB组、test组,其中,testA组与testB组是test组的子组,除了组中的主机,还有三台主机没有任何分组,直接写在了清单中。

    现在,我们获取一下groups变量的值,看看会返回哪些信息,随便操作清单中的任意一台主机即可,示例如下
    1. # ansible test70 -m debug -a "msg={{groups}}"
    2. test70 | SUCCESS => {
    3.     "changed": false,
    4.     "msg": {
    5.         "all": [
    6.             "10.1.1.60",
    7.             "test70.zsythink.net",
    8.             "test71",
    9.             "test60",
    10.             "test61",
    11.             "test70"
    12.         ],
    13.         "test": [
    14.             "test60",
    15.             "test61",
    16.             "test70"
    17.         ],
    18.         "testA": [
    19.             "test60",
    20.             "test61"
    21.         ],
    22.         "testB": [
    23.             "test70"
    24.         ],
    25.         "ungrouped": [
    26.             "10.1.1.60",
    27.             "test70.zsythink.net",
    28.             "test71"
    29.         ]
    30.     }
    31. }
    复制代码
    从上述返回信息可以看出,所有主机默认被分成了组名为"all"的组,testA组中有两台主机,testB组中有一台主机,由于testA组和testB组都属于test组的子组,所以testA组与testB组中的主机都属于test组,由于有三台主机在清单中并未分组,所以,ansible自动将没有分组的主机分到了名为"ungrouped"的组中,即组名为"未分组"的组。

    我们还能够通过组名,获取到指定组的分组信息,假设,我想要获取到上例中test组中的主机名称,则可以使用如下方法。
    1. # ansible test70 -m debug -a "msg={{groups.test}}"
    复制代码
    当然,语法也可以改为如下
    1. # ansible test70 -m debug -a "msg={{groups['test']}}"
    复制代码
    聪明如你一定已经会举一反三了,所以,如果我们想要获取到所有未分组主机的主机名,则可以使用如下方法
    1. # ansible test70 -m debug -a "msg={{groups.ungrouped}}"
    复制代码
    内置变量group_names
    见名知义,我们可以通过内置变量group_names获取到当前主机所在分组的组名,比如,我的清单配置如下
    1. 10.1.1.60
    2. test70.zsythink.net ansible_host=10.1.1.70
    3. test71 anisble_host=10.1.1.71

    4. [testA]
    5. test60 ansible_host=10.1.1.60
    6. test61 ansible_host=10.1.1.61

    7. [testB]
    8. test70 ansible_host=10.1.1.70

    9. [test:children]
    10. testA
    11. testB
    复制代码
    那么,当我操作test70主机时,group_names变量值如下
    1. # ansible test70 -m debug -a "msg={{group_names}}"
    2. test70 | SUCCESS => {
    3.     "changed": false,
    4.     "msg": [
    5.         "test",
    6.         "testB"
    7.     ]
    8. }
    复制代码
    如上例返回值所示,test70主机属于testB组,而testB组又是test组的子组,所以test70主机同时属于testB组和test组,所以,最终返回的信息中包括test与testB

    当我们操作未分组的主机时,group_names的值为"ungrouped",示例如下
    1. # ansible 10.1.1.60 -m debug -a "msg={{group_names}}"
    2. 10.1.1.60 | SUCCESS => {
    3.     "changed": false,
    4.     "msg": [
    5.         "ungrouped"
    6.     ]
    7. }
    复制代码
    内置变量inventory_dir
    我们可以通过inventory_dir变量获取到ansible主机中清单文件的存放路径,我使用的是默认的清单文件/etc/ansible/hosts,所以,inventory_dir变量对应的值为/etc/ansible,如下例所示
    1. # ansible test71 -m debug -a "msg={{inventory_dir}}"
    2. test71 | SUCCESS => {
    3.     "changed": false,
    4.     "msg": "/etc/ansible"
    5. }
    复制代码
    ansible中还有其他的一些变量的使用方法,但是需要结合其他的一些知识点,所以之后遇到了实际的使用场景,我们再进行介绍吧。
    这篇文章就先总结到这里,希望能够对你有所帮助。

    帖子永久地址: 

    黑帽联盟 - 论坛版权1、本主题所有言论和图片纯属会员个人意见,与本论坛立场无关
    2、本站所有主题由该帖子作者发表,该帖子作者与黑帽联盟享有帖子相关版权
    3、其他单位或个人使用、转载或引用本文时必须同时征得该帖子作者和黑帽联盟的同意
    4、帖子作者须承担一切因本文发表而直接或间接导致的民事或刑事法律责任
    5、本帖部分内容转载自其它媒体,但并不代表本站赞同其观点和对其真实性负责
    6、如本帖侵犯到任何版权问题,请立即告知本站,本站将及时予与删除并致以最深的歉意
    7、黑帽联盟管理员和版主有权不事先通知发贴者而删除本文

    您需要登录后才可以回帖 登录 | 会员注册

    发布主题 !fastreply! 收藏帖子 返回列表 搜索
    回顶部