#!/usr/bin/python3 # -*- coding: utf-8 -*- # Copyright (c) 2019 Huawei Technologies Co., Ltd. # A-Tune is licensed under the Mulan PSL v1. # You can use this software according to the terms and conditions of the Mulan PSL v1. # You may obtain a copy of Mulan PSL v1 at: # http://license.coscl.org.cn/MulanPSL # THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR # IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR # PURPOSE. # See the Mulan PSL v1 for more details. # Create: 2019-10-29 """ The sub class of the monitor, used to collect the vm stat info. """ import sys import logging import subprocess import getopt import re if __name__ == "__main__": sys.path.insert(0, "./../../") from monitor.common import * logger = logging.getLogger(__name__) class MemVmstat(Monitor): """To collect the vm stat info""" _module = "MEM" _purpose = "VMSTAT" _option = "{int} 2" def __init__(self, user=None): Monitor.__init__(self, user) self.__cmd = "vmstat" self.__interval = 1 self.decode.__func__.__doc__ = Monitor.decode.__doc__ % ( "--fields=procs.r/procs.b/memory.swpd/memory.free/memory.buff/memory.cache/swap.si/swap.so/io.bi/io.bo/system.in/system.cs/cpu.us/cpu.sy/cpu.id/cpu.wa/cpu.st") def _get(self, para=None): if para is not None: opts, args = getopt.getopt(para.split(), None, ['interval=']) for opt, val in opts: if opt in ('--interval'): if val.isdigit(): self.__interval = int(val) else: err = ValueError( "Invalid parameter: {opt}={val}".format( opt=opt, val=val)) logger.error( "{}.{}: {}".format( self.__class__.__name__, sys._getframe().f_code.co_name, str(err))) raise err continue output = subprocess.check_output( "{cmd} {opt}".format( cmd=self.__cmd, opt=self._option.format( int=self.__interval)).split()) return output.decode() def decode(self, info, para): if para is None: return info keyword = {"procs.r": 0, "procs.b": 1, "memory.swpd": 2, "memory.free": 3, "memory.buff": 4, "memory.cache": 5, "swap.si": 6, "swap.so": 7, "io.bi": 8, "io.bo": 9, "system.in": 10, "system.cs": 11, "cpu.us": 12, "cpu.sy": 13, "cpu.id": 14, "cpu.wa": 15, "cpu.st": 16, "util.swap": "util.swap", "util.cpu": "util.cpu"} keys = [] ret = "" opts, args = getopt.getopt(para.split(), None, ['fields=']) for opt, val in opts: if opt in ('--fields'): keys.append(keyword[val]) continue pattern = re.compile( "^\ ?(\d*)\ {1,}(\d*)\ {1,}(\d*)\ {1,}(\d*)\ {1,}(\d*)\ {1,}(\d*)\ {1,}(\d*)\ {1,}(\d*)\ {1,}(\d*)\ {1,}(\d*)\ {1,}(\d*)\ {1,}(\d*)\ {1,}(\d*)\ {1,}(\d*)\ {1,}(\d*)\ {1,}(\d*)\ {1,}(\d*)", re.ASCII | re.MULTILINE) searchObj = pattern.findall(info) if len(searchObj) == 0: err = LookupError("Fail to find data") logger.error( "{}.{}: {}".format( self.__class__.__name__, sys._getframe().f_code.co_name, str(err))) raise err for i in keys: if type(i).__name__ == 'int': ret = ret + " " + searchObj[-1][i] elif i == "util.swap": util = int(searchObj[-1][keyword["swap.si"]]) + \ int(searchObj[-1][keyword["swap.so"]]) ret = ret + " " + str(util) elif i == "util.cpu": util = int(searchObj[-1][keyword["cpu.us"]]) + \ int(searchObj[-1][keyword["cpu.sy"]]) + \ int(searchObj[-1][keyword["cpu.st"]]) ret = ret + " " + str(util) return ret if __name__ == "__main__": if len(sys.argv) != 3: print('usage: ' + sys.argv[0] + ' fmt path') sys.exit(-1) ct = MemVmstat("UT") ct.report( sys.argv[1], sys.argv[2], "--interval=2;--fields=memory.free --fields=cpu.us --fields=cpu.id --fields=util.cpu")