Windows: executable/path handling (#37762)

Windows executable paths can have spaces in them, which was leading to
errors when constructing Executable objects: the parser was intended
to handle cases where users could provide an executable along with one
or more space-delimited arguments.

* Executable now assumes that it is constructed with a string argument
  that represents the path to the executable, but no additional arguments.
* Invocations of Executable.__init__ that depended on this have been
  updated (this includes the core, tests, and one instance of builtin
  repository package).
* The error handling for failed invocations of Executable.__call__ now
  includes a check for whether the executable name/path contains a
  space, to help users debug cases where they (now incorrectly)
  concatenate the path and the arguments.
This commit is contained in:
markus-ferrell 2023-08-14 19:29:12 -04:00 committed by GitHub
parent 843e1e80f0
commit c202a045e6
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
6 changed files with 92 additions and 61 deletions

View file

@ -10,11 +10,12 @@
dependencies. dependencies.
""" """
import os import os
from pathlib import PurePath
import llnl.util.filesystem import llnl.util.filesystem
#: This file lives in $prefix/lib/spack/spack/__file__ #: This file lives in $prefix/lib/spack/spack/__file__
prefix = llnl.util.filesystem.ancestor(__file__, 4) prefix = str(PurePath(llnl.util.filesystem.ancestor(__file__, 4)))
#: synonym for prefix #: synonym for prefix
spack_root = prefix spack_root = prefix
@ -88,7 +89,7 @@ def _get_user_cache_path():
return os.path.expanduser(os.getenv("SPACK_USER_CACHE_PATH") or "~%s.spack" % os.sep) return os.path.expanduser(os.getenv("SPACK_USER_CACHE_PATH") or "~%s.spack" % os.sep)
user_cache_path = _get_user_cache_path() user_cache_path = str(PurePath(_get_user_cache_path()))
#: junit, cdash, etc. reports about builds #: junit, cdash, etc. reports about builds
reports_path = os.path.join(user_cache_path, "reports") reports_path = os.path.join(user_cache_path, "reports")

View file

@ -5,6 +5,7 @@
import os import os
import sys import sys
from pathlib import PurePath
import pytest import pytest
@ -16,13 +17,16 @@
def test_read_unicode(tmpdir, working_env): def test_read_unicode(tmpdir, working_env):
with tmpdir.as_cwd():
script_name = "print_unicode.py" script_name = "print_unicode.py"
# read the unicode back in and see whether things work # read the unicode back in and see whether things work
if sys.platform == "win32": if sys.platform == "win32":
script = ex.Executable("%s %s" % (sys.executable, script_name)) script = ex.Executable("%s" % (sys.executable))
script_args = [script_name]
else: else:
script = ex.Executable("./%s" % script_name) script = ex.Executable("./%s" % script_name)
with tmpdir.as_cwd(): script_args = []
os.environ["LD_LIBRARY_PATH"] = spack.main.spack_ld_library_path os.environ["LD_LIBRARY_PATH"] = spack.main.spack_ld_library_path
# make a script that prints some unicode # make a script that prints some unicode
with open(script_name, "w") as f: with open(script_name, "w") as f:
@ -38,7 +42,7 @@ def test_read_unicode(tmpdir, working_env):
fs.set_executable(script_name) fs.set_executable(script_name)
filter_shebangs_in_directory(".", [script_name]) filter_shebangs_in_directory(".", [script_name])
assert "\xc3" == script(output=str).strip() assert "\xc3" == script(*script_args, output=str).strip()
def test_which_relative_path_with_slash(tmpdir, working_env): def test_which_relative_path_with_slash(tmpdir, working_env):
@ -68,7 +72,7 @@ def test_which_with_slash_ignores_path(tmpdir, working_env):
path = str(tmpdir.join("exe")) path = str(tmpdir.join("exe"))
wrong_path = str(tmpdir.join("bin", "exe")) wrong_path = str(tmpdir.join("bin", "exe"))
os.environ["PATH"] = os.path.dirname(wrong_path) os.environ["PATH"] = str(PurePath(wrong_path).parent)
with tmpdir.as_cwd(): with tmpdir.as_cwd():
if sys.platform == "win32": if sys.platform == "win32":

View file

@ -1066,8 +1066,8 @@ def environment_after_sourcing_files(
Keyword Args: Keyword Args:
env (dict): the initial environment (default: current environment) env (dict): the initial environment (default: current environment)
shell (str): the shell to use (default: ``/bin/bash``) shell (str): the shell to use (default: ``/bin/bash`` or ``cmd.exe`` (Windows))
shell_options (str): options passed to the shell (default: ``-c``) shell_options (str): options passed to the shell (default: ``-c`` or ``/C`` (Windows))
source_command (str): the command to run (default: ``source``) source_command (str): the command to run (default: ``source``)
suppress_output (str): redirect used to suppress output of command suppress_output (str): redirect used to suppress output of command
(default: ``&> /dev/null``) (default: ``&> /dev/null``)
@ -1075,15 +1075,23 @@ def environment_after_sourcing_files(
only when the previous command succeeds (default: ``&&``) only when the previous command succeeds (default: ``&&``)
""" """
# Set the shell executable that will be used to source files # Set the shell executable that will be used to source files
if sys.platform == "win32":
shell_cmd = kwargs.get("shell", "cmd.exe")
shell_options = kwargs.get("shell_options", "/C")
suppress_output = kwargs.get("suppress_output", "")
source_command = kwargs.get("source_command", "")
else:
shell_cmd = kwargs.get("shell", "/bin/bash") shell_cmd = kwargs.get("shell", "/bin/bash")
shell_options = kwargs.get("shell_options", "-c") shell_options = kwargs.get("shell_options", "-c")
source_command = kwargs.get("source_command", "source")
suppress_output = kwargs.get("suppress_output", "&> /dev/null") suppress_output = kwargs.get("suppress_output", "&> /dev/null")
source_command = kwargs.get("source_command", "source")
concatenate_on_success = kwargs.get("concatenate_on_success", "&&") concatenate_on_success = kwargs.get("concatenate_on_success", "&&")
shell = Executable(" ".join([shell_cmd, shell_options])) shell = Executable(shell_cmd)
def _source_single_file(file_and_args, environment): def _source_single_file(file_and_args, environment):
shell_options_list = shell_options.split()
source_file = [source_command] source_file = [source_command]
source_file.extend(x for x in file_and_args) source_file.extend(x for x in file_and_args)
source_file = " ".join(source_file) source_file = " ".join(source_file)
@ -1101,7 +1109,14 @@ def _source_single_file(file_and_args, environment):
source_file_arguments = " ".join( source_file_arguments = " ".join(
[source_file, suppress_output, concatenate_on_success, dump_environment_cmd] [source_file, suppress_output, concatenate_on_success, dump_environment_cmd]
) )
output = shell(source_file_arguments, output=str, env=environment, ignore_quotes=True) output = shell(
*shell_options_list,
source_file_arguments,
output=str,
env=environment,
ignore_quotes=True,
)
return json.loads(output) return json.loads(output)
current_environment = kwargs.get("env", dict(os.environ)) current_environment = kwargs.get("env", dict(os.environ))

View file

@ -5,14 +5,13 @@
import os import os
import re import re
import shlex
import subprocess import subprocess
import sys import sys
from pathlib import Path, PurePath
import llnl.util.tty as tty import llnl.util.tty as tty
import spack.error import spack.error
from spack.util.path import Path, format_os_path, path_to_os_path, system_path_filter
__all__ = ["Executable", "which", "ProcessError"] __all__ = ["Executable", "which", "ProcessError"]
@ -21,11 +20,12 @@ class Executable:
"""Class representing a program that can be run on the command line.""" """Class representing a program that can be run on the command line."""
def __init__(self, name): def __init__(self, name):
# necesary here for the shlex call to succeed file_path = str(Path(name))
name = format_os_path(name, mode=Path.unix) if sys.platform != "win32" and name.startswith("."):
self.exe = shlex.split(str(name)) # pathlib strips the ./ from relative paths so it must be added back
# filter back to platform dependent path file_path = os.path.join(".", file_path)
self.exe = path_to_os_path(*self.exe) self.exe = [file_path]
self.default_env = {} self.default_env = {}
from spack.util.environment import EnvironmentModifications # no cycle from spack.util.environment import EnvironmentModifications # no cycle
@ -35,12 +35,10 @@ def __init__(self, name):
if not self.exe: if not self.exe:
raise ProcessError("Cannot construct executable for '%s'" % name) raise ProcessError("Cannot construct executable for '%s'" % name)
@system_path_filter
def add_default_arg(self, arg): def add_default_arg(self, arg):
"""Add a default argument to the command.""" """Add a default argument to the command."""
self.exe.append(arg) self.exe.append(arg)
@system_path_filter
def add_default_env(self, key, value): def add_default_env(self, key, value):
"""Set an environment variable when the command is run. """Set an environment variable when the command is run.
@ -70,7 +68,7 @@ def name(self):
Returns: Returns:
str: The basename of the executable str: The basename of the executable
""" """
return os.path.basename(self.path) return PurePath(self.path).name
@property @property
def path(self): def path(self):
@ -79,7 +77,7 @@ def path(self):
Returns: Returns:
str: The path to the executable str: The path to the executable
""" """
return self.exe[0] return str(PurePath(self.exe[0]))
def __call__(self, *args, **kwargs): def __call__(self, *args, **kwargs):
"""Run this executable in a subprocess. """Run this executable in a subprocess.
@ -235,7 +233,11 @@ def streamify(arg, mode):
return result return result
except OSError as e: except OSError as e:
raise ProcessError("%s: %s" % (self.exe[0], e.strerror), "Command: " + cmd_line_string) message = "Command: " + cmd_line_string
if " " in self.exe[0]:
message += "\nDid you mean to add a space to the command?"
raise ProcessError("%s: %s" % (self.exe[0], e.strerror), message)
except subprocess.CalledProcessError as e: except subprocess.CalledProcessError as e:
if fail_on_error: if fail_on_error:
@ -269,39 +271,48 @@ def __str__(self):
return " ".join(self.exe) return " ".join(self.exe)
@system_path_filter
def which_string(*args, **kwargs): def which_string(*args, **kwargs):
"""Like ``which()``, but return a string instead of an ``Executable``.""" """Like ``which()``, but return a string instead of an ``Executable``."""
path = kwargs.get("path", os.environ.get("PATH", "")) path = kwargs.get("path", os.environ.get("PATH", ""))
required = kwargs.get("required", False) required = kwargs.get("required", False)
if isinstance(path, list):
paths = [Path(str(x)) for x in path]
if isinstance(path, str): if isinstance(path, str):
path = path.split(os.pathsep) paths = [Path(x) for x in path.split(os.pathsep)]
for name in args: def get_candidate_items(search_item):
win_candidates = [] if sys.platform == "win32" and not search_item.suffix:
if sys.platform == "win32" and (not name.endswith(".exe") and not name.endswith(".bat")): return [search_item.parent / (search_item.name + ext) for ext in [".exe", ".bat"]]
win_candidates = [name + ext for ext in [".exe", ".bat"]]
candidate_names = [name] if not win_candidates else win_candidates
return [Path(search_item)]
def add_extra_search_paths(paths):
with_parents = []
with_parents.extend(paths)
if sys.platform == "win32": if sys.platform == "win32":
new_path = path[:] for p in paths:
for p in path: if p.name == "bin":
if os.path.basename(p) == "bin": with_parents.append(p.parent)
new_path.append(os.path.dirname(p)) return with_parents
path = new_path
for candidate_name in candidate_names: for search_item in args:
if os.path.sep in candidate_name: search_paths = []
exe = os.path.abspath(candidate_name) search_paths.extend(paths)
if os.path.isfile(exe) and os.access(exe, os.X_OK): if search_item.startswith("."):
return exe # we do this because pathlib will strip any leading ./
else: search_paths.insert(0, Path.cwd())
for directory in path: search_paths = add_extra_search_paths(search_paths)
directory = path_to_os_path(directory).pop()
exe = os.path.join(directory, candidate_name) search_item = Path(search_item)
if os.path.isfile(exe) and os.access(exe, os.X_OK): candidate_items = get_candidate_items(Path(search_item))
return exe
for candidate_item in candidate_items:
for directory in search_paths:
exe = directory / candidate_item
if exe.is_file() and os.access(str(exe), os.X_OK):
return str(exe)
if required: if required:
raise CommandNotFoundError("spack requires '%s'. Make sure it is in your path." % args[0]) raise CommandNotFoundError("spack requires '%s'. Make sure it is in your path." % args[0])
@ -326,7 +337,7 @@ def which(*args, **kwargs):
Executable: The first executable that is found in the path Executable: The first executable that is found in the path
""" """
exe = which_string(*args, **kwargs) exe = which_string(*args, **kwargs)
return Executable(shlex.quote(exe)) if exe else None return Executable(exe) if exe else None
class ProcessError(spack.error.SpackError): class ProcessError(spack.error.SpackError):

View file

@ -109,7 +109,7 @@ def install(self, spec, prefix):
print(cmake) print(cmake)
print(cmake.exe) print(cmake.exe)
check( check(
cmake.exe[0].startswith(spec["cmake"].prefix.bin), cmake.path.startswith(spec["cmake"].prefix.bin),
"Wrong cmake was in environment: %s" % cmake, "Wrong cmake was in environment: %s" % cmake,
) )

View file

@ -168,15 +168,15 @@ def test_env(self):
def configure(self, spec, prefix): def configure(self, spec, prefix):
# change into directory with source code # change into directory with source code
with working_dir(self.get_oommf_source_root()): with working_dir(self.get_oommf_source_root()):
configure = Executable("./oommf.tcl pimake distclean") configure = Executable("./oommf.tcl")
configure() configure("pimake", "distclean")
configure2 = Executable("./oommf.tcl pimake upgrade") configure2 = Executable("./oommf.tcl")
configure2() configure2("pimake", "upgrade")
def build(self, spec, prefix): def build(self, spec, prefix):
with working_dir(self.get_oommf_source_root()): with working_dir(self.get_oommf_source_root()):
make = Executable("./oommf.tcl pimake ") make = Executable("./oommf.tcl")
make() make("pimake")
def install(self, spec, prefix): def install(self, spec, prefix):
# keep a copy of all the tcl files and everything oommf created. # keep a copy of all the tcl files and everything oommf created.