Ansible AdHoc & playbook API + 动态生成Inventory +结果关注

为以后用的方便,记录一下(主要是怕忘,又得折腾半天)

直接贴代码,没太多注释,看不懂的看下源码。Pycharm+b

Ansible 2.0 之后的 API 比 2.0 之前要复杂,但使用起来的自由度更好,可根据自己需求修改 Ansible API 的使用方法;功能也更强大。

我主要是使用这个 API 配合 Djcelery 实现监控系统的数据采集功能,好处是不再需要每中服务器再开发一个agent。这样使用的问题可能主要是程序的负载性能及程序并发性能,我还没测试。

-- coding:utf8 --

import os
import sys
import logging
logger = logging.getLogger(‘django‘)

from collections import namedtuple

from ansible.inventory import Inventory
from ansible.vars import VariableManager
from ansible.parsing.dataloader import DataLoader
from ansible.inventory.group import Group
from ansible.inventory.host import Host
from ansible.playbook.play import Play
from ansible.executor.playbook_executor import PlaybookExecutor
from ansible.executor.task_queue_manager import TaskQueueManager
from ansible.plugins.callback import CallbackBase

OPTIONS = namedtuple(‘OPTIONS‘,

[‘module_path‘, ‘extra_vars‘, ‘forks‘, ‘become‘, ‘become_method‘, ‘become_user‘,

‘become_ask_pass‘, ‘connection‘, ‘timeout‘, ‘poll_interval‘, ‘check‘, ‘diff‘]

)

options = OPTIONS(module_path=None, extra_vars=None, forks=5, become=None, become_method=None, become_user=None,

become_ask_pass=None, connection=‘smart‘, timeout=10, poll_interval=15, check=False, diff=3)

class Options(object):

"""
Initialize options class to replace Ansible OptParser
"""

def __init__(self, module_path=None, extra_vars=None, forks=10, become=None, become_method=None, become_user=None,
             become_ask_pass=None, connection=‘smart‘, timeout=2, poll_interval=15, check=False, diff=3,
             listtasks=None, listhosts=None, listtags=None, syntax=None):
    self.module_path = module_path
    self.extra_vars = extra_vars
    self.forks = forks
    self.become = become
    self.become_method = become_method
    self.become_user = become_user
    self.become_ask_pass = become_ask_pass
    self.connection = connection
    self.timeout = timeout
    self.poll_interval = poll_interval
    self.check = check
    self.diff = diff
    self.listhosts = listhosts
    self.listtasks = listtasks
    self.listtags = listtags
    self.syntax = syntax

class ResultsCallback(CallbackBase):

‘‘‘
Callback the result of execute AdHoc and playbook
‘‘‘

def __init__(self, *args, **kwargs):
    super(ResultsCallback, self).__init__(*args, **kwargs)
    self.host_ok = {}
    self.host_unreachable = {}
    self.host_failed = {}

def v2_runner_on_unreachable(self, result):
    self.host_unreachable[result._host.get_name()] = result

def v2_runner_on_ok(self, result,  *args, **kwargs):
    self.host_ok[result._host.get_name()] = result

def v2_runner_on_failed(self, result,  *args, **kwargs):
    self.host_failed[result._host.get_name()] = result

def v2_runner_on_async_poll(self, result):
    self.host_ok.setdefault(‘async_poll‘, result)

def v2_runner_on_async_ok(self, result):
    self.host_ok.setdefault(‘async_ok‘, result)

def v2_runner_on_async_failed(self, result):
    self.host_failed.setdefault(‘async_failed‘, result)

class DynamicInventory(Inventory):

def __init__(self, resource, loader, variable_manager):

    ‘‘‘
    @resource:
    {
        "group1": {
            "hosts": [{"hostname": "10.0.0.0", "port": "22", "username": "test", "password": "pass"}, ...],
            "vars": {"var1": value1, "var2": value2, ...}
        }
    }
    ‘‘‘

    self.resource = resource
    self.inventory = Inventory(loader=loader, variable_manager=variable_manager, host_list=[])
    self.gen_inventory()

def Dynamic_add_group(self, hosts, groupname, groupvars=None):
    NewGroup = Group(name=groupname)

    # if group variables exists, add them to group
    if groupvars:
        for key, value in groupvars.iteritems():
            NewGroup.set_variable(key, value)

    # add hosts to group
    for host in hosts:
        # set connection variables
        hostname = host.get("hostname")
        hostport = host.get("port")
        username = host.get("username")
        password = host.get("password")
        # ssh_key = host.get("ssh_key")
        GeneralHost = Host(name=hostname, port=hostport)
        GeneralHost.set_variable(‘ansible_ssh_host‘, hostname)
        GeneralHost.set_variable(‘ansible_ssh_port‘, hostport)
        GeneralHost.set_variable(‘ansible_ssh_user‘, username)
        GeneralHost.set_variable(‘ansible_ssh_pass‘, password)
        # GeneralHost.set_variable(‘ansible_ssh_private_key_file‘, ssh_key)

        # set other variables
        for key, value in host.iteritems():
            if key not in ["hostname", "port", "username", "password"]:
            # if key not in ["hostname", "port"]:
                GeneralHost.set_variable(key, value)
        # add to group
        NewGroup.add_host(GeneralHost)

    return self.inventory.add_group(NewGroup)

