188 lines
6.7 KiB
Python
Executable File
188 lines
6.7 KiB
Python
Executable File
# 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
|