ansible——playbook lookups从插件加载变量

之前说过playbook的变量引用,这是传参的一种方式。
playbook还支持很多插件从外部读取数据,比如从文件中读取、从数据库中读取。
lookups的所有操作都是在中控机上进行。目前有58个插件
官网文档:https://docs.ansible.com/ansible/latest/plugins/lookup.html#plugin-list
下面介绍几个常用的。

 

1.lookups file

通过file插件可以读取文件。其内部是使用python打开文件然后把结果返回给变量。

[root@centos3 yaml_test]# cat file_lookup.yaml 
---
- hosts: all
  gather_facts: False
  vars: 
    contents: "{{ lookup('file','/etc/hostname') }}"
  tasks:
    - name: debug lookups
      debug: msg="The contents is {% for i in contents.split("\n") %} {{ i }} {% endfor %}"

执行结果:

file这个插件读取的是中控机本地的文件,然后拿到数据以供任务使用.
这里使用的是jinjia2的语法.

 

2.lookups password

passwd这个插件会对传入的内容进行加密处理.
定义:pass_lookup.yaml

---
- hosts: all
  vars:
    contents: "{{ lookup('password', 'ansible_book') }}"
  tasks:
    - name: debug lookups
      debug: msg="The contents is {{ contents }}"

执行结果:

上面的例子就是将ansible_book加密,然后以作他用.

 

3.lookups pipe

pipe插件运行命令并返回结果。

---
- hosts: all
  vars:
    contents: "{{ lookup('pipe', 'date +%Y-%m-%d') }}"
  tasks:
    - name: debug lookups
      debug: msg="The contents is {% for i in contents.split("\n") %} {{ i }} {% endfor %}

执行结果:

pipe看起和shell一样,也是用来执行命令,但是shell是在管理节点上执行,而pipe是在中控机上执行。
pipe这个插件底层使用的是subprocess这个python库来实现的
subprocess是用来生成新的进程,连接它们的输入输出。Popen类是用来创建和管理进程。
虽然增加一个进程不一定能够解决ansible进程和pipe进程的拥堵问题,但是一般情况还是适用的。

 

4.lookups redis_kv

redis_kv是用来从本地redis中读取数据。
ansible默认支持的是python2,因此你需要在python2中添加redis这个python包。

定义:  redis_lookup.yaml

---
- hosts: all
  vars:
    contents: "{{ lookup('redis_kv', 'redis://127.0.0.1:6379,ansible') }}"

  tasks:
    - name: debug lookups
      debug: msg="The contents is {% for i in contents.split("\n") %} {{ i }} {% endfor %}"

执行结果:

 查看源码:

#ansible/lib/ansible/plugins/lookup/_redis_kv.py
# (c) 2012, Jan-Piet Mens <jpmens(at)gmail.com>
# (c) 2017 Ansible Project
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
from __future__ import (absolute_import, division, print_function)  #程序检测
__metaclass__ = type

import os
import re

HAVE_REDIS = False
try:
    import redis  #python是否已经安装了python模块
    HAVE_REDIS = True
except ImportError:
    pass

from ansible.errors import AnsibleError
from ansible.plugins.lookup import LookupBase

class LookupModule(LookupBase):

    def run(self, terms, variables, **kwargs):  #term是连接参数

        if not HAVE_REDIS:  #redis模块检测
            raise AnsibleError("Can't LOOKUP(redis_kv): module redis is not installed")

        ret = []
        for term in terms:   #返回连接路径
            (url, key) = term.split(',')
            if url == "":
                url = 'redis://localhost:6379'  

            # urlsplit on Python 2.6.1 is broken. Hmm. Probably also the reason
            # Redis' from_url() doesn't work here.

            p = '(?P<scheme>[^:]+)://?(?P<host>[^:/ ]+).?(?P<port>[0-9]*).*'

            try:   #正则匹配来获取主机信息和端口号
                m = re.search(p, url)
                host = m.group('host')
                port = int(m.group('port'))
            except AttributeError:
                raise AnsibleError("Bad URI in redis lookup")

            try:
                conn = redis.Redis(host=host, port=port)  #连接获取对应的值
                res = conn.get(key)
                if res is None:
                    res = ""
                ret.append(res)
            except Exception:
                ret.append("")  # connection failed or key not found
        return ret  #将值返回

整个过程并不复杂,解析参数、获取连接然后取值。
与callback类型的插件一样,都是自定义一个类,名字名字应该是固定格式,LookupModule,
这个类必须继承LookupBase这个基础类,往前继承AnsiblePlugin这个基础类。
如果你要自定义一个lookup插件,自定义LookupModule这个类就行了。

posted @ 2020-04-07 23:52  明王不动心  阅读(611)  评论(0编辑  收藏  举报