# Copyright (c) 2014 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 logging import os import re import shutil import tempfile import xml.etree.ElementTree as ET import common from autotest_lib.client.bin import test, utils from autotest_lib.client.common_lib import error from autotest_lib.client.common_lib import file_utils from autotest_lib.client.cros import constants class ChromeBinaryTest(test.test): """ Base class for tests to run chrome test binaries without signing in and running Chrome. """ CHROME_TEST_DEP = 'chrome_test' CHROME_SANDBOX = '/opt/google/chrome/chrome-sandbox' COMPONENT_LIB = '/opt/google/chrome/lib' home_dir = None cr_source_dir = None test_binary_dir = None def setup(self): """ Sets up a test. """ self.job.setup_dep([self.CHROME_TEST_DEP]) def initialize(self): """ Initializes members after setup(). """ test_dep_dir = os.path.join(self.autodir, 'deps', self.CHROME_TEST_DEP) self.job.install_pkg(self.CHROME_TEST_DEP, 'dep', test_dep_dir) self.cr_source_dir = '%s/test_src' % test_dep_dir self.test_binary_dir = '%s/out/Release' % self.cr_source_dir # If chrome is a component build then need to create a symlink such # that the _unittest binaries can find the chrome component libraries. Release_lib = os.path.join(self.test_binary_dir, 'lib') if os.path.isdir(self.COMPONENT_LIB): logging.info('Detected component build. This assumes binary ' 'compatibility between chrome and *unittest.') if not os.path.islink(Release_lib): os.symlink(self.COMPONENT_LIB, Release_lib) self.home_dir = tempfile.mkdtemp() def cleanup(self): """ Cleans up working directory after run. """ if self.home_dir: shutil.rmtree(self.home_dir, ignore_errors=True) def get_chrome_binary_path(self, binary_to_run): """ Gets test binary's full path. @returns full path of the test binary to run. """ return os.path.join(self.test_binary_dir, binary_to_run) def parse_fail_reason(self, err, gtest_xml): """ Parses reason of failure from CmdError and gtest result. @param err: CmdError raised from utils.system(). @param gtest_xml: filename of gtest result xml. @returns reason string """ reasons = {} # Parse gtest result. if os.path.exists(gtest_xml): tree = ET.parse(gtest_xml) root = tree.getroot() for suite in root.findall('testsuite'): for case in suite.findall('testcase'): failure = case.find('failure') if failure is None: continue testname = '%s.%s' % (suite.get('name'), case.get('name')) reasons[testname] = failure.attrib['message'] # Parse messages from chrome's test_launcher. # This provides some information not available from gtest, like timeout. for line in err.result_obj.stdout.splitlines(): m = re.match(r'\[\d+/\d+\] (\S+) \(([A-Z ]+)\)$', line) if not m: continue testname, reason = m.group(1, 2) # Existing reason from gtest has more detail, don't overwrite. if testname not in reasons: reasons[testname] = reason if reasons: message = '%d failures' % len(reasons) for testname, reason in sorted(reasons.items()): message += '; <%s>: %s' % (testname, reason.replace('\n', '; ')) return message return 'Unable to parse fail reason: ' + str(err) def run_chrome_test_binary(self, binary_to_run, extra_params='', prefix='', as_chronos=True, timeout=None): """ Runs chrome test binary. @param binary_to_run: The name of the browser test binary. @param extra_params: Arguments for the browser test binary. @param prefix: Prefix to the command that invokes the test binary. @param as_chronos: Boolean indicating if the tests should run in a chronos shell. @param timeout: timeout in seconds @raises: error.TestFail if there is error running the command. @raises: CmdTimeoutError: the command timed out and |timeout| is specified and not None. """ gtest_xml = tempfile.mktemp(prefix='gtest_xml', suffix='.xml') binary_path = self.get_chrome_binary_path(binary_to_run) env_vars = ' '.join([ 'HOME=' + self.home_dir, 'CR_SOURCE_ROOT=' + self.cr_source_dir, 'CHROME_DEVEL_SANDBOX=' + self.CHROME_SANDBOX, 'GTEST_OUTPUT=xml:' + gtest_xml, ]) cmd = ' '.join([env_vars, prefix, binary_path, extra_params]) try: if as_chronos: utils.system("su chronos -c '%s'" % cmd, timeout=timeout) else: utils.system(cmd, timeout=timeout) except error.CmdError as e: return_code = e.result_obj.exit_status if return_code == 126: path_permission = '; '.join( file_utils.recursive_path_permission(binary_path)) fail_reason = ('Cannot execute command %s. Permissions: %s' % (binary_path, path_permission)) elif return_code == 127: fail_reason = ('Command not found: %s' % binary_path) else: fail_reason = self.parse_fail_reason(e, gtest_xml) raise error.TestFail(fail_reason) def nuke_chrome(func): """ Decorator to nuke the Chrome browser processes. """ def wrapper(*args, **kargs): """ Nukes Chrome browser processes before invoking func(). Also, restarts Chrome after func() returns. """ open(constants.DISABLE_BROWSER_RESTART_MAGIC_FILE, 'w').close() try: try: utils.nuke_process_by_name(name=constants.BROWSER, with_prejudice=True) except error.AutoservPidAlreadyDeadError: pass return func(*args, **kargs) finally: # Allow chrome to be restarted again later. os.unlink(constants.DISABLE_BROWSER_RESTART_MAGIC_FILE) return wrapper