"""SCons.Executor A module for executing actions with specific lists of target and source Nodes. """ # # Copyright (c) 2001, 2002, 2003, 2004 The SCons Foundation # # Permission is hereby granted, free of charge, to any person obtaining # a copy of this software and associated documentation files (the # "Software"), to deal in the Software without restriction, including # without limitation the rights to use, copy, modify, merge, publish, # distribute, sublicense, and/or sell copies of the Software, and to # permit persons to whom the Software is furnished to do so, subject to # the following conditions: # # The above copyright notice and this permission notice shall be included # in all copies or substantial portions of the Software. # # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY # KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE # WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE # LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION # OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION # WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. # __revision__ = "/home/scons/scons/branch.0/baseline/src/engine/SCons/Executor.py 0.96.1.D001 2004/08/23 09:55:29 knight" from SCons.Debug import logInstanceCreation import SCons.Util class Executor: """A class for controlling instances of executing an action. This largely exists to hold a single association of an action, environment, list of environment override dictionaries, targets and sources for later processing as needed. """ def __init__(self, action, env=None, overridelist=[], targets=[], sources=[]): if __debug__: logInstanceCreation(self) self.action = action self.env = env self.overridelist = overridelist self.targets = targets self.sources = sources[:] def get_build_env(self): """Fetch or create the appropriate build Environment for this Executor. """ try: return self.build_env except AttributeError: # Create the build environment instance with appropriate # overrides. These get evaluated against the current # environment's construction variables so that users can # add to existing values by referencing the variable in # the expansion. overrides = {} for odict in self.overridelist: overrides.update(odict) try: generate_build_dict = self.targets[0].generate_build_dict except (AttributeError, IndexError): pass else: overrides.update(generate_build_dict()) import SCons.Defaults env = self.env or SCons.Defaults.DefaultEnvironment() self.build_env = env.Override(overrides) # Now update the build environment with the things that we # don't want expanded against the current construction # variables. self.build_env._update(SCons.Util.subst_dict(self.targets, self.sources)) return self.build_env def get_action_list(self, target): """Fetch or create the appropriate action list (for this target). There is an architectural mistake here: we cache the action list for the Executor and re-use it regardless of which target is being asked for. In practice, this doesn't seem to be a problem because executing the action list will update all of the targets involved, so only one target's pre- and post-actions will win, anyway. This is probably a bug we should fix... """ try: al = self.action_list except AttributeError: al = self.action.get_actions() self.action_list = al try: # XXX shouldn't reach into node attributes like this return target.pre_actions + al + target.post_actions except AttributeError: return al def __call__(self, target, errfunc, **kw): """Actually execute the action list.""" action_list = self.get_action_list(target) if not action_list: return env = self.get_build_env() for action in action_list: apply(action, (self.targets, self.sources, env, errfunc), kw) def cleanup(self): try: del self.build_env except AttributeError: pass def add_sources(self, sources): """Add source files to this Executor's list. This is necessary for "multi" Builders that can be called repeatedly to build up a source file list for a given target.""" slist = filter(lambda x, s=self.sources: x not in s, sources) self.sources.extend(slist) def __str__(self): try: return self.string except AttributeError: action = self.action self.string = action.genstring(self.targets, self.sources, self.get_build_env()) return self.string def get_raw_contents(self): """Fetch the raw signature contents. This, along with get_contents(), is the real reason this class exists, so we can compute this once and cache it regardless of how many target or source Nodes there are. """ try: return self.raw_contents except AttributeError: action = self.action self.raw_contents = action.get_raw_contents(self.targets, self.sources, self.get_build_env()) return self.raw_contents def get_contents(self): """Fetch the signature contents. This, along with get_raw_contents(), is the real reason this class exists, so we can compute this once and cache it regardless of how many target or source Nodes there are. """ try: return self.contents except AttributeError: action = self.action self.contents = action.get_contents(self.targets, self.sources, self.get_build_env()) return self.contents def get_timestamp(self): """Fetch a time stamp for this Executor. We don't have one, of course (only files do), but this is the interface used by the timestamp module. """ return 0