如何利用ansible callback插件對執行結果進行解析

来源:http://www.cnblogs.com/ivictor/archive/2016/12/30/6235420.html
-Advertisement-
Play Games

最近在寫一個批量巡檢工具,利用ansible將腳本推到各個機器上執行,然後將執行的結果以json格式返回來。 如下所示: # ansible node2 -m script -a /root/python/health_check.py 然後將結果重定向到一個文本文件中,再通過另外一個腳本,對該文本 ...


最近在寫一個批量巡檢工具,利用ansible將腳本推到各個機器上執行,然後將執行的結果以json格式返回來。

 

如下所示:

# ansible node2 -m script -a /root/python/health_check.py

node2 | SUCCESS => {
    "changed": true, 
    "rc": 0, 
    "stderr": "Shared connection to 192.168.244.20 closed.\r\n", 
    "stdout": "{'cpu_iowait': '0.00', 'swap_out': 0, 'cpu_usr': '0.00', 'cpu_idle': '100.00', 'swap_total': '1999', 'swap_used': '78'
, 'load_average_5': '0.11', 'mem_util': '92.0', 'uptime': '5', 'load_average_1': '0.03', 'cpu_sys': '0.00', 'mem_total': '475', 'swap_in': 0, 'load_average_15': '0.06', 'disk': ['Filesystem      Size  Used Avail Use% Mounted on\\n', '/dev/sda3        18G  8.6G  8.1G  52% /\\n', 'tmpfs           238M     0  238M   0% /dev/shm\\n', '/dev/sda1       190M   27M  154M  15% /boot\\n'], 'numa': '1'}\r\n",     
"stdout_lines": [ "{'cpu_iowait': '0.00', 'swap_out': 0, 'cpu_usr': '0.00', 'cpu_idle': '100.00', 'swap_total': '1999', 'swap_used': '78', 'loa d_average_5': '0.11', 'mem_util': '92.0', 'uptime': '5', 'load_average_1': '0.03', 'cpu_sys': '0.00', 'mem_total': '475', 'swap_in': 0, 'load_average_15': '0.06', 'disk': ['Filesystem Size Used Avail Use% Mounted on\\n', '/dev/sda3 18G 8.6G 8.1G 52% /\\n', 'tmpfs 238M 0 238M 0% /dev/shm\\n', '/dev/sda1 190M 27M 154M 15% /boot\\n'], 'numa': '1'}" ] }

 

然後將結果重定向到一個文本文件中,再通過另外一個腳本,對該文本文件進行解析彙總,最後實現的結果如下:

ip              uptime          cpu_usr         cpu_sys         cpu_iowait      cpu_idle        load_average_1  load_average_5  ...           
192.168.244.30  24              0               0               6               94              0.02            0.08            ...              
192.168.244.20  24              0               0               0               100             0               0.01            ...           

 

但總感覺這種方式有點low,對返回結果進行解析,這似乎是一個比較普遍的需求吧?

 

沒道理,官方會對這種需求視而不見的,其實,官方提供了一個callback插件,來實現回調功能,裡面定義了若幹場景,譬如主機不可達,執行任務失敗,執行任務成功等,分別對應不同的方法,這樣就可以實現在不同的場景觸發不同的操作,譬如,如果執行playbook失敗了就發送郵件等,執行成功了將返回的結果保存到資料庫中。

 

官方給了一個樣例,具體可見:https://github.com/ansible/ansible/blob/devel/lib/ansible/plugins/callback/log_plays.py

 

基於上面這個樣例,自己進行了定製性開發。本來想在callback插件中實現所有功能,但callback插件調試相當麻煩,不允許使用print函數,而且如果出現問題了,譬如列表下標越界,也只是在執行ansible時給出報錯信息,並沒有指出具體的報錯行數。

 

最後,放棄了自己ALL IN ONE的想法,只是將返回的結果解析後保存到sqlite3資料庫中,後續再基於資料庫中的數據進行彙總。

 

代碼如下:

from __future__ import (absolute_import, division, print_function)
__metaclass__ = type

import os
import time
import json
import sqlite3
from ansible.module_utils._text import to_bytes
from ansible.plugins.callback import CallbackBase


