#!/usr/bin/env python # Copyright 2016 The Chromium OS Authors. All rights reserved. # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file. import argparse import contextlib import logging import os import re import shutil import stat import subprocess import tempfile import zipfile _CONTROLFILE_TEMPLATE = """\ # Copyright 2016 The Chromium OS Authors. All rights reserved. # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file. # This file has been automatically generated. Do not edit! AUTHOR = 'ARC Team' NAME = '{name}' ATTRIBUTES = '{attributes}' DEPENDENCIES = '{dependencies}' JOB_RETRIES = {retries} TEST_TYPE = 'server' TIME = 'LENGTHY' DOC = ('Run package {package} of the ' 'Android {revision} Compatibility Test Suite (CTS), build {build},' 'using {abi} ABI in the ARC container.') def run_CTS(machine): host = hosts.create_host(machine) job.run_test( 'cheets_CTS', host=host, iterations=1,{retry} tag='{tag}', target_package='{package}', {source}, timeout={timeout}) parallel_simple(run_CTS, machines)""" _CTS_TIMEOUT = { 'android.core.tests.libcore.package.org': 90, 'android.core.vm-tests-tf': 90, 'android.host.security': 90, 'android.media': 360, 'android.mediastress': 480, 'android.print': 180, 'com.android.cts.filesystemperf': 180, 'com.drawelements.deqp.gles3': 240, 'com.drawelements.deqp.gles31': 90, 'com.drawelements.deqp.gles31.copy_image_mixed': 120, 'com.drawelements.deqp.gles31.copy_image_non_compressed': 240, } _SUITE_SMOKE = [ 'android.core.tests.libcore.package.harmony_java_math' ] # Any test in SMOKE (VMTest) should also be in CQ (HWTest). _SUITE_BVT_CQ = _SUITE_SMOKE + [ 'com.android.cts.dram' ] _SUITE_ARC_BVT_CQ = _SUITE_BVT_CQ _SUITE_BVT_PERBUILD = [ 'android.signature', 'android.speech', 'android.systemui', 'android.telecom', 'android.telephony', 'android.theme', 'android.transition', 'android.tv', 'android.uiautomation', 'android.usb', 'android.voicesettings', 'com.android.cts.filesystemperf', 'com.android.cts.jank', 'com.android.cts.opengl', 'com.android.cts.simplecpu', ] def get_tradefed_build(line): """Gets the build of Android CTS from tradefed. @param line Tradefed identification output on startup. Example: Android CTS 6.0_r6 build:2813453 @return Tradefed CTS build. Example: 2813453. """ # Sample string: Android CTS 6.0_r6 build:2813453. m = re.search(r'(?<=build:)(.*)', line) if m: return m.group(0) logging.warning('Could not identify build in line "%s".', line) return '<unknown>' def get_tradefed_revision(line): """Gets the revision of Android CTS from tradefed. @param line Tradefed identification output on startup. Example: Android CTS 6.0_r6 build:2813453 @return Tradefed CTS revision. Example: 6.0_r6. """ m = re.search(r'(?<=Android CTS )(.*) build:', line) if m: return m.group(1) logging.warning('Could not identify revision in line "%s".', line) return None def get_bundle_abi(filename): """Makes an educated guess about the ABI. In this case we chose to guess by filename, but we could also parse the xml files in the package. (Maybe this needs to be done in the future.) """ if filename.endswith('_x86-arm.zip'): return 'arm' if filename.endswith('_x86-x86.zip'): return 'x86' raise Exception('Could not determine ABI from "%s".' % filename) def get_bundle_revision(filename): """Makes an educated guess about the revision. In this case we chose to guess by filename, but we could also parse the xml files in the package. """ m = re.search(r'(?<=android-cts-)(.*)-linux', filename) if m is not None: return m.group(1) return None def get_extension(package, abi, revision): """Defines a unique string. Notice we chose package revision first, then abi, as the package revision changes at least on a monthly basis. This ordering makes it simpler to add/remove packages. """ return '%s.%s.%s' % (revision, abi, package) def get_controlfile_name(package, abi, revision): """Defines the control file name.""" return 'control.%s' % get_extension(package, abi, revision) def get_public_extension(package, abi): return '%s.%s' % (abi, package) def get_public_controlfile_name(package, abi): """Defines the public control file name.""" return 'control.%s' % get_public_extension(package, abi) def get_attribute_suites(package, abi): """Defines the suites associated with a package.""" attributes = 'suite:arc-cts' # Get a minmum amount of coverage on cq (one quick CTS package only). if package in _SUITE_SMOKE: attributes += ', suite:smoke' if package in _SUITE_BVT_CQ: attributes += ', suite:bvt-cq' if package in _SUITE_ARC_BVT_CQ: attributes += ', suite:arc-bvt-cq' if package in _SUITE_BVT_PERBUILD and abi == 'arm': attributes += ', suite:bvt-perbuild' # Adding arc-cts-stable runs all packages twice on stable. return attributes def get_dependencies(abi): """Defines lab dependencies needed to schedule a package. Currently we only care about x86 ABI tests, which must run on Intel boards. """ # TODO(ihf): Enable this once crbug.com/618509 has labeled all lab DUTs. if abi == 'x86': return 'arc, cts_abi_x86' return 'arc' # suite_scheduler retries _RETRIES = 2 def get_controlfile_content(package, abi, revision, build, uri): """Returns the text inside of a control file.""" # For test_that the NAME should be the same as for the control file name. # We could try some trickery here to get shorter extensions for a default # suite/ARM. But with the monthly uprevs this will quickly get confusing. name = 'cheets_CTS.%s' % get_extension(package, abi, revision) if is_public(uri): name = 'cheets_CTS.%s' % get_public_extension(package, abi) dependencies = get_dependencies(abi) retries = _RETRIES # We put all revisions and all abi in the same tag for wmatrix, otherwise # the list at https://wmatrix.googleplex.com/tests would grow way too large. # Example: for all values of |REVISION| and |ABI| we map control file # cheets_CTS.REVISION.ABI.PACKAGE on wmatrix to cheets_CTS.PACKAGE. # This way people can find results independent of |REVISION| and |ABI| at # the same URL. tag = package timeout = 3600 if package in _CTS_TIMEOUT: timeout = 60 * _CTS_TIMEOUT[package] if is_public(uri): source = "bundle='%s'" % abi attributes = 'suite:cts' else: source = "uri='%s'" % uri attributes = '%s, test_name:%s' % (get_attribute_suites(package, abi), name) # cheets_CTS internal retries limited due to time constraints on cq. retry = '' if (package in (_SUITE_SMOKE + _SUITE_BVT_CQ + _SUITE_ARC_BVT_CQ) or package in _SUITE_BVT_PERBUILD and abi == 'arm'): retry = '\n max_retry=3,' return _CONTROLFILE_TEMPLATE.format( name=name, attributes=attributes, dependencies=dependencies, retries=retries, package=package, revision=revision, build=build, abi=abi, tag=tag, source=source, retry=retry, timeout=timeout) def get_tradefed_data(path): """Queries tradefed to provide us with a list of packages.""" tradefed = os.path.join(path, 'android-cts/tools/cts-tradefed') # Forgive me for I have sinned. Same as: chmod +x tradefed. os.chmod(tradefed, os.stat(tradefed).st_mode | stat.S_IEXEC) cmd_list = [tradefed, 'list', 'packages'] logging.info('Calling tradefed for list of packages.') # TODO(ihf): Get a tradefed command which terminates then refactor. p = subprocess.Popen(cmd_list, stdout=subprocess.PIPE) packages = [] build = '<unknown>' line = '' revision = None # The process does not terminate, but we know the last test starts with zzz. while not line.startswith('zzz'): line = p.stdout.readline().strip() if line: print line if line.startswith('Android CTS '): logging.info('Unpacking: %s.', line) build = get_tradefed_build(line) revision = get_tradefed_revision(line) elif re.search(r'^(android\.|com\.|zzz\.)', line): packages.append(line) p.kill() p.wait() return packages, build, revision def is_public(uri): if uri.startswith('gs:'): return False elif uri.startswith('https://dl.google.com'): return True raise def download(uri, destination): """Download |uri| to local |destination|.""" if is_public(uri): subprocess.check_call(['wget', uri, '-P', destination]) else: subprocess.check_call(['gsutil', 'cp', uri, destination]) @contextlib.contextmanager def pushd(d): """Defines pushd.""" current = os.getcwd() os.chdir(d) try: yield finally: os.chdir(current) def unzip(filename, destination): """Unzips a zip file to the destination directory.""" with pushd(destination): # We are trusting Android to have a sane zip file for us. with zipfile.ZipFile(filename) as zf: zf.extractall() @contextlib.contextmanager def TemporaryDirectory(prefix): """Poor man's python 3.2 import.""" tmp = tempfile.mkdtemp(prefix=prefix) try: yield tmp finally: shutil.rmtree(tmp) def main(uris): """Downloads each package in |uris| and generates control files.""" for uri in uris: abi = get_bundle_abi(uri) with TemporaryDirectory(prefix='cts-android_') as tmp: logging.info('Downloading to %s.', tmp) bundle = os.path.join(tmp, 'cts-android.zip') download(uri, tmp) bundle = os.path.join(tmp, os.path.basename(uri)) logging.info('Extracting %s.', bundle) unzip(bundle, tmp) packages, build, revision = get_tradefed_data(tmp) if not revision: raise Exception('Could not determine revision.') # And write all control files. logging.info('Writing all control files.') for package in packages: name = get_controlfile_name(package, abi, revision) if is_public(uri): name = get_public_controlfile_name(package, abi) content = get_controlfile_content(package, abi, revision, build, uri) with open(name, 'w') as f: f.write(content) if __name__ == '__main__': logging.basicConfig(level=logging.INFO) parser = argparse.ArgumentParser( description='Create control files for a CTS bundle on GS.', formatter_class=argparse.RawTextHelpFormatter) parser.add_argument( 'uris', nargs='+', help='List of Google Storage URIs to CTS bundles. Example:\n' 'gs://chromeos-arc-images/cts/bundle/2016-06-02/' 'android-cts-6.0_r6-linux_x86-arm.zip') args = parser.parse_args() main(args.uris)