黑帽联盟

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

[基础服务] ansible笔记(46):常用技巧(二)

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

920

主题

37

听众

1364

积分

超级版主

Rank: 8Rank: 8

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

    [LV.9]以坛为家II

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

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

    前一篇文章中,我们总结了一些常用小技巧,这篇文章继续。

    小技巧:向列表中追加项
    在某些场景下,我们需要在剧本运行的过程中,向列表中加入指定的值,也就是说,我们需要利用task在list中加入新元素,那么我们该怎样实现呢?方法很简单,利用python中列表合并的特性,即可在ansible运行时,对list追加新元素,先来看一个小示例。

    在python中,我们可以直接将两个列表相加,这样两个列表会合并成一个列表,我们在ansible中也可以这样做,示例如下:
    1. - hosts: test71
    2.   gather_facts: no
    3.   vars:
    4.    l1:
    5.      - 1
    6.      - 2
    7.    l2:
    8.      - a
    9.      - b
    10.      - 2
    11.   tasks:
    12.   - set_fact:
    13.       l3: "{{ l1 + l2 }}"
    14.   - debug:
    15.       var: l3
    复制代码
    如上例所示,我们定义了两个列表变量,l1和l2,然后在tasks中,使用"set_fact"模块,设置了l3变量,l3变量的值为l1和l2相加后的结果,这其实就是利用了python中列表合并的方法,在python中,我们可以使用加号将两个列表合并成一个列表,所以,执行上例playbook,debug模块的输出的结果如下:
    1. TASK [debug] ***********************
    2. ok: [test71] => {
    3.     "l3": [
    4.         1,
    5.         2,
    6.         "a",
    7.         "b",
    8.         2
    9.     ]
    10. }
    复制代码
    如上述信息所示,l3这个列表其实就是l1和l2两个列表合并后的结果,其实,我们可以利用列表合并的这一特性,在列表中追加元素,示例如下:
    1. - hosts: test71
    2.   gather_facts: no
    3.   vars:
    4.    tlist:
    5.      - 1
    6.      - 2
    7.   tasks:
    8.   - set_fact:
    9.       tlist: "{{ tlist + ['a'] }}"
    10.   - debug:
    11.       var: tlist
    复制代码
    如上例所示,当我们想要向tlist列表追加新元素'a'时,只需要先将字符串'a'变成一个列表,然后利用加号,将tlist列表和['a']列表合并,再将合并后的结果赋值给tlist变量,即可实现我们想要的效果,那么执行上例playbook,debug模块的输出信息如下:
    1. TASK [debug] ***********************
    2. ok: [test71] => {
    3.     "tlist": [
    4.         1,
    5.         2,
    6.         "a"
    7.     ]
    8. }
    复制代码
    正如我们所愿,新元素'a'追加到了tlist列表中。
    其实,我们也可以使用jinja2的语法,完成上述追加元素的过程,示例如下:
    1. - hosts: test71
    2.   gather_facts: no
    3.   vars:
    4.    tlist:
    5.      - 1
    6.      - 2
    7.   tasks:
    8.   - set_fact:
    9.       tlist: "{% set tlist = tlist + ['a'] %}{{tlist}}"
    10.   - debug:
    11.       var: tlist
    复制代码
    执行上例playbook,也可以实现完全相同的效果,jinja2的语法在前文中已经总结过,此处不再赘述。

    在python中,使用列表的extend()函数,也可以实现列表合并的效果,与使用"+"的效果完全相同,比如,将列表A和列表B合并,可以使用A.extend(B),在ansible中,我们也可以这样干,示例如下:
    1. - hosts: test71
    2.   gather_facts: no
    3.   vars:
    4.    tlist:
    5.      - 1
    6.      - 2
    7.   tasks:
    8.   - set_fact:
    9.       tlist: "{{ tlist.extend([3,'a']) }}{{tlist}}"
    10.   - debug:
    11.       var: tlist
    复制代码
    说到extend()函数,用过python的小伙伴可能会联想到另外一个函数,append()函数,append()函数可以直接在列表中追加元素,示例如下:
    1. - hosts: test71
    2.   gather_facts: no
    3.   vars:
    4.    tlist:
    5.      - 1
    6.      - 2
    7.   tasks:
    8.   - set_fact:
    9.       tlist: "{{ tlist.append('a') }}{{tlist}}"
    10.   - debug:
    11.       var: tlist
    复制代码
    细心如你一定发现了,当使用append函数时,我们并没有将'a'转换成列表,而是直接将字符串'a'追加到了tlist列表中,执行上例playbook,输出如下:
    1. TASK [debug] ***********************
    2. ok: [test71] => {
    3.     "tlist": [
    4.         1,
    5.         2,
    6.         "a"
    7.     ]
    8. }
    复制代码
    你一定看出了append函数和extend函数的区别,没错,extend函数操作的两个对象必须都是列表,而append函数只是单纯的将一个对象当做新元素追加到列表中,这个新元素可以时数值或者字符串,也可以是一个列表,或者字典,示例如下:
    1. - hosts: test71
    2.   gather_facts: no
    3.   vars:
    4.    tlist:
    5.      - 1
    6.      - 2
    7.   tasks:
    8.   - set_fact:
    9.       tlist: "{{ tlist.append(['a','b']) }}{{tlist}}"
    10.   - debug:
    11.       var: tlist
    复制代码
    执行上例playbook,debug模块输出如下:
    1. TASK [debug] ***********************
    2. ok: [test71] => {
    3.     "tlist": [
    4.         1,
    5.         2,
    6.         [
    7.             "a",
    8.             "b"
    9.         ]
    10.     ]
    11. }
    复制代码
    小技巧:在列表中插入项
    刚才一直在聊怎样向list中追加元素,追加的元素总是在列表的最后,但是有时,我们想要在列表的指定位置中插入新元素,该怎么办呢?聪明如你,一定已经想到了,既然追加元素能够利用python中的函数,那么插入元素肯定也可以使用同样的套路啊,没错,我们一起来看一个小示例。

    我们可以利用python的insert()函数,在list的指定位置插入新元素,示例如下:
    1. - hosts: test71
    2.   gather_facts: no
    3.   vars:
    4.    tlist:
    5.      - 11
    6.      - 2
    7.      - 11
    8.   tasks:
    9.   - set_fact:
    10.       tlist: "{{ tlist.insert(1,'a') }}{{tlist}}"
    11.   - debug:
    12.       var: tlist
    复制代码
    上例表示,在tlist列表的索引1位置插入元素'a',索引从0开始(0表示列表的第一项,索引可以为负数,为-1时表示列表中倒数第二个位置),执行上例playbook,debug模块的输出信息如下:
    1. TASK [debug] ***********************
    2. ok: [test71] => (item=11) => {
    3.     "msg": 11
    4. }
    5. ok: [test71] => (item=a) => {
    6.     "msg": "a"
    7. }
    8. ok: [test71] => (item=2) => {
    9.     "msg": 2
    10. }
    11. ok: [test71] => (item=11) => {
    12.     "msg": 11
    13. }
    复制代码
    小技巧:从列表中删除项
    当我们需要在ansible中删除list中的元素时,有哪些小技巧可以使用呢?其实大致的思路与上文并没有什么不同,无非是利用python中的函数,但是,删除时有一些特殊的注意点,所以需要更加灵活一点,先来看一些小示例吧。

    我们可以利用python的pop()函数,根据指定的索引删除元素。
    但是需要注意,在python中,pop()函数会根据索引删除元素的同时,将被删除的元素值作为返回值,示例如下:
    1. - hosts: test71
    2.   gather_facts: no
    3.   vars:
    4.    tlist:
    5.      - 11
    6.      - 2
    7.      - 'a'
    8.      - 'b'
    9.      - 3
    10.   tasks:
    11.   - debug:
    12.       msg: "{{tlist.pop(2)}}"
    复制代码
    上例中,我们使用pop函数删除了tlist列表中的第三个元素(索引为2),也就是元素'a',执行上例playbook,会发现debug模块的输出信息就是被删除的元素'a',输出如下:
    1. TASK [debug] ***********************
    2. ok: [test71] => {
    3.     "msg": "a"
    4. }
    复制代码
    如你所见,pop()函数在删除元素的同时,会将删除的元素值作为返回值,所以,我们并不能生搬硬套上文中的套路,如果生搬硬套上文中的套路,得到的结果并不是我们想要的,你可以按照上文的套路尝试一下,就会明白为什么不行了,所以,我们需要变通一下,变通后的写法如下:
    利用pop()函数删除列表中指定位置的元素
    1. - hosts: test71
    2.   gather_facts: no
    3.   vars:
    4.    tlist:
    5.      - 11
    6.      - 2
    7.      - 'a'
    8.      - 'b'
    9.      - 3
    10.   tasks:
    11.   - set_fact:
    12.       tlist: "{{ [ tlist.pop(2),tlist ][1] }}"
    13.   - debug:
    14.       var: tlist
    复制代码
    如上例所示,我们先利用pop()函数删除了列表中指定位置的元素,然后将"pop()函数的返回值"和"删除指定元素后的列表"组成了一个新的大列表,并且获取到了新的大列表中索引为1的元素(即删除指定元素后的列表),并且将这个删除元素后的列表通过set_fact模块赋值给了原来的列表变量,从而实现了我们的目的,上例debug模块的输出信息如下:
    1. TASK [debug] ***********************
    2. ok: [test71] => {
    3.     "tlist": [
    4.         11,
    5.         2,
    6.         "b",
    7.         3
    8.     ]
    9. }
    复制代码
    我们可以利用python的remove()函数,根据指定的值删除元素(第一个与指定值匹配的元素会被删除),示例如下
    1. - hosts: test71
    2.   gather_facts: no
    3.   vars:
    4.    tlist:
    5.      - 11
    6.      - 'b'
    7.      - 'a'
    8.      - 'b'
    9.      - 3
    10.   tasks:
    11.   - set_fact:
    12.       tlist: "{{ tlist.remove('b') }}{{tlist}}"
    13.   - debug:
    14.       var: tlist
    复制代码
    如上例所示,我们想要删除tlist列表中元素值为'b'的元素,于是,我们在remove函数中指定的值为'b',执行上例playbook,debug模块的输出信息如下:
    1. TASK [debug] ***********************
    2. ok: [test71] => (item=11) => {
    3.     "msg": 11
    4. }
    5. ok: [test71] => (item=a) => {
    6.     "msg": "a"
    7. }
    8. ok: [test71] => (item=b) => {
    9.     "msg": "b"
    10. }
    11. ok: [test71] => (item=3) => {
    12.     "msg": 3
    13. }
    复制代码
    如你所见,tlist中有两个元素的值都为'b',但是只有第一个'b'被删除了,因为remove函数只会删除第一个匹配到的项,需要注意,使用这种方法删除元素时,如果指定的要删除的元素并不存在于列表中,则会报错。

    正如刚才所说,remove函数只会删除第一个匹配到的项,于是,我使用了一个笨办法,利用jinja2的for循环,删除列表中所有匹配到的项,示例playbook如下:
    1. - hosts: test71
    2.   gather_facts: no
    3.   vars:
    4.     tlist:
    5.     - 11
    6.     - 'b'
    7.     - 'a'
    8.     - 'b'
    9.     - 11
    10.     - 'a11'
    11.     - '1a'
    12.     - '11'
    13.   tasks:
    14.   - set_fact:
    15.       tlist: "{%for i in tlist%}{% if 11 in tlist%}{{tlist.remove(11)}}{%endif%}{%endfor%}{{tlist}}"
    16.   - debug:
    17.       var: tlist
    复制代码
    上例中,结合jinja2语法中的for循环和if判断,循环的删除了列表中所有的数值为11的元素,执行上例playbook,debug模块的输出信息如下:
    1. TASK [debug] ***********************
    2. ok: [test71] => {
    3.     "tlist": [
    4.         "b",
    5.         "a",
    6.         "b",
    7.         "a11",
    8.         "1a",
    9.         "11"
    10.     ]
    11. }
    复制代码
    如你所见,由于最后一个元素"11"是字符串类型,而我们的匹配条件是匹配数字类型的11,所以最后一个元素"11"并没有被匹配到,也就没有被删除。

    当我在网络上借鉴别人的做法时,我发现了如下一种写法,如下写法也可以删除列表中的元素,示例如下:
    注:如下写法只适合所有元素都是字符串类型的列表,而且,如下写法可以通过正则表达式,删除掉所有匹配的元素。
    1. - hosts: test71
    2.   gather_facts: no
    3.   vars:
    4.     tlist:
    5.     - '11'
    6.     - 'b'
    7.     - '1a'
    8.     - '1b'
    9.     - 'c'
    10.     - 'a1c'
    11.     - 'b1'
    12.   tasks:
    13.   - set_fact:
    14.       tlist: "{{ tlist | reject('search','1') | list }}"
    15.   - debug:
    16.       var: tlist
    复制代码
    上例表示,删除tlist列表中所有包含字符'1'的元素,执行上例playbook,debug输出信息如下:
    1. TASK [debug] ***********************
    2. ok: [test71] => {
    3.     "tlist": [
    4.         "b",
    5.         "c"
    6.     ]
    7. }
    复制代码
    从输出结果可以看出,所有包含字符'1'的元素都被删除了。
    上例playbook中,reject是jinja2中的一个内建的filter,search是ansible实现的一个tests,list也是一个过滤器,上例通过将它们结合的方式,删除了列表中所有符合条件的元素,使用这种方法有个前提条件,就是列表中的所有元素都必须是字符串,当列表中的所有元素都是字符串时,使用这种方法,可以通过正则表达式灵活的删除元素,比如,删除所有'1'开头的元素,如下:
    1. - hosts: test71
    2.   gather_facts: no
    3.   vars:
    4.     tlist: ['11','b','1a','1b','c','a1c1','b11']
    5.   tasks:
    6.   - set_fact:
    7.       tlist: "{{ tlist | reject('search','^1') | list }}"
    8.   - debug:
    9.       var: tlist
    10. 上例playbook执行后,debug输出如下:
    11. TASK [debug] ***********************
    12. ok: [test71] => {
    13.     "tlist": [
    14.         "b",
    15.         "c",
    16.         "a1c1",
    17.         "b11"
    18.     ]
    19. }
    复制代码
    如你所见,所有以字符'1'开头的元素都被删除了,很方便吧,快动手试试吧。

    其实,上文中总结的小技巧,无非是将一些零散的知识点灵活的组合在一起罢了,比如filter、tests、在ansible中使用python的特性、jinja2的语法等等,这些知识我们在前文中都总结过,你肯定不会陌生的。

    之前有个朋友留言,问了一个问题:
    目标主机有8台,现在需要获取这8台主机上的某个fact(同一个fact),怎样通过循环,将这8台主机的fact的值写入到一个列表中。
    我觉得这个问题就可以使用到上文的小技巧来解决,我给出的代码如下:
    1. - hosts: 8hostsgroup
    2.   vars:
    3.    testarr: []
    4.   tasks:
    5.   - set_fact:
    6.       testarr: "{{ testarr + [hostvars[item].ansible_uptime_seconds] }}"
    7.     with_items: "{{play_hosts}}"
    8.     run_once: true
    9.   - debug:
    10.       msg: "{{ testarr }}"
    11.     run_once: true
    复制代码
    上例中获取到fact信息是ansible_uptime_seconds,你也可以改成你想要获取的fact信息。


    这篇文章总结了一些小技巧,利用这些小技巧,可以在playbook运行的过程中,对列表追加元素、插入元素、或者删除元素,文章就先写到这里,希望能够对你有所帮助~
    帖子永久地址: 

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

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

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