2017-12-28 23:35:22 +08:00
|
|
|
#!/usr/bin/env python3
|
2017-10-23 22:17:06 +08:00
|
|
|
import argparse
|
2017-12-28 23:35:22 +08:00
|
|
|
import json
|
2017-08-24 21:03:21 +08:00
|
|
|
import os
|
|
|
|
import sys
|
|
|
|
|
2017-12-28 23:35:22 +08:00
|
|
|
def load_file(fname):
|
2017-08-24 21:03:21 +08:00
|
|
|
with open(fname) as f:
|
2017-12-28 23:35:22 +08:00
|
|
|
data = f.read()
|
|
|
|
try:
|
|
|
|
return json.loads(data)
|
|
|
|
except json.decoder.JSONDecodeError:
|
|
|
|
return {"results": [], "headers": []}
|
2017-08-24 21:03:21 +08:00
|
|
|
|
|
|
|
def strip_integers(row):
|
2017-12-28 23:35:22 +08:00
|
|
|
return {k: v for k, v in row.items() if type(v) == str}
|
2017-08-24 21:03:21 +08:00
|
|
|
|
2017-12-28 23:35:22 +08:00
|
|
|
def find_item(results_prev, header_cur, row_cur):
|
|
|
|
row_cur = strip_integers(row_cur)
|
|
|
|
row_prev = None
|
|
|
|
for result in results_prev:
|
|
|
|
s = strip_integers(result)
|
|
|
|
if s == row_cur:
|
|
|
|
row_prev = result
|
|
|
|
break
|
|
|
|
if row_prev is None: return None
|
|
|
|
if not header_cur in row_prev: return None
|
|
|
|
return row_prev[header_cur]
|
2017-08-24 21:03:21 +08:00
|
|
|
|
2017-12-28 23:35:22 +08:00
|
|
|
def compare_values(headers_cur, results_cur, headers_prev, results_prev):
|
|
|
|
ret = [list(map(lambda x: " ".join(x.split("_")).capitalize(),
|
|
|
|
headers_cur))]
|
|
|
|
for row_cur in results_cur:
|
2017-08-24 21:03:21 +08:00
|
|
|
ret.append([])
|
2017-10-19 18:40:54 +08:00
|
|
|
performance_change = False
|
2017-12-28 23:35:22 +08:00
|
|
|
for header in headers_cur:
|
|
|
|
item_cur = row_cur[header]
|
2017-08-24 21:03:21 +08:00
|
|
|
if type(item_cur) == str:
|
|
|
|
item = " ".join(item_cur.split("_")).capitalize()
|
|
|
|
else:
|
2017-12-28 23:35:22 +08:00
|
|
|
value_cur = item_cur["median"]
|
|
|
|
item_prev = find_item(results_prev, header, row_cur)
|
|
|
|
if header != "max_memory":
|
2017-09-13 22:57:15 +08:00
|
|
|
fmt = "{:.3f}ms"
|
|
|
|
scale = 1000.0
|
2017-10-25 22:38:33 +08:00
|
|
|
treshold = 0.050
|
2017-09-13 22:57:15 +08:00
|
|
|
else:
|
|
|
|
fmt = "{:.2f}MiB"
|
|
|
|
scale = 1.0 / 1024.0
|
2017-10-25 22:38:33 +08:00
|
|
|
treshold = 0.025
|
2017-12-28 23:35:22 +08:00
|
|
|
# TODO: add statistics check
|
2017-08-24 21:03:21 +08:00
|
|
|
if item_prev != None:
|
2017-12-28 23:35:22 +08:00
|
|
|
value_prev = item_prev["median"]
|
|
|
|
if value_prev != 0.0:
|
|
|
|
diff = (value_cur - value_prev) / value_prev
|
2017-08-24 21:03:21 +08:00
|
|
|
else:
|
|
|
|
diff = 0.0
|
2017-12-28 23:35:22 +08:00
|
|
|
if diff < -treshold and value_cur > 0.0005:
|
2017-10-19 18:40:54 +08:00
|
|
|
performance_change = True
|
2017-08-24 21:03:21 +08:00
|
|
|
sign = " {icon arrow-down color=green}"
|
2017-12-28 23:35:22 +08:00
|
|
|
elif diff > treshold and value_cur > 0.0005:
|
2017-10-19 18:40:54 +08:00
|
|
|
performance_change = True
|
2017-08-24 21:03:21 +08:00
|
|
|
sign = " {icon arrow-up color=red}"
|
|
|
|
else:
|
|
|
|
sign = ""
|
2017-09-13 22:57:15 +08:00
|
|
|
fmt += " //({:+.2%})//{}"
|
2017-12-28 23:35:22 +08:00
|
|
|
item = fmt.format(value_cur * scale, diff, sign)
|
2017-08-24 21:03:21 +08:00
|
|
|
else:
|
2017-10-23 22:17:06 +08:00
|
|
|
fmt += " //(new)// {{icon plus color=blue}}"
|
2017-12-28 23:35:22 +08:00
|
|
|
item = fmt.format(value_cur * scale)
|
2017-10-19 18:40:54 +08:00
|
|
|
performance_change = True
|
|
|
|
ret[-1].append(item)
|
2017-12-28 23:35:22 +08:00
|
|
|
if not performance_change: ret.pop()
|
2017-08-24 21:03:21 +08:00
|
|
|
return ret
|
|
|
|
|
|
|
|
def generate_remarkup(data):
|
|
|
|
ret = "==== Macro benchmark summary: ====\n\n"
|
2017-10-19 18:40:54 +08:00
|
|
|
if len(data) > 1:
|
2017-11-07 00:36:10 +08:00
|
|
|
ret += "<table>\n"
|
|
|
|
for row in data:
|
|
|
|
ret += " <tr>\n"
|
|
|
|
for item in row:
|
|
|
|
if row == data[0]:
|
|
|
|
fmt = " <th>{}</th>\n"
|
|
|
|
else:
|
|
|
|
fmt = " <td>{}</td>\n"
|
|
|
|
ret += fmt.format(item)
|
|
|
|
ret += " </tr>\n"
|
|
|
|
ret += "</table>\n"
|
2017-10-19 18:40:54 +08:00
|
|
|
else:
|
2017-11-07 00:36:10 +08:00
|
|
|
ret += "No performance change detected.\n"
|
2017-08-24 21:03:21 +08:00
|
|
|
return ret
|
|
|
|
|
2017-10-23 22:17:06 +08:00
|
|
|
if __name__ == "__main__":
|
|
|
|
parser = argparse.ArgumentParser(description="Process macro benchmark summary.")
|
|
|
|
parser.add_argument("--current", nargs = "+", required = True,
|
|
|
|
help = "current summary files")
|
|
|
|
parser.add_argument("--previous", nargs = "+", required = True,
|
|
|
|
help = "previous summary files")
|
|
|
|
parser.add_argument("--output", default = "",
|
|
|
|
help = "output file, if not specified the script outputs to stdout")
|
2017-08-24 21:03:21 +08:00
|
|
|
|
2017-10-23 22:17:06 +08:00
|
|
|
args = parser.parse_args()
|
2017-08-24 21:03:21 +08:00
|
|
|
|
2017-12-28 23:35:22 +08:00
|
|
|
headers_cur, headers_prev = None, None
|
|
|
|
results_cur, results_prev = [], []
|
|
|
|
for current in args.current:
|
|
|
|
data = load_file(current)
|
|
|
|
if headers_cur is None:
|
|
|
|
headers_cur = data["headers"]
|
|
|
|
results_cur += data["results"]
|
|
|
|
for previous in args.previous:
|
|
|
|
data = load_file(previous)
|
|
|
|
if headers_prev is None:
|
|
|
|
headers_prev = data["headers"]
|
|
|
|
results_prev += data["results"]
|
2017-08-24 21:03:21 +08:00
|
|
|
|
2017-12-28 23:35:22 +08:00
|
|
|
markup = generate_remarkup(compare_values(headers_cur, results_cur,
|
|
|
|
headers_prev, results_prev))
|
2017-08-24 21:03:21 +08:00
|
|
|
|
2017-10-23 22:17:06 +08:00
|
|
|
if args.output == "":
|
|
|
|
sys.stdout.write(markup)
|
|
|
|
sys.exit(0)
|
2017-08-24 21:03:21 +08:00
|
|
|
|
2017-10-23 22:17:06 +08:00
|
|
|
with open(args.output, "w") as f:
|
|
|
|
f.write(markup)
|