revsetbenchmarks.py
319 lines
| 9.7 KiB
| text/x-python
|
PythonLexer
/ contrib / revsetbenchmarks.py
Pierre-Yves David
|
r20848 | #!/usr/bin/env python | ||
# Measure the performance of a list of revsets against multiple revisions | ||||
# defined by parameter. Checkout one by one and run perfrevset with every | ||||
# revset in the list to benchmark its performance. | ||||
# | ||||
Pierre-Yves David
|
r25535 | # You should run this from the root of your mercurial repository. | ||
Pierre-Yves David
|
r20848 | # | ||
Pierre-Yves David
|
r25535 | # call with --help for details | ||
Pierre-Yves David
|
r20848 | |||
import sys | ||||
Pierre-Yves David
|
r21548 | import os | ||
Pierre-Yves David
|
r25530 | import re | ||
Pierre-Yves David
|
r25539 | import math | ||
Durham Goode
|
r20893 | from subprocess import check_call, Popen, CalledProcessError, STDOUT, PIPE | ||
Pierre-Yves David
|
r21287 | # cannot use argparse, python 2.7 only | ||
from optparse import OptionParser | ||||
Pierre-Yves David
|
r25544 | DEFAULTVARIANTS = ['plain', 'min', 'max', 'first', 'last', | ||
'reverse', 'reverse+first', 'reverse+last', | ||||
'sort', 'sort+first', 'sort+last'] | ||||
Pierre-Yves David
|
r25540 | |||
Durham Goode
|
r20893 | def check_output(*args, **kwargs): | ||
kwargs.setdefault('stderr', PIPE) | ||||
kwargs.setdefault('stdout', PIPE) | ||||
proc = Popen(*args, **kwargs) | ||||
output, error = proc.communicate() | ||||
if proc.returncode != 0: | ||||
Pierre-Yves David
|
r21202 | raise CalledProcessError(proc.returncode, ' '.join(args[0])) | ||
Durham Goode
|
r20893 | return output | ||
Pierre-Yves David
|
r20848 | |||
Pierre-Yves David
|
r20850 | def update(rev): | ||
"""update the repo to a revision""" | ||||
try: | ||||
check_call(['hg', 'update', '--quiet', '--check', str(rev)]) | ||||
Yuya Nishihara
|
r26034 | check_output(['make', 'local'], | ||
stderr=None) # suppress output except for error/warning | ||||
Gregory Szorc
|
r25660 | except CalledProcessError as exc: | ||
Pierre-Yves David
|
r20850 | print >> sys.stderr, 'update to revision %s failed, aborting' % rev | ||
sys.exit(exc.returncode) | ||||
Pierre-Yves David
|
r25528 | |||
def hg(cmd, repo=None): | ||||
"""run a mercurial command | ||||
<cmd> is the list of command + argument, | ||||
<repo> is an optional repository path to run this command in.""" | ||||
fullcmd = ['./hg'] | ||||
if repo is not None: | ||||
fullcmd += ['-R', repo] | ||||
fullcmd += ['--config', | ||||
'extensions.perf=' + os.path.join(contribdir, 'perf.py')] | ||||
fullcmd += cmd | ||||
return check_output(fullcmd, stderr=STDOUT) | ||||
Pierre-Yves David
|
r21549 | def perf(revset, target=None): | ||
Pierre-Yves David
|
r20851 | """run benchmark for this very revset""" | ||
try: | ||||
Pierre-Yves David
|
r25528 | output = hg(['perfrevset', revset], repo=target) | ||
Pierre-Yves David
|
r25530 | return parseoutput(output) | ||
Gregory Szorc
|
r25660 | except CalledProcessError as exc: | ||
Pierre-Yves David
|
r25529 | print >> sys.stderr, 'abort: cannot run revset benchmark: %s' % exc.cmd | ||
if exc.output is None: | ||||
Mads Kiilerich
|
r26781 | print >> sys.stderr, '(no output)' | ||
Pierre-Yves David
|
r25529 | else: | ||
print >> sys.stderr, exc.output | ||||
Pierre-Yves David
|
r25646 | return None | ||
Pierre-Yves David
|
r20851 | |||
Pierre-Yves David
|
r25530 | outputre = re.compile(r'! wall (\d+.\d+) comb (\d+.\d+) user (\d+.\d+) ' | ||
'sys (\d+.\d+) \(best of (\d+)\)') | ||||
def parseoutput(output): | ||||
"""parse a textual output into a dict | ||||
We cannot just use json because we want to compare with old | ||||
versions of Mercurial that may not support json output. | ||||
""" | ||||
match = outputre.search(output) | ||||
if not match: | ||||
print >> sys.stderr, 'abort: invalid output:' | ||||
print >> sys.stderr, output | ||||
sys.exit(1) | ||||
return {'comb': float(match.group(2)), | ||||
'count': int(match.group(5)), | ||||
'sys': float(match.group(3)), | ||||
'user': float(match.group(4)), | ||||
'wall': float(match.group(1)), | ||||
} | ||||
Pierre-Yves David
|
r20852 | def printrevision(rev): | ||
"""print data about a revision""" | ||||
Pierre-Yves David
|
r25538 | sys.stdout.write("Revision ") | ||
Pierre-Yves David
|
r20852 | sys.stdout.flush() | ||
check_call(['hg', 'log', '--rev', str(rev), '--template', | ||||
Pierre-Yves David
|
r25546 | '{if(tags, " ({tags})")} ' | ||
Pierre-Yves David
|
r25538 | '{rev}:{node|short}: {desc|firstline}\n']) | ||
Pierre-Yves David
|
r20852 | |||
Pierre-Yves David
|
r25532 | def idxwidth(nbidx): | ||
"""return the max width of number used for index | ||||
Augie Fackler
|
r25533 | This is similar to log10(nbidx), but we use custom code here | ||
because we start with zero and we'd rather not deal with all the | ||||
extra rounding business that log10 would imply. | ||||
""" | ||||
Pierre-Yves David
|
r25532 | nbidx -= 1 # starts at 0 | ||
idxwidth = 0 | ||||
while nbidx: | ||||
idxwidth += 1 | ||||
nbidx //= 10 | ||||
if not idxwidth: | ||||
idxwidth = 1 | ||||
return idxwidth | ||||
Pierre-Yves David
|
r25539 | def getfactor(main, other, field, sensitivity=0.05): | ||
"""return the relative factor between values for 'field' in main and other | ||||
Mads Kiilerich
|
r26781 | Return None if the factor is insignificant (less than <sensitivity> | ||
Pierre-Yves David
|
r25539 | variation).""" | ||
factor = 1 | ||||
if main is not None: | ||||
factor = other[field] / main[field] | ||||
low, high = 1 - sensitivity, 1 + sensitivity | ||||
if (low < factor < high): | ||||
return None | ||||
return factor | ||||
def formatfactor(factor): | ||||
"""format a factor into a 4 char string | ||||
22% | ||||
156% | ||||
x2.4 | ||||
x23 | ||||
x789 | ||||
x1e4 | ||||
x5x7 | ||||
""" | ||||
if factor is None: | ||||
return ' ' | ||||
elif factor < 2: | ||||
return '%3i%%' % (factor * 100) | ||||
elif factor < 10: | ||||
return 'x%3.1f' % factor | ||||
elif factor < 1000: | ||||
return '%4s' % ('x%i' % factor) | ||||
else: | ||||
order = int(math.log(factor)) + 1 | ||||
while 1 < math.log(factor): | ||||
factor //= 0 | ||||
return 'x%ix%i' % (factor, order) | ||||
Pierre-Yves David
|
r25541 | def formattiming(value): | ||
"""format a value to strictly 8 char, dropping some precision if needed""" | ||||
if value < 10**7: | ||||
return ('%.6f' % value)[:8] | ||||
else: | ||||
# value is HUGE very unlikely to happen (4+ month run) | ||||
return '%i' % value | ||||
Pierre-Yves David
|
r25539 | _marker = object() | ||
Pierre-Yves David
|
r25540 | def printresult(variants, idx, data, maxidx, verbose=False, reference=_marker): | ||
Pierre-Yves David
|
r25531 | """print a line of result to stdout""" | ||
Pierre-Yves David
|
r25532 | mask = '%%0%ii) %%s' % idxwidth(maxidx) | ||
Pierre-Yves David
|
r25646 | |||
Pierre-Yves David
|
r25540 | out = [] | ||
for var in variants: | ||||
Pierre-Yves David
|
r25646 | if data[var] is None: | ||
out.append('error ') | ||||
out.append(' ' * 4) | ||||
continue | ||||
Pierre-Yves David
|
r25541 | out.append(formattiming(data[var]['wall'])) | ||
Pierre-Yves David
|
r25540 | if reference is not _marker: | ||
factor = None | ||||
if reference is not None: | ||||
factor = getfactor(reference[var], data[var], 'wall') | ||||
out.append(formatfactor(factor)) | ||||
if verbose: | ||||
Pierre-Yves David
|
r25541 | out.append(formattiming(data[var]['comb'])) | ||
out.append(formattiming(data[var]['user'])) | ||||
out.append(formattiming(data[var]['sys'])) | ||||
Pierre-Yves David
|
r25540 | out.append('%6d' % data[var]['count']) | ||
Pierre-Yves David
|
r25534 | print mask % (idx, ' '.join(out)) | ||
Pierre-Yves David
|
r25530 | |||
Pierre-Yves David
|
r25540 | def printheader(variants, maxidx, verbose=False, relative=False): | ||
header = [' ' * (idxwidth(maxidx) + 1)] | ||||
for var in variants: | ||||
if not var: | ||||
var = 'iter' | ||||
Pierre-Yves David
|
r25541 | if 8 < len(var): | ||
var = var[:3] + '..' + var[-3:] | ||||
header.append('%-8s' % var) | ||||
Pierre-Yves David
|
r25540 | if relative: | ||
header.append(' ') | ||||
if verbose: | ||||
Pierre-Yves David
|
r25541 | header.append('%-8s' % 'comb') | ||
header.append('%-8s' % 'user') | ||||
header.append('%-8s' % 'sys') | ||||
Pierre-Yves David
|
r25540 | header.append('%6s' % 'count') | ||
Pierre-Yves David
|
r25534 | print ' '.join(header) | ||
Pierre-Yves David
|
r25530 | |||
Pierre-Yves David
|
r20853 | def getrevs(spec): | ||
"""get the list of rev matched by a revset""" | ||||
try: | ||||
out = check_output(['hg', 'log', '--template={rev}\n', '--rev', spec]) | ||||
Gregory Szorc
|
r25660 | except CalledProcessError as exc: | ||
Pierre-Yves David
|
r20853 | print >> sys.stderr, "abort, can't get revision from %s" % spec | ||
sys.exit(exc.returncode) | ||||
return [r for r in out.split() if r] | ||||
Pierre-Yves David
|
r25540 | def applyvariants(revset, variant): | ||
if variant == 'plain': | ||||
return revset | ||||
Pierre-Yves David
|
r25543 | for var in variant.split('+'): | ||
revset = '%s(%s)' % (var, revset) | ||||
return revset | ||||
Pierre-Yves David
|
r25540 | |||
Pierre-Yves David
|
r25607 | helptext="""This script will run multiple variants of provided revsets using | ||
different revisions in your mercurial repository. After the benchmark are run | ||||
Mads Kiilerich
|
r26781 | summary output is provided. Use it to demonstrate speed improvements or pin | ||
Pierre-Yves David
|
r25607 | point regressions. Revsets to run are specified in a file (or from stdin), one | ||
revsets per line. Line starting with '#' will be ignored, allowing insertion of | ||||
comments.""" | ||||
parser = OptionParser(usage="usage: %prog [options] <revs>", | ||||
description=helptext) | ||||
Pierre-Yves David
|
r21287 | parser.add_option("-f", "--file", | ||
Mads Kiilerich
|
r23139 | help="read revset from FILE (stdin if omitted)", | ||
Pierre-Yves David
|
r22555 | metavar="FILE") | ||
Pierre-Yves David
|
r21549 | parser.add_option("-R", "--repo", | ||
help="run benchmark on REPO", metavar="REPO") | ||||
Pierre-Yves David
|
r21287 | |||
Pierre-Yves David
|
r25537 | parser.add_option("-v", "--verbose", | ||
action='store_true', | ||||
help="display all timing data (not just best total time)") | ||||
Pierre-Yves David
|
r25540 | parser.add_option("", "--variants", | ||
default=','.join(DEFAULTVARIANTS), | ||||
help="comma separated list of variant to test " | ||||
"(eg: plain,min,sorted) (plain = no modification)") | ||||
Pierre-Yves David
|
r21287 | (options, args) = parser.parse_args() | ||
Pierre-Yves David
|
r20848 | |||
Pierre-Yves David
|
r25535 | if not args: | ||
Pierre-Yves David
|
r21287 | parser.print_help() | ||
Pierre-Yves David
|
r21286 | sys.exit(255) | ||
Pierre-Yves David
|
r21548 | # the directory where both this script and the perf.py extension live. | ||
contribdir = os.path.dirname(__file__) | ||||
Pierre-Yves David
|
r21287 | |||
Pierre-Yves David
|
r20848 | revsetsfile = sys.stdin | ||
Pierre-Yves David
|
r21287 | if options.file: | ||
revsetsfile = open(options.file) | ||||
Pierre-Yves David
|
r20848 | |||
Pierre-Yves David
|
r22556 | revsets = [l.strip() for l in revsetsfile if not l.startswith('#')] | ||
Pierre-Yves David
|
r25642 | revsets = [l for l in revsets if l] | ||
Pierre-Yves David
|
r20848 | |||
print "Revsets to benchmark" | ||||
print "----------------------------" | ||||
for idx, rset in enumerate(revsets): | ||||
print "%i) %s" % (idx, rset) | ||||
print "----------------------------" | ||||
Pierre-Yves David
|
r25535 | revs = [] | ||
for a in args: | ||||
revs.extend(getrevs(a)) | ||||
Pierre-Yves David
|
r20848 | |||
Pierre-Yves David
|
r25540 | variants = options.variants.split(',') | ||
Pierre-Yves David
|
r20855 | results = [] | ||
Pierre-Yves David
|
r20848 | for r in revs: | ||
print "----------------------------" | ||||
Pierre-Yves David
|
r20852 | printrevision(r) | ||
Pierre-Yves David
|
r20848 | print "----------------------------" | ||
Pierre-Yves David
|
r20850 | update(r) | ||
Pierre-Yves David
|
r20855 | res = [] | ||
results.append(res) | ||||
Pierre-Yves David
|
r25540 | printheader(variants, len(revsets), verbose=options.verbose) | ||
Pierre-Yves David
|
r20848 | for idx, rset in enumerate(revsets): | ||
Pierre-Yves David
|
r25540 | varres = {} | ||
for var in variants: | ||||
varrset = applyvariants(rset, var) | ||||
data = perf(varrset, target=options.repo) | ||||
varres[var] = data | ||||
res.append(varres) | ||||
printresult(variants, idx, varres, len(revsets), | ||||
verbose=options.verbose) | ||||
Pierre-Yves David
|
r20855 | sys.stdout.flush() | ||
Pierre-Yves David
|
r20848 | print "----------------------------" | ||
Pierre-Yves David
|
r20855 | |||
print """ | ||||
Result by revset | ||||
================ | ||||
""" | ||||
Pierre-Yves David
|
r25538 | print 'Revision:' | ||
Pierre-Yves David
|
r20855 | for idx, rev in enumerate(revs): | ||
sys.stdout.write('%i) ' % idx) | ||||
sys.stdout.flush() | ||||
printrevision(rev) | ||||
for ridx, rset in enumerate(revsets): | ||||
print "revset #%i: %s" % (ridx, rset) | ||||
Pierre-Yves David
|
r25540 | printheader(variants, len(results), verbose=options.verbose, relative=True) | ||
Pierre-Yves David
|
r25539 | ref = None | ||
Pierre-Yves David
|
r20855 | for idx, data in enumerate(results): | ||
Pierre-Yves David
|
r25540 | printresult(variants, idx, data[ridx], len(results), | ||
verbose=options.verbose, reference=ref) | ||||
Pierre-Yves David
|
r25539 | ref = data[ridx] | ||
Pierre-Yves David
|
r20855 | |||