# 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 re import subprocess import sys from pathlib import Path from mach.util import to_optional_path, win_to_msys_path from mozbuild.bootstrap import bootstrap_all_toolchains_for, bootstrap_toolchain from mozfile import which from packaging.version import Version from mozboot import rust from mozboot.util import ( MINIMUM_RUST_VERSION, get_mach_virtualenv_binary, http_download_and_save, ) NO_MERCURIAL = """ Could not find Mercurial (hg) in the current shell's path. Try starting a new shell and running the bootstrapper again. """ MERCURIAL_UNABLE_UPGRADE = """ You are currently running Mercurial %s. Running %s or newer is recommended for performance and stability reasons. Unfortunately, this bootstrapper currently does not know how to automatically upgrade Mercurial on your machine. You can usually install Mercurial through your package manager or by downloading a package from http://mercurial.selenic.com/. """ MERCURIAL_UPGRADE_FAILED = """ We attempted to upgrade Mercurial to a modern version (%s or newer). However, you appear to have version %s still. It's possible your package manager doesn't support a modern version of Mercurial. It's also possible Mercurial is not being installed in the search path for this shell. Try creating a new shell and run this bootstrapper again. If it continues to fail, consider installing Mercurial by following the instructions at http://mercurial.selenic.com/. """ MERCURIAL_INSTALL_PROMPT = """ Mercurial releases a new version every 3 months and your distro's package may become out of date. This may cause incompatibility with some Mercurial extensions that rely on new Mercurial features. As a result, you may not have an optimal version control experience. To have the best Mercurial experience possible, we recommend installing Mercurial via the "pip" Python packaging utility. This will likely result in files being placed in /usr/local/bin and /usr/local/lib. How would you like to continue? 1. Install a modern Mercurial via pip [default] 2. Install a legacy Mercurial via the distro package manager 3. Do not install Mercurial Your choice: """ PYTHON_UNABLE_UPGRADE = """ You are currently running Python %s. Running %s or newer (but not 3.x) is required. Unfortunately, this bootstrapper does not currently know how to automatically upgrade Python on your machine. Please search the Internet for how to upgrade your Python and try running this bootstrapper again to ensure your machine is up to date. """ RUST_INSTALL_COMPLETE = """ Rust installation complete. You should now have rustc and cargo in %(cargo_bin)s The installer tries to add these to your default shell PATH, so restarting your shell and running this script again may work. If it doesn't, you'll need to add the new command location manually. If restarting doesn't work, edit your shell initialization script, which may be called ~/.bashrc or ~/.bash_profile or ~/.profile, and add the following line: %(cmd)s Then restart your shell and run the bootstrap script again. """ RUST_NOT_IN_PATH = """ You have some rust files in %(cargo_bin)s but they're not part of this shell's PATH. To add these to the PATH, edit your shell initialization script, which may be called ~/.bashrc or ~/.bash_profile or ~/.profile, and add the following line: %(cmd)s Then restart your shell and run the bootstrap script again. """ RUSTUP_OLD = """ We found an executable called `rustup` which we normally use to install and upgrade Rust programming language support, but we didn't understand its output. It may be an old version, or not be the installer from https://rustup.rs/ Please move it out of the way and run the bootstrap script again. Or if you prefer and know how, use the current rustup to install a compatible version of the Rust programming language yourself. """ RUST_UPGRADE_FAILED = """ We attempted to upgrade Rust to a modern version (%s or newer). However, you appear to still have version %s. It's possible rustup failed. It's also possible the new Rust is not being installed in the search path for this shell. Try creating a new shell and run this bootstrapper again. If this continues to fail and you are sure you have a modern Rust on your system, ensure it is on the $PATH and try again. If that fails, you'll need to install Rust manually. We recommend the installer from https://rustup.rs/ for installing Rust, but you may be able to get a recent enough version from a software install tool or package manager on your system, or directly from https://rust-lang.org/ """ BROWSER_ARTIFACT_MODE_MOZCONFIG = """ # Automatically download and use compiled C++ components: ac_add_options --enable-artifact-builds """.strip() JS_MOZCONFIG_TEMPLATE = """\ # Build only the SpiderMonkey JS test shell ac_add_options --enable-project=js """ # Upgrade Mercurial older than this. # This should match the OLDEST_NON_LEGACY_VERSION in # version-control-tools/hgext/configwizard/__init__.py. MODERN_MERCURIAL_VERSION = Version("4.9") # Upgrade rust older than this. MODERN_RUST_VERSION = Version(MINIMUM_RUST_VERSION) class BaseBootstrapper(object): """Base class for system bootstrappers.""" def __init__(self, no_interactive=False, no_system_changes=False): self.package_manager_updated = False self.no_interactive = no_interactive self.no_system_changes = no_system_changes self.state_dir = None self.srcdir = None def validate_environment(self): """ Called once the current firefox checkout has been detected. Platform-specific implementations should check the environment and offer advice/warnings to the user, if necessary. """ def suggest_install_distutils(self): """Called if distutils.{sysconfig,spawn} can't be imported.""" print( "Does your distro require installing another package for distutils?", file=sys.stderr, ) def suggest_install_pip3(self): """Called if pip3 can't be found.""" print( "Try installing pip3 with your system's package manager.", file=sys.stderr ) def install_system_packages(self): """ Install packages shared by all applications. These are usually packages required by the development (like mercurial) or the build system (like autoconf). """ raise NotImplementedError( "%s must implement install_system_packages()" % __name__ ) def install_browser_packages(self, mozconfig_builder): """ Install packages required to build Firefox for Desktop (application 'browser'). """ raise NotImplementedError( "Cannot bootstrap Firefox for Desktop: " "%s does not yet implement install_browser_packages()" % __name__ ) def ensure_browser_packages(self): """ Install pre-built packages needed to build Firefox for Desktop (application 'browser') Currently this is not needed and kept for compatibility with Firefox for Android. """ pass def ensure_js_packages(self): """ Install pre-built packages needed to build SpiderMonkey JavaScript Engine Currently this is not needed and kept for compatibility with Firefox for Android. """ pass def ensure_browser_artifact_mode_packages(self): """ Install pre-built packages needed to build Firefox for Desktop (application 'browser') Currently this is not needed and kept for compatibility with Firefox for Android. """ pass def generate_browser_mozconfig(self): """ Print a message to the console detailing what the user's mozconfig should contain. Firefox for Desktop can in simple cases determine its build environment entirely from configure. """ pass def install_js_packages(self, mozconfig_builder): """ Install packages required to build SpiderMonkey JavaScript Engine (application 'js'). """ return self.install_browser_packages(mozconfig_builder) def generate_js_mozconfig(self): """ Create a reasonable starting point for a JS shell build. """ return JS_MOZCONFIG_TEMPLATE def install_browser_artifact_mode_packages(self, mozconfig_builder): """ Install packages required to build Firefox for Desktop (application 'browser') in Artifact Mode. """ raise NotImplementedError( "Cannot bootstrap Firefox for Desktop Artifact Mode: " "%s does not yet implement install_browser_artifact_mode_packages()" % __name__ ) def generate_browser_artifact_mode_mozconfig(self): """ Print a message to the console detailing what the user's mozconfig should contain. Firefox for Desktop Artifact Mode needs to enable artifact builds and a path where the build artifacts will be written to. """ return BROWSER_ARTIFACT_MODE_MOZCONFIG def install_mobile_android_packages(self, mozconfig_builder): """ Install packages required to build GeckoView (application 'mobile/android'). """ raise NotImplementedError( "Cannot bootstrap GeckoView/Firefox for Android: " "%s does not yet implement install_mobile_android_packages()" % __name__ ) def ensure_mobile_android_packages(self): """ Install pre-built packages required to run GeckoView (application 'mobile/android') """ raise NotImplementedError( "Cannot bootstrap GeckoView/Firefox for Android: " "%s does not yet implement ensure_mobile_android_packages()" % __name__ ) def ensure_mobile_android_artifact_mode_packages(self): """ Install pre-built packages required to run GeckoView Artifact Build (application 'mobile/android') """ self.ensure_mobile_android_packages() def generate_mobile_android_mozconfig(self): """ Print a message to the console detailing what the user's mozconfig should contain. GeckoView/Firefox for Android needs an application and an ABI set, and it needs paths to the Android SDK and NDK. """ raise NotImplementedError( "%s does not yet implement generate_mobile_android_mozconfig()" % __name__ ) def install_mobile_android_artifact_mode_packages(self, mozconfig_builder): """ Install packages required to build GeckoView/Firefox for Android (application 'mobile/android', also known as Fennec) in Artifact Mode. """ raise NotImplementedError( "Cannot bootstrap GeckoView/Firefox for Android Artifact Mode: " "%s does not yet implement install_mobile_android_artifact_mode_packages()" % __name__ ) def generate_mobile_android_artifact_mode_mozconfig(self): """ Print a message to the console detailing what the user's mozconfig should contain. GeckoView/Firefox for Android Artifact Mode needs an application and an ABI set, and it needs paths to the Android SDK. """ raise NotImplementedError( "%s does not yet implement generate_mobile_android_artifact_mode_mozconfig()" % __name__ ) def ensure_sccache_packages(self): """ Install sccache. """ pass def install_toolchain_artifact(self, toolchain_job, no_unpack=False): if no_unpack: return self.install_toolchain_artifact_impl( self.state_dir, toolchain_job, no_unpack ) bootstrap_toolchain(toolchain_job) def install_toolchain_artifact_impl( self, install_dir: Path, toolchain_job, no_unpack=False ): if type(self.srcdir) is str: mach_binary = Path(self.srcdir) / "mach" else: mach_binary = (self.srcdir / "mach").resolve() if not mach_binary.exists(): raise ValueError(f"mach not found at {mach_binary}") if not self.state_dir: raise ValueError( "Need a state directory (e.g. ~/.mozbuild) to download " "artifacts" ) python_location = get_mach_virtualenv_binary() if not python_location.exists(): raise ValueError(f"python not found at {python_location}") cmd = [ str(python_location), str(mach_binary), "artifact", "toolchain", "--bootstrap", "--from-build", toolchain_job, ] if no_unpack: cmd += ["--no-unpack"] subprocess.check_call(cmd, cwd=str(install_dir)) def auto_bootstrap(self, application, exclude=[]): args = ["--with-ccache=sccache"] if application.endswith("_artifact_mode"): args.append("--enable-artifact-builds") application = application[: -len("_artifact_mode")] args.append("--enable-project={}".format(application.replace("_", "/"))) if exclude: args.append( "--enable-bootstrap={}".format(",".join(f"-{x}" for x in exclude)) ) bootstrap_all_toolchains_for(args) def run_as_root(self, command, may_use_sudo=True): if os.geteuid() != 0: if may_use_sudo and which("sudo"): command.insert(0, "sudo") else: command = ["su", "root", "-c", " ".join(command)] print("Executing as root:", subprocess.list2cmdline(command)) subprocess.check_call(command, stdin=sys.stdin) def prompt_int(self, prompt, low, high, default=None): """Prompts the user with prompt and requires an integer between low and high. If the user doesn't select an option and a default isn't provided, then the lowest option is used. This is because some option must be implicitly selected if mach is invoked with "--no-interactive" """ if default is not None: assert isinstance(default, int) assert low <= default <= high else: default = low if self.no_interactive: print(prompt) print('Selecting "{}" because context is not interactive.'.format(default)) return default while True: choice = input(prompt) if choice == "" and default is not None: return default try: choice = int(choice) if low <= choice <= high: return choice except ValueError: pass print("ERROR! Please enter a valid option!") def prompt_yesno(self, prompt): """Prompts the user with prompt and requires a yes/no answer.""" if self.no_interactive: print(prompt) print('Selecting "Y" because context is not interactive.') return True while True: choice = input(prompt + " (Yn): ").strip().lower()[:1] if choice == "": return True elif choice in ("y", "n"): return choice == "y" print("ERROR! Please enter y or n!") def _ensure_package_manager_updated(self): if self.package_manager_updated: return self._update_package_manager() self.package_manager_updated = True def _update_package_manager(self): """Updates the package manager's manifests/package list. This should be defined in child classes. """ def _parse_version_impl(self, path: Path, name, env, version_param): """Execute the given path, returning the version. Invokes the path argument with the --version switch and returns a Version representing the output if successful. If not, returns None. An optional name argument gives the expected program name returned as part of the version string, if it's different from the basename of the executable. An optional env argument allows modifying environment variable during the invocation to set options, PATH, etc. """ if not name: name = path.name if name.lower().endswith(".exe"): name = name[:-4] process = subprocess.run( [str(path), version_param], env=env, universal_newlines=True, stdout=subprocess.PIPE, stderr=subprocess.STDOUT, ) if process.returncode != 0: # This can happen e.g. if the user has an inactive pyenv shim in # their path. Just silently treat this as a failure to parse the # path and move on. return None match = re.search(name + " ([a-z0-9\.]+)", process.stdout) if not match: print("ERROR! Unable to identify %s version." % name) return None return Version(match.group(1)) def _parse_version(self, path: Path, name=None, env=None): return self._parse_version_impl(path, name, env, "--version") def _hg_cleanenv(self, load_hgrc=False): """Returns a copy of the current environment updated with the HGPLAIN and HGRCPATH environment variables. HGPLAIN prevents Mercurial from applying locale variations to the output making it suitable for use in scripts. HGRCPATH controls the loading of hgrc files. Setting it to the empty string forces that no user or system hgrc file is used. """ env = os.environ.copy() env["HGPLAIN"] = "1" if not load_hgrc: env["HGRCPATH"] = "" return env def is_mercurial_modern(self): hg = to_optional_path(which("hg")) if not hg: print(NO_MERCURIAL) return False, False, None our = self._parse_version(hg, "version", self._hg_cleanenv()) if not our: return True, False, None return True, our >= MODERN_MERCURIAL_VERSION, our def ensure_mercurial_modern(self): installed, modern, version = self.is_mercurial_modern() if modern: print("Your version of Mercurial (%s) is sufficiently modern." % version) return installed, modern self._ensure_package_manager_updated() if installed: print("Your version of Mercurial (%s) is not modern enough." % version) print( "(Older versions of Mercurial have known security vulnerabilities. " "Unless you are running a patched Mercurial version, you may be " "vulnerable." ) else: print("You do not have Mercurial installed") if self.upgrade_mercurial(version) is False: return installed, modern installed, modern, after = self.is_mercurial_modern() if installed and not modern: print(MERCURIAL_UPGRADE_FAILED % (MODERN_MERCURIAL_VERSION, after)) return installed, modern def upgrade_mercurial(self, current): """Upgrade Mercurial. Child classes should reimplement this. Return False to not perform a version check after the upgrade is performed. """ print(MERCURIAL_UNABLE_UPGRADE % (current, MODERN_MERCURIAL_VERSION)) def warn_if_pythonpath_is_set(self): if "PYTHONPATH" in os.environ: print( "WARNING: Your PYTHONPATH environment variable is set. This can " "cause flaky installations of the requirements, and other unexpected " "issues with mach. It is recommended to unset this variable." ) def is_rust_modern(self, cargo_bin: Path): rustc = to_optional_path(which("rustc", extra_search_dirs=[str(cargo_bin)])) if not rustc: print("Could not find a Rust compiler.") return False, None our = self._parse_version(rustc) if not our: return False, None return our >= MODERN_RUST_VERSION, our def cargo_home(self): cargo_home = Path(os.environ.get("CARGO_HOME", Path("~/.cargo").expanduser())) cargo_bin = cargo_home / "bin" return cargo_home, cargo_bin def print_rust_path_advice(self, template, cargo_home: Path, cargo_bin: Path): # Suggest ~/.cargo/env if it exists. if (cargo_home / "env").exists(): cmd = f"source {cargo_home}/env" else: # On Windows rustup doesn't write out ~/.cargo/env # so fall back to a manual PATH update. Bootstrap # only runs under msys, so a unix-style shell command # is appropriate there. cargo_bin = win_to_msys_path(cargo_bin) cmd = f"export PATH={cargo_bin}:$PATH" print(template % {"cargo_bin": cargo_bin, "cmd": cmd}) def ensure_rust_modern(self): cargo_home, cargo_bin = self.cargo_home() modern, version = self.is_rust_modern(cargo_bin) rustup = to_optional_path(which("rustup", extra_search_dirs=[str(cargo_bin)])) if modern: print("Your version of Rust (%s) is new enough." % version) elif version: print("Your version of Rust (%s) is too old." % version) if rustup and not modern: rustup_version = self._parse_version(rustup) if not rustup_version: print(RUSTUP_OLD) sys.exit(1) print("Found rustup. Will try to upgrade.") self.upgrade_rust(rustup) modern, after = self.is_rust_modern(cargo_bin) if not modern: print(RUST_UPGRADE_FAILED % (MODERN_RUST_VERSION, after)) sys.exit(1) elif not rustup: # No rustup. Download and run the installer. print("Will try to install Rust.") self.install_rust() modern, version = self.is_rust_modern(cargo_bin) rustup = to_optional_path( which("rustup", extra_search_dirs=[str(cargo_bin)]) ) self.ensure_rust_targets(rustup, version) def ensure_rust_targets(self, rustup: Path, rust_version): """Make sure appropriate cross target libraries are installed.""" target_list = subprocess.check_output( [str(rustup), "target", "list"], universal_newlines=True ) targets = [ line.split()[0] for line in target_list.splitlines() if "installed" in line or "default" in line ] print("Rust supports %s targets." % ", ".join(targets)) # Support 32-bit Windows on 64-bit Windows. win32 = "i686-pc-windows-msvc" win64 = "x86_64-pc-windows-msvc" if rust.platform() == win64 and win32 not in targets: subprocess.check_call([str(rustup), "target", "add", win32]) if "mobile_android" in self.application: # Let's add the most common targets. if rust_version < Version("1.33"): arm_target = "armv7-linux-androideabi" else: arm_target = "thumbv7neon-linux-androideabi" android_targets = ( arm_target, "aarch64-linux-android", "i686-linux-android", "x86_64-linux-android", ) for target in android_targets: if target not in targets: subprocess.check_call([str(rustup), "target", "add", target]) def upgrade_rust(self, rustup: Path): """Upgrade Rust. Invoke rustup from the given path to update the rust install.""" subprocess.check_call([str(rustup), "update"]) # This installs rustfmt when not already installed, or nothing # otherwise, while the update above would have taken care of upgrading # it. subprocess.check_call([str(rustup), "component", "add", "rustfmt"]) def install_rust(self): """Download and run the rustup installer.""" import errno import stat import tempfile platform = rust.platform() url = rust.rustup_url(platform) checksum = rust.rustup_hash(platform) if not url or not checksum: print("ERROR: Could not download installer.") sys.exit(1) print("Downloading rustup-init... ", end="") fd, rustup_init = tempfile.mkstemp(prefix=Path(url).name) rustup_init = Path(rustup_init) os.close(fd) try: http_download_and_save(url, rustup_init, checksum) mode = rustup_init.stat().st_mode rustup_init.chmod(mode | stat.S_IRWXU) print("Ok") print("Running rustup-init...") subprocess.check_call( [ str(rustup_init), "-y", "--default-toolchain", "stable", "--default-host", platform, "--component", "rustfmt", ] ) cargo_home, cargo_bin = self.cargo_home() self.print_rust_path_advice(RUST_INSTALL_COMPLETE, cargo_home, cargo_bin) finally: try: rustup_init.unlink() except OSError as e: if e.errno != errno.ENOENT: raise