2020-10-01 19:05:27 +02:00
|
|
|
|
module main
|
|
|
|
|
|
|
|
|
|
import os
|
|
|
|
|
import flag
|
|
|
|
|
import time
|
2020-10-01 22:25:29 +02:00
|
|
|
|
import term
|
|
|
|
|
import math
|
2020-10-01 19:05:27 +02:00
|
|
|
|
import scripting
|
2020-10-01 22:25:29 +02:00
|
|
|
|
import v.util
|
2020-10-01 19:05:27 +02:00
|
|
|
|
|
|
|
|
|
struct CmdResult {
|
|
|
|
|
mut:
|
2020-10-15 15:17:52 +02:00
|
|
|
|
runs int
|
|
|
|
|
cmd string
|
|
|
|
|
icmd int
|
2020-10-01 19:05:27 +02:00
|
|
|
|
outputs []string
|
2020-10-15 15:17:52 +02:00
|
|
|
|
oms map[string][]int
|
2020-10-01 22:25:29 +02:00
|
|
|
|
summary map[string]Aints
|
2020-10-01 19:05:27 +02:00
|
|
|
|
timings []int
|
2020-10-01 22:25:29 +02:00
|
|
|
|
atiming Aints
|
2020-10-01 19:05:27 +02:00
|
|
|
|
}
|
2020-10-15 15:17:52 +02:00
|
|
|
|
|
2020-10-01 19:05:27 +02:00
|
|
|
|
struct Context {
|
|
|
|
|
mut:
|
2020-10-15 15:17:52 +02:00
|
|
|
|
count int
|
|
|
|
|
series int
|
|
|
|
|
warmup int
|
|
|
|
|
show_help bool
|
|
|
|
|
show_output bool
|
2020-10-02 09:57:58 +02:00
|
|
|
|
fail_on_regress_percent int
|
2020-10-15 15:17:52 +02:00
|
|
|
|
fail_on_maxtime int // in ms
|
|
|
|
|
verbose bool
|
|
|
|
|
commands []string
|
|
|
|
|
results []CmdResult
|
2020-11-29 15:13:45 +01:00
|
|
|
|
cmd_template string // {T} will be substituted with the current command
|
|
|
|
|
cmd_params map[string][]string
|
2020-10-15 15:17:52 +02:00
|
|
|
|
cline string // a terminal clearing line
|
2020-10-01 19:05:27 +02:00
|
|
|
|
}
|
|
|
|
|
|
2020-10-01 22:25:29 +02:00
|
|
|
|
struct Aints {
|
2020-10-15 15:17:52 +02:00
|
|
|
|
values []int
|
2020-10-01 22:25:29 +02:00
|
|
|
|
mut:
|
2020-10-15 15:17:52 +02:00
|
|
|
|
imin int
|
|
|
|
|
imax int
|
2020-10-01 22:25:29 +02:00
|
|
|
|
average f64
|
2020-10-15 15:17:52 +02:00
|
|
|
|
stddev f64
|
2020-10-01 22:25:29 +02:00
|
|
|
|
}
|
2020-10-15 15:17:52 +02:00
|
|
|
|
|
2020-10-01 22:25:29 +02:00
|
|
|
|
fn new_aints(vals []int) Aints {
|
2020-10-15 15:17:52 +02:00
|
|
|
|
mut res := Aints{
|
|
|
|
|
values: vals
|
|
|
|
|
}
|
2020-10-01 22:25:29 +02:00
|
|
|
|
mut sum := i64(0)
|
2020-10-15 15:17:52 +02:00
|
|
|
|
mut imin := math.max_i32
|
2020-10-01 22:25:29 +02:00
|
|
|
|
mut imax := -math.max_i32
|
|
|
|
|
for i in vals {
|
2020-10-15 15:17:52 +02:00
|
|
|
|
sum += i
|
2020-10-01 22:25:29 +02:00
|
|
|
|
if i < imin {
|
|
|
|
|
imin = i
|
|
|
|
|
}
|
|
|
|
|
if i > imax {
|
|
|
|
|
imax = i
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
res.imin = imin
|
|
|
|
|
res.imax = imax
|
|
|
|
|
if vals.len > 0 {
|
|
|
|
|
res.average = sum / f64(vals.len)
|
|
|
|
|
}
|
|
|
|
|
//
|
|
|
|
|
mut devsum := f64(0.0)
|
|
|
|
|
for i in vals {
|
|
|
|
|
x := f64(i) - res.average
|
|
|
|
|
devsum += (x * x)
|
|
|
|
|
}
|
2020-10-15 15:17:52 +02:00
|
|
|
|
res.stddev = math.sqrt(devsum / f64(vals.len))
|
2020-10-01 22:25:29 +02:00
|
|
|
|
return res
|
|
|
|
|
}
|
2020-10-15 15:17:52 +02:00
|
|
|
|
|
|
|
|
|
fn (a Aints) str() string {
|
|
|
|
|
return util.bold('${a.average:9.3f}') +
|
|
|
|
|
'ms ± σ: ${a.stddev:-5.1f}ms, min … max: ${a.imin}ms … ${a.imax}ms'
|
|
|
|
|
}
|
2020-10-01 22:25:29 +02:00
|
|
|
|
|
2020-10-02 09:57:58 +02:00
|
|
|
|
const (
|
2020-10-15 15:17:52 +02:00
|
|
|
|
max_fail_percent = 100000
|
|
|
|
|
max_time = 60 * 1000 // ms
|
2020-10-09 10:06:00 +02:00
|
|
|
|
performance_regression_label = 'Performance regression detected, failing since '
|
2020-10-02 09:57:58 +02:00
|
|
|
|
)
|
2020-10-15 15:17:52 +02:00
|
|
|
|
|
|
|
|
|
fn main() {
|
2020-10-01 19:05:27 +02:00
|
|
|
|
mut context := Context{}
|
2020-10-01 19:46:45 +02:00
|
|
|
|
context.parse_options()
|
|
|
|
|
context.run()
|
|
|
|
|
context.show_diff_summary()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn (mut context Context) parse_options() {
|
2020-10-01 19:05:27 +02:00
|
|
|
|
mut fp := flag.new_flag_parser(os.args)
|
|
|
|
|
fp.application(os.file_name(os.executable()))
|
|
|
|
|
fp.version('0.0.1')
|
2020-10-02 12:28:05 +02:00
|
|
|
|
fp.description('Repeat command(s) and collect statistics. NB: you have to quote each command, if it contains spaces.')
|
2020-10-01 19:05:27 +02:00
|
|
|
|
fp.arguments_description('CMD1 CMD2 ...')
|
|
|
|
|
fp.skip_executable()
|
|
|
|
|
fp.limit_free_args_to_at_least(1)
|
2020-10-02 12:28:05 +02:00
|
|
|
|
context.count = fp.int('count', `c`, 10, 'Repetition count.')
|
|
|
|
|
context.series = fp.int('series', `s`, 2, 'Series count. `-s 2 -c 4 a b` => aaaabbbbaaaabbbb, while `-s 3 -c 2 a b` => aabbaabbaabb.')
|
|
|
|
|
context.warmup = fp.int('warmup', `w`, 2, 'Warmup runs. These are done *only at the start*, and are ignored.')
|
2020-10-01 19:05:27 +02:00
|
|
|
|
context.show_help = fp.bool('help', `h`, false, 'Show this help screen.')
|
2020-10-02 17:10:25 +02:00
|
|
|
|
context.show_output = fp.bool('output', `O`, false, 'Show command stdout/stderr in the progress indicator for each command. NB: slower, for verbose commands.')
|
2020-10-01 19:05:27 +02:00
|
|
|
|
context.verbose = fp.bool('verbose', `v`, false, 'Be more verbose.')
|
2020-10-09 10:06:00 +02:00
|
|
|
|
context.fail_on_maxtime = fp.int('max_time', `m`, max_time, 'Fail with exit code 2, when first cmd takes above M milliseconds (regression).')
|
|
|
|
|
context.fail_on_regress_percent = fp.int('fail_percent', `f`, max_fail_percent, 'Fail with exit code 3, when first cmd is X% slower than the rest (regression).')
|
2020-11-29 15:13:45 +01:00
|
|
|
|
context.cmd_template = fp.string('template', `t`, '{T}', 'Command template. {T} will be substituted with the current command.')
|
|
|
|
|
cmd_params := fp.string_multi('parameter', `p`, 'A parameter substitution list. `{p}=val1,val2,val2` means that {p} in the template, will be substituted with each of val1, val2, val3.')
|
|
|
|
|
for p in cmd_params {
|
|
|
|
|
parts := p.split(':')
|
|
|
|
|
if parts.len > 1 {
|
|
|
|
|
context.cmd_params[parts[0]] = parts[1].split(',')
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-10-01 19:05:27 +02:00
|
|
|
|
if context.show_help {
|
|
|
|
|
println(fp.usage())
|
|
|
|
|
exit(0)
|
|
|
|
|
}
|
|
|
|
|
if context.verbose {
|
|
|
|
|
scripting.set_verbose(true)
|
|
|
|
|
}
|
2020-11-29 15:13:45 +01:00
|
|
|
|
commands := fp.finalize() or {
|
2020-10-01 19:05:27 +02:00
|
|
|
|
eprintln('Error: ' + err)
|
|
|
|
|
exit(1)
|
|
|
|
|
}
|
2020-11-29 15:13:45 +01:00
|
|
|
|
context.commands = context.expand_all_commands(commands)
|
2020-10-15 15:17:52 +02:00
|
|
|
|
context.results = []CmdResult{len: context.commands.len, init: CmdResult{}}
|
2020-10-01 22:25:29 +02:00
|
|
|
|
context.cline = '\r' + term.h_divider('')
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn (mut context Context) clear_line() {
|
|
|
|
|
print(context.cline)
|
2020-10-01 19:46:45 +02:00
|
|
|
|
}
|
|
|
|
|
|
2020-11-29 15:13:45 +01:00
|
|
|
|
fn (mut context Context) expand_all_commands(commands []string) []string {
|
|
|
|
|
mut all_commands := []string{}
|
|
|
|
|
for cmd in commands {
|
|
|
|
|
maincmd := context.cmd_template.replace('{T}', cmd)
|
|
|
|
|
mut substituted_commands := [maincmd]
|
|
|
|
|
for paramk, paramlist in context.cmd_params {
|
|
|
|
|
for paramv in paramlist {
|
|
|
|
|
mut new_substituted_commands := []string{}
|
|
|
|
|
for cscmd in substituted_commands {
|
|
|
|
|
scmd := cscmd.replace(paramk, paramv)
|
|
|
|
|
new_substituted_commands << scmd
|
|
|
|
|
}
|
|
|
|
|
substituted_commands << new_substituted_commands
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
all_commands << substituted_commands
|
|
|
|
|
}
|
|
|
|
|
mut unique := map[string]int{}
|
|
|
|
|
for x in all_commands {
|
|
|
|
|
if x.contains('{') && x.contains('}') {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
unique[x] = 1
|
|
|
|
|
}
|
|
|
|
|
return unique.keys()
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-01 19:46:45 +02:00
|
|
|
|
fn (mut context Context) run() {
|
2020-10-02 12:28:05 +02:00
|
|
|
|
mut run_warmups := 0
|
2020-10-15 15:17:52 +02:00
|
|
|
|
for si in 1 .. context.series + 1 {
|
2020-10-02 12:28:05 +02:00
|
|
|
|
for icmd, cmd in context.commands {
|
|
|
|
|
mut runs := 0
|
|
|
|
|
mut duration := 0
|
|
|
|
|
mut sum := 0
|
|
|
|
|
mut oldres := ''
|
|
|
|
|
println('Series: ${si:4}/${context.series:-4}, command: $cmd')
|
|
|
|
|
if context.warmup > 0 && run_warmups < context.commands.len {
|
2020-10-15 15:17:52 +02:00
|
|
|
|
for i in 1 .. context.warmup + 1 {
|
2020-10-02 12:28:05 +02:00
|
|
|
|
print('\r warming up run: ${i:4}/${context.warmup:-4} for ${cmd:-50s} took ${duration:6} ms ...')
|
|
|
|
|
mut sw := time.new_stopwatch({})
|
2020-10-15 15:17:52 +02:00
|
|
|
|
os.exec(cmd) or {
|
|
|
|
|
continue
|
|
|
|
|
}
|
2020-10-02 12:28:05 +02:00
|
|
|
|
duration = int(sw.elapsed().milliseconds())
|
|
|
|
|
}
|
|
|
|
|
run_warmups++
|
|
|
|
|
}
|
|
|
|
|
context.clear_line()
|
2020-10-15 15:17:52 +02:00
|
|
|
|
for i in 1 .. (context.count + 1) {
|
|
|
|
|
avg := f64(sum) / f64(i)
|
2020-10-02 12:28:05 +02:00
|
|
|
|
print('\rAverage: ${avg:9.3f}ms | run: ${i:4}/${context.count:-4} | took ${duration:6} ms')
|
2020-10-02 17:10:25 +02:00
|
|
|
|
if context.show_output {
|
2020-10-15 15:17:52 +02:00
|
|
|
|
print(' | result: ${oldres:s}')
|
2020-10-02 12:28:05 +02:00
|
|
|
|
}
|
2020-10-01 19:13:19 +02:00
|
|
|
|
mut sw := time.new_stopwatch({})
|
2020-10-09 10:06:00 +02:00
|
|
|
|
res := scripting.exec(cmd) or {
|
2020-10-02 12:28:05 +02:00
|
|
|
|
continue
|
|
|
|
|
}
|
2020-10-01 19:13:19 +02:00
|
|
|
|
duration = int(sw.elapsed().milliseconds())
|
2020-10-02 12:28:05 +02:00
|
|
|
|
if res.exit_code != 0 {
|
|
|
|
|
eprintln('${i:10} non 0 exit code for cmd: $cmd')
|
|
|
|
|
continue
|
|
|
|
|
}
|
2020-10-15 15:17:52 +02:00
|
|
|
|
context.results[icmd].outputs <<
|
|
|
|
|
res.output.trim_right('\r\n').replace('\r\n', '\n').split('\n')
|
2020-10-02 12:28:05 +02:00
|
|
|
|
context.results[icmd].timings << duration
|
|
|
|
|
sum += duration
|
|
|
|
|
runs++
|
|
|
|
|
oldres = res.output.replace('\n', ' ')
|
2020-10-01 19:13:19 +02:00
|
|
|
|
}
|
2020-10-02 12:28:05 +02:00
|
|
|
|
context.results[icmd].cmd = cmd
|
|
|
|
|
context.results[icmd].icmd = icmd
|
2020-10-06 07:12:09 +02:00
|
|
|
|
context.results[icmd].runs += runs
|
2020-10-02 12:28:05 +02:00
|
|
|
|
context.results[icmd].atiming = new_aints(context.results[icmd].timings)
|
|
|
|
|
context.clear_line()
|
|
|
|
|
print('\r')
|
2020-10-15 15:17:52 +02:00
|
|
|
|
mut m := map[string][]int{}
|
2020-10-02 12:28:05 +02:00
|
|
|
|
for o in context.results[icmd].outputs {
|
|
|
|
|
x := o.split(':')
|
|
|
|
|
if x.len > 1 {
|
|
|
|
|
k := x[0]
|
|
|
|
|
v := x[1].trim_left(' ').int()
|
|
|
|
|
m[k] << v
|
|
|
|
|
}
|
2020-10-09 10:06:00 +02:00
|
|
|
|
}
|
2020-10-02 12:28:05 +02:00
|
|
|
|
mut summary := map[string]Aints{}
|
2020-10-15 15:17:52 +02:00
|
|
|
|
for k, v in m {
|
2020-10-06 07:12:09 +02:00
|
|
|
|
// show a temporary summary for the current series/cmd cycle
|
2020-10-02 12:28:05 +02:00
|
|
|
|
s := new_aints(v)
|
|
|
|
|
println(' $k: $s')
|
|
|
|
|
summary[k] = s
|
2020-10-01 19:05:27 +02:00
|
|
|
|
}
|
2020-10-06 07:12:09 +02:00
|
|
|
|
// merge current raw results to the previous ones
|
|
|
|
|
old_oms := context.results[icmd].oms
|
2020-10-15 15:17:52 +02:00
|
|
|
|
mut new_oms := map[string][]int{}
|
|
|
|
|
for k, v in m {
|
2020-10-06 07:12:09 +02:00
|
|
|
|
if old_oms[k].len == 0 {
|
|
|
|
|
new_oms[k] = v
|
|
|
|
|
} else {
|
|
|
|
|
new_oms[k] << old_oms[k]
|
|
|
|
|
new_oms[k] << v
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
context.results[icmd].oms = new_oms
|
2020-10-15 15:17:52 +02:00
|
|
|
|
// println('')
|
2020-10-01 22:25:29 +02:00
|
|
|
|
}
|
2020-10-01 20:06:32 +02:00
|
|
|
|
}
|
2020-10-06 07:12:09 +02:00
|
|
|
|
// create full summaries, taking account of all runs
|
2020-10-15 15:17:52 +02:00
|
|
|
|
for icmd in 0 .. context.results.len {
|
2020-10-06 07:12:09 +02:00
|
|
|
|
mut new_full_summary := map[string]Aints{}
|
2020-10-15 15:17:52 +02:00
|
|
|
|
for k, v in context.results[icmd].oms {
|
2020-10-06 07:12:09 +02:00
|
|
|
|
new_full_summary[k] = new_aints(v)
|
|
|
|
|
}
|
|
|
|
|
context.results[icmd].summary = new_full_summary
|
|
|
|
|
}
|
2020-10-01 19:46:45 +02:00
|
|
|
|
}
|
2020-10-15 15:17:52 +02:00
|
|
|
|
|
2020-10-01 19:46:45 +02:00
|
|
|
|
fn (mut context Context) show_diff_summary() {
|
2020-10-15 15:17:52 +02:00
|
|
|
|
context.results.sort_with_compare(fn (a &CmdResult, b &CmdResult) int {
|
2020-10-01 22:25:29 +02:00
|
|
|
|
if a.atiming.average < b.atiming.average {
|
|
|
|
|
return -1
|
|
|
|
|
}
|
|
|
|
|
if a.atiming.average > b.atiming.average {
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
return 0
|
|
|
|
|
})
|
2020-10-02 12:28:05 +02:00
|
|
|
|
println('Summary (commands are ordered by ascending mean time), after $context.series series of $context.count repetitions:')
|
2020-10-01 22:25:29 +02:00
|
|
|
|
base := context.results[0].atiming.average
|
2020-10-02 09:57:58 +02:00
|
|
|
|
mut first_cmd_percentage := f64(100.0)
|
2020-10-01 22:25:29 +02:00
|
|
|
|
for i, r in context.results {
|
2020-10-02 09:57:58 +02:00
|
|
|
|
cpercent := (r.atiming.average / base) * 100 - 100
|
|
|
|
|
first_marker := if r.icmd == 0 { util.bold('>') } else { ' ' }
|
|
|
|
|
if r.icmd == 0 {
|
|
|
|
|
first_cmd_percentage = cpercent
|
|
|
|
|
}
|
2020-10-21 13:36:16 +02:00
|
|
|
|
println(' $first_marker${(i + 1):3} | ${cpercent:6.1f}% slower | ${r.cmd:-55s} | $r.atiming')
|
2020-10-02 09:57:58 +02:00
|
|
|
|
}
|
2020-10-09 10:06:00 +02:00
|
|
|
|
$if debugcontext ? {
|
|
|
|
|
println('context: $context')
|
|
|
|
|
}
|
|
|
|
|
if int(base) > context.fail_on_maxtime {
|
|
|
|
|
print(performance_regression_label)
|
2020-10-15 15:17:52 +02:00
|
|
|
|
println('average time: ${base:6.1f} ms > $context.fail_on_maxtime ms threshold.')
|
2020-10-09 10:06:00 +02:00
|
|
|
|
exit(2)
|
|
|
|
|
}
|
2020-10-02 09:57:58 +02:00
|
|
|
|
if context.fail_on_regress_percent == max_fail_percent || context.results.len < 2 {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
fail_threshold_max := f64(context.fail_on_regress_percent)
|
|
|
|
|
if first_cmd_percentage > fail_threshold_max {
|
2020-10-09 10:06:00 +02:00
|
|
|
|
print(performance_regression_label)
|
2020-10-02 09:57:58 +02:00
|
|
|
|
println('${first_cmd_percentage:5.1f}% > ${fail_threshold_max:5.1f}% threshold.')
|
2020-10-09 10:06:00 +02:00
|
|
|
|
exit(3)
|
2020-10-01 22:25:29 +02:00
|
|
|
|
}
|
2020-10-01 19:05:27 +02:00
|
|
|
|
}
|