因公司服務器上部署應用較多,在有大併發訪問、業務邏輯有問題的狀況下反覆互相調用或者有異常流量訪問的時候,須要對業務應用進行故障定位,因此利用python調用iftop命令來獲取應用進程流量,結合zabbix,可幫助定位分析問題。,如下是腳本內容,大概思路是:python
#!/usr/bin/python #coding=utf-8 #針對業務監聽的端口流量進行統計,忽略對隨機端口流量統計 #若針對忽然流量增大,找到其進程進行告警,能夠不作統計,獲取到流量進行判斷,若大於多少閥值,則輸出 import os def change_unit(unit): if "Mb" in unit: flow = float(unit.strip("Mb")) * 1024 return flow elif "Kb" in unit: flow = float(unit.strip("Kb")) return flow elif "b" in unit: flow = float(unit.strip("b")) / 1024 return flow def get_flow(): #iftop參數:-t 使用不帶ncurses的文本界面,-P顯示主機以及端口信息,-N只顯示鏈接端口號,不顯示端口對應的服務名稱,-n 將輸出的主機信息都經過IP顯示,不進行DNS解析,-s num num秒後打印一次文本輸出而後退出 mes = os.popen("iftop -t -P -N -n -s 2 2>/dev/null |grep -A 1 -E '^ [0-9]'").read() #以換行符進行分割 iftop_list = mes.split("\n") count = len(iftop_list) #定義字典 存放主機信息和進出流量 flow_dict = {} #定義列表,存放主機信息 host_ips = [] # 把主機加入數組,新的主機查詢是否在列表裏面,沒有的話,把主機信息加入host_ips,並新組裝一個字典值加入flow_dict字典,若是host_ips存在主機信息,則把字典值取出來,從新計算增長流量數值,再加入字典flow_dict #這裏的 count/2 是iftop獲取到的數據,是進出流量爲一組,則有count/2 個流量鏈接,可執行os.popen 裏面的iftop命令便可明白 for i in range(count/2): flow_msg = "" #獲取發送的ip地址(本地ip地址),端口(本地端口),發送的流量,以換行符分割後,數據偶數位爲本地發送流量信息 location_li_s = iftop_list[i*2] send_flow_lists = location_li_s.split(" ") #去空元素 while '' in send_flow_lists: send_flow_lists.remove('') host_ip = send_flow_lists[1] send_flow = send_flow_lists[3] send_flow_float = change_unit(send_flow) #print send_flow_lists #獲取接收的流量 location_li_r = iftop_list[i*2+1] rec_flow_lists = location_li_r.split(" ") while '' in rec_flow_lists: rec_flow_lists.remove('') rec_flow = rec_flow_lists[3] rec_flow_float = change_unit(rec_flow) #去掉本地linux 大於10000的隨機端口,由於公司業務應用無大於10000,也可把這裏去掉 port = host_ip.split(":")[1] if int(port) < 10000: #主機信息若不存在列表則加入host_ips,若存在,則字典取值,對進出流量進行相加 if host_ip not in host_ips: host_ips.append(host_ip) flow_msg = str(float('%2.f' % send_flow_float)) + ":" + str(float('%.2f' % rec_flow_float)) flow_dict[host_ip]=flow_msg else: flow_dict_msg = flow_dict[host_ip] flow_dict_msg_li = flow_dict_msg.split(":") #獲取字典裏的發送接收流量 flow_dict_msg_send = float(flow_dict_msg_li[0]) flow_dict_msg_rec = float(flow_dict_msg_li[1]) #字典裏面的發送接收流量和獲取到的新流量相加 flow_add_send = flow_dict_msg_send + send_flow_float flow_add_rec = flow_dict_msg_rec + rec_flow_float #把新得出的結果,更新到字典 flow_msg = str(float('%.2f' % flow_add_send)) + ":" + str(float('%.2f' % flow_add_rec)) flow_dict[host_ip]=flow_msg for key in flow_dict: flow_li = flow_dict[key].split(":") #flow_li[0]爲發送流量,flow_li[1]爲接收流量,單位是Kb print key + "|" + flow_li[0] + "|" + flow_li[1] get_flow()