def __init__(self, num):
"""Create the request and reply queues, and 'num' worker threads."""
- # Ideally we wouldn't have to artificially limit the number of
- # tasks that can be posted to the request queue. But this can
- # result in a large number of pending tasks, which at the time
- # of this writing causes the taskmaster's next_task method to
- # take a very long time.
- self.requestQueue = Queue.Queue(num)
+ self.requestQueue = Queue.Queue()
self.resultsQueue = Queue.Queue()
# Create worker threads
self.taskmaster = taskmaster
self.tp = ThreadPool(num)
+ self.jobs = 0
+ self.maxjobs = num
+
def start(self):
"""Start the job. This will begin pulling tasks from the
taskmaster and executing them, and return when there are no
an exception), then the job will stop."""
while 1:
- task = self.taskmaster.next_task()
- if task is None:
- break
+ if self.jobs < self.maxjobs:
+ task = self.taskmaster.next_task()
+ if task is None:
+ break
- # prepare task for execution
- try:
- task.prepare()
- except KeyboardInterrupt:
- raise
- except:
- # Let the failed() callback function arrange for the
- # build to stop if that's appropriate.
- task.failed()
+ # prepare task for execution
+ try:
+ task.prepare()
+ except KeyboardInterrupt:
+ raise
+ except:
+ # Let the failed() callback function arrange for the
+ # build to stop if that's appropriate.
+ task.failed()
- # dispatch task
- self.tp.put(task)
+ # dispatch task
+ self.tp.put(task)
+ self.jobs = self.jobs + 1
while 1:
try:
task, ok = self.tp.get_nowait()
except Queue.Empty:
- if not self.taskmaster.is_blocked():
+ if not (self.jobs is self.maxjobs or self.taskmaster.is_blocked()):
break
task, ok = self.tp.get()
+ self.jobs = self.jobs - 1
if ok:
task.executed()
else:
"""Explicit stop-the-build failure."""
for t in self.targets:
t.set_state(SCons.Node.failed)
+ self.tm.failed(self.node)
self.tm.stop()
def fail_continue(self):
self.ready = None
self.pending = []
+ def failed(self, node):
+ try:
+ tlist = node.builder.targets(node)
+ except AttributeError:
+ tlist = [node]
+ for t in tlist:
+ self.executing.remove(t)
+ for side_effect in node.side_effects:
+ self.executing.remove(side_effect)
+
def executed(self, node):
try:
tlist = node.builder.targets(node)
assert built_text == "MyTM.stop()"
assert tm.next_task() is None
+ def test_failed(self):
+ """Test when a task has failed
+ """
+ n1 = Node("n1")
+ tm = SCons.Taskmaster.Taskmaster([n1])
+ t = tm.next_task()
+ assert tm.executing == [n1], tm.executing
+ tm.failed(n1)
+ assert tm.executing == [], tm.executing
+
def test_executed(self):
"""Test when a task has been executed
"""
__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+import os.path
import string
import sys
import TestSCons
for t in target:
shutil.copy(str(source[0]), str(t))
-t = env.Command(target=['foo/foo1.out', 'foo/foo2.out'], source='foo/foo.in', action=copyn)
+t = env.Command(target=['foo/foo1.out', 'foo/foo2.out'],
+ source='foo/foo.in',
+ action=copyn)
env.Install('out', t)
""" % python)
test.fail_test(start2 < finish1)
# Make sure that a parallel build using a list builder
-# succeedes.
+# succeeds.
test.run(arguments='-j 2 out')
+
+# Test that a failed build with -j works properly.
+
+
# Test SetJobs() with no -j:
test.write('SConstruct', """
MyBuild = Builder(action = r'%s build.py $TARGETS')
test.fail_test(start2 < finish1)
+# Test that a failed build with -j works properly.
+
+test.write('copy.py', r"""\
+import sys
+import time
+time.sleep(1)
+open(sys.argv[1], 'w').write(open(sys.argv[2], 'r').read())
+""")
+
+test.write('fail.py', r"""\
+import sys
+sys.exit(1)
+""")
+
+test.write('SConstruct', """
+MyCopy = Builder(action = r'%s copy.py $TARGET $SOURCE')
+Fail = Builder(action = r'%s fail.py $TARGETS $SOURCE')
+env = Environment(BUILDERS = { 'MyCopy' : MyCopy, 'Fail' : Fail })
+env.Fail(target = 'f3', source = 'f3.in')
+env.MyCopy(target = 'f4', source = 'f4.in')
+env.MyCopy(target = 'f5', source = 'f5.in')
+env.MyCopy(target = 'f6', source = 'f6.in')
+""" % (python, python))
+
+test.write('f3.in', "f3.in\n")
+test.write('f4.in', "f4.in\n")
+test.write('f5.in', "f5.in\n")
+test.write('f6.in', "f6.in\n")
+
+test.run(arguments = '-j 2 .',
+ status = 2,
+ stderr = "scons: *** [f3] Error 1\n")
+
+test.fail_test(os.path.exists(test.workpath('f3')))
+test.fail_test(test.read(test.workpath('f4')) != 'f4.in\n')
+test.fail_test(os.path.exists(test.workpath('f5')))
+test.fail_test(os.path.exists(test.workpath('f6')))
+
test.pass_test()
-