-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathbenchmark.py
executable file
·203 lines (155 loc) · 6.23 KB
/
benchmark.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
#!/usr/bin/env python3
import os
import re
import glob
import time
import shutil
import tempfile
import fileinput
import subprocess
from typing import Union
# Number of benchmark iterations to run
ITERATIONS = 60
# Confidence interval to present
PRECISION_PERCENT = 50
# Number of throwaway iterations to run before starting the benchmark
WARMUP_RUNS = 10
BINDIR = os.path.join(
os.path.dirname(os.path.realpath(__file__)), ".benchmark-binaries"
)
os.makedirs(BINDIR, exist_ok=True)
def build_latest_commit(clonedir: str):
# From: https://stackoverflow.com/a/949391/473672
latest_sha = (
subprocess.check_output(["git", "rev-parse", "--verify", "--short", "HEAD"])
.strip()
.decode("utf-8")
)
latest_commit_bin_name = f"riff-latest-commit-{latest_sha}"
# Remove any outdated riff-latest-commit built
for committed in glob.glob(os.path.join(BINDIR, "riff-latest-commit-*")):
if os.path.basename(committed) != latest_commit_bin_name:
print(f"Removing build for earlier unreleased commit: {committed}")
os.remove(committed)
# Do we already have our build? If so, do nothing.
full_path = os.path.join(BINDIR, latest_commit_bin_name)
if os.path.isfile(full_path):
return
build_binary(clonedir, latest_sha, full_path)
def replace_line_in_file(filename: str, line_prefix: str, replacement: str) -> None:
# Inspired by: https://stackoverflow.com/a/290494/473672
for line in fileinput.input(filename, inplace=True):
if line.startswith(line_prefix):
print(replacement)
continue
print(line, end="")
def cargo_build(cwd=None):
main_rs_path = "src/main.rs"
if cwd:
main_rs_path = f"{cwd}/{main_rs_path}"
# Accept any warnings when building. I'd prefer to do this using the cargo
# build command line, but was unable to figure out how. Better suggestions
# welcome.
replace_line_in_file(main_rs_path, "#![deny(warnings)]", "#![allow(warnings)]")
subprocess.run(["cargo", "build", "--release"], check=True, cwd=cwd)
# Restore the warnings line
replace_line_in_file(main_rs_path, "#![allow(warnings)]", "#![deny(warnings)]")
def atoi(text: str) -> Union[str, int]:
return int(text) if text.isdigit() else text
def natural_keys(text: str):
"""
From: https://stackoverflow.com/a/5967539/473672
"""
return [atoi(c) for c in re.split(r"(\d+)", text)]
def gather_binaries():
"""
Gather binaries to benchmark in BINDIR
"""
# Query git for all Rust versions released and store freshly built binaries
# for those in BINDIR
tags = [tag.decode() for tag in subprocess.check_output(["git", "tag"]).split()]
# Only include version number tags
tags = list(filter(lambda tag: re.match(r"^[0-9.]+$", tag), tags))
# Ignore Ruby tags
rust_tags = list(filter(lambda tag: not tag.startswith("0"), tags))
rust_tags = list(filter(lambda tag: not tag.startswith("1"), rust_tags))
# Just do the four last releases
rust_tags = list(sorted(rust_tags, key=natural_keys))[-4:]
# Build binaries we need
with tempfile.TemporaryDirectory(prefix="riff-benchmark") as clonedir:
subprocess.run(["git", "clone", "-b", "master", ".", clonedir], check=True)
build_latest_commit(clonedir)
for tag in rust_tags:
binary_name = os.path.join(BINDIR, f"riff-{tag}")
if os.path.isfile(binary_name):
continue
print()
print(f"Building missing binary: {binary_name}")
build_binary(clonedir, tag, binary_name)
# Build the current version
print()
print("Building current sources...")
cargo_build()
shutil.copy("target/release/riff", os.path.join(BINDIR, "riff-current"))
def build_binary(clonedir: str, tag: str, binary_name: str):
"""
In clonedir, check out tag, build a binary and put it in binary_name.
"""
# Detached HEAD warning disabling:
# https://stackoverflow.com/a/45652159/473672
subprocess.run(
["git", "-c", "advice.detachedHead=false", "checkout", tag],
cwd=clonedir,
check=True,
)
cargo_build(cwd=clonedir)
shutil.copy(os.path.join(clonedir, "target", "release", "riff"), binary_name)
def print_timings(binary: str, testdata_filename: str):
"""
Run the indicated binary and print timings for it
"""
# Ensure we throw away an integer number of iterations
assert ((100 - PRECISION_PERCENT) * ITERATIONS) % 200 == 0
THROW_AWAY_AT_EACH_END = ((100 - PRECISION_PERCENT) * ITERATIONS) // 200
# Do some warmup runs
for _ in range(WARMUP_RUNS):
with open(testdata_filename, encoding="utf-8") as testdata:
subprocess.check_call(binary, stdin=testdata, stdout=subprocess.DEVNULL)
# Do the actual benchmarking runs
deltas = []
for _ in range(ITERATIONS):
with open(testdata_filename, encoding="utf-8") as testdata:
t0 = time.time()
subprocess.check_call(binary, stdin=testdata, stdout=subprocess.DEVNULL)
t1 = time.time()
dt_seconds = t1 - t0
deltas.append(dt_seconds)
deltas.sort()
from_ms = deltas[THROW_AWAY_AT_EACH_END] * 1000
to_ms = deltas[-THROW_AWAY_AT_EACH_END - 1] * 1000
mid_ms = (from_ms + to_ms) / 2
spread_ms = to_ms - from_ms
plusminus_ms = spread_ms / 2
print(f"{mid_ms:.1f}ms±{plusminus_ms:.1f}ms: {binary}")
def time_binaries():
"""
Print timings for all binaries in BINDIR
"""
print()
print("=== BENCHMARKING ===")
with tempfile.NamedTemporaryFile(
prefix="riff-benchmark", suffix=".gitlog"
) as testdata:
subprocess.check_call(
["git", "log", "--color=always", "-p", "master"], stdout=testdata
)
binaries = sorted(glob.glob(os.path.join(BINDIR, "*")), key=natural_keys)
# Do riff-current last: https://stackoverflow.com/a/20320940/473672
binaries.sort(key=lambda s: s.endswith("riff-current"))
# 5 = four versions back, plus the most recent commit and any
# non-commited changes
for binary in binaries[-6:]:
print_timings(binary, testdata.name)
print_timings("/bin/cat", testdata.name)
gather_binaries()
time_binaries()