Fix is_pseudo_derived_builder(). (Kevin Quick)
authorstevenknight <stevenknight@fdb21ef1-2011-0410-befe-b5e4ea1792b1>
Wed, 15 Sep 2004 04:12:13 +0000 (04:12 +0000)
committerstevenknight <stevenknight@fdb21ef1-2011-0410-befe-b5e4ea1792b1>
Wed, 15 Sep 2004 04:12:13 +0000 (04:12 +0000)
git-svn-id: http://scons.tigris.org/svn/scons/trunk@1070 fdb21ef1-2011-0410-befe-b5e4ea1792b1

src/CHANGES.txt
src/engine/SCons/Node/FS.py
src/engine/SCons/Node/FSTests.py
test/NodeOps.py [new file with mode: 0644]

index feb988a236157b545ed2518edcb671a1c9ded239..814107bda5dee3f2abeb66debebfe2e1b16d0933 100644 (file)
@@ -63,6 +63,8 @@ RELEASE 0.97 - XXX
     module prior to Python 2.3, which would generate an exception for
     a nonexistent file even when ignore_errors was set.)
 
+  - Only replace a Node's builder with a non-null source builder.
+
   From Christoph Wiedemann:
 
   - Add an Environment.SetDefault() method that only sets values if
index 6d2dd8b43a4ab036b5f99d7370004bb279fa96b2..b59c375e464a04de4ff1af21a03357f32d03c26b 100644 (file)
@@ -1618,7 +1618,8 @@ class File(Base):
                             rcspath = os.path.join(dir, rcspath)
                         if os.path.exists(rcspath):
                             scb = get_DefaultRCSBuilder()
-                self.builder = scb
+                if scb is not None:
+                    self.builder_set(scb)
             self.sbuilder = scb
         return not scb is None
 
index 3e803096babf024c8256783f4b7b809f6e76cdc0..4c086ddd2fa49ca2099c50be301d24929c97b67e 100644 (file)
@@ -1426,6 +1426,7 @@ class has_src_builderTestCase(unittest.TestCase):
         f5 = fs.File('f5', sub2)
         f6 = fs.File('f6', sub2)
         f7 = fs.File('f7', sub2)
+        f8 = fs.File('f8', sub2)
 
         h = f1.has_src_builder()
         assert not h, h
@@ -1451,6 +1452,7 @@ class has_src_builderTestCase(unittest.TestCase):
         assert f3.builder is b1, f3.builder
 
         f7.set_src_builder(b1)
+        f8.builder_set(b1)
 
         test.write(['sub2', 'SCCS', 's.f5'], "sub2/SCCS/s.f5\n")
         test.write(['sub2', 'RCS', 'f6,v'], "sub2/RCS/f6,v\n")
@@ -1470,6 +1472,10 @@ class has_src_builderTestCase(unittest.TestCase):
         assert h, h
         h = f7.has_builder()
         assert h, h
+        h = f8.has_src_builder()
+        assert not h, h
+        h = f8.has_builder()
+        assert h, h
 
 class prepareTestCase(unittest.TestCase):
     def runTest(self):
