| # Copyright (c) 2017 The WebRTC project authors. All Rights Reserved. |
| # |
| # Use of this source code is governed by a BSD-style license |
| # that can be found in the LICENSE file in the root of the source |
| # tree. An additional intellectual property rights grant can be found |
| # in the file PATENTS. All contributing project authors may |
| # be found in the AUTHORS file in the root of the source tree. |
| |
| """Plots statistics from WebRTC integration test logs. |
| |
| Usage: $ python plot_webrtc_test_logs.py filename.txt |
| """ |
| |
| import numpy |
| import sys |
| import re |
| |
| import matplotlib.pyplot as plt |
| |
| # Log events. |
| EVENT_START = \ |
| 'RUN ] CodecSettings/VideoProcessorIntegrationTestParameterized.' |
| EVENT_END = 'OK ] CodecSettings/VideoProcessorIntegrationTestParameterized.' |
| |
| # Metrics to plot, tuple: (name to parse in file, label to use when plotting). |
| BITRATE = ('Target bitrate', 'target bitrate (kbps)') |
| WIDTH = ('Width', 'width') |
| HEIGHT = ('Height', 'height') |
| FILENAME = ('Filename', 'clip') |
| CODEC_TYPE = ('Codec type', 'Codec') |
| ENCODER_IMPLEMENTATION_NAME = ('Encoder implementation name', 'enc name') |
| DECODER_IMPLEMENTATION_NAME = ('Decoder implementation name', 'dec name') |
| CODEC_IMPLEMENTATION_NAME = ('Codec implementation name', 'codec name') |
| CORES = ('# CPU cores used', 'CPU cores used') |
| DENOISING = ('Denoising', 'denoising') |
| RESILIENCE = ('Resilience', 'resilience') |
| ERROR_CONCEALMENT = ('Error concealment', 'error concealment') |
| QP = ('Average QP', 'avg QP') |
| PSNR = ('PSNR avg', 'PSNR (dB)') |
| SSIM = ('SSIM avg', 'SSIM') |
| ENC_BITRATE = ('Encoded bitrate', 'encoded bitrate (kbps)') |
| FRAMERATE = ('Frame rate', 'fps') |
| NUM_FRAMES = ('# processed frames', 'num frames') |
| NUM_DROPPED_FRAMES = ('# dropped frames', 'num dropped frames') |
| NUM_FRAMES_TO_TARGET = ('# frames to convergence', |
| 'frames to reach target rate') |
| ENCODE_TIME = ('Encoding time', 'encode time (us)') |
| ENCODE_TIME_AVG = ('Encoding time', 'encode time (us) avg') |
| DECODE_TIME = ('Decoding time', 'decode time (us)') |
| DECODE_TIME_AVG = ('Decoding time', 'decode time (us) avg') |
| FRAME_SIZE = ('Frame sizes', 'frame size (bytes)') |
| FRAME_SIZE_AVG = ('Frame sizes', 'frame size (bytes) avg') |
| AVG_KEY_FRAME_SIZE = ('Average key frame size', 'avg key frame size (bytes)') |
| AVG_NON_KEY_FRAME_SIZE = ('Average non-key frame size', |
| 'avg non-key frame size (bytes)') |
| |
| # Settings. |
| SETTINGS = [ |
| WIDTH, |
| HEIGHT, |
| FILENAME, |
| NUM_FRAMES, |
| ENCODE_TIME, |
| DECODE_TIME, |
| FRAME_SIZE, |
| ] |
| |
| # Settings, options for x-axis. |
| X_SETTINGS = [ |
| CORES, |
| FRAMERATE, |
| DENOISING, |
| RESILIENCE, |
| ERROR_CONCEALMENT, |
| BITRATE, # TODO(asapersson): Needs to be last. |
| ] |
| |
| # Settings, options for subplots. |
| SUBPLOT_SETTINGS = [ |
| CODEC_TYPE, |
| ENCODER_IMPLEMENTATION_NAME, |
| DECODER_IMPLEMENTATION_NAME, |
| CODEC_IMPLEMENTATION_NAME, |
| ] + X_SETTINGS |
| |
| # Results. |
| RESULTS = [ |
| PSNR, |
| SSIM, |
| ENC_BITRATE, |
| NUM_DROPPED_FRAMES, |
| NUM_FRAMES_TO_TARGET, |
| ENCODE_TIME_AVG, |
| DECODE_TIME_AVG, |
| QP, |
| AVG_KEY_FRAME_SIZE, |
| AVG_NON_KEY_FRAME_SIZE, |
| ] |
| |
| METRICS_TO_PARSE = SETTINGS + SUBPLOT_SETTINGS + RESULTS |
| |
| Y_METRICS = [res[1] for res in RESULTS] |
| |
| # Parameters for plotting. |
| FIG_SIZE_SCALE_FACTOR_X = 1.6 |
| FIG_SIZE_SCALE_FACTOR_Y = 1.8 |
| GRID_COLOR = [0.45, 0.45, 0.45] |
| |
| |
| def ParseSetting(filename, setting): |
| """Parses setting from file. |
| |
| Args: |
| filename: The name of the file. |
| setting: Name of setting to parse (e.g. width). |
| |
| Returns: |
| A list holding parsed settings, e.g. ['width: 128.0', 'width: 160.0'] """ |
| |
| settings = [] |
| |
| settings_file = open(filename) |
| while True: |
| line = settings_file.readline() |
| if not line: |
| break |
| if re.search(r'%s' % EVENT_START, line): |
| # Parse event. |
| parsed = {} |
| while True: |
| line = settings_file.readline() |
| if not line: |
| break |
| if re.search(r'%s' % EVENT_END, line): |
| # Add parsed setting to list. |
| if setting in parsed: |
| s = setting + ': ' + str(parsed[setting]) |
| if s not in settings: |
| settings.append(s) |
| break |
| |
| TryFindMetric(parsed, line, settings_file) |
| |
| settings_file.close() |
| return settings |
| |
| |
| def ParseMetrics(filename, setting1, setting2): |
| """Parses metrics from file. |
| |
| Args: |
| filename: The name of the file. |
| setting1: First setting for sorting metrics (e.g. width). |
| setting2: Second setting for sorting metrics (e.g. CPU cores used). |
| |
| Returns: |
| A dictionary holding parsed metrics. |
| |
| For example: |
| metrics[key1][key2][measurement] |
| |
| metrics = { |
| "width: 352": { |
| "CPU cores used: 1.0": { |
| "encode time (us)": [0.718005, 0.806925, 0.909726, 0.931835, 0.953642], |
| "PSNR (dB)": [25.546029, 29.465518, 34.723535, 36.428493, 38.686551], |
| "bitrate (kbps)": [50, 100, 300, 500, 1000] |
| }, |
| "CPU cores used: 2.0": { |
| "encode time (us)": [0.718005, 0.806925, 0.909726, 0.931835, 0.953642], |
| "PSNR (dB)": [25.546029, 29.465518, 34.723535, 36.428493, 38.686551], |
| "bitrate (kbps)": [50, 100, 300, 500, 1000] |
| }, |
| }, |
| "width: 176": { |
| "CPU cores used: 1.0": { |
| "encode time (us)": [0.857897, 0.91608, 0.959173, 0.971116, 0.980961], |
| "PSNR (dB)": [30.243646, 33.375592, 37.574387, 39.42184, 41.437897], |
| "bitrate (kbps)": [50, 100, 300, 500, 1000] |
| }, |
| } |
| } """ |
| |
| metrics = {} |
| |
| # Parse events. |
| settings_file = open(filename) |
| while True: |
| line = settings_file.readline() |
| if not line: |
| break |
| if re.search(r'%s' % EVENT_START, line): |
| # Parse event. |
| parsed = {} |
| while True: |
| line = settings_file.readline() |
| if not line: |
| break |
| if re.search(r'%s' % EVENT_END, line): |
| # Add parsed values to metrics. |
| key1 = setting1 + ': ' + str(parsed[setting1]) |
| key2 = setting2 + ': ' + str(parsed[setting2]) |
| if key1 not in metrics: |
| metrics[key1] = {} |
| if key2 not in metrics[key1]: |
| metrics[key1][key2] = {} |
| |
| for label in parsed: |
| if label not in metrics[key1][key2]: |
| metrics[key1][key2][label] = [] |
| metrics[key1][key2][label].append(parsed[label]) |
| |
| break |
| |
| TryFindMetric(parsed, line, settings_file) |
| |
| settings_file.close() |
| return metrics |
| |
| |
| def TryFindMetric(parsed, line, settings_file): |
| for metric in METRICS_TO_PARSE: |
| name = metric[0] |
| label = metric[1] |
| if re.search(r'%s' % name, line): |
| found, value = GetMetric(name, line) |
| if not found: |
| # TODO(asapersson): Change format. |
| # Try find min, max, average stats. |
| found, minimum = GetMetric("Min", settings_file.readline()) |
| if not found: |
| return |
| found, maximum = GetMetric("Max", settings_file.readline()) |
| if not found: |
| return |
| found, average = GetMetric("Average", settings_file.readline()) |
| if not found: |
| return |
| |
| parsed[label + ' min'] = minimum |
| parsed[label + ' max'] = maximum |
| parsed[label + ' avg'] = average |
| |
| parsed[label] = value |
| return |
| |
| |
| def GetMetric(name, string): |
| # Float (e.g. bitrate = 98.8253). |
| pattern = r'%s\s*[:=]\s*([+-]?\d+\.*\d*)' % name |
| m = re.search(r'%s' % pattern, string) |
| if m is not None: |
| return StringToFloat(m.group(1)) |
| |
| # Alphanumeric characters (e.g. codec type : VP8). |
| pattern = r'%s\s*[:=]\s*(\w+)' % name |
| m = re.search(r'%s' % pattern, string) |
| if m is not None: |
| return True, m.group(1) |
| |
| return False, -1 |
| |
| |
| def StringToFloat(value): |
| try: |
| value = float(value) |
| except ValueError: |
| print "Not a float, skipped %s" % value |
| return False, -1 |
| |
| return True, value |
| |
| |
| def Plot(y_metric, x_metric, metrics): |
| """Plots y_metric vs x_metric per key in metrics. |
| |
| For example: |
| y_metric = 'PSNR (dB)' |
| x_metric = 'bitrate (kbps)' |
| metrics = { |
| "CPU cores used: 1.0": { |
| "PSNR (dB)": [25.546029, 29.465518, 34.723535, 36.428493, 38.686551], |
| "bitrate (kbps)": [50, 100, 300, 500, 1000] |
| }, |
| "CPU cores used: 2.0": { |
| "PSNR (dB)": [25.546029, 29.465518, 34.723535, 36.428493, 38.686551], |
| "bitrate (kbps)": [50, 100, 300, 500, 1000] |
| }, |
| } |
| """ |
| for key in sorted(metrics): |
| data = metrics[key] |
| if y_metric not in data: |
| print "Failed to find metric: %s" % y_metric |
| continue |
| |
| y = numpy.array(data[y_metric]) |
| x = numpy.array(data[x_metric]) |
| if len(y) != len(x): |
| print "Length mismatch for %s, %s" % (y, x) |
| continue |
| |
| label = y_metric + ' - ' + str(key) |
| |
| plt.plot(x, y, label=label, linewidth=1.5, marker='o', markersize=5, |
| markeredgewidth=0.0) |
| |
| |
| def PlotFigure(settings, y_metrics, x_metric, metrics, title): |
| """Plots metrics in y_metrics list. One figure is plotted and each entry |
| in the list is plotted in a subplot (and sorted per settings). |
| |
| For example: |
| settings = ['width: 128.0', 'width: 160.0']. Sort subplot per setting. |
| y_metrics = ['PSNR (dB)', 'PSNR (dB)']. Metric to plot per subplot. |
| x_metric = 'bitrate (kbps)' |
| |
| """ |
| |
| plt.figure() |
| plt.suptitle(title, fontsize='large', fontweight='bold') |
| settings.sort() |
| rows = len(settings) |
| cols = 1 |
| pos = 1 |
| while pos <= rows: |
| plt.rc('grid', color=GRID_COLOR) |
| ax = plt.subplot(rows, cols, pos) |
| plt.grid() |
| plt.setp(ax.get_xticklabels(), visible=(pos == rows), fontsize='large') |
| plt.setp(ax.get_yticklabels(), fontsize='large') |
| setting = settings[pos - 1] |
| Plot(y_metrics[pos - 1], x_metric, metrics[setting]) |
| if setting.startswith(WIDTH[1]): |
| plt.title(setting, fontsize='medium') |
| plt.legend(fontsize='large', loc='best') |
| pos += 1 |
| |
| plt.xlabel(x_metric, fontsize='large') |
| plt.subplots_adjust(left=0.06, right=0.98, bottom=0.05, top=0.94, hspace=0.08) |
| |
| |
| def GetTitle(filename, setting): |
| title = '' |
| if setting != CODEC_IMPLEMENTATION_NAME[1] and setting != CODEC_TYPE[1]: |
| codec_types = ParseSetting(filename, CODEC_TYPE[1]) |
| for i in range(0, len(codec_types)): |
| title += codec_types[i] + ', ' |
| |
| if setting != CORES[1]: |
| cores = ParseSetting(filename, CORES[1]) |
| for i in range(0, len(cores)): |
| title += cores[i].split('.')[0] + ', ' |
| |
| if setting != FRAMERATE[1]: |
| framerate = ParseSetting(filename, FRAMERATE[1]) |
| for i in range(0, len(framerate)): |
| title += framerate[i].split('.')[0] + ', ' |
| |
| if (setting != CODEC_IMPLEMENTATION_NAME[1] and |
| setting != ENCODER_IMPLEMENTATION_NAME[1]): |
| enc_names = ParseSetting(filename, ENCODER_IMPLEMENTATION_NAME[1]) |
| for i in range(0, len(enc_names)): |
| title += enc_names[i] + ', ' |
| |
| if (setting != CODEC_IMPLEMENTATION_NAME[1] and |
| setting != DECODER_IMPLEMENTATION_NAME[1]): |
| dec_names = ParseSetting(filename, DECODER_IMPLEMENTATION_NAME[1]) |
| for i in range(0, len(dec_names)): |
| title += dec_names[i] + ', ' |
| |
| filenames = ParseSetting(filename, FILENAME[1]) |
| title += filenames[0].split('_')[0] |
| |
| num_frames = ParseSetting(filename, NUM_FRAMES[1]) |
| for i in range(0, len(num_frames)): |
| title += ' (' + num_frames[i].split('.')[0] + ')' |
| |
| return title |
| |
| |
| def ToString(input_list): |
| return ToStringWithoutMetric(input_list, ('', '')) |
| |
| |
| def ToStringWithoutMetric(input_list, metric): |
| i = 1 |
| output_str = "" |
| for m in input_list: |
| if m != metric: |
| output_str = output_str + ("%s. %s\n" % (i, m[1])) |
| i += 1 |
| return output_str |
| |
| |
| def GetIdx(text_list): |
| return int(raw_input(text_list)) - 1 |
| |
| |
| def main(): |
| filename = sys.argv[1] |
| |
| # Setup. |
| idx_metric = GetIdx("Choose metric:\n0. All\n%s" % ToString(RESULTS)) |
| if idx_metric == -1: |
| # Plot all metrics. One subplot for each metric. |
| # Per subplot: metric vs bitrate (per resolution). |
| cores = ParseSetting(filename, CORES[1]) |
| setting1 = CORES[1] |
| setting2 = WIDTH[1] |
| sub_keys = [cores[0]] * len(Y_METRICS) |
| y_metrics = Y_METRICS |
| x_metric = BITRATE[1] |
| else: |
| resolutions = ParseSetting(filename, WIDTH[1]) |
| idx = GetIdx("Select metric for x-axis:\n%s" % ToString(X_SETTINGS)) |
| if X_SETTINGS[idx] == BITRATE: |
| idx = GetIdx("Plot per:\n%s" % ToStringWithoutMetric(SUBPLOT_SETTINGS, |
| BITRATE)) |
| idx_setting = METRICS_TO_PARSE.index(SUBPLOT_SETTINGS[idx]) |
| # Plot one metric. One subplot for each resolution. |
| # Per subplot: metric vs bitrate (per setting). |
| setting1 = WIDTH[1] |
| setting2 = METRICS_TO_PARSE[idx_setting][1] |
| sub_keys = resolutions |
| y_metrics = [RESULTS[idx_metric][1]] * len(sub_keys) |
| x_metric = BITRATE[1] |
| else: |
| # Plot one metric. One subplot for each resolution. |
| # Per subplot: metric vs setting (per bitrate). |
| setting1 = WIDTH[1] |
| setting2 = BITRATE[1] |
| sub_keys = resolutions |
| y_metrics = [RESULTS[idx_metric][1]] * len(sub_keys) |
| x_metric = X_SETTINGS[idx][1] |
| |
| metrics = ParseMetrics(filename, setting1, setting2) |
| |
| # Stretch fig size. |
| figsize = plt.rcParams["figure.figsize"] |
| figsize[0] *= FIG_SIZE_SCALE_FACTOR_X |
| figsize[1] *= FIG_SIZE_SCALE_FACTOR_Y |
| plt.rcParams["figure.figsize"] = figsize |
| |
| PlotFigure(sub_keys, y_metrics, x_metric, metrics, |
| GetTitle(filename, setting2)) |
| |
| plt.show() |
| |
| |
| if __name__ == '__main__': |
| main() |