perfkitbenchmarker/timing_util.py (103 lines of code) (raw):

# Copyright 2014 PerfKitBenchmarker Authors. All rights reserved. # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. """Utilities for generating timing samples.""" from collections import OrderedDict from contextlib import contextmanager import time from absl import flags from perfkitbenchmarker import sample MEASUREMENTS_FLAG_NAME = 'timing_measurements' # Valid options that can be included in the flag's list value. MEASUREMENTS_NONE = 'none' MEASUREMENTS_END_TO_END_RUNTIME = 'end_to_end_runtime' MEASUREMENTS_RUNTIMES = 'runtimes' MEASUREMENTS_TIMESTAMPS = 'timestamps' MEASUREMENTS_ALL = OrderedDict([ ( MEASUREMENTS_NONE, ( 'No measurements included (same as providing an empty list, and' ' cannot be combined with other options).' ), ), ( MEASUREMENTS_END_TO_END_RUNTIME, 'Includes an end-to-end runtime measurement.', ), ( MEASUREMENTS_RUNTIMES, ( 'Includes runtimes of all measured intervals, including the' ' end-to-end runtime, the time taken by the benchmark module' ' Prepare, Run, and Cleanup functions, and other important' ' intervals.' ), ), ( MEASUREMENTS_TIMESTAMPS, 'Includes start and stop timestamps of all measured intervals.', ), ]) def EndToEndRuntimeMeasurementEnabled(): """Returns whether end-to-end runtime measurement is globally enabled.""" return ( MEASUREMENTS_END_TO_END_RUNTIME in flags.FLAGS.timing_measurements or RuntimeMeasurementsEnabled() ) def RuntimeMeasurementsEnabled(): """Returns whether runtime measurements are globally enabled.""" return MEASUREMENTS_RUNTIMES in flags.FLAGS.timing_measurements def TimestampMeasurementsEnabled(): """Returns whether timestamps measurements are globally enabled.""" return MEASUREMENTS_TIMESTAMPS in flags.FLAGS.timing_measurements def ValidateMeasurementsFlag(options_list): """Verifies correct usage of the measurements configuration flag. The user of the flag must provide at least one option. All provided options must be valid. The NONE option cannot be combined with other options. Args: options_list: A list of strings parsed from the provided value for the flag. Returns: True if the list of options provided as the value for the flag meets all the documented requirements. Raises: flags.ValidationError: If the list of options provided as the value for the flag does not meet the documented requirements. """ for option in options_list: if option not in MEASUREMENTS_ALL: raise flags.ValidationError( '%s: Invalid value for --%s' % (option, MEASUREMENTS_FLAG_NAME) ) if option == MEASUREMENTS_NONE and len(options_list) != 1: raise flags.ValidationError( '%s: Cannot combine with other --%s options' % (option, MEASUREMENTS_FLAG_NAME) ) return True flags.DEFINE_list( MEASUREMENTS_FLAG_NAME, MEASUREMENTS_END_TO_END_RUNTIME, 'Comma-separated list of values from <%s> that selects which timing ' 'measurements to enable. Measurements will be included as samples in the ' 'benchmark results. %s' % ( '|'.join(MEASUREMENTS_ALL), ' '.join([ '%s: %s' % (option, description) for option, description in MEASUREMENTS_ALL.items() ]), ), ) flags.register_validator(MEASUREMENTS_FLAG_NAME, ValidateMeasurementsFlag) def _GenerateIntervalSamples(interval, include_timestamps): """Generates Samples for a single interval timed by IntervalTimer.Measure. Args: interval: A (name, start_time, stop_time) tuple from a call to IntervalTimer.Measure. include_timestamps: A Boolean that controls whether Samples containing the start and stop timestamps are added to the generated list. Returns: A list of 0 to 3 Samples as specified by the args. When included, the Samples appear in the order of runtime, start timestamp, stop timestamp. """ samples = [] name = interval[0] start_time = interval[1] stop_time = interval[2] elapsed_time = stop_time - start_time samples.append(sample.Sample(name + ' Runtime', elapsed_time, 'seconds')) if include_timestamps: samples.append( sample.Sample(name + ' Start Timestamp', start_time, 'seconds') ) samples.append( sample.Sample(name + ' Stop Timestamp', stop_time, 'seconds') ) return samples class IntervalTimer: """Class that can measure time and generate samples for each measurement. Attributes: intervals: A list of one 3-tuple per measured interval. Each tuple is of the form (name string, start_time float, stop_time float). """ def __init__(self): self.intervals = [] @contextmanager def Measure(self, name): """Records the start and stop times of the enclosed interval. Args: name: A string that names the interval. """ start_time = time.time() yield stop_time = time.time() self.intervals.append((name, start_time, stop_time)) def GenerateSamples(self): """Generates Samples based on the times recorded in all calls to Measure. Returns: A list of Samples. The list contains Samples for each interval that was wrapped by a call to Measure, with per-interval Samples generated as specified by the args in the order of runtime, start timestamp, stop timestamp. All Samples for one interval appear before any Samples from the next interval. """ include_timestamps = TimestampMeasurementsEnabled() return [ sample for interval in self.intervals for sample in _GenerateIntervalSamples(interval, include_timestamps) ]