Anton Shestakov <av6@dwimlabs.net>, Mon, 18 Jul 2016 18:45:54 +0800
viewer: pass marks to getresults() as a positional argument
bench.py
Permissions: -rwxr-xr-x
from argparse import ArgumentParser from collections import OrderedDict from subprocess import check_output, CalledProcessError, STDOUT from settings import DBPATH, HG, LOCKFILE, TESTHGREPO, TESTREPO, rel REVSETWIP = '(parents(not public()) or not public() or . or (head() and branch(default))) and (not obsolete() or unstable()^) and not closed()' ('blame', 'hg blame README'), ('grepall', 'hg grep "version" --all README'), ('grepallf', 'hg grep "version" --all --follow README'), ('diff', 'hg diff -r "tip~100:tip" README'), ('diffg', 'hg diff -r "tip~100:tip" --git README'), ('stcp', 'hg status --copies README'), ('logfile', 'hg log README'), ('logfilecp', 'hg log --copies README'), ('log1000', 'hg log -l1000'), ('revsetor', 'hg log -r "0|1|2|3|4|5|6|7|8|9"'), ('revsetwip', 'hg log -r "' + REVSETWIP + '"'), parser = ArgumentParser(description='Benchmark revisions and put results in the db.') group = parser.add_mutually_exclusive_group() group.add_argument('revsets', metavar='REVSET', default=('last(all(), 120)',), nargs='*', help='update these revisions') group.add_argument('--auto', metavar='MAXREVS', type=int, help='guess revisions, up to MAXREVS') parser.add_argument('--retry', action='store_true', help='try and reduce existing timings') def test(mark, mintime=1.0, mintries=3, dropcache=True): cmd = [rel(TESTHGREPO, 'hg'), '-R', TESTREPO] cmd += ['blame', rel(TESTREPO, 'README')] cmd += ['grep', '--all', 'version', rel(TESTREPO, 'README')] cmd += ['grep', '--all', '--follow', 'version', rel(TESTREPO, 'README')] cmd += ['status', '--copies', rel(TESTREPO, 'README')] cmd += ['diff', '-r', 'tip~100:tip', rel(TESTREPO, 'README')] cmd += ['diff', '-r', 'tip~100:tip', '--git', rel(TESTREPO, 'README')] cmd += ['log', rel(TESTREPO, 'README')] elif mark == 'logfilecp': cmd += ['log', '--copies', rel(TESTREPO, 'README')] cmd += ['log', '-r', '0|1|2|3|4|5|6|7|8|9'] elif mark == 'revsetwip': cmd += ['log', '-r', REVSETWIP] while sum(results) < mintime or len(results) < mintries: shutil.rmtree(rel(TESTREPO, '.hg', 'cache'), ignore_errors=True) except CalledProcessError: results.append(time.time() - start) cmd = [HG, 'log', '-R', TESTHGREPO, '-T', '{node}\n'] output = check_output(cmd) """ Pick one continuous span of nodes that still need testing. """ cmd = [HG, 'log', '-R', TESTHGREPO, '-T', '{node}\n', '-r', 'sort(all(), rev)'] output = check_output(cmd) conn = sqlite3.connect(DBPATH) while len(todo) < maxrevs: 'SELECT COUNT(*) FROM results' if count < len(MARKS) * len(('without cache', 'with cache')): def guessspikes(maxrevs): cmd = [HG, 'log', '-R', TESTHGREPO, '-T', '{node}\n', '-r', 'sort(all(), -rev)'] output = check_output(cmd) conn = sqlite3.connect(DBPATH) 'SELECT MIN(time), MAX(time) FROM results' ' WHERE mark = ? AND cache = ?', (mark, False)).fetchone() 'SELECT MIN(time), MAX(time) FROM results' ' WHERE mark = ? AND cache = ?', 'SELECT mark, time, cache FROM results WHERE node = ?', for mark, t, cache in resultsq: results.setdefault(node, {}).setdefault(mark, [None, None]) results[node][mark][0] = t results[node][mark][1] = t for i in range(1, len(nodes) - 1): for cache in (False, True): eps = abs(results[node1][mark][cache] - results[node3][mark][cache]) delta = results[node2][mark][cache] - results[node1][mark][cache] l = limits[mark][2:4] if cache else limits[mark][0:2] if delta > eps * 10 and delta > (l[1] - l[0]) * 0.1: except (KeyError, TypeError): check_output(['make', '--directory', TESTHGREPO, 'clean'], stderr=STDOUT) check_output([HG, 'update', '-R', TESTHGREPO, '--clean', node], stderr=STDOUT) check_output(['make', '--directory', TESTHGREPO, 'local'], stderr=STDOUT) conn = sqlite3.connect(DBPATH) 'CREATE TABLE IF NOT EXISTS results (' ' node CHAR(40) NOT NULL,' ' mark VARCHAR(40) NOT NULL,' 'CREATE INDEX IF NOT EXISTS idx_results_node' 'CREATE INDEX IF NOT EXISTS idx_results_limits' ' ON results (mark, cache, node, time ASC)') def dbupdate(revsets, retry=False): conn = sqlite3.connect(DBPATH) nodes = getnodes(revsets) numwidth = len(str(len(nodes))) markwidth = max(len(mark) for mark in MARKS) for i, node in enumerate(nodes, 1): for cache in (False, True): 'SELECT time FROM results' ' WHERE node = ? AND mark = ? AND cache = ?', (node, mark, cache)).fetchall() oldtime = old[0][0] if old else None if oldtime is not None and not retry: time = test(mark, dropcache=not cache) '%0*d/%0*d %s %-*s %s (%s)', numwidth, i, numwidth, len(nodes), node, markwidth, mark, time, status) 'INSERT INTO results (node, mark, time, cache)' (node, mark, time, cache)) 'UPDATE results SET time = ?' ' WHERE node = ? AND mark = ? AND cache = ?', (time, node, mark, cache)) flags = os.O_WRONLY | os.O_CREAT | os.O_EXCL return os.fdopen(os.open(LOCKFILE, flags), 'w') if e.errno == errno.EEXIST: logging.error('cannot lock data directory') args.revsets = guessspikes(args.auto) args.revsets = guessnew(args.auto) dbupdate(args.revsets, args.retry) if __name__ == '__main__': logging.basicConfig(format='%(levelname).1s %(asctime)s %(message)s', level=logging.INFO) args = parser.parse_args()