forked from Eriner/zsh-framework-benchmark
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathrun.zsh
167 lines (151 loc) · 4.49 KB
/
run.zsh
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
() {
builtin emulate -L zsh -o EXTENDED_GLOB
if (( SHLVL > 1 )); then
print -u2 "::error::Cannot be executed on a nested shell. SHLVL is ${SHLVL}."
return 1
fi
if (( ! ${+commands[expect]} )); then
print -u2 "::error::expect command required."
return 1
fi
if ! autoload -Uz is-at-least || ! is-at-least '5.2'; then
print -u2 "::error::Zsh >= 5.2 required."
return 1
fi
local -r run_dir=${PWD:A}
local test_dir=${run_dir}/results
local work_dir=${${TMPDIR:-${TMPPREFIX}}:A}/${RANDOM}
local -i keep_frameworks=0
local -i iterations=100
# adding vanilla first, because it should always be the baseline
local -r available_frameworks=(vanilla frameworks/(^vanilla.*)(N:t:r))
local frameworks=()
# ensure to use dot ('.') as decimal separator, because some locale (ex: it_IT) use comma (',')
unset LC_NUMERIC
local -r usage="source run.zsh [options]
Options:
-h Show this help
-k Keep the frameworks after the tests are complete (default: delete)
-p <path> Set the path to where the frameworks should be installed (default: results)
-w <path> Set the working directory (default: temp directory)
-n <number> Set the number of iterations to run for each framework (default: 100)
-f <framework> Specify framework to benchmark (default: all; can specify more than once)"
while (( # )); do
case ${1} in
-h) print ${usage}
return 0
;;
-k) keep_frameworks=1
shift
;;
-p) shift
test_dir=${1:A}
shift
;;
-w) shift
work_dir=${1:A}
shift
;;
-n) shift
iterations=${1}
shift
;;
-f) shift
if [[ ${available_frameworks[(r)${1}]} == ${1} ]]; then
frameworks+=(${1})
else
print -u2 "::error::Framework \"${1}\" unknown. Available frameworks are: ${available_frameworks}"
return 1
fi
shift
;;
*) print -u2 "::error::Invalid option \"${1}\"\n"
print -u2 ${usage}
return 1
;;
esac
done
if (( # )); then
print -u2 ${usage}
return 1
fi
command mkdir -p ${test_dir} || return 1
command mkdir -p ${work_dir} || return 1
if (( ! ${#frameworks} )); then
frameworks=(${available_frameworks})
fi
set_up() {
local -r home_dir=${test_dir}/${1}
if [[ -e ${home_dir} && ${keep_frameworks} -eq 1 ]]; then
print "${1} already installed"
return 0
fi
# first delete any old instances of the frameworks
command rm -rf ${home_dir} || return 1
# setup the directory for the framework
command mkdir -p ${home_dir} || return 1
# source the installer
print "::group::Setting up ${1} ..."
{
( cd ${home_dir} && source ${run_dir}/frameworks/${1}.zsh || return 1 )
} always {
print '\n::endgroup::'
}
}
benchmark() {
local -r home_dir=${test_dir}/${1} timediv=1000000
# warmup
builtin pushd -q ${work_dir}
{
repeat 3 do
HOME=${home_dir} ${run_dir}/expect-warmup || return 1
done >! ${test_dir}/${1}_out.log 2>! ${test_dir}/${1}_err.log
repeat ${iterations} do
HOME=${home_dir} ${run_dir}/expect-run || return 1
done >> ${test_dir}/${1}_out.log 2>! ${test_dir}/${1}.log
} always {
builtin popd -q
}
if command grep -v '^[0-9]\+$' ${test_dir}/${1}.log; then
print -u2 "::error::Unexpected output when benchmarking ${1}"
return 1
fi
command awk -v framework=${1} -v timediv=${timediv} '
count == 0 || $1 < min { min = $1 }
count == 0 || $1 > max { max = $1 }
{
sum += $1
sumsq += $1^2
count++
}
END {
if (count > 0) {
mean = sum/count
if (min < max) { stddev = sqrt(sumsq/count - mean^2) } else { stddev = 0 }
}
print framework "," mean/timediv "," stddev/timediv "," min/timediv "," max/timediv
}' ${test_dir}/${1}.log | command tee -a ${results_file}
}
# Useful for debugging.
local -r results_file=${test_dir}/results.csv
print "Results: ${results_file}\n"
print "This may take a LONG time, as it runs each framework startup ${iterations} times.
Average startup times for each framework will be printed as the tests progress.\n"
print "Using Zsh ${ZSH_VERSION}\n"
{
local framework
for framework in ${frameworks}; do
set_up ${framework} || return 1
done
print -P "\n%F{green}Benchmarking ...%f"
print 'framework,mean,stddev,min,max' | command tee ${results_file}
for framework in ${frameworks}; do
benchmark ${framework} || return 1
done
} always {
# cleanup frameworks unless '-k' was provided
if (( ! keep_frameworks )); then
command rm -rf ${test_dir}/*(/) || return 1
fi
}
} "${@}"