黑帽联盟

标题: ansible笔记(46):常用技巧(二) [打印本页]

作者: yun    时间: 2019-9-11 16:38
标题: ansible笔记(46):常用技巧(二)
本帖最后由 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运行的过程中,对列表追加元素、插入元素、或者删除元素,文章就先写到这里,希望能够对你有所帮助~





欢迎光临 黑帽联盟 (https://bbs.cnblackhat.com/) Powered by Discuz! X2.5