Pythonモニタサーバの利器--psutil


サーバのモニタリングは、一般的なモニタリングソフトウェアをインストールするほか、shellやPythonスクリプトを実行する必要がある場合があります.shellの下ではシステムが持参したps/free/top/dfなどのshellコマンド、Pythonはsubprocessなどのモジュールを呼び出してshellコマンドを実行することができますが、これは面倒です.ここには使いやすいサードパーティモジュールがあります:psutil.psutilはプラットフォームにまたがるライブラリで、Pythonで実行プロセスとシステム利用率(CPU、メモリ、ディスク、ネットワーク、センサ)に関する情報を取得します.主にシステム監視、分析、プロセスリソースの制限、プロセスの実行管理に使用されます.UNIXコマンドラインツールでは、ps、top、lsof、netstat、ifconfig、who、df、kill、free、nice、iostat、iotop、uptime、pidof、tty、taskset、pmapなど、多くの機能を実現しています.psutilは現在、以下のプラットフォームをサポートしています.
Linux  
Windows
OSX,
FreeBSD, OpenBSD, NetBSD
Sun Solaris
AIX
...

Python 2がインストールされています.6~3.6の32-bitおよび64-bitアーキテクチャ.PyPyで運転することもできます.インストール文の例は、Pythonバージョン3.6の環境で実行されています.# pip3 install psutil共通モジュールpsutilバージョン情報の取得
In [1]: import psutil
In [2]: psutil.version_info
Out[2]: (5, 4, 3

CPU情報の取得
In [3]: psutil.cpu_count() #   CPU  
Out[3]: 4
In [4]: psutil.cpu_count(logical=False) #   CPU  
Out[4]: 2
In [5]: psutil.cpu_times() # CPU   、  、    
Out[5]: scputimes(user=240773.0, nice=0.0, system=96416.32, idle=1161930.41)
In [9]: psutil.cpu_percent(percpu=True) #     CPU    ,  TOP  
Out[9]: [43.3, 22.0, 42.0, 23.0]
In [10]: top = [psutil.cpu_percent(interval=i, percpu=True) for i in range(10)] #          ,       
In [11]: top
Out[11]:
[[40.8, 19.7, 38.5, 20.7],
[25.7, 5.9, 13.0, 5.0],
[35.0, 15.6, 30.0, 14.4],
[23.7, 7.0, 18.3, 7.4],
[38.5, 17.0, 34.2, 17.5],
[37.2, 19.6, 36.3, 20.0],
[29.6, 16.6, 28.8, 16.8],
[37.7, 19.0, 35.4, 18.7],
[30.8, 16.3, 26.9, 16.5],
[44.2, 27.9, 41.5, 28.6]]

メモリ情報の取得
In [13]: psutil.virtual_memory() #        ,  bytes,   8G  
Out[13]: svmem(total=8589934592, available=1891045376, percent=78.0, used=6053986304, free=15130624, active=1878392832, inactive=1875914752, wired=2299678720)
In [14]: psutil.swap_memory() #   swap     
Out[14]: sswap(total=2147483648, used=1340866560, free=806617088, percent=62.4, sin=126090076160, sout=3524710400)

ディスク情報の取得
In [17]: psutil.disk_partitions() #        
Out[17]: [sdiskpart(device='/dev/disk1', mountpoint='/', fstype='hfs', opts='rw,local,rootfs,dovolfs,journaled,multilabel')]
In [20]: psutil.disk_usage('/') #         ,     25.4%
Out[20]: sdiskusage(total=499055067136, used=126482944000, free=372309979136, percent=25.4)
In [22]: psutil.disk_io_counters() #  IO  
Out[22]: sdiskio(read_count=7364142, write_count=6510641, read_bytes=282106464256, write_bytes=261763244544, read_time=2608778, write_time=1095259)

ネットワーク情報の取得
In [23]: psutil.net_if_stats() #         
Out[23]:
{'awdl0': snicstats(isup=True, duplex=, speed=0, mtu=1484),
'bridge0': snicstats(isup=True, duplex=, speed=0, mtu=1500),
'en0': snicstats(isup=True, duplex=, speed=0, mtu=1500),
'en1': snicstats(isup=True, duplex=, speed=0, mtu=1500),
'en2': snicstats(isup=True, duplex=, speed=0, mtu=1500),
'gif0': snicstats(isup=False, duplex=, speed=0, mtu=1280),
'lo0': snicstats(isup=True, duplex=, speed=0, mtu=16384),
'p2p0': snicstats(isup=True, duplex=, speed=0, mtu=2304),
'stf0': snicstats(isup=False, duplex=, speed=0, mtu=1280),
'utun0': snicstats(isup=True, duplex=, speed=0, mtu=2000),
'utun1': snicstats(isup=True, duplex=, speed=0, mtu=1352)}
In [25]: psutil.net_if_stats().get("en0") #      en0   
Out[25]: snicstats(isup=True, duplex=, speed=0, mtu=1500)
In [26]: psutil.net_if_addrs() #            
Out[26]:
{'awdl0': [snic(family=, address='36:7d:f3:80:6e:4e', netmask=None, broadcast=None, ptp=None),
snic(family=, address='fe80::347d:f3ff:fe80:6e4e%awdl0', netmask='ffff:ffff:ffff:ffff::', broadcast=None, ptp=None)],
'bridge0': [snic(family=, address='4a:00:02:c0:33:70', netmask=None, broadcast=None, ptp=None)],
'en0': [snic(family=, address='192.168.0.101', netmask='255.255.255.0', broadcast='192.168.0.255', ptp=None),
snic(family=, address='ac:bc:32:91:32:8b', netmask=None, broadcast=None, ptp=None),
snic(family=, address='fe80::1476:ce7e:210a:2e32%en0', netmask='ffff:ffff:ffff:ffff::', broadcast=None, ptp=None)],
'en1': [snic(family=, address='4a:00:02:c0:33:70', netmask=None, broadcast=None, ptp=None)],
'en2': [snic(family=, address='4a:00:02:c0:33:71', netmask=None, broadcast=None, ptp=None)],
'lo0': [snic(family=, address='127.0.0.1', netmask='255.0.0.0', broadcast=None, ptp=None),
snic(family=, address='::1', netmask='ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff', broadcast=None, ptp=None),
snic(family=, address='fe80::1%lo0', netmask='ffff:ffff:ffff:ffff::', broadcast=None, ptp=None)],
'p2p0': [snic(family=, address='0e:bc:32:91:32:8b', netmask=None, broadcast=None, ptp=None)],
'utun0': [snic(family=, address='fe80::583c:77a0:6b93:b045%utun0', netmask='ffff:ffff:ffff:ffff::', broadcast=None, ptp=None)],
'utun1': [snic(family=, address='10.5.200.244', netmask=None, broadcast=None, ptp='10.5.200.244')]}
#   en0     ,     mac ipv6  
In [40]: for addr in psutil.net_if_addrs().get("en0"):
...: print(addr.address)
192.168.0.101
ac:bc:32:91:32:8b
fe80::1476:ce7e:210a:2e32%en0
In [43]: psutil.net_io_counters() #         /    
Out[43]: snetio(bytes_sent=174614221, bytes_recv=586279725, packets_sent=863903, packets_recv=873583, errin=0, errout=0, dropin=0, dropout=0)
In [45]: psutil.net_connections() #         ,      root  。

プロセス情報の取得:
In [46]: psutil.pids() #       ID
In [47]: psutil.Process(61) #     PID     
Out[47]: psutil.Process(pid=61, name='dsAccessService', started='2018-02-26 09:57:04')
In [49]: psutil.Process(45573).exe() #      exe  
Out[49]: '/usr/local/Cellar/python3/3.6.3/Frameworks/Python.framework/Versions/3.6/Resources/Python.app/Contents/MacOS/Python'
In [50]: psutil.Process(45573).name() #       
Out[50]: 'Python'
In [52]: psutil.Process(45573).cmdline() #          
Out[52]:
['/usr/local/Cellar/python3/3.6.3/Frameworks/Python.framework/Versions/3.6/Resources/Python.app/Contents/MacOS/Python',
'/usr/local/bin/ptipython']
In [56]: psutil.Process(45573).num_threads() #          
Out[56]: 3
In [57]: psutil.Process(45573).environ() #            

まとめ:psutilモジュールを使用すると、システムのモニタリングを比較的包括的に行うことができます.Pythonでモニタリングシステムやスクリプトツールを作ることを考えている場合は、このモジュールを考慮することができます.