# Copyright (c) 2013 The Chromium OS Authors. All rights reserved. # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file. import logging import csv import cStringIO import random import re import collections from autotest_lib.client.common_lib.cros import path_utils class ResourceMonitorRawResult(object): """Encapsulates raw resource_monitor results.""" def __init__(self, raw_results_filename): self._raw_results_filename = raw_results_filename def get_parsed_results(self): """Constructs parsed results from the raw ones. @return ResourceMonitorParsedResult object """ return ResourceMonitorParsedResult(self.raw_results_filename) @property def raw_results_filename(self): """@return string filename storing the raw top command output.""" return self._raw_results_filename class IncorrectTopFormat(Exception): """Thrown if top output format is not as expected""" pass def _extract_value_before_single_keyword(line, keyword): """Extract word occurring immediately before the specified keyword. @param line string the line in which to search for the keyword. @param keyword string the keyword to look for. Can be a regexp. @return string the word just before the keyword. """ pattern = ".*?(\S+) " + keyword matches = re.match(pattern, line) if matches is None or len(matches.groups()) != 1: raise IncorrectTopFormat return matches.group(1) def _extract_values_before_keywords(line, *args): """Extract the words occuring immediately before each specified keyword in args. @param line string the string to look for the keywords. @param args variable number of string args the keywords to look for. @return string list the words occuring just before each keyword. """ line_nocomma = re.sub(",", " ", line) line_singlespace = re.sub("\s+", " ", line_nocomma) return [_extract_value_before_single_keyword( line_singlespace, arg) for arg in args] def _find_top_output_identifying_pattern(line): """Return true iff the line looks like the first line of top output. @param line string to look for the pattern @return boolean """ pattern ="\s*top\s*-.*up.*users.*" matches = re.match(pattern, line) return matches is not None class ResourceMonitorParsedResult(object): """Encapsulates logic to parse and represent top command results.""" _columns = ["Time", "UserCPU", "SysCPU", "NCPU", "Idle", "IOWait", "IRQ", "SoftIRQ", "Steal", "MemUnits", "UsedMem", "FreeMem", "SwapUnits", "UsedSwap", "FreeSwap"] UtilValues = collections.namedtuple('UtilValues', ' '.join(_columns)) def __init__(self, raw_results_filename): """Construct a ResourceMonitorResult. @param raw_results_filename string filename of raw batch top output. """ self._raw_results_filename = raw_results_filename self.parse_resource_monitor_results() def parse_resource_monitor_results(self): """Extract utilization metrics from output file.""" self._utils_over_time = [] with open(self._raw_results_filename, "r") as results_file: while True: curr_line = '\n' while curr_line != '' and \ not _find_top_output_identifying_pattern(curr_line): curr_line = results_file.readline() if curr_line == '': break try: time, = _extract_values_before_keywords(curr_line, "up") # Ignore one line. _ = results_file.readline() # Get the cpu usage. curr_line = results_file.readline() (cpu_user, cpu_sys, cpu_nice, cpu_idle, io_wait, irq, sirq, steal) = _extract_values_before_keywords(curr_line, "us", "sy", "ni", "id", "wa", "hi", "si", "st") # Get memory usage. curr_line = results_file.readline() (mem_units, mem_free, mem_used) = _extract_values_before_keywords( curr_line, "Mem", "free", "used") # Get swap usage. curr_line = results_file.readline() (swap_units, swap_free, swap_used) = _extract_values_before_keywords( curr_line, "Swap", "free", "used") curr_util_values = ResourceMonitorParsedResult.UtilValues( Time=time, UserCPU=cpu_user, SysCPU=cpu_sys, NCPU=cpu_nice, Idle=cpu_idle, IOWait=io_wait, IRQ=irq, SoftIRQ=sirq, Steal=steal, MemUnits=mem_units, UsedMem=mem_used, FreeMem=mem_free, SwapUnits=swap_units, UsedSwap=swap_used, FreeSwap=swap_free) self._utils_over_time.append(curr_util_values) except IncorrectTopFormat: logging.error( "Top output format incorrect. Aborting parse.") return def __repr__(self): output_stringfile = cStringIO.StringIO() self.save_to_file(output_stringfile) return output_stringfile.getvalue() def save_to_file(self, file): """Save parsed top results to file @param file file object to write to """ if len(self._utils_over_time) < 1: logging.warning("Tried to save parsed results, but they were " "empty. Skipping the save.") return csvwriter = csv.writer(file, delimiter=',') csvwriter.writerow(self._utils_over_time[0]._fields) for row in self._utils_over_time: csvwriter.writerow(row) def save_to_filename(self, filename): """Save parsed top results to filename @param filename string filepath to write to """ out_file = open(filename, "wb") self.save_to_file(out_file) out_file.close() class ResourceMonitorConfig(object): """Defines a single top run.""" DEFAULT_MONITOR_PERIOD = 3 def __init__(self, monitor_period=DEFAULT_MONITOR_PERIOD, rawresult_output_filename=None): """Construct a ResourceMonitorConfig. @param monitor_period float seconds between successive top refreshes. @param rawresult_output_filename string filename to output the raw top results to """ if monitor_period < 0.1: logging.info('Monitor period must be at least 0.1s.' ' Given: %r. Defaulting to 0.1s', monitor_period) monitor_period = 0.1 self._monitor_period = monitor_period self._server_outfile = rawresult_output_filename class ResourceMonitor(object): """Delegate to run top on a client. Usage example (call from a test): rmc = resource_monitor.ResourceMonitorConfig(monitor_period=1, rawresult_output_filename=os.path.join(self.resultsdir, 'topout.txt')) with resource_monitor.ResourceMonitor(self.context.client.host, rmc) as rm: rm.start() <operation_to_monitor> rm_raw_res = rm.stop() rm_res = rm_raw_res.get_parsed_results() rm_res.save_to_filename( os.path.join(self.resultsdir, 'resource_mon.csv')) """ def __init__(self, client_host, config): """Construct a ResourceMonitor. @param client_host: SSHHost object representing a remote ssh host """ self._client_host = client_host self._config = config self._command_top = path_utils.must_be_installed( 'top', host=self._client_host) self._top_pid = None def __enter__(self): return self def __exit__(self, exc_type, exc_value, traceback): if self._top_pid is not None: self._client_host.run('kill %s && rm %s' % (self._top_pid, self._client_outfile), ignore_status=True) return True def start(self): """Run top and save results to a temp file on the client.""" if self._top_pid is not None: logging.debug("Tried to start monitoring before stopping. " "Ignoring request.") return # Decide where to write top's output to (on the client). random_suffix = random.random() self._client_outfile = '/tmp/topcap-%r' % random_suffix # Run top on the client. top_command = '%s -b -d%d > %s' % (self._command_top, self._config._monitor_period, self._client_outfile) logging.info('Running top.') self._top_pid = self._client_host.run_background(top_command) logging.info('Top running with pid %s', self._top_pid) def stop(self): """Stop running top and return the results. @return ResourceMonitorRawResult object """ logging.debug("Stopping monitor") if self._top_pid is None: logging.debug("Tried to stop monitoring before starting. " "Ignoring request.") return # Stop top on the client. self._client_host.run('kill %s' % self._top_pid, ignore_status=True) # Get the top output file from the client onto the server. if self._config._server_outfile is None: self._config._server_outfile = self._client_outfile self._client_host.get_file( self._client_outfile, self._config._server_outfile) # Delete the top output file from client. self._client_host.run('rm %s' % self._client_outfile, ignore_status=True) self._top_pid = None logging.info("Saved resource monitor results at %s", self._config._server_outfile) return ResourceMonitorRawResult(self._config._server_outfile)