skia2/tools/skqp/cut_release.py
Hal Canary a06f9d0787 SkQP: model-creation improvements
- cut_release/find_commit: use cookies, handle errors.

  - tools/skqp/make_skqp_model.cpp: better error handling

Change-Id: I702e9a13d3a2123c30e3a870fcb942759cfc47c3
Reviewed-on: https://skia-review.googlesource.com/c/skia/+/257682
Reviewed-by: Ben Wagner <bungeman@google.com>
Commit-Queue: Hal Canary <halcanary@google.com>
2019-12-04 17:08:36 +00:00

173 lines
5.3 KiB
Python
Executable File

#! /usr/bin/env python
# Copyright 2018 Google LLC.
# Use of this source code is governed by a BSD-style license that can be found in the LICENSE file.
import json
import md5
import multiprocessing
import os
import shutil
import sys
import tempfile
import urllib
import urllib2
from subprocess import check_call, check_output
assert '/' in [os.sep, os.altsep] and os.pardir == '..'
ASSETS = 'platform_tools/android/apps/skqp/src/main/assets'
BUCKET = 'skia-skqp-assets'
def urlopen(url):
cookie = os.environ.get('SKIA_GOLD_COOKIE', '')
return urllib2.urlopen(urllib2.Request(url, headers={'Cookie': cookie}))
def make_skqp_model(arg):
name, urls, exe = arg
tmp = tempfile.mkdtemp()
for url in urls:
urllib.urlretrieve(url, tmp + '/' + url[url.rindex('/') + 1:])
check_call([exe, tmp, ASSETS + '/gmkb/' + name])
shutil.rmtree(tmp)
sys.stdout.write(name + ' ')
sys.stdout.flush()
def goldgetter(meta, exe):
assert os.path.exists(exe)
jobs = []
for rec in meta:
urls = [d['URL'] for d in rec['digests']
if d['status'] == 'positive' and
(set(d['paramset']['config']) & set(['vk', 'gles']))]
if urls:
jobs.append((rec['testName'], urls, exe))
pool = multiprocessing.Pool(processes=20)
pool.map(make_skqp_model, jobs)
sys.stdout.write('\n')
return set((n for n, _, _ in jobs))
def gold(first_commit, last_commit):
c1, c2 = (check_output(['git', 'rev-parse', c]).strip()
for c in (first_commit, last_commit))
f = urlopen('https://public-gold.skia.org/json/export?' + urllib.urlencode([
('fbegin', c1),
('fend', c2),
('query', 'config=gles&config=vk&source_type=gm'),
('pos', 'true'),
('neg', 'false'),
('unt', 'false')
]))
j = json.load(f)
f.close()
return j
def gset(path):
s = set()
if os.path.isfile(path):
with open(path, 'r') as f:
for line in f:
s.add(line.strip())
return s
def make_rendertest_list(models, good, bad):
assert good.isdisjoint(bad)
do_score = good & models
no_score = bad | (good - models)
to_delete = models & bad
for d in to_delete:
path = ASSETS + '/gmkb/' + d
if os.path.isdir(path):
shutil.rmtree(path)
results = dict()
for n in do_score:
results[n] = 0
for n in no_score:
results[n] = -1
return ''.join('%s,%d\n' % (n, results[n]) for n in sorted(results))
def get_digest(path):
m = md5.new()
with open(path, 'r') as f:
m.update(f.read())
return m.hexdigest()
def upload_cmd(path, digest):
return ['gsutil', 'cp', path, 'gs://%s/%s' % (BUCKET, digest)]
def upload_model():
bucket_url = 'gs://%s/' % BUCKET
extant = set((u.replace(bucket_url, '', 1)
for u in check_output(['gsutil', 'ls', bucket_url]).splitlines() if u))
cmds = []
filelist = []
for dirpath, _, filenames in os.walk(ASSETS + '/gmkb'):
for filename in filenames:
path = os.path.join(dirpath, filename)
digest = get_digest(path)
if digest not in extant:
cmds.append(upload_cmd(path, digest))
filelist.append('%s;%s\n' % (digest, os.path.relpath(path, ASSETS)))
tmp = tempfile.mkdtemp()
filelist_path = tmp + '/x'
with open(filelist_path, 'w') as o:
for l in filelist:
o.write(l)
filelist_digest = get_digest(filelist_path)
if filelist_digest not in extant:
cmds.append(upload_cmd(filelist_path, filelist_digest))
pool = multiprocessing.Pool(processes=20)
pool.map(check_call, cmds)
shutil.rmtree(tmp)
return filelist_digest
def remove(x):
if os.path.isdir(x) and not os.path.islink(x):
shutil.rmtree(x)
if os.path.exists(x):
os.remove(x)
def main(first_commit, last_commit):
check_call(upload_cmd('/dev/null', get_digest('/dev/null')))
os.chdir(os.path.dirname(__file__) + '/../..')
remove(ASSETS + '/files.checksum')
for d in [ASSETS + '/gmkb', ASSETS + '/skqp', ]:
remove(d)
os.mkdir(d)
check_call([sys.executable, 'tools/git-sync-deps'],
env=dict(os.environ, GIT_SYNC_DEPS_QUIET='T'))
build = 'out/ndebug'
check_call(['bin/gn', 'gen', build,
'--args=cc="clang" cxx="clang++" is_debug=false'])
check_call(['ninja', '-C', build,
'jitter_gms', 'list_gpu_unit_tests', 'make_skqp_model'])
models = goldgetter(gold(first_commit, last_commit), build + '/make_skqp_model')
check_call([build + '/jitter_gms', 'tools/skqp/bad_gms.txt'])
with open(ASSETS + '/skqp/rendertests.txt', 'w') as o:
o.write(make_rendertest_list(models, gset('good.txt'), gset('bad.txt')))
remove('good.txt')
remove('bad.txt')
with open(ASSETS + '/skqp/unittests.txt', 'w') as o:
o.write(check_output([build + '/list_gpu_unit_tests']))
with open(ASSETS + '/files.checksum', 'w') as o:
o.write(upload_model() + '\n')
sys.stdout.write(ASSETS + '/files.checksum\n')
sys.stdout.write(ASSETS + '/skqp/rendertests.txt\n')
sys.stdout.write(ASSETS + '/skqp/unittests.txt\n')
if __name__ == '__main__':
if len(sys.argv) != 3:
sys.stderr.write('Usage:\n %s C1 C2\n\n' % sys.argv[0])
sys.exit(1)
main(sys.argv[1], sys.argv[2])