def gen_inventory(self):

    """
    Dynamic add host to inventory.
    """

    if isinstance(self.resource, list):
        self.Dynamic_add_group(self.resource, ‘default_group‘)
    elif isinstance(self.resource, dict):
        for groupname, hosts_and_vars in self.resource.iteritems():
            self.Dynamic_add_group(hosts_and_vars.get("hosts"), groupname, hosts_and_vars.get("vars"))

class AnsibleAPI(object):

def __init__(self, resource):

    ‘‘‘
    @resource type: dict
    {
        "group1": {
            "hosts": [{"hostname": "10.0.0.0", "port": "22", "username": "test", "password": "pass"}, ...],
            "vars": {"var1": value1, "var2": value2, ...}
        }
    }
    @resource: list
    {
        "hosts": [{"hostname": "10.0.0.0", "port": "22", "username": "test", "password": "pass"}
    }
    ‘‘‘
    self.variable_manager = VariableManager()
    self.loader = DataLoader()
    self.options = Options()
    self.passwords = dict()
    self.results_raw = {‘success‘: {}, ‘failed‘: {}, ‘unreachable‘: {}}
    self.resource = resource
    self.callback = ResultsCallback()
    self.inventory = DynamicInventory(self.resource, self.loader, self.variable_manager).inventory
    self.variable_manager.set_inventory(self.inventory)

    if isinstance(self.resource, list):
        self.host_list = map(str, self.inventory.get_hosts(pattern=‘default_group‘))

    if isinstance(self.resource, dict):
        self.host_list = []
        for groupname in self.resource.keys():
            self.host_list.extend(self.inventory.get_group(groupname).get_hosts())
        self.host_list = map(str, self.host_list)

def PrivateAdHoc(self, module_name, module_args=‘‘):

    """
    run module from andible ad-hoc.
    module_name: ansible module_name
    module_args: ansible module args
    """

    # create play with tasks
    play_source = dict(
            name="Ansible Play",
            hosts=self.host_list,
            gather_facts=‘no‘,
            tasks=[dict(action=dict(module=module_name, args=module_args))]
    )
    play = Play().load(play_source, variable_manager=self.variable_manager, loader=self.loader)

    # Actually run it
    tqm = None
    try:
        tqm = TaskQueueManager(
                inventory=self.inventory,
                variable_manager=self.variable_manager,
                loader=self.loader,
                options=self.options,
                passwords=self.passwords,
                stdout_callback=self.callback
        )
        result = tqm.run(play)
    finally:
        if tqm is not None:
            tqm.cleanup()

    return result

def PrivatePlaybook(self, playbooks):

    """
    run ansible palybook and get ran result
    """

    try:
        if not os.path.exists(playbooks):
            logger.error(‘No such file: %s‘ % playbooks)
            sys.exit()
        self.executor = PlaybookExecutor(playbooks=[playbooks, ],
                                         inventory=self.inventory,
                                         variable_manager=self.variable_manager,
                                         loader=self.loader,
                                         options=self.options,
                                         passwords=self.passwords
                                         )

        print self.inventory.get_group_dict()

        self.executor._tqm._stdout_callback = self.callback
        result = self.executor.run()

    except Exception as e:
        print "Error: %s" % e
    finally:
        return result

def playbook_result(self):

    for host, result_object in self.executor._tqm._stdout_callback.host_ok.iteritems():
        self.results_raw[‘success‘][host] = result_object._result

    for host, result_object in self.executor._tqm._stdout_callback.host_failed.iteritems():
        self.results_raw[‘failed‘][host] = result_object._result

    for host, result_object in self.executor._tqm._stdout_callback.host_unreachable.iteritems():
        self.results_raw[‘failed‘][host] = result_object._result

    return self.results_raw

def AdHoc_result(self):

    for host, result in self.callback.host_ok.items():
        self.results_raw[‘success‘][host] = result._result

    for host, result in self.callback.host_failed.items():
        try:
            self.results_raw[‘failed‘][host] = result._result[‘stderr‘]
        except KeyError:
            self.results_raw[‘failed‘][host] = result._result[‘msg‘]

    for host, result in self.callback.host_unreachable.items():
        self.results_raw[‘unreachable‘][host] = result._result[‘msg‘]

    return self.results_raw

if name == ‘main‘:

print ‘complete‘
    #### 以下是测试成功示例,主要关注如何传参
# example = AnsibleAPI([
    # dict(hostname=‘10.10.181.132‘, port=20003, username=‘root‘, password=123456),
# ])
# example.PrivateAdHoc(‘service‘, module_args=‘name=crond state=reloaded‘)
# example.PrivateAdHoc(‘setup‘)
# print example.AdHoc_result()
# example = AnsibleAPI(dict(memory=dict(hosts=[dict(hostname=‘10.10.181.132‘, port=20003, username=‘root‘, password=123456)], vars=dict())))
# print example.PrivatePlaybook(‘/Users/fanolee/PycharmProjects/AutoOPPlatform/monitor/ansible/playbooks/entry_files/memory/memory.yml‘)
# print example.playboot_result()

原文地址:http://blog.51cto.com/784687488/2171821

时间: 2024-11-11 10:54:56

Ansible AdHoc & playbook API + 动态生成Inventory +结果关注的相关文章

Ansible 自动化运维工具之inventory和常用模块介绍

一.inventory介绍 前面我们介绍过ansible的hosts文件是存放被管理主机的,被管理主机比较少的情况下,直接在hosts中定义即可,但是以后很定会管理多台主机,而ansible可管理的主机集合就叫做inventory.在ansible中,描述你主机的默认方法是将它们列在一个文本文件中,这个文件叫inventory文件. 一个简单的inventory文件可能只包含一组主机名的列表,如下: ftp.testansible.com samba.testansible.com mail.t

Python调用ansible API系列(四)动态生成hosts文件

方法一:通过最原始的操作文件的方式 #!/usr/bin/env python # -*- coding: utf-8 -*- """ 通过操作文件形式动态生成ansible的hosts文件 """ import sys class Inventory: def __init__(self): # ansible的hosts文件路径 self._hostsfile = "./aaa" self._data = self._ge

Ansible的playbook的基本使用与快速入门

1.使用playbook有什么好处2.认识playbook(自动部署nginx)3.YAML语法4.playbook文件结构5.在变更时执行操作(handlers)6.任务控制(tags)7.playbook文件调试8.案例:自动部署tomcat9.Playbook变量定义与使用 playbok文件复用 1. 使用playbook有什么好处 Ansible真正的能力在于使用playbook,也就称为剧本 当我们在某个目录下执行某个命令的时候,那么我们可以使用ansible的ad hoc的模式快速

自动化运维工具Ansible之Python API

Ansible 的Python API使用起来相当简单快捷,使用API可以将某些运维操作封装成一个带有WEB界面的操作,免去了每次执行某个操作的时候都需要SSH运行Ansible命令. 官方给出的一个简单示例: import ansible.runner          runner = ansible.runner.Runner(        module_name='ping',        module_args='',        pattern='web*',        f

ansible 2.7 API

# coding:utf-8 # @Time : 2019-01-14 15:22 # @Author : 小贰 # @FileName: ansible_sync_hosts.py # @function: About ansible 2.7 API import json from collections import namedtuple from ansible.parsing.dataloader import DataLoader from ansible.vars.manager

[搬运自我的CSDN博客] python抓取javascript动态生成HTML内容的实践

<注:CSDN博客在美国访问特别卡,所以转移到cnblogs来发文章> 本实验在Ubuntu14.04上完成.使用的浏览器是火狐(Firefox 33.0),python版本是2.7.6. 大家都知道用urllib配合正则表达式抓取静态HTML的内容很方便,但是如果网页中有javascript动态生成的内容,urllib就无能为力了. 此时我们要借助一个额外的工具:selenium.它的工作原理是操纵(火狐)浏览器浏览目标网页,等待网页中的javascript全部执行完毕后再对HTML源码进行

如何用js为动态生成的元素绑定事件

今天在开发项目的时候,需要为用js动态生成的元素绑定一个click事件,但是在页面加载之前,这个元素不存在,没法获取这个元素,然后为其添加事件.查阅jquery api 发现可以用on()事件(注意:新版本的jquery已经把live()去除,现在用on()),为动态生产的元素绑定事件,然后触发该事件执行.比如下面一个例子: 假设我们要给div动态添加的span绑定click事件形成如下结果 x //做法如下: ("#choose_result").on("click&quo

ABP展现层——动态生成WebApi

ABP展现层——动态生成WebApi 点这里进入ABP系列文章总目录 ABP(现代ASP.NET样板开发框架)系列之20.ABP展现层——动态生成WebApi ABP是“ASP.NET Boilerplate Project (ASP.NET样板项目)”的简称. ABP的官方网站:http://www.aspnetboilerplate.com ABP在Github上的开源项目:https://github.com/aspnetboilerplate 建立动态WebApi控制器 Abp框架能够通

IT咨询服务-客户案例(四):根据图片等素材,动态生成个性化图片

最近,慕名而来一个客户. 客户的目标    网站来一个用户,选择模版,上传图片等素材,生成自定义的图片,或者静态网站,然后分享到社交网站.有点类似于,网上制作名片,选择模版,输入个人信息,生成名片的设计图.可以下载名片设计图,或者直接让别人打印名片,寄给自己.     客户做这个产品的目的,是自己用,还是卖给别人,我是不用关心的.我需要做的,就是帮助客户实现他的目标. 客户的背景   计算机专业毕业,有iOS UI设计经验,了解Object-C和Java,对于JavaScript等Web前端技术