diff --git a/test/NodeOps.py b/test/NodeOps.py
new file mode 100644 (file)
index 0000000..e884400
--- /dev/null
@@ -0,0 +1,279 @@
+#!/usr/bin/env python
+#
+# __COPYRIGHT__
+#
+# 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__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+
+# This test is used to verify that the Buildability of a set of nodes
+# is unaffected by various querying operations on those nodes:
+#
+# 1) Calling exists() on a Node (e.g. from find_file) in a BuildDir
+#    will cause that node to be duplicated into the builddir.
+#    However, this should *not* occur during a dryrun (-n).  When not
+#    performed during a dryrun, this should not affect buildability.
+# 2) Calling is_derived() should not affect buildability.
+# 3) Calling is_pseudo_derived() may cause the sbuilder to be set, and
+#    it may caues the builder to be set as well, but it should not
+#    adversely affect buildability.
+
+import sys
+import TestSCons
+import os
+import string
+
+if sys.platform == 'win32':
+    _obj = '.obj'
+    fooflags = '/nologo -DFOO'
+    barflags = '/nologo -DBAR'
+else:
+    _obj = '.o'
+    fooflags = '-DFOO'
+    barflags = '-DBAR'
+    
+if os.name == 'posix':
+    os.environ['LD_LIBRARY_PATH'] = '.'
+if string.find(sys.platform, 'irix') > -1:
+    os.environ['LD_LIBRARYN32_PATH'] = '.'
+
+test = TestSCons.TestSCons()
+
+test.subdir('bld', 'src')
+
+sconstruct = r"""
+foo = Environment(SHCXXFLAGS = '%(fooflags)s', WIN32_INSERT_DEF=1)
+bar = Environment(SHCXXFLAGS = '%(barflags)s', WIN32_INSERT_DEF=1)
+src = Dir('src')
+BuildDir('bld', src, duplicate=1)
+Nodes=[]
+Nodes.extend(foo.SharedObject(target = 'foo%(_obj)s', source = 'prog.cpp'))
+Nodes.extend(bar.SharedObject(target = 'bar%(_obj)s', source = 'prog.cpp'))
+SConscript('bld/SConscript', ['Nodes'])
+if %(_E)s:
+  import os
+  derived = map(lambda N: N.is_derived(), Nodes)
+  p_derived = map(lambda N: N.is_pseudo_derived(), Nodes)
+  real1 = map(lambda N: os.path.exists(str(N)), Nodes)
+  exists = map(lambda N: N.exists(), Nodes)
+  real2 = map(lambda N: os.path.exists(str(N)), Nodes)
+  for N,D,P,R,E,F in map(None, Nodes, derived, p_derived,
+                               real1, exists, real2):
+    print '%%s: %%s %%s %%s %%s %%s'%%(N,D,P,R,E,F)
+foo.SharedLibrary(target = 'foo', source = 'foo%(_obj)s')
+bar.SharedLibrary(target = 'bar', source = 'bar%(_obj)s')
+
+fooMain = foo.Copy(LIBS='foo', LIBPATH='.')
+foo_obj = fooMain.Object(target='foomain', source='main.c')
+fooMain.Program(target='fooprog', source=foo_obj)
+
+barMain = bar.Copy(LIBS='bar', LIBPATH='.')
+bar_obj = barMain.Object(target='barmain', source='main.c')
+barMain.Program(target='barprog', source=bar_obj)
+
+gooMain = foo.Copy(LIBS='goo', LIBPATH='bld')
+goo_obj = gooMain.Object(target='goomain', source='main.c')
+gooMain.Program(target='gooprog', source=goo_obj)
+"""
+           
+
+test.write('foo.def', r"""
+LIBRARY        "foo"
+DESCRIPTION    "Foo Shared Library"
+
+EXPORTS
+   doIt
+""")
+
+test.write('bar.def', r"""
+LIBRARY        "bar"
+DESCRIPTION    "Bar Shared Library"
+
+EXPORTS
+   doIt
+""")
+
+test.write('prog.cpp', r"""
+#include <stdio.h>
+
+extern "C" void
+doIt()
+{
+#ifdef FOO
+       printf("prog.cpp:  FOO\n");
+#endif
+#ifdef BAR
+       printf("prog.cpp:  BAR\n");
+#endif
+}
+""")
+
+test.write(['src', 'SConscript'], r"""
+import os
+Import('*')
+
+# Interoperability with old Python versions.
+True, False = 1, 0
+
+def mycopy(env, source, target):
+    open(str(target[0]),'w').write(open(str(source[0]),'r').read())
+    
+goo = Environment(CPPFLAGS = '%(fooflags)s')
+Nodes.append(File('goof.in'))
+before = os.path.exists('goof.in')  # doesn't exist yet in BuildDir
+via_node = Nodes[-1].exists()       # side effect causes copy from src
+after = os.path.exists('goof.in')
+import SCons.Script
+if SCons.Script.options.noexec:
+    if (before,via_node,after) != (False,False,False):
+        import sys
+        sys.stderr.write('BuildDir exits() populated during dryrun!\n')
+        sys.exit(-2)
+else:
+    if (before,via_node,after) != (False,True,True):
+        import sys
+        sys.stderr.write('BuildDir exists() population did not occur!\n'%%(before,via_node,after))
+        sys.exit(-2)
+Nodes.extend(goo.Command(target='goof.c', source='goof.in', action=mycopy))
+Nodes.extend(goo.Object(target='goo%(_obj)s',source='goof.c'))
+goo.Library(target = 'goo', source = 'goo%(_obj)s')
+""" % locals() )
+
+test.write(['src', 'goof.in'], r"""
+#include <stdio.h>
+
+void
+doIt()
+{
+#ifdef FOO
+       printf("prog.cpp:  GOO\n");
+#endif
+}
+""")
+
+test.write('main.c', r"""
+
+void doIt();
+
+int
+main(int argc, char* argv[])
+{
+    doIt();
+    return 0;
+}
+""")
+
+build_nodes = ['fooprog', 'libfoo.so', 'foo.o',
+               'barprog', 'libbar.so', 'bar.o',
+
+               'gooprog',
+               os.path.join('bld','goo.o'),
+               os.path.join('bld','goof.c'),
+               os.path.join('bld','goof.in'),
+
+               ]
+
+### First pass, make sure everything goes quietly
+
+for name in build_nodes:
+    test.fail_test(os.path.exists(test.workpath(name)))
+
+_E=0
+test.write('SConstruct', sconstruct % locals() )
+
+test.run(arguments = '.',
+         stderr=TestSCons.noisy_ar,
+         match=TestSCons.match_re_dotall)
+
+test.run(program = test.workpath('fooprog'), stdout = "prog.cpp:  FOO\n")
+test.run(program = test.workpath('barprog'), stdout = "prog.cpp:  BAR\n")
+test.run(program = test.workpath('gooprog'), stdout = "prog.cpp:  GOO\n")
+
+for name in build_nodes:
+    test.fail_test(not os.path.exists(test.workpath(name)))
+
+test.unlink('bld/goof.in')  # will be repopulated during clean operation
+test.run(arguments = '-c')
+test.unlink('bld/goof.in')
+for name in build_nodes:
+    test.fail_test(os.path.exists(test.workpath(name)))
+
+### Next pass: add internal Node ops that may have side effects to
+### ensure that those side-effects don't interfere with building
+
+for name in build_nodes:
+    test.fail_test(os.path.exists(test.workpath(name)))
+
+_E=1
+test.write('SConstruct', sconstruct % locals() )
+
+test.run(arguments = '.',
+         stderr=TestSCons.noisy_ar,
+         match=TestSCons.match_re_dotall)
+
+test.run(program = test.workpath('fooprog'), stdout = "prog.cpp:  FOO\n")
+test.run(program = test.workpath('barprog'), stdout = "prog.cpp:  BAR\n")
+test.run(program = test.workpath('gooprog'), stdout = "prog.cpp:  GOO\n")
+
+for name in build_nodes:
+    test.fail_test(not os.path.exists(test.workpath(name)))
+
+test.unlink('bld/goof.in')  # will be repopulated during clean operation
+test.run(arguments = '-c')
+test.unlink('bld/goof.in')
+for name in build_nodes:
+    test.fail_test(os.path.exists(test.workpath(name)))
+
+### Next pass: try a dry-run first and verify that it doesn't change
+### the buildability.
+
+for name in build_nodes:
+    test.fail_test(os.path.exists(test.workpath(name)))
+
+_E=1
+test.write('SConstruct', sconstruct % locals() )
+
+test.run(arguments = '-n .',
+         stderr=TestSCons.noisy_ar,
+         match=TestSCons.match_re_dotall)
+
+for name in build_nodes:
+    test.fail_test(os.path.exists(test.workpath(name)))
+
+test.run(arguments = '.',
+         stderr=TestSCons.noisy_ar,
+         match=TestSCons.match_re_dotall)
+
+test.run(program = test.workpath('fooprog'), stdout = "prog.cpp:  FOO\n")
+test.run(program = test.workpath('barprog'), stdout = "prog.cpp:  BAR\n")
+test.run(program = test.workpath('gooprog'), stdout = "prog.cpp:  GOO\n")
+
+for name in build_nodes:
+    test.fail_test(not os.path.exists(test.workpath(name)))
+
+test.unlink('bld/goof.in')  # will be repopulated during clean operation
+test.run(arguments = '-c')
+test.unlink('bld/goof.in')
+for name in build_nodes:
+    test.fail_test(os.path.exists(test.workpath(name)))
+
+
+test.pass_test()