|
|
|
#coding: utf-8
|
|
|
|
#-------------------------------------------------------------------
|
|
|
|
# 宝塔Linux面板
|
|
|
|
#-------------------------------------------------------------------
|
|
|
|
# Copyright (c) 2015-2099 宝塔软件(http://bt.cn) All rights reserved.
|
|
|
|
#-------------------------------------------------------------------
|
|
|
|
# Author: hwliang<hwl@bt.cn>
|
|
|
|
#-------------------------------------------------------------------
|
|
|
|
|
|
|
|
import sys
|
|
|
|
import time
|
|
|
|
import os
|
|
|
|
import struct
|
|
|
|
|
|
|
|
os.chdir('/www/server/mdserver-web')
|
|
|
|
if 'class/' in sys.path: sys.path.insert(0,"class/")
|
|
|
|
import copy
|
|
|
|
try:
|
|
|
|
import pcap
|
|
|
|
except ImportError:
|
|
|
|
if os.path.exists('/usr/bin/apt'):
|
|
|
|
os.system("apt install libpcap-dev -y")
|
|
|
|
elif os.path.exists('/usr/bin/dnf'):
|
|
|
|
red_file = '/etc/redhat-release'
|
|
|
|
if os.path.exists(red_file):
|
|
|
|
f = open(red_file,'r')
|
|
|
|
red_body = f.read()
|
|
|
|
f.close()
|
|
|
|
if red_body.find('CentOS Linux release 8.') != -1:
|
|
|
|
rpm_file = '/root/libpcap-1.9.1.rpm'
|
|
|
|
down_url = "wget -O {} https://repo.almalinux.org/almalinux/8/PowerTools/x86_64/os/Packages/libpcap-devel-1.9.1-5.el8.x86_64.rpm --no-check-certificate -T 10".format(rpm_file)
|
|
|
|
print(down_url)
|
|
|
|
os.system(down_url)
|
|
|
|
os.system("rpm -ivh {}".format(rpm_file))
|
|
|
|
if os.path.exists(rpm_file): os.remove(rpm_file)
|
|
|
|
else:
|
|
|
|
os.system("dnf install libpcap-devel -y")
|
|
|
|
else:
|
|
|
|
os.system("dnf install libpcap-devel -y")
|
|
|
|
elif os.path.exists('/usr/bin/yum'):
|
|
|
|
os.system("yum install libpcap-devel -y")
|
|
|
|
|
|
|
|
os.system("pip install pypcap")
|
|
|
|
try:
|
|
|
|
import pcap
|
|
|
|
except ImportError:
|
|
|
|
print("pypcap module install failed.")
|
|
|
|
sys.exit()
|
|
|
|
|
|
|
|
class process_network_total:
|
|
|
|
__pid_file = 'logs/process_network_total.pid'
|
|
|
|
__inode_list = {}
|
|
|
|
__net_process_list = {}
|
|
|
|
__net_process_size = {}
|
|
|
|
__last_stat = 0
|
|
|
|
__last_write_time = 0
|
|
|
|
__end_time = 0
|
|
|
|
|
|
|
|
def start(self,timeout = 0):
|
|
|
|
stime = time.time()
|
|
|
|
self.__end_time = timeout + stime
|
|
|
|
self.__last_stat = stime
|
|
|
|
try:
|
|
|
|
p = pcap.pcap() # 监听所有网卡
|
|
|
|
p.setfilter('tcp') # 只监听TCP数据包
|
|
|
|
for p_time,p_data in p:
|
|
|
|
self.handle_packet(p_data)
|
|
|
|
# 过期停止监听
|
|
|
|
if timeout > 0:
|
|
|
|
if p_time > self.__end_time:
|
|
|
|
self.rm_pid_file()
|
|
|
|
break
|
|
|
|
except:
|
|
|
|
self.rm_pid_file()
|
|
|
|
|
|
|
|
def handle_packet(self, pcap_data):
|
|
|
|
# 获取IP协议头
|
|
|
|
ip_header = pcap_data[14:34]
|
|
|
|
# 解析src/dst地址
|
|
|
|
src_ip = ip_header[12:16]
|
|
|
|
dst_ip = ip_header[16:20]
|
|
|
|
# 解析sport/dport端口
|
|
|
|
src_port = pcap_data[34:36]
|
|
|
|
dst_port = pcap_data[36:38]
|
|
|
|
|
|
|
|
src = src_ip + b':' + src_port
|
|
|
|
dst = dst_ip + b':' + dst_port
|
|
|
|
# 计算数据包长度
|
|
|
|
pack_size = len(pcap_data)
|
|
|
|
# 统计进程流量
|
|
|
|
self.total_net_process(dst,src,pack_size)
|
|
|
|
|
|
|
|
def total_net_process(self,dst,src,pack_size):
|
|
|
|
self.get_tcp_stat()
|
|
|
|
direction = None
|
|
|
|
mtime = time.time()
|
|
|
|
if dst in self.__net_process_list:
|
|
|
|
pid = self.__net_process_list[dst]
|
|
|
|
direction = 'down'
|
|
|
|
elif src in self.__net_process_list:
|
|
|
|
pid = self.__net_process_list[src]
|
|
|
|
direction = 'up'
|
|
|
|
else:
|
|
|
|
if mtime - self.__last_stat > 3:
|
|
|
|
self.__last_stat = mtime
|
|
|
|
self.get_tcp_stat(True)
|
|
|
|
if dst in self.__net_process_list:
|
|
|
|
pid = self.__net_process_list[dst]
|
|
|
|
direction = 'down'
|
|
|
|
elif src in self.__net_process_list:
|
|
|
|
pid = self.__net_process_list[src]
|
|
|
|
direction = 'up'
|
|
|
|
|
|
|
|
if not direction: return False
|
|
|
|
if not pid: return False
|
|
|
|
if not pid in self.__net_process_size:
|
|
|
|
self.__net_process_size[pid] = {}
|
|
|
|
self.__net_process_size[pid]['down'] = 0
|
|
|
|
self.__net_process_size[pid]['up'] = 0
|
|
|
|
self.__net_process_size[pid]['up_package'] = 0
|
|
|
|
self.__net_process_size[pid]['down_package'] = 0
|
|
|
|
|
|
|
|
self.__net_process_size[pid][direction] += pack_size
|
|
|
|
self.__net_process_size[pid][direction + '_package'] += 1
|
|
|
|
|
|
|
|
# 写入到文件
|
|
|
|
if mtime - self.__last_write_time > 1:
|
|
|
|
self.__last_write_time = mtime
|
|
|
|
self.write_net_process()
|
|
|
|
|
|
|
|
def write_net_process(self):
|
|
|
|
w_file = '/dev/shm/mw_net_process'
|
|
|
|
process_size = copy.deepcopy(self.__net_process_size)
|
|
|
|
net_process = []
|
|
|
|
for pid in process_size.keys():
|
|
|
|
net_process.append(str(pid) + " " + str(process_size[pid]['down']) + " " + str(process_size[pid]['up']) + " " + str(process_size[pid]['down_package']) + " " + str(process_size[pid]['up_package']))
|
|
|
|
|
|
|
|
f = open(w_file,'w+',encoding='utf-8')
|
|
|
|
f.write('\n'.join(net_process))
|
|
|
|
f.close()
|
|
|
|
|
|
|
|
def hex_to_ip(self, hex_ip):
|
|
|
|
hex_ip,hex_port = hex_ip.split(':')
|
|
|
|
ip = '.'.join([str(int(hex_ip[i:i+2], 16)) for i in range(0, len(hex_ip), 2)][::-1])
|
|
|
|
port = int(hex_port, 16)
|
|
|
|
return ip,port
|
|
|
|
|
|
|
|
def get_tcp_stat(self,force = False):
|
|
|
|
if not force and self.__net_process_list: return self.__net_process_list
|
|
|
|
self.__net_process_list = {}
|
|
|
|
tcp_stat_file = '/proc/net/tcp'
|
|
|
|
tcp_stat = open(tcp_stat_file, 'rb')
|
|
|
|
tcp_stat_list = tcp_stat.read().decode('utf-8').split('\n')
|
|
|
|
tcp_stat.close()
|
|
|
|
tcp_stat_list = tcp_stat_list[1:]
|
|
|
|
if force: self.get_process_inodes(force)
|
|
|
|
for i in tcp_stat_list:
|
|
|
|
tcp_tmp = i.split()
|
|
|
|
if len(tcp_tmp) < 10: continue
|
|
|
|
inode = tcp_tmp[9]
|
|
|
|
if inode == '0': continue
|
|
|
|
local_ip,local_port = self.hex_to_ip(tcp_tmp[1])
|
|
|
|
if local_ip == '127.0.0.1': continue
|
|
|
|
remote_ip,remote_port = self.hex_to_ip(tcp_tmp[2])
|
|
|
|
if local_ip == remote_ip: continue
|
|
|
|
if remote_ip == '0.0.0.0': continue
|
|
|
|
|
|
|
|
pid = self.inode_to_pid(inode,force)
|
|
|
|
if not pid: continue
|
|
|
|
|
|
|
|
key = self.get_ip_pack(local_ip) + b':' + self.get_port_pack(local_port)
|
|
|
|
self.__net_process_list[key] = pid
|
|
|
|
return self.__net_process_list
|
|
|
|
|
|
|
|
|
|
|
|
def get_port_pack(self,port):
|
|
|
|
return struct.pack('H',int(port))[::-1]
|
|
|
|
|
|
|
|
def get_ip_pack(self,ip):
|
|
|
|
ip_arr = ip.split('.')
|
|
|
|
ip_pack = b''
|
|
|
|
for i in ip_arr:
|
|
|
|
ip_pack += struct.pack('B',int(i))
|
|
|
|
return ip_pack
|
|
|
|
|
|
|
|
def inode_to_pid(self,inode,force = False):
|
|
|
|
inode_list = self.get_process_inodes()
|
|
|
|
if inode in inode_list:
|
|
|
|
return inode_list[inode]
|
|
|
|
return None
|
|
|
|
|
|
|
|
def get_process_inodes(self,force = False):
|
|
|
|
if not force and self.__inode_list: return self.__inode_list
|
|
|
|
proc_path = '/proc'
|
|
|
|
inode_list = {}
|
|
|
|
for pid in os.listdir(proc_path):
|
|
|
|
try:
|
|
|
|
if not pid.isdigit(): continue
|
|
|
|
inode_path = proc_path + '/' + pid + '/fd'
|
|
|
|
for fd in os.listdir(inode_path):
|
|
|
|
try:
|
|
|
|
fd_file = inode_path + '/' + fd
|
|
|
|
fd_link = os.readlink(fd_file)
|
|
|
|
if fd_link.startswith('socket:['):
|
|
|
|
inode = fd_link[8:-1]
|
|
|
|
inode_list[inode] = pid
|
|
|
|
except:
|
|
|
|
continue
|
|
|
|
except:
|
|
|
|
continue
|
|
|
|
self.__inode_list = inode_list
|
|
|
|
return inode_list
|
|
|
|
|
|
|
|
def get_process_name(self,pid):
|
|
|
|
pid_path = '/proc/' + pid + '/comm'
|
|
|
|
if not os.path.exists(pid_path): return ''
|
|
|
|
pid_file = open(pid_path, 'rb')
|
|
|
|
pid_name = pid_file.read().decode('utf-8').strip()
|
|
|
|
pid_file.close()
|
|
|
|
return pid_name
|
|
|
|
|
|
|
|
|
|
|
|
def write_pid(self):
|
|
|
|
self_pid = os.getpid()
|
|
|
|
pid_file = open(self.__pid_file,'w')
|
|
|
|
pid_file.write(str(self_pid))
|
|
|
|
pid_file.close()
|
|
|
|
|
|
|
|
def rm_pid_file(self):
|
|
|
|
if os.path.exists(self.__pid_file):
|
|
|
|
os.remove(self.__pid_file)
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
if len(sys.argv) > 1:
|
|
|
|
timeout = int(sys.argv[-1])
|
|
|
|
else:
|
|
|
|
timeout = 0
|
|
|
|
p = process_network_total()
|
|
|
|
p.write_pid()
|
|
|
|
p.start(timeout)
|