SSL bootstrap.py
Interaktion und PortierbarkeitPython
# 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 http://mozilla.org/MPL/2.0/.
import os import platform import re import shutil import stat import subprocess import sys import time from collections import OrderedDict from pathlib import Path from typing import Optional
# Use distro package to retrieve linux platform information import distro from mach.site import MachSiteManager from mach.telemetry import initialize_telemetry_setting from mach.util import (
UserError,
get_state_dir,
to_optional_path,
to_optional_str,
win_to_msys_path,
) from mozbuild.base import MozbuildObject from mozfile import which from packaging.version import Version
from mozboot.archlinux import ArchlinuxBootstrapper from mozboot.base import MODERN_RUST_VERSION from mozboot.centosfedora import CentOSFedoraBootstrapper from mozboot.debian import DebianBootstrapper from mozboot.freebsd import FreeBSDBootstrapper from mozboot.gentoo import GentooBootstrapper from mozboot.mozconfig import MozconfigBuilder from mozboot.mozillabuild import MozillaBuildBootstrapper from mozboot.openbsd import OpenBSDBootstrapper from mozboot.opensuse import OpenSUSEBootstrapper from mozboot.osx import OSXBootstrapper, OSXBootstrapperLight from mozboot.solus import SolusBootstrapper from mozboot.void import VoidBootstrapper from mozboot.windows import WindowsBootstrapper
APPLICATION_CHOICE = """
Note on Artifact Mode:
Artifact builds download prebuilt C++ components rather than building
them locally. Artifact builds are faster!
Please choose the version of Firefox you want to build (see note above):
%s
Your choice: """
APPLICATIONS = OrderedDict(
[
("Firefox for Desktop Artifact Mode", "browser_artifact_mode"),
("Firefox for Desktop", "browser"),
("GeckoView/Firefox for Android Artifact Mode", "mobile_android_artifact_mode"),
("GeckoView/Firefox for Android", "mobile_android"),
("SpiderMonkey JavaScript engine", "js"),
]
)
FINISHED = """
Your system should be ready to build %s! """
MOZCONFIG_SUGGESTION_TEMPLATE = """
Paste the lines between the chevrons (>>> and <<<) into
%s:
>>>
%s
<<< """
MOZCONFIG_MISMATCH_WARNING_TEMPLATE = """
WARNING! Mismatch detected between the selected build target and the
mozconfig file %s:
Current config
>>>
%s
<<<
Expected config
>>>
%s
<<< """
CONFIGURE_MERCURIAL = """
Mozilla recommends a number of changes to Mercurial to enhance your
experience with it.
Would you like to run a configuration wizard to ensure Mercurial is
optimally configured? (This will also ensure 'version-control-tools'is up-to-date)"""
CONFIGURE_GIT = """
Mozilla recommends using git-cinnabar to work with mozilla-central (or
mozilla-unified).
Would you like to run a few configuration steps to ensure Git is
optimally configured?"""
ADD_GIT_CINNABAR_PATH = """
To add git-cinnabar to the PATH, edit your shell initialization script, which
may be called {prefix}/.bash_profile or {prefix}/.profile, and add the following
lines:
export PATH="{cinnabar_dir}:$PATH"
Then restart your shell. """
OLD_REVISION_WARNING = """
WARNING! You appear to be running `mach bootstrap` from an old revision.
bootstrap is meant primarily for getting developer environments up-to-date to
build the latest version of tree. Running bootstrap on old revisions may fail andisnot guaranteed to bring your machine to any working state in particular.
Proceed at your own peril. """
# Version 2.24 changes the "core.commitGraph" setting to be "True" by default.
MINIMUM_RECOMMENDED_GIT_VERSION = Version("2.24")
OLD_GIT_WARNING = """
You are running an older version of git ("{old_version}").
We recommend upgrading to at least version "{minimum_recommended_version}" to improve
performance. """.strip()
# Dev Drives were added in 22621.2338 and should be available in all subsequent versions
DEV_DRIVE_MINIMUM_VERSION = Version("10.0.22621.2338")
DEV_DRIVE_SUGGESTION = """
Mach has detected that the Firefox source repository ({}) is located on an {} drive.
Your current version of Windows ({}) supports ReFS drives (Dev Drive).
It has been shown that Firefox builds are 5-10% faster on
ReFS, it is recommended that you create an ReFS drive and move the Firefox
source repository to it before proceeding.
If you wish disregard this recommendation, you can hide this message by setting 'MACH_HIDE_DEV_DRIVE_SUGGESTION=1'in your environment variables (and restarting your shell)."""
DEV_DRIVE_DETECTION_ERROR = """
Error encountered while checking for Dev Drive.
Reason: {} (skipping) """
if result.returncode:
print("Failed to run 'hg config'. hg configuration checks will be skipped.") return
import json
try:
json_data = json.loads(result.stdout) except json.JSONDecodeError as e:
print(
f"Error parsing 'hg config' JSON: {e}\n\n"
f"hg configuration checks will be skipped."
) return
mismatched_paths = []
pattern = re.compile(r"(.*\.mozbuild)[\\/](.*)") for entry in json_data: ifnot entry["name"].startswith("extensions."): continue
extension_path = entry["value"]
match = pattern.search(extension_path) if match:
extension = entry["name"]
source_path = entry["source"]
state_dir_from_hgrc = Path(match.group(1))
extension_suffix = match.group(2)
if state_dir != state_dir_from_hgrc.expanduser():
expected_extension_path = state_dir / extension_suffix
mismatched_paths.append(
f"Extension: '{extension}' found in config file '{source_path}'\n"
f" Current: {extension_path}\n"
f" Expected: {expected_extension_path}\n"
)
if mismatched_paths:
hgrc_state_dir_mismatch_error_message = (
f"Paths for extensions in your hgrc file appear to be referencing paths that are not in "
f"the current '.mozbuild' state directory.\nYou may have set the `MOZBUILD_STATE_PATH` "
f"environment variable and/or moved the `.mozbuild` directory. You should update the "
f"paths for the following extensions manually to be inside '{state_dir}'\n"
f"(If you instead wish to hide this error, set 'MACH_IGNORE_HGRC_STATE_DIR_MISMATCH=1' "
f"in your environment variables and restart your shell before rerunning mach).\n\n"
f"You can either use the command 'hg config --edit' to make changes to your hg "
f"configuration or manually edit the 'config file' specified for each extension "
f"below:\n\n"
)
hgrc_state_dir_mismatch_error_message += "".join(mismatched_paths)
if sys.platform.startswith("linux"): # distro package provides reliable ids for popular distributions so # we use those instead of the full distribution name
dist_id, version, codename = distro.linux_distribution(
full_distribution_name=False
)
if dist_id in FEDORA_DISTROS:
cls = CentOSFedoraBootstrapper
args["distro"] = dist_id elif dist_id in DEBIAN_DISTROS:
cls = DebianBootstrapper
args["distro"] = dist_id
args["codename"] = codename elif dist_id in ("gentoo", "funtoo"):
cls = GentooBootstrapper elif dist_id in ("solus"):
cls = SolusBootstrapper elif dist_id in ("arch") or Path("/etc/arch-release").exists():
cls = ArchlinuxBootstrapper elif dist_id in ("void"):
cls = VoidBootstrapper elif dist_id in ( "opensuse", "opensuse-leap", "opensuse-tumbleweed", "suse",
):
cls = OpenSUSEBootstrapper else: raise NotImplementedError( "Bootstrap support for this Linux " "distro not yet available: " + dist_id
)
args["version"] = version
args["dist_id"] = dist_id
elif sys.platform.startswith("darwin"): # TODO Support Darwin platforms that aren't OS X.
osx_version = platform.mac_ver()[0] if platform.machine() == "arm64"or _macos_is_running_under_rosetta():
cls = OSXBootstrapperLight else:
cls = OSXBootstrapper
args["version"] = osx_version
elif sys.platform.startswith("win32") or sys.platform.startswith("msys"): if"MOZILLABUILD"in os.environ:
cls = MozillaBuildBootstrapper else:
cls = WindowsBootstrapper if cls isNone: raise NotImplementedError( "Bootstrap support is not yet available ""for your OS."
)
self.instance = cls(**args)
def maybe_install_private_packages_or_exit(self, application, checkout_type): # Install the clang packages needed for building the style system, as # well as the version of NodeJS that we currently support. # Also install the clang static-analysis package by default # The best place to install our packages is in the state directory # we have. We should have created one above in non-interactive mode.
self.instance.auto_bootstrap(application, self.exclude)
self.instance.install_toolchain_artifact("fix-stacks")
self.instance.install_toolchain_artifact("minidump-stackwalk") ifnot self.instance.artifact_mode:
self.instance.install_toolchain_artifact("clang-tools/clang-tidy")
self.instance.ensure_sccache_packages() # Like 'ensure_browser_packages' or 'ensure_mobile_android_packages'
getattr(self.instance, "ensure_%s_packages" % application)()
def check_code_submission(self, checkout_root: Path): if self.instance.no_interactive or which("moz-phab"): return
# Skip moz-phab install until bug 1696357 is fixed and makes it to a moz-phab # release. if sys.platform.startswith("darwin") and platform.machine() == "arm64": return
ifnot self.instance.prompt_yesno("Will you be submitting commits to Mozilla?"): return
if sys.platform.startswith("darwin") andnot os.environ.get( "MACH_I_DO_WANT_TO_USE_ROSETTA"
): # If running on arm64 mac, check whether we're running under # Rosetta and advise against it. if _macos_is_running_under_rosetta():
print( "Python is being emulated under Rosetta. Please use a native " "Python instead. If you still really want to go ahead, set " "the MACH_I_DO_WANT_TO_USE_ROSETTA environment variable.",
file=sys.stderr,
) return 1
self.instance.state_dir = state_dir
# We need to enable the loading of hgrc in case extensions are # required to open the repo.
(checkout_type, checkout_root) = current_firefox_checkout(
env=self.instance._hg_cleanenv(load_hgrc=True),
hg=hg,
)
self.instance.srcdir = checkout_root
self.instance.validate_environment()
self._validate_python_environment(checkout_root)
if sys.platform.startswith("win"):
self._check_for_dev_drive(checkout_root)
if self.instance.no_system_changes:
self.maybe_install_private_packages_or_exit(application, checkout_type)
self._output_mozconfig(application, mozconfig_builder)
sys.exit(0)
self.instance.install_system_packages()
# Like 'install_browser_packages' or 'install_mobile_android_packages'.
getattr(self.instance, "install_%s_packages" % application)(mozconfig_builder)
# Possibly configure Mercurial, but not if the current checkout or repo # type is Git. if checkout_type == "hg":
hg_installed, hg_modern = self.instance.ensure_mercurial_modern()
if hg_installed and checkout_type == "hg": ifnot self.instance.no_interactive:
configure_hg = self.instance.prompt_yesno(prompt=CONFIGURE_MERCURIAL) else:
configure_hg = self.hg_configure
if configure_hg:
configure_mercurial(hg, state_dir)
# Offer to configure Git, if the current checkout or repo type is Git. elif git and checkout_type == "git":
should_configure_git = False ifnot self.instance.no_interactive:
should_configure_git = self.instance.prompt_yesno(prompt=CONFIGURE_GIT) else: # Assuming default configuration setting applies to all VCS.
should_configure_git = self.hg_configure
if should_configure_git:
configure_git(
git,
to_optional_path(which("git-cinnabar")),
state_dir,
checkout_root,
)
self.maybe_install_private_packages_or_exit(application, checkout_type)
self.check_code_submission(checkout_root) # Wait until after moz-phab setup to check telemetry so that employees # will be automatically opted-in. ifnot self.instance.no_interactive andnot settings.mach_telemetry.is_set_up:
initialize_telemetry_setting(settings, str(checkout_root), str(state_dir))
if file_system_type == "ReFS":
print(" The Firefox source repository is on a Dev Drive.") else:
print(
DEV_DRIVE_SUGGESTION.format(
topsrcdir, file_system_type, current_windows_version
)
) if self.instance.no_interactive: pass else:
input("\nPress enter to continue.")
except subprocess.CalledProcessError as error:
print(
DEV_DRIVE_DETECTION_ERROR.format(f"CalledProcessError: {error.stderr}")
) pass
def _read_default_mozconfig(self):
path = self._default_mozconfig_path() with open(path, "r") as mozconfig_file: return mozconfig_file.read()
def _write_default_mozconfig(self, raw_mozconfig):
path = self._default_mozconfig_path() with open(path, "w") as mozconfig_file:
mozconfig_file.write(raw_mozconfig)
print(f'Your requested configuration has been written to "{path}".')
if current_mozconfig_path: # mozconfig file exists if self._default_mozconfig_path().exists() and Path.samefile(
Path(current_mozconfig_path), self._default_mozconfig_path()
): # This mozconfig file may be created by bootstrap.
self._check_default_mozconfig_mismatch(
current_mozconfig_info, application, raw_mozconfig
) elif raw_mozconfig: # The mozconfig file is created by user.
self._show_mozconfig_suggestion(raw_mozconfig) elif raw_mozconfig: # No mozconfig file exists yet
self._write_default_mozconfig(raw_mozconfig)
def _validate_python_environment(self, topsrcdir):
valid = True
pip3 = to_optional_path(which("pip3")) ifnot pip3:
print("ERROR: Could not find pip3.", file=sys.stderr)
self.instance.suggest_install_pip3()
valid = False ifnot valid:
print( "ERROR: Your Python installation will not be able to run " "`mach bootstrap`. `mach bootstrap` cannot maintain your " "Python environment for you; fix the errors shown here, and " "then re-run `mach bootstrap`.",
file=sys.stderr,
)
sys.exit(1)
def update_vct(hg: Path, root_state_dir: Path): """Ensure version-control-tools in the state directory is up to date."""
vct_dir = root_state_dir / "version-control-tools"
def current_firefox_checkout(env, hg: Optional[Path] = None): """Determine whether we're in a Firefox checkout.
Returns one of None, ``git``, or ``hg``. """
HG_ROOT_REVISIONS = set(
[ # From mozilla-unified. "8ba995b74e18334ab3707f27e9eb8f4e37ba3d29"
]
)
path = Path.cwd() while path:
hg_dir = path / ".hg"
git_dir = path / ".git"
known_file = path / "config" / "milestone.txt" if hg and hg_dir.exists(): # Verify the hg repo is a Firefox repo by looking at rev 0. try:
node = subprocess.check_output(
[str(hg), "log", "-r", "0", "--template", "{node}"],
cwd=str(path),
env=env,
universal_newlines=True,
) if node in HG_ROOT_REVISIONS:
_warn_if_risky_revision(path) return"hg", path # Else the root revision is different. There could be nested # repos. So keep traversing the parents. except subprocess.CalledProcessError: pass
# Just check for known-good files in the checkout, to prevent attempted # foot-shootings. Determining a canonical git checkout of mozilla-unified # is...complicated elif git_dir.exists() or hg_dir.exists(): if known_file.exists():
_warn_if_risky_revision(path) return ("git"if git_dir.exists() else"hg"), path elif known_file.exists(): return"SOURCE", path
ifnot len(path.parents): break
path = path.parent
raise UserError( "Could not identify the root directory of your checkout! " "Are you running `mach bootstrap` in an hg or git clone?"
)
def update_git_tools(git: Optional[Path], root_state_dir: Path): """Update git tools, hooks and extensions""" # Ensure git-cinnabar is up to date.
cinnabar_dir = root_state_dir / "git-cinnabar"
cinnabar_exe = cinnabar_dir / "git-cinnabar"
if sys.platform.startswith(("win32", "msys")):
cinnabar_exe = cinnabar_exe.with_suffix(".exe")
# Older versions of git-cinnabar can't do self-update. So if we start # from such a version, we remove it and start over. # The first version that supported self-update is also the first version # that wasn't a python script, so we can just look for a hash-bang. # Or, on Windows, the .exe didn't exist.
start_over = cinnabar_dir.exists() andnot cinnabar_exe.exists() if cinnabar_exe.exists(): try: with cinnabar_exe.open("rb") as fh:
start_over = fh.read(2) == b"#!" except Exception: # If we couldn't read the binary, let's just try to start over.
start_over = True
if start_over: # git sets pack files read-only, which causes problems removing # them on Windows. To work around that, we use an error handler # on rmtree that retries to remove the file after chmod'ing it. def onerror(func, path, exc): if func == os.unlink:
os.chmod(path, stat.S_IRWXU)
func(path) else: raise exc
shutil.rmtree(str(cinnabar_dir), onerror=onerror)
# If we already have an executable, ask it to update itself.
exists = cinnabar_exe.exists() if exists: try:
subprocess.check_call([str(cinnabar_exe), "self-update"]) except subprocess.CalledProcessError as e:
print(e)
# git-cinnabar 0.6.0rc1 self-update had a bug that could leave an empty # file. If that happens, install from scratch. ifnot exists or cinnabar_exe.stat().st_size == 0: import ssl from urllib.request import urlopen
match = re.search(
r"(\d+\.\d+\.\d+)",
subprocess.check_output([git_str, "--version"], universal_newlines=True),
) ifnot match: raise Exception("Could not find git version")
git_version = Version(match.group(1))
if git_version < MINIMUM_RECOMMENDED_GIT_VERSION:
print(
OLD_GIT_WARNING.format(
old_version=git_version,
minimum_recommended_version=MINIMUM_RECOMMENDED_GIT_VERSION,
)
)
if git_version >= Version("2.17"): # "core.untrackedCache" has a bug before 2.17
subprocess.check_call(
[git_str, "config", "core.untrackedCache", "true"], cwd=str(top_src_dir)
)
ifnot cinnabar: if"MOZILLABUILD"in os.environ: # Slightly modify the path on Windows to be correct # for the copy/paste into the .bash_profile
cinnabar_dir = win_to_msys_path(cinnabar_dir)
def _warn_if_risky_revision(path: Path): # Warn the user if they're trying to bootstrap from an obviously old # version of tree as reported by the version control system (a month in # this case). This is an approximate calculation but is probably good # enough for our purposes.
NUM_SECONDS_IN_MONTH = 60 * 60 * 24 * 30 from mozversioncontrol import get_repository_object
repo = get_repository_object(path) if (time.time() - repo.get_commit_time()) >= NUM_SECONDS_IN_MONTH:
print(OLD_REVISION_WARNING)
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung ist noch experimentell.