mirror of
https://github.com/servo/servo.git
synced 2025-06-06 16:45:39 +00:00
Requires servo/servo#37045 for deps and config. Testing: No need for tests to test tests. Fixes: servo/servo#37041 --------- Signed-off-by: zefr0x <zer0-x.7ty50@aleeas.com>
230 lines
7.5 KiB
Python
230 lines
7.5 KiB
Python
# This Source Code Form is subject to the terms of the Mozilla Public
|
|
# License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
# file, You can obtain one at https://mozilla.org/MPL/2.0/.
|
|
|
|
import hashlib
|
|
import os
|
|
import subprocess
|
|
import sys
|
|
import runpy
|
|
|
|
SCRIPT_PATH = os.path.abspath(os.path.dirname(__file__))
|
|
TOP_DIR = os.path.abspath(os.path.join(SCRIPT_PATH, ".."))
|
|
WPT_PATH = os.path.join(TOP_DIR, "tests", "wpt")
|
|
WPT_TOOLS_PATH = os.path.join(WPT_PATH, "tests", "tools")
|
|
WPT_RUNNER_PATH = os.path.join(WPT_TOOLS_PATH, "wptrunner")
|
|
WPT_SERVE_PATH = os.path.join(WPT_TOOLS_PATH, "wptserve")
|
|
|
|
SEARCH_PATHS = [
|
|
os.path.join("python", "mach"),
|
|
os.path.join("third_party", "mozdebug"),
|
|
]
|
|
|
|
# Individual files providing mach commands.
|
|
MACH_MODULES = [
|
|
os.path.join("python", "servo", "bootstrap_commands.py"),
|
|
os.path.join("python", "servo", "build_commands.py"),
|
|
os.path.join("python", "servo", "testing_commands.py"),
|
|
os.path.join("python", "servo", "post_build_commands.py"),
|
|
os.path.join("python", "servo", "package_commands.py"),
|
|
os.path.join("python", "servo", "devenv_commands.py"),
|
|
]
|
|
|
|
CATEGORIES = {
|
|
"bootstrap": {
|
|
"short": "Bootstrap Commands",
|
|
"long": "Bootstrap the build system",
|
|
"priority": 90,
|
|
},
|
|
"build": {
|
|
"short": "Build Commands",
|
|
"long": "Interact with the build system",
|
|
"priority": 80,
|
|
},
|
|
"post-build": {
|
|
"short": "Post-build Commands",
|
|
"long": "Common actions performed after completing a build.",
|
|
"priority": 70,
|
|
},
|
|
"testing": {
|
|
"short": "Testing",
|
|
"long": "Run tests.",
|
|
"priority": 60,
|
|
},
|
|
"devenv": {
|
|
"short": "Development Environment",
|
|
"long": "Set up and configure your development environment.",
|
|
"priority": 50,
|
|
},
|
|
"build-dev": {
|
|
"short": "Low-level Build System Interaction",
|
|
"long": "Interact with specific parts of the build system.",
|
|
"priority": 20,
|
|
},
|
|
"package": {
|
|
"short": "Package",
|
|
"long": "Create objects to distribute",
|
|
"priority": 15,
|
|
},
|
|
"misc": {
|
|
"short": "Potpourri",
|
|
"long": "Potent potables and assorted snacks.",
|
|
"priority": 10,
|
|
},
|
|
"disabled": {
|
|
"short": "Disabled",
|
|
"long": "The disabled commands are hidden by default. Use -v to display them. These commands are unavailable "
|
|
'for your current context, run "mach <command>" to see why.',
|
|
"priority": 0,
|
|
},
|
|
}
|
|
|
|
|
|
def _process_exec(args, cwd):
|
|
try:
|
|
subprocess.check_output(args, stderr=subprocess.STDOUT, cwd=cwd)
|
|
except subprocess.CalledProcessError as exception:
|
|
print(exception.output.decode(sys.stdout.encoding))
|
|
print(f"Process failed with return code: {exception.returncode}")
|
|
sys.exit(1)
|
|
|
|
|
|
def install_virtual_env_requirements(project_path: str, marker_path: str):
|
|
requirements_paths = [
|
|
os.path.join(project_path, "python", "requirements.txt"),
|
|
os.path.join(
|
|
project_path,
|
|
WPT_TOOLS_PATH,
|
|
"requirements_tests.txt",
|
|
),
|
|
os.path.join(
|
|
project_path,
|
|
WPT_RUNNER_PATH,
|
|
"requirements.txt",
|
|
),
|
|
]
|
|
|
|
requirements_hasher = hashlib.sha256()
|
|
for path in requirements_paths:
|
|
with open(path, "rb") as file:
|
|
requirements_hasher.update(file.read())
|
|
|
|
try:
|
|
with open(marker_path, "r") as marker_file:
|
|
marker_hash = marker_file.read()
|
|
except FileNotFoundError:
|
|
marker_hash = None
|
|
|
|
requirements_hash = requirements_hasher.hexdigest()
|
|
|
|
if marker_hash != requirements_hash:
|
|
print(" * Installing Python requirements...")
|
|
pip_install_command = ["uv", "pip", "install"]
|
|
for requirements in requirements_paths:
|
|
pip_install_command.extend(["-r", requirements])
|
|
_process_exec(pip_install_command, cwd=project_path)
|
|
with open(marker_path, "w") as marker_file:
|
|
marker_file.write(requirements_hash)
|
|
|
|
|
|
def _activate_virtualenv(topdir):
|
|
virtualenv_path = os.path.join(topdir, ".venv")
|
|
|
|
with open(".python-version", "r") as python_version_file:
|
|
required_python_version = python_version_file.read().strip()
|
|
marker_path = os.path.join(virtualenv_path, f"requirements.{required_python_version}.sha256")
|
|
|
|
if os.environ.get("VIRTUAL_ENV") != virtualenv_path:
|
|
if not os.path.exists(marker_path):
|
|
print(" * Setting up virtual environment...")
|
|
_process_exec(["uv", "venv"], cwd=topdir)
|
|
|
|
script_dir = "Scripts" if _is_windows() else "bin"
|
|
runpy.run_path(os.path.join(virtualenv_path, script_dir, "activate_this.py"))
|
|
|
|
install_virtual_env_requirements(topdir, marker_path)
|
|
|
|
# Turn off warnings about deprecated syntax in our indirect dependencies.
|
|
# TODO: Find a better approach for doing this.
|
|
import warnings
|
|
|
|
warnings.filterwarnings("ignore", category=SyntaxWarning, module=r".*.venv")
|
|
|
|
|
|
def _ensure_case_insensitive_if_windows():
|
|
# The folder is called 'python'. By deliberately checking for it with the wrong case, we determine if the file
|
|
# system is case sensitive or not.
|
|
if _is_windows() and not os.path.exists("Python"):
|
|
print("Cannot run mach in a path on a case-sensitive file system on Windows.")
|
|
print("For more details, see https://github.com/pypa/virtualenv/issues/935")
|
|
sys.exit(1)
|
|
|
|
|
|
def _is_windows():
|
|
return sys.platform == "win32"
|
|
|
|
|
|
def bootstrap_command_only(topdir):
|
|
# we should activate the venv before importing servo.boostrap
|
|
# because the module requires non-standard python packages
|
|
_activate_virtualenv(topdir)
|
|
|
|
# We cannot import these modules until the virtual environment
|
|
# is active because they depend on modules installed via the
|
|
# virtual environment.
|
|
# pylint: disable=import-outside-toplevel
|
|
import servo.platform
|
|
import servo.util
|
|
|
|
try:
|
|
force = "-f" in sys.argv or "--force" in sys.argv
|
|
skip_platform = "--skip-platform" in sys.argv
|
|
skip_lints = "--skip-lints" in sys.argv
|
|
servo.platform.get().bootstrap(force, skip_platform, skip_lints)
|
|
except NotImplementedError as exception:
|
|
print(exception)
|
|
return 1
|
|
|
|
return 0
|
|
|
|
|
|
def bootstrap(topdir):
|
|
_ensure_case_insensitive_if_windows()
|
|
|
|
topdir = os.path.abspath(topdir)
|
|
|
|
# We don't support paths with spaces for now
|
|
# https://github.com/servo/servo/issues/9616
|
|
if " " in topdir and (not _is_windows()):
|
|
print("Cannot run mach in a path with spaces.")
|
|
print("Current path:", topdir)
|
|
sys.exit(1)
|
|
|
|
_activate_virtualenv(topdir)
|
|
|
|
def populate_context(context, key=None):
|
|
if key is None:
|
|
return
|
|
if key == "topdir":
|
|
return topdir
|
|
raise AttributeError(key)
|
|
|
|
sys.path[0:0] = [os.path.join(topdir, path) for path in SEARCH_PATHS]
|
|
sys.path[0:0] = [WPT_PATH, WPT_RUNNER_PATH, WPT_SERVE_PATH]
|
|
|
|
import mach.main
|
|
|
|
mach = mach.main.Mach(os.getcwd())
|
|
mach.populate_context_handler = populate_context
|
|
|
|
for category, meta in CATEGORIES.items():
|
|
mach.define_category(category, meta["short"], meta["long"], meta["priority"])
|
|
|
|
for path in MACH_MODULES:
|
|
# explicitly provide a module name
|
|
# workaround for https://bugzilla.mozilla.org/show_bug.cgi?id=1549636
|
|
file = os.path.basename(path)
|
|
module_name = os.path.splitext(file)[0]
|
|
mach.load_commands_from_file(os.path.join(topdir, path), module_name)
|
|
|
|
return mach
|