# 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/.
# This module provides mixins to perform process execution.
import logging import os import signal import subprocess import sys from pathlib import Path from typing import Optional
from mozbuild import shellutil from mozprocess.processhandler import ProcessHandlerMixin
from .logging import LoggingMixin
# Perform detection of operating system environment. This is used by command # execution. We only do this once to save redundancy. Yes, this can fail module # loading. That is arguably OK. if"SHELL"in os.environ:
_current_shell = os.environ["SHELL"] elif"MOZILLABUILD"in os.environ:
mozillabuild = os.environ["MOZILLABUILD"] if (Path(mozillabuild) / "msys2").exists():
_current_shell = mozillabuild + "/msys2/usr/bin/sh.exe" else:
_current_shell = mozillabuild + "/msys/bin/sh.exe" elif"COMSPEC"in os.environ:
_current_shell = os.environ["COMSPEC"] elif sys.platform != "win32": # Fall back to a standard shell.
_current_shell = "/bin/sh" else: raise Exception("Could not detect environment shell!")
_in_msys = False
if (
os.environ.get("MSYSTEM", None) in ("MINGW32", "MINGW64") or"MOZILLABUILD"in os.environ
):
_in_msys = True
class ProcessExecutionMixin(LoggingMixin): """Mix-in that provides process execution functionality."""
def run_process(
self,
args=None,
cwd: Optional[str] = None,
append_env=None,
explicit_env=None,
log_name=None,
log_level=logging.INFO,
line_handler=None,
require_unix_environment=False,
ensure_exit_code=0,
ignore_children=False,
pass_thru=False,
python_unbuffered=True,
): """Runs a single process to completion.
Takes a list of arguments to run where the first item is the
executable. Runs the command in the specified directory and with optional environment variables.
append_env -- Dict of environment variables to append to the current
set of environment variables.
explicit_env -- Dict of environment variables to set for the new
process. Any existing environment variables will be ignored.
require_unix_environment ifTrue will ensure the command is executed
within a UNIX environment. Basically, if we are on Windows, it will
execute the command via an appropriate UNIX-like shell.
ignore_children is proxied to mozprocess's ignore_children.
ensure_exit_code is used to ensure the exit code of a process matches
what is expected. If it is an integer, we raise an Exception if the
exit code does not match this value. If it isTrue, we ensure the exit
code is 0. If it isFalse, we don't perform any exit code validation.
pass_thru is a special execution mode where the child process inherits
this process's standard file handles (stdin, stdout, stderr) as well as
additional file descriptors. It should be used for interactive processes
where buffering from mozprocess could be an issue. pass_thru does not
use mozprocess. Therefore, arguments like log_name, line_handler, and ignore_children have no effect.
When python_unbuffered is set, the PYTHONUNBUFFERED environment variable
will be set in the child process. This is normally advantageous (see bug
1627873) but is detrimental in certain circumstances (specifically, we
have seen issues when using pass_thru mode to open a Python subshell, as in bug 1628838). This variable should be set to False to avoid bustage in those circumstances. """
args = self._normalize_command(args, require_unix_environment)
self.log(
logging.INFO, "new_process",
{"args": " ".join(shellutil.quote(arg) for arg in args)}, "{args}",
)
def handleLine(line): # Converts str to unicode on Python 2 and bytes to str on Python 3. if isinstance(line, bytes):
line = line.decode(sys.stdout.encoding or"utf-8", "replace")
if line_handler: try:
line_handler(line) except LineHandlingEarlyReturn: return
if pass_thru:
proc = subprocess.Popen(args, cwd=cwd, env=use_env, close_fds=False)
status = None # Leave it to the subprocess to handle Ctrl+C. If it terminates as # a result of Ctrl+C, proc.wait() will return a status code, and, # we get out of the loop. If it doesn't, like e.g. gdb, we continue # waiting. while status isNone: try:
status = proc.wait() except KeyboardInterrupt: pass else:
p = ProcessHandlerMixin(
args,
cwd=cwd,
env=use_env,
processOutputLine=[handleLine],
universal_newlines=True,
ignore_children=ignore_children,
)
p.run()
p.processOutput()
status = None
sig = None while status isNone: try: if sig isNone:
status = p.wait() else:
status = p.kill(sig=sig) except KeyboardInterrupt: if sig isNone:
sig = signal.SIGINT elif sig == signal.SIGINT: # If we've already tried SIGINT, escalate (if possible). # Note: SIGKILL is not available on Windows.
getattr(signal, "SIGKILL", sig)
if ensure_exit_code isFalse: return status
if ensure_exit_code isTrue:
ensure_exit_code = 0
if status != ensure_exit_code: raise Exception(
f"Process executed with non-0 exit code {status}: {' '.join(shellutil.quote(arg) for arg in args)}"
)
return status
def _normalize_command(self, args, require_unix_environment): """Adjust command arguments to run in the necessary environment.
This exists mainly to facilitate execution of programs requiring a *NIX
shell when running on Windows. The caller specifies whether a shell
environment is required. If it isand we are running on Windows but
aren't running in the UNIX-like msys environment, then we rewrite the
command to execute via a shell. """ assert isinstance(args, list) and len(args)
# Always munge Windows-style into Unix style for the command.
prog = args[0].replace("\\", "/")
# PyMake removes the C: prefix. But, things seem to work here # without it. Not sure what that's about.
# We run everything through the msys shell. We need to use # '-c' and pass all the arguments as one argument because that is # how sh works.
cline = subprocess.list2cmdline([prog] + args[1:]) return [_current_shell, "-c", cline]
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.