class CallbackModule(CallbackBase):
    """
    logs playbook results, per host, in /var/log/ansible/hosts
    """
    CALLBACK_VERSION = 2.0
    CALLBACK_TYPE = 'notification'
    CALLBACK_NAME = 'performance_check'
    CALLBACK_NEEDS_WHITELIST = False

    def __init__(self):
        super(CallbackModule, self).__init__()

    def runner_on_failed(self, host, res, ignore_errors=False):
        pass

    def runner_on_ok(self, host, res):
        performance_data=PerformanceData()
        create_table_sql = 'CREATE TABLE performance_data(ip varchar(20) primary key, uptime varchar(20),cpu_usr DECIMAL,cpu_sys DECI
MAL, cpu_iowait DECIMAL,cpu_idle DECIMAL,load_average_1 DECIMAL,load_average_5 DECIMAL,load_average_15 DECIMAL, mem_total INTEGER,mem_util DECIMAL,swap_total INTEGER,swap_used INTEGER,swap_in INTEGER,swap_out INTEGER,
numa TINYINT)'        
insert_sql =
'insert into performance_data values (?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?)' insert_value = str_to_json(host,res) performance_data.create_table(create_table_sql) performance_data.insert_command(insert_sql,insert_value) performance_data.quit() def runner_on_skipped(self, host, item=None): #self.log(host, 'SKIPPED', '...') pass def runner_on_unreachable(self, host, res): #self.log(host, 'UNREACHABLE', res) pass def runner_on_async_failed(self, host, res, jid): #self.log(host, 'ASYNC_FAILED', res) pass def playbook_on_import_for_host(self, host, imported_file): pass def playbook_on_not_import_for_host(self, host, missing_file): pass class PerformanceData(): def __init__(self): self.conn = sqlite3.connect("/tmp/data.db") self.cursor = self.conn.cursor() def create_table(self,create_table_sql): self.cursor.execute(create_table_sql) def insert_command(self,insert_sql,insert_value): self.cursor.execute(insert_sql,insert_value) def query(self,query_sql): self.cursor.execute(query_sql) results=self.cursor.fetchall() return results def quit(self): self.conn.commit() self.conn.close() def str_to_json(host,res): result= res["stdout"].strip(" ").replace("'",'"').strip('\n').strip('"') results= '{"'+host+'":'+result+'}' result_with_host = json.loads(results) value=result_with_host[host] return (host,value['uptime'],float(value['cpu_usr']),float(value['cpu_sys']),float(value['cpu_iowait']), float(value['cpu_idle']), float(value['load_average_1']), float(value['load_average_5']), float(value['load_average_15 ']), int(value['mem_total']), float(value['mem_util']),int(value['swap_total']),int(value['swap_used']),int(value['swap_in' ]), int(value['swap_out']), int(value['numa']) )

 

這裡一併附上,上述解析文本的腳本,似乎更能實現我ALL IN ONE的想法,哈哈~

#coding: utf8
import re,json,sqlite3
def get_ip_success():
    with open(r'C:\Users\Administrator\Desktop\2.txt') as f:
        ip_unreachable = []
        ip_failed = []
        ip_success=[]
        line_num=0
        for line in f.readlines():
            if re.search('UNREACHABLE', line):
                ip=line.split()[0]
                ip_unreachable.append(ip)
                flag=0
            elif re.search('FAILED',line):
                ip = line.split()[0]
                ip_failed.append(ip)
                flag=0
            elif re.search('SUCCESS',line):
                ip = line.split()[0]
                flag=1
                line_num=1
            elif flag == 1 and line_num == 7:
                line= line.strip(" ").replace("'",'"').strip('\n').strip('"')
                stdout_lines= '{"'+ip+'":'+line+'}'
                stdout_lines_with_ip = json.loads(stdout_lines)
                ip_success.append(stdout_lines_with_ip)
            line_num =line_num + 1
    return ip_success

def os_status_generator(ip_success):
    for os_status in ip_success:
        for key,value in os_status.iteritems():
            yield (key,value['uptime'],float(value['cpu_usr']),float(value['cpu_sys']),float(value['cpu_iowait']),
               float(value['cpu_idle']), float(value['load_average_1']), float(value['load_average_5']), float(value['load_average_15']),
               int(value['mem_total']), float(value['mem_util']),int(value['swap_total']),int(value['swap_used']),int(value['swap_in']),
               int(value['swap_out']), int(value['numa'])
               )

