aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--mesonbuild/programs.py48
-rwxr-xr-xrun_mypy.py1
2 files changed, 28 insertions, 21 deletions
diff --git a/mesonbuild/programs.py b/mesonbuild/programs.py
index 0bfe773..b436a15 100644
--- a/mesonbuild/programs.py
+++ b/mesonbuild/programs.py
@@ -31,14 +31,17 @@ if T.TYPE_CHECKING:
class ExternalProgram:
+
+ """A program that is found on the system."""
+
windows_exts = ('exe', 'msc', 'com', 'bat', 'cmd')
- # An 'ExternalProgram' always runs on the build machine
for_machine = MachineChoice.BUILD
def __init__(self, name: str, command: T.Optional[T.List[str]] = None,
silent: bool = False, search_dir: T.Optional[str] = None,
extra_search_dirs: T.Optional[T.List[str]] = None):
self.name = name
+ self.path = None # type: T.Optional[str]
if command is not None:
self.command = mesonlib.listify(command)
if mesonlib.is_windows():
@@ -61,15 +64,16 @@ class ExternalProgram:
if self.found():
break
- # Set path to be the last item that is actually a file (in order to
- # skip options in something like ['python', '-u', 'file.py']. If we
- # can't find any components, default to the last component of the path.
- self.path = self.command[-1]
- for i in range(len(self.command) - 1, -1, -1):
- arg = self.command[i]
- if arg is not None and os.path.isfile(arg):
- self.path = arg
- break
+ if self.found():
+ # Set path to be the last item that is actually a file (in order to
+ # skip options in something like ['python', '-u', 'file.py']. If we
+ # can't find any components, default to the last component of the path.
+ for arg in reversed(self.command):
+ if arg is not None and os.path.isfile(arg):
+ self.path = arg
+ break
+ else:
+ self.path = self.command[-1]
if not silent:
# ignore the warning because derived classes never call this __init__
@@ -94,7 +98,7 @@ class ExternalProgram:
return ' '.join(self.command)
@classmethod
- def from_bin_list(cls, env: 'Environment', for_machine: MachineChoice, name):
+ def from_bin_list(cls, env: 'Environment', for_machine: MachineChoice, name: str) -> 'ExternalProgram':
# There is a static `for_machine` for this class because the binary
# aways runs on the build platform. (It's host platform is our build
# platform.) But some external programs have a target platform, so this
@@ -129,20 +133,22 @@ class ExternalProgram:
return os.pathsep.join(paths)
@staticmethod
- def from_entry(name, command):
+ def from_entry(name: str, command: T.Union[str, T.List[str]]) -> 'ExternalProgram':
if isinstance(command, list):
if len(command) == 1:
command = command[0]
# We cannot do any searching if the command is a list, and we don't
# need to search if the path is an absolute path.
if isinstance(command, list) or os.path.isabs(command):
+ if isinstance(command, str):
+ command = [command]
return ExternalProgram(name, command=command, silent=True)
assert isinstance(command, str)
# Search for the command using the specified string!
return ExternalProgram(command, silent=True)
@staticmethod
- def _shebang_to_cmd(script: str) -> T.Optional[list]:
+ def _shebang_to_cmd(script: str) -> T.Optional[T.List[str]]:
"""
Check if the file has a shebang and manually parse it to figure out
the interpreter to use. This is useful if the script is not executable
@@ -184,11 +190,11 @@ class ExternalProgram:
commands = mesonlib.python_command + commands[1:]
return commands + [script]
except Exception as e:
- mlog.debug(e)
+ mlog.debug(str(e))
mlog.debug(f'Unusable script {script!r}')
return None
- def _is_executable(self, path):
+ def _is_executable(self, path: str) -> bool:
suffix = os.path.splitext(path)[-1].lower()[1:]
execmask = stat.S_IXUSR | stat.S_IXGRP | stat.S_IXOTH
if mesonlib.is_windows():
@@ -217,7 +223,7 @@ class ExternalProgram:
return [trial_ext]
return None
- def _search_windows_special_cases(self, name: str, command: str) -> list:
+ def _search_windows_special_cases(self, name: str, command: str) -> T.List[T.Optional[str]]:
'''
Lots of weird Windows quirks:
1. PATH search for @name returns files with extensions from PATHEXT,
@@ -260,7 +266,7 @@ class ExternalProgram:
return commands
return [None]
- def _search(self, name: str, search_dir: T.Optional[str]) -> list:
+ def _search(self, name: str, search_dir: T.Optional[str]) -> T.List[T.Optional[str]]:
'''
Search in the specified dir for the specified executable by name
and if not found search in PATH
@@ -285,7 +291,7 @@ class ExternalProgram:
def get_command(self) -> T.List[str]:
return self.command[:]
- def get_path(self) -> str:
+ def get_path(self) -> T.Optional[str]:
return self.path
def get_name(self) -> str:
@@ -314,16 +320,16 @@ class EmptyExternalProgram(ExternalProgram): # lgtm [py/missing-call-to-init]
such as a cross file exe_wrapper to represent that it's not required.
'''
- def __init__(self):
+ def __init__(self) -> None:
self.name = None
self.command = []
self.path = None
- def __repr__(self):
+ def __repr__(self) -> str:
r = '<{} {!r} -> {!r}>'
return r.format(self.__class__.__name__, self.name, self.command)
- def found(self):
+ def found(self) -> bool:
return True
diff --git a/run_mypy.py b/run_mypy.py
index a31afb2..75c825e 100755
--- a/run_mypy.py
+++ b/run_mypy.py
@@ -38,6 +38,7 @@ modules = [
'mesonbuild/msetup.py',
'mesonbuild/mtest.py',
'mesonbuild/optinterpreter.py',
+ 'mesonbuild/programs.py',
'run_mypy.py',
'run_single_test.py',