diff options
| author | Dylan Baker <dylan@pnwbakers.com> | 2020-04-16 21:42:38 -0700 | 
|---|---|---|
| committer | Dylan Baker <dylan@pnwbakers.com> | 2020-04-30 10:01:14 -0700 | 
| commit | 2c0eaf5c4f4493146355eeb8521c17a3c2ef5acd (patch) | |
| tree | d77467eafcd7ff4432cd65fa204369809a29476b /mesonbuild/interpreter.py | |
| parent | cd566d2bd5f2f1faa3576b51b6b47c74a7ed0392 (diff) | |
| download | meson-2c0eaf5c4f4493146355eeb8521c17a3c2ef5acd.tar.gz | |
interpreter: Allow install_script to use additional input types
This adds support for Files, CustomTarget, Indexs of CustomTargets,
ConfigureFiles, ExternalPrograms, and Executables.
Fixes: #1234
Fixes: #3552
Fixes: #6175
Diffstat (limited to 'mesonbuild/interpreter.py')
| -rw-r--r-- | mesonbuild/interpreter.py | 70 | 
1 files changed, 59 insertions, 11 deletions
| diff --git a/mesonbuild/interpreter.py b/mesonbuild/interpreter.py index dd1e57bad..2b699f8fa 100644 --- a/mesonbuild/interpreter.py +++ b/mesonbuild/interpreter.py @@ -1893,31 +1893,79 @@ class MesonMain(InterpreterObject):                               'backend': self.backend_method,                               }) -    def _find_source_script(self, name, args): +    def _find_source_script(self, prog: T.Union[str, ExecutableHolder], args): +        if isinstance(prog, ExecutableHolder): +            prog_path = self.interpreter.backend.get_target_filename(prog.held_object) +            return build.RunScript([prog_path], args) +        elif isinstance(prog, ExternalProgramHolder): +            return build.RunScript(prog.get_command(), args) +          # Prefer scripts in the current source directory          search_dir = os.path.join(self.interpreter.environment.source_dir,                                    self.interpreter.subdir) -        key = (name, search_dir) +        key = (prog, search_dir)          if key in self._found_source_scripts:              found = self._found_source_scripts[key]          else: -            found = dependencies.ExternalProgram(name, search_dir=search_dir) +            found = dependencies.ExternalProgram(prog, search_dir=search_dir)              if found.found():                  self._found_source_scripts[key] = found              else:                  m = 'Script or command {!r} not found or not executable' -                raise InterpreterException(m.format(name)) +                raise InterpreterException(m.format(prog))          return build.RunScript(found.get_command(), args) -    @permittedKwargs({}) -    def add_install_script_method(self, args, kwargs): +    def _process_script_args( +            self, name: str, args: T.List[T.Union[ +                str, mesonlib.File, CustomTargetHolder, +                CustomTargetIndexHolder, ConfigureFileHolder, +                ExternalProgramHolder, ExecutableHolder, +            ]]) -> T.List[str]: +        script_args = []  # T.List[str] +        new = False +        for a in args: +            a = unholder(a) +            if isinstance(a, str): +                script_args.append(a) +            elif isinstance(a, mesonlib.File): +                new = True +                script_args.append(a.rel_to_builddir(self.interpreter.environment.source_dir)) +            elif isinstance(a, (build.BuildTarget, build.CustomTarget, build.CustomTargetIndex)): +                new = True +                script_args.extend([os.path.join(a.get_subdir(), o) for o in a.get_outputs()]) + +                # This feels really hacky, but I'm not sure how else to fix +                # this without completely rewriting install script handling. +                # This is complicated by the fact that the install target +                # depends on all. +                if isinstance(a, build.CustomTargetIndex): +                    a.target.build_by_default = True +                else: +                    a.build_by_default = True +            elif isinstance(a, build.ConfigureFile): +                new = True +                script_args.append(os.path.join(a.subdir, a.targetname)) +            elif isinstance(a, dependencies.ExternalProgram): +                script_args.extend(a.command) +                new = True +            else: +                raise InterpreterException( +                    'Arguments to {} must be strings, Files, CustomTargets, ' +                    'Indexes of CustomTargets, or ConfigureFiles'.format(name)) +        if new: +            FeatureNew('Calling "{}" with File, CustomTaget, Index of CustomTarget, ConfigureFile, Executable, or ExternalProgram'.format(name), '0.55.0').use( +                self.interpreter.subproject) +        return script_args + +    @permittedKwargs(set()) +    def add_install_script_method(self, args: 'T.Tuple[T.Union[str, ExecutableHolder], T.Union[str, mesonlib.File, CustomTargetHolder, CustomTargetIndexHolder, ConfigureFileHolder], ...]', kwargs):          if len(args) < 1:              raise InterpreterException('add_install_script takes one or more arguments') -        check_stringlist(args, 'add_install_script args must be strings') -        script = self._find_source_script(args[0], args[1:]) +        script_args = self._process_script_args('add_install_script', args[1:]) +        script = self._find_source_script(args[0], script_args)          self.build.install_scripts.append(script) -    @permittedKwargs({}) +    @permittedKwargs(set())      def add_postconf_script_method(self, args, kwargs):          if len(args) < 1:              raise InterpreterException('add_postconf_script takes one or more arguments') @@ -1925,13 +1973,13 @@ class MesonMain(InterpreterObject):          script = self._find_source_script(args[0], args[1:])          self.build.postconf_scripts.append(script) -    @permittedKwargs({}) +    @permittedKwargs(set())      def add_dist_script_method(self, args, kwargs):          if len(args) < 1:              raise InterpreterException('add_dist_script takes one or more arguments')          if len(args) > 1:              FeatureNew('Calling "add_dist_script" with multiple arguments', '0.49.0').use(self.interpreter.subproject) -        check_stringlist(args, 'add_dist_script argument must be a string') +        check_stringlist(args, 'add_dist_script argumetn must be a string')          if self.interpreter.subproject != '':              raise InterpreterException('add_dist_script may not be used in a subproject.')          script = self._find_source_script(args[0], args[1:]) | 