class OsStatus():
    def __init__(self,ip_success):
        try:
            self.conn = sqlite3.connect(":memory:")
            self.cursor = self.conn.cursor()
            self.cursor.execute('''CREATE TABLE os_status
                     (ip varchar(20) primary key, uptime varchar(20),cpu_usr DECIMAL,cpu_sys DECIMAL,cpu_iowait DECIMAL,cpu_idle DECIMAL,
                      load_average_1 DECIMAL,load_average_5 DECIMAL,load_average_15 DECIMAL,mem_total INTEGER,mem_util DECIMAL,
                      swap_total INTEGER,swap_used INTEGER,swap_in INTEGER,swap_out INTEGER,numa TINYINT)''')
            self.cursor.executemany("insert into os_status values (?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?)",os_status_generator(ip_success) )
        except Exception as e:
            print e;

    def query(self,sql):
        self.cursor.execute(sql)
        results=self.cursor.fetchall()
        column_size=len(results[0])
        column_name= [column[0] for column in self.cursor.description]
        for i in range(column_size):
            print column_name[i].ljust(15),
        print
        for each_result in results:
            for i in range(column_size):
                print str(each_result[i]).ljust(15),
            print

    def quit(self):
        try:
            self.cursor.close()
            self.conn.close()
        except Exception as e:
            print e;

ip_success=get_ip_success()
os_status=OsStatus(ip_success)
sql = "select * from os_status"
os_status.query(sql)

 

最後,再提一下ansible中如何開啟callback插件功能,預設是關閉的。

開啟兩個選項:

callback_plugins   = /usr/share/ansible/plugins/callback
bin_ansible_callbacks = True

這兩個是必需的,另外一個選項是

callback_whitelist = performance_check

其中,performance_check對應的是上面callback插件中定義的“CALLBACK_NAME”,

另一個相關參數是“CALLBACK_NEEDS_WHITELIST”,如果設置為False,則無需設置callback_whitelist選項,反之,則必須在callback_whitelist選項中指定“CALLBACK_NAME”。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1.介紹 雙因素認證:雙因素身份認證就是通過你所知道再加上你所能擁有的這二個要素組合到一起才能發揮作用的身份認證系統。雙因素認證是一種採用時間同步技術的系統,採用了基於時間、事件和密鑰三變數而產生的一次性密碼來代替傳統的靜態密碼。每個動態密碼卡都有一個唯一的密鑰,該密鑰同時存放在伺服器端,每次認證時 ...
  • SOLUTION VERIFIED April 27 2013 KB26727 Environment Red Hat Enterprise Linux 5 Red Hat Enterprise Linux 6 Issue How to create a bridge using a tagged ...
  • not on top of a bond Environment Red Hat Enterprise Linux 7 NetworkManager Issue Need an 802.1q VLAN in RHEL 7 when using NetworkManager. Resolution R ...
  • 我新裝的centos7主機無法使用裡面自帶的網卡,查詢後發現網卡型號為BCM4312。我在看資料安裝的過程中遇到了些問題,糾結了好久,現在分享下要註意的點,為後來的遇到同樣問題的人提供點幫助。現在開始說正事: 若要安裝以 Broadcom BCM4311、BCM4312、BCM4313、BCM432 ...
  • environment Red Hat Enterprise Linux 5.4 or later Red Hat Enterprise Linux 6.0 or later KVM virtual machines question How do I configure a bridged net ...
  • 修改環境變數PATH 最近為root添加一個環境變數發現sudo su進去沒有變化所以總結了一下所有設置環境變數的方法: 查看PATH: 1. 直接在命令行修改,就可以使用,但是只有在當前的視窗和用戶有用, 關閉以後就失效了,所以如果是臨時使用可以這樣設置 2. 修改家目錄下.bashrc文件,只對 ...
  • linux用戶分類 超級用戶:UID=0,root 普通用戶:UID 500起,由超級用戶或具有超級用戶許可權的用戶創建的用戶。 虛擬用戶:UID 1—499,為了滿足文件或服務啟動的需要而存在,一般都不能登錄,只是傀儡用戶。 用戶關聯的四個文件: /etc/passwd:用戶的賬號文件 /etc/s... ...
  • 準備工作: 實現lnmp環境 給php添加模塊,so庫 下載擴展包:memcache-2.2.5.tgz wget http://pecl.php.net/get/memcache-2.2.5.tgz ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...