calc = SCons.Sig.Calculator(self.module)
for node in nodes:
- self.failUnless(not current(calc, node), "none of the nodes should be current")
+ self.failUnless(not current(calc, node),
+ "none of the nodes should be current")
# simulate a build:
self.files[1].modify('built', 222)
write(calc, nodes)
for node in nodes:
- self.failUnless(current(calc, node), "all of the nodes should be current")
+ self.failUnless(current(calc, node),
+ "all of the nodes should be current")
def test_modify(self):
self.failUnless(current(calc, nodes[4]))
self.failUnless(current(calc, nodes[5]))
self.failUnless(not current(calc, nodes[6]), "modified directly")
- self.failUnless(not current(calc, nodes[7]), "indirect source modified")
+ self.failUnless(current(calc, nodes[7]), "indirect source modified")
self.failUnless(not current(calc, nodes[8]), "modified directory")
self.failUnless(not current(calc, nodes[9]), "direct source modified")
self.failUnless(not current(calc, nodes[10]), "indirect source modified")
self.failUnless(current(calc, nodes[8]))
self.failUnless(not current(calc, nodes[9]), "deleted")
self.failUnless(current(calc, nodes[10]),
- "current even though it's source was deleted")
+ "current even though its source was deleted")
class MD5TestCase(unittest.TestCase, SigTestBase):
"""Test MD5 signatures"""
module = SCons.Sig.TimeStamp
+class CalcTestCase(unittest.TestCase):
+
+ def runTest(self):
+ class MySigModule:
+ def collect(self, signatures):
+ return reduce(lambda x, y: x + y, signatures)
+ def current(self, newsig, oldsig):
+ return newsig == oldsig
+ def signature(self, node):
+ return node.get_signature()
+
+ class MyNode:
+ def __init__(self, name, sig):
+ self.name = name
+ self.sig = sig
+ self.kids = []
+ self.builder = None
+ self.use_signature = 1
+ def children(self):
+ return self.kids
+ def has_signature(self):
+ return self.sig != None
+ def get_signature(self):
+ return self.sig
+ def get_oldentry(self):
+ return 0, self.sig
+ def get_timestamp(self):
+ return 1
+
+ self.module = MySigModule()
+ self.nodeclass = MyNode
+ self.test_Calc___init__()
+ self.test_Calc_collect()
+ self.test_Calc_get_signature()
+ self.test_Calc_current()
+
+ def test_Calc___init__(self):
+ self.calc = SCons.Sig.Calculator(self.module)
+ assert self.calc.module == self.module
+
+ def test_Calc_collect(self):
+ n1 = self.nodeclass('n1', 11)
+ n2 = self.nodeclass('n2', 22)
+ n3 = self.nodeclass('n3', 33)
+ n1.builder = 1
+ n1.kids = [n2, n3]
+
+ assert self.calc.collect(n1) == 55
+
+ def test_Calc_get_signature(self):
+ class NE(self.nodeclass):
+ def exists(self):
+ return 0
+ def has_signature(self):
+ return None
+ class NN(self.nodeclass):
+ def exists(self):
+ return 1
+ def has_signature(self):
+ return None
+
+ n1 = self.nodeclass('n1', 11)
+ n1.use_signature = 0
+ assert self.calc.get_signature(n1) is None
+
+ n2 = self.nodeclass('n2', 22)
+ assert self.calc.get_signature(n2) == 22
+
+ n3 = self.nodeclass('n3', 33)
+ n4 = self.nodeclass('n4', None)
+ n4.builder = 1
+ n4.kids = [n2, n3]
+ assert self.calc.get_signature(n4) == 55
+
+ n5 = NE('n5', 55)
+ assert self.calc.get_signature(n5) is None
+
+ n6 = NN('n6', 66)
+ assert self.calc.get_signature(n6) == 66
+
+ def test_Calc_current(self):
+ class N0(self.nodeclass):
+ def current(self):
+ return 0
+ class N1(self.nodeclass):
+ def current(self):
+ return 1
+ class NN(self.nodeclass):
+ def current(self):
+ return None
+
+ n0 = N0('n0', 11)
+ assert not self.calc.current(n0, 10)
+ assert not self.calc.current(n0, 11)
+
+ n1 = N1('n1', 22)
+ assert self.calc.current(n1, 20)
+ assert self.calc.current(n1, 22)
+
+ nn = NN('nn', 33)
+ assert not self.calc.current(nn, 30)
+ assert self.calc.current(nn, 33)
+
def suite():
suite = unittest.TestSuite()
suite.addTest(MD5TestCase())
suite.addTest(TimeStampTestCase())
+ suite.addTest(CalcTestCase())
return suite
if __name__ == "__main__":
self.module = module
- def collect(self, node, signatures):
+ def collect(self, node):
"""
- Collect the signatures of the node's sources.
+ Collect the signatures of a node's sources.
node - the node whose sources will be collected
- signatures - the dictionary that the signatures will be
- gathered into.
+
+ This no longer handles the recursive descent of the
+ node's children's signatures. We expect that they're
+ already built and updated by someone else, if that's
+ what's wanted.
"""
- for source_node in node.children():
- if not signatures.has_key(source_node):
- signature = self.get_signature(source_node)
- signatures[source_node] = signature
- self.collect(source_node, signatures)
+ sigs = map(lambda n,s=self: s.get_signature(n), node.children())
+ return self.module.collect(filter(lambda x: not x is None, sigs))
def get_signature(self, node):
"""
elif node.has_signature():
sig = node.get_signature()
elif node.builder:
- signatures = {}
- self.collect(node, signatures)
- signatures = filter(lambda x: not x is None, signatures.values())
- sig = self.module.collect(signatures)
+ sig = self.collect(node)
else:
if not node.exists():
return None
test.up_to_date(arguments = '.')
-# make sure the programs don't get rebuilt, because nothing changed:
+# make sure the programs didn't get rebuilt, because nothing changed:
oldtime1 = os.path.getmtime(test.workpath('foo1'))
oldtime2 = os.path.getmtime(test.workpath('foo2'))
oldtime3 = os.path.getmtime(test.workpath('foo3'))
+
time.sleep(2) # introduce a small delay, to make the test valid
-test.run(arguments = '.')
+
+test.run(arguments = 'foo1 foo2 foo3')
+
+test.fail_test(not (oldtime1 == os.path.getmtime(test.workpath('foo1'))))
+test.fail_test(not (oldtime2 == os.path.getmtime(test.workpath('foo2'))))
+test.fail_test(not (oldtime3 == os.path.getmtime(test.workpath('foo3'))))
+
+test.write('f1.c', """
+int
+main(int argc, char *argv[])
+{
+ argv[argc++] = "--";
+ printf("f1.c Y\n");
+ exit (0);
+}
+""")
+
+test.write('f3b.c', """
+void
+f3b(void)
+{
+ printf("f3b.c Y\n");
+}
+""")
+
+test.run(arguments = 'foo1 foo2 foo3')
+
+test.run(program = test.workpath('foo1'), stdout = "f1.c Y\n")
+test.run(program = test.workpath('foo2'), stdout = "f2a.c\nf2b.c\nf2c.c\n")
+test.run(program = test.workpath('foo3'), stdout = "f3a.c\nf3b.c Y\nf3c.c\n")
+
+test.up_to_date(arguments = 'foo1 foo2 foo3')
+
+test.write('f1.c', """
+int
+main(int argc, char *argv[])
+{
+ argv[argc++] = "--";
+ printf("f1.c Z\n");
+ exit (0);
+}
+""")
+
+test.write('f3b.c', """
+void
+f3b(void)
+{
+ printf("f3b.c Z\n");
+}
+""")
+
+test.run(arguments = 'foo1 foo2 foo3')
+
+test.run(program = test.workpath('foo1'), stdout = "f1.c Z\n")
+test.run(program = test.workpath('foo2'), stdout = "f2a.c\nf2b.c\nf2c.c\n")
+test.run(program = test.workpath('foo3'), stdout = "f3a.c\nf3b.c Z\nf3c.c\n")
+
+test.up_to_date(arguments = 'foo1 foo2 foo3')
+
+# make sure the programs didn't get rebuilt, because nothing changed:
+oldtime1 = os.path.getmtime(test.workpath('foo1'))
+oldtime2 = os.path.getmtime(test.workpath('foo2'))
+oldtime3 = os.path.getmtime(test.workpath('foo3'))
+
+time.sleep(2) # introduce a small delay, to make the test valid
+
+test.run(arguments = 'foo1 foo2 foo3')
+
test.fail_test(not (oldtime1 == os.path.getmtime(test.workpath('foo1'))))
test.fail_test(not (oldtime2 == os.path.getmtime(test.workpath('foo2'))))
test.fail_test(not (oldtime3 == os.path.getmtime(test.workpath('foo3'))))