# Copyright (c) 2012 Google Inc. All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are
# met:
#
# * Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
# * Redistributions in binary form must reproduce the above
# copyright notice, this list of conditions and the following disclaimer
# in the documentation and/or other materials provided with the
# distribution.
# * Neither the name of Google Inc. nor the names of its
# contributors may be used to endorse or promote products derived from
# this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
import os
import posixpath
import sys
from blinkpy.common.memoized import memoized
def add_typ_dir_to_sys_path():
path_to_typ = get_typ_dir()
if path_to_typ not in sys.path:
# `//third_party/catapult/third_party/typ/` has a `tools/` module that
# conflicts with `wpt/tools/`. Always place `typ/` last in the path to
# ensure WPT always takes precedence.
sys.path.append(path_to_typ)
def add_bindings_scripts_dir_to_sys_path():
path_to_bindings_scripts = get_bindings_scripts_dir()
if path_to_bindings_scripts not in sys.path:
sys.path.insert(0, path_to_bindings_scripts)
def add_build_scripts_dir_to_sys_path():
path_to_build_scripts = get_build_scripts_dir()
if path_to_build_scripts not in sys.path:
sys.path.insert(0, path_to_build_scripts)
def add_blinkpy_thirdparty_dir_to_sys_path():
path = get_blinkpy_thirdparty_dir()
if path not in sys.path:
sys.path.insert(0, path)
def add_testing_dir_to_sys_path():
path = get_testing_dir()
if path not in sys.path:
sys.path.insert(0, path)
def add_build_android_to_sys_path():
path = get_build_android_dir()
if path not in sys.path:
sys.path.insert(0, path)
def add_build_ios_to_sys_path():
path = get_build_ios_dir()
if path not in sys.path:
sys.path.insert(0, path)
def bootstrap_wpt_imports():
"""Bootstrap the availability of all wpt-vended packages."""
path = get_wpt_tools_wpt_dir()
# Do not add the Chromium-vended version of WPT to the path if there's
# already a WPT root there.
#
# This WPT detection is admittedly crude, but it's meant to detect
# `/tmp/wpt` created by `LocalWPT`.
if path not in sys.path and not any(
os.path.basename(path).lower() == 'wpt' for path in sys.path):
sys.path.insert(0, path)
# This module is under `//third_party/wpt_tools/wpt/tools`, and has the side
# effect of inserting wpt-related directories into `sys.path`.
from tools import localpaths # pylint: disable=unused-import
def add_depot_tools_dir_to_os_path():
path = get_depot_tools_dir()
if path not in os.environ['PATH']:
os.environ['PATH'] += os.pathsep + path
def get_bindings_scripts_dir():
return os.path.join(get_source_dir(), 'bindings', 'scripts')
def get_blink_dir():
return os.path.dirname(
os.path.dirname(
os.path.dirname(os.path.dirname(os.path.realpath(__file__)))))
def get_chromium_src_dir():
return os.path.dirname(os.path.dirname(get_blink_dir()))
def get_depot_tools_dir():
return os.path.join(get_chromium_src_dir(), 'third_party', 'depot_tools')
def get_source_dir():
return os.path.join(get_chromium_src_dir(), 'third_party', 'blink',
'renderer')
def get_testing_dir():
return os.path.join(get_chromium_src_dir(), 'testing')
def get_build_android_dir():
return os.path.join(get_chromium_src_dir(), 'build', 'android')
def get_build_ios_dir():
return os.path.join(get_chromium_src_dir(), 'ios', 'build', 'bots',
'scripts')
def get_typ_dir():
return os.path.join(get_chromium_src_dir(), 'third_party', 'catapult',
'third_party', 'typ')
def get_blinkpy_thirdparty_dir():
return os.path.join(get_blink_tools_dir(), 'blinkpy', 'third_party')
def get_blink_tools_dir():
return os.path.join(get_chromium_src_dir(), 'third_party', 'blink',
'tools')
def get_wpt_tools_wpt_dir():
return os.path.join(get_chromium_src_dir(), 'third_party', 'wpt_tools',
'wpt')
def get_build_scripts_dir():
return os.path.join(get_source_dir(), 'build', 'scripts')
def add_blink_tools_dir_to_sys_path():
path = get_blink_tools_dir()
if path not in sys.path:
sys.path.insert(0, path)
# web_tests path relative to the repository root.
# Path separators are always '/', and this contains the trailing '/'.
RELATIVE_WEB_TESTS = 'third_party/blink/web_tests/'
RELATIVE_WPT_TESTS = 'third_party/blink/web_tests/external/wpt/'
WEB_TESTS_LAST_COMPONENT = 'web_tests'
class PathFinder(object):
def __init__(self, filesystem, sys_path=None, env_path=None):
self._filesystem = filesystem
self._dirsep = filesystem.sep
self._sys_path = sys_path or sys.path
self._env_path = env_path or os.environ['PATH'].split(os.pathsep)
@memoized
def chromium_base(self):
return self._filesystem.dirname(
self._filesystem.dirname(self._blink_base()))
def web_tests_dir(self):
return self.path_from_chromium_base('third_party', 'blink',
'web_tests')
def wpt_tests_dir(self):
return self.path_from_chromium_base('third_party', 'blink',
'web_tests', 'external', 'wpt')
def perf_tests_dir(self):
return self.path_from_chromium_base('third_party', 'blink',
'perf_tests')
def wpt_prefix(self):
# Always use '/' instead of the platform dependent separator.
# This should be only used with a test id.
return posixpath.join('external', 'wpt', '')
@memoized
def _blink_base(self):
"""Returns the absolute path to the top of the Blink directory."""
module_path = self._filesystem.path_to_module(self.__module__)
tools_index = module_path.rfind('tools')
assert tools_index != -1, 'could not find location of this checkout from %s' % module_path
return self._filesystem.normpath(module_path[0:tools_index - 1])
def path_from_chromium_base(self, *comps):
return self._filesystem.join(self.chromium_base(), *comps)
def _blink_source_dir(self):
return self._filesystem.join(self.chromium_base(), 'third_party',
'blink', 'renderer')
def path_from_blink_source(self, *comps):
return self._filesystem.join(self._blink_source_dir(), *comps)
def path_from_blink_tools(self, *comps):
return self._filesystem.join(
self._filesystem.join(self.chromium_base(), 'third_party', 'blink',
'tools'), *comps)
def path_from_web_tests(self, *comps):
return self._filesystem.join(self.web_tests_dir(), *comps)
def path_from_wpt_tests(self, *comps):
return self._filesystem.join(self.wpt_tests_dir(), *comps)
def strip_web_tests_path(self, web_test_abs_path):
web_tests_path = self.path_from_web_tests('')
if web_test_abs_path.startswith(web_tests_path):
return web_test_abs_path[len(web_tests_path):]
return web_test_abs_path
def strip_wpt_path(self, wpt_path):
"""Remove the prefix before all WPT paths.
ResultDB identifies WPTs as web tests with the prefix "external/wpt",
but wptrunner expects paths relative to the WPT root, which is already
"<web-tests-dir>/external/wpt". This function removes the redundant
path fragment.
"""
if self.is_wpt_path(wpt_path):
return wpt_path[len(self.wpt_prefix()):]
# Path is absolute or does not start with the prefix.
# Assume the path already points to a valid WPT and pass through.
return wpt_path
def is_wpt_path(self, test_path):
return test_path.startswith(self.wpt_prefix())
@memoized
def depot_tools_base(self):
"""Returns the path to depot_tools, or None if not found.
Expects depot_tools to be //third_party/depot_tools.
src.git's DEPS defines depot_tools to be there.
"""
depot_tools = self.path_from_chromium_base('third_party',
'depot_tools')
return depot_tools if self._filesystem.isdir(depot_tools) else None
def path_from_depot_tools_base(self, *comps):
return self._filesystem.join(self.depot_tools_base(), *comps)