diff options
Diffstat (limited to 'test/NodeOps.py')
| -rw-r--r-- | test/NodeOps.py | 350 |
1 files changed, 350 insertions, 0 deletions
diff --git a/test/NodeOps.py b/test/NodeOps.py new file mode 100644 index 0000000..5062b72 --- /dev/null +++ b/test/NodeOps.py @@ -0,0 +1,350 @@ +#!/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 VariantDir +# 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. + +import sys +import TestSCons +import os + +_exe = TestSCons._exe +lib_ = TestSCons.lib_ +_lib = TestSCons._lib +_obj = TestSCons._obj +dll_ = TestSCons.dll_ +_dll = TestSCons._dll + +if os.name == 'posix': + os.environ['LD_LIBRARY_PATH'] = '.' +if sys.platform.find('irix') > -1: + os.environ['LD_LIBRARYN32_PATH'] = '.' + +test = TestSCons.TestSCons() + +test.subdir('bld', 'src', ['src', 'subsrcdir']) + +sconstruct = r""" +foo = Environment(SHOBJPREFIX='', WINDOWS_INSERT_DEF=1) +foo.Append(SHCXXFLAGS = '-DFOO') +bar = Environment(SHOBJPREFIX='', WINDOWS_INSERT_DEF=1) +bar.Append(SHCXXFLAGS = '-DBAR') +src = Dir('src') +VariantDir('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 = [N.is_derived() for N in Nodes] + real1 = [os.path.exists(str(N)) for N in Nodes] + exists = [N.exists() for N in Nodes] + real2 = [os.path.exists(str(N)) for N in Nodes] + for N,D,R,E,F in zip(Nodes, derived, real1, exists, real2): + print '%%s: %%s %%s %%s %%s'%%(N,D,R,E,F) +foo.SharedLibrary(target = 'foo', source = 'foo%(_obj)s') +bar.SharedLibrary(target = 'bar', source = 'bar%(_obj)s') + +fooMain = foo.Clone(LIBS='foo', LIBPATH='.') +foo_obj = fooMain.Object(target='foomain', source='main.c') +fooMain.Program(target='fooprog', source=foo_obj) + +barMain = bar.Clone(LIBS='bar', LIBPATH='.') +bar_obj = barMain.Object(target='barmain', source='main.c') +barMain.Program(target='barprog', source=bar_obj) + +gooMain = foo.Clone(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" + +EXPORTS + doIt +""") + +test.write('bar.def', r""" +LIBRARY "bar" + +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 +} +""") + +sconscript = r""" +import os +Import('*') + +def mycopy(env, source, target): + open(str(target[0]),'w').write(open(str(source[0]),'r').read()) + +def exists_test(node): + before = os.path.exists(str(node)) # doesn't exist yet in VariantDir + via_node = node.exists() # side effect causes copy from src + after = os.path.exists(str(node)) + node.is_derived() + import SCons.Script + if GetOption('no_exec'): + if (before,via_node,after) != (False,False,False): + import sys + sys.stderr.write('VariantDir exists() populated during dryrun!\n') + sys.exit(-2) + else: + if (before,via_node,after) != (False,True,True): + import sys + sys.stderr.write('VariantDir exists() population did not occur! (%%s:%%s,%%s,%%s)\n'%%(str(node),before,via_node,after)) + sys.exit(-2) + +goo = Environment() +goo.Append(CFLAGS = '-DFOO') +goof_in = File('goof.in') +if %(_E)s: + exists_test(goof_in) +Nodes.append(goof_in) +Nodes.extend(goo.Command(target='goof.c', source='goof.in', action=mycopy)) +boo_src = File('subsrcdir/boo.c') +if %(_E)s: + exists_test(boo_src) +boo_objs = goo.Object(target='subsrcdir/boo%(_obj)s', source = boo_src) +Nodes.extend(boo_objs) +Nodes.extend(goo.Object(target='goo%(_obj)s',source='goof.c')) +goo.Library(target = 'goo', source = ['goo%(_obj)s'] + boo_objs) +""" + +test.write(['src', 'goof.in'], r""" +#include <stdio.h> + +extern char *boo_sub(); + +void +doIt() +{ +#ifdef FOO + printf("prog.cpp: %s\n", boo_sub()); +#endif +} +""") + +test.write(['src', 'subsrcdir', 'boo.c'], r""" +char * +boo_sub() +{ + return "GOO"; +} +""") + +test.write('main.c', r""" +void doIt(); + +int +main(int argc, char* argv[]) +{ + doIt(); + return 0; +} +""") + +builddir_srcnodes = [ os.path.join('bld', 'goof.in'), + os.path.join('bld', 'subsrcdir', 'boo.c'), + ] + +sub_build_nodes = [ os.path.join('bld', 'subsrcdir','boo' + _obj), + os.path.join('bld', 'goo' + _obj), + os.path.join('bld', 'goof.c'), + os.path.join('bld', lib_ + 'goo' + _lib), +] + +build_nodes = ['fooprog' + _exe, + dll_ + 'foo' + _dll, + 'foo' + _obj, + 'barprog' + _exe, + dll_ + 'bar' + _dll, + 'bar' + _obj, + + 'gooprog' + _exe, + + ] + builddir_srcnodes + sub_build_nodes + +def cleanup_test(): + "cleanup after running a test" + for F in builddir_srcnodes: + test.unlink(F) # will be repopulated during clean operation + test.run(arguments = '-c') + for F in builddir_srcnodes: + test.unlink(F) + for name in build_nodes: + test.must_not_exist(test.workpath(name)) + + +### First pass, make sure everything goes quietly + +for name in build_nodes: + test.must_not_exist(test.workpath(name)) + +_E=0 +test.write('SConstruct', sconstruct % locals() ) +test.write(['src', 'SConscript'], sconscript % 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.must_exist(test.workpath(name)) + +cleanup_test() + +### 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.must_not_exist(test.workpath(name)) + +_E=1 +test.write('SConstruct', sconstruct % locals() ) +test.write(['src', 'SConscript'], sconscript % 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.must_exist(test.workpath(name)) + +cleanup_test() + +### Next pass: try a dry-run first and verify that it doesn't change +### the buildability. + +for name in build_nodes: + test.must_not_exist(test.workpath(name)) + +_E=1 +test.write('SConstruct', sconstruct % locals() ) +test.write(['src', 'SConscript'], sconscript % locals() ) + +test.run(arguments = '-n .', + stderr=TestSCons.noisy_ar, + match=TestSCons.match_re_dotall) + +for name in build_nodes: + test.must_not_exist(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.must_exist(test.workpath(name)) + +cleanup_test() + +### Next pass: do an up-build from a VariantDir src + + +for name in build_nodes: + test.must_not_exist(test.workpath(name)) + +_E=0 +test.write('SConstruct', sconstruct % locals() ) +test.write(['src', 'SConscript'], sconscript % locals() ) + +test.run(chdir='src', arguments = '-u', + stderr=TestSCons.noisy_ar, + match=TestSCons.match_re_dotall) + +for name in build_nodes: + if name in sub_build_nodes or name in builddir_srcnodes: + test.must_exist(test.workpath(name)) + else: + test.must_not_exist(test.workpath(name)) + +cleanup_test() + +### Next pass: do an up-build from a VariantDir src with Node Ops +### side-effects + +for name in build_nodes: + test.must_not_exist(test.workpath(name)) + +_E=1 +test.write('SConstruct', sconstruct % locals() ) +test.write(['src', 'SConscript'], sconscript % locals() ) + +test.run(chdir='src', arguments = '-u', + stderr=TestSCons.noisy_ar, + match=TestSCons.match_re_dotall) + +for name in build_nodes: + if name in sub_build_nodes or name in builddir_srcnodes: + test.must_exist(test.workpath(name)) + else: + test.must_not_exist(test.workpath(name)) + +cleanup_test() + +test.pass_test() + +# Local Variables: +# tab-width:4 +# indent-tabs-mode:nil +# End: +# vim: set expandtab tabstop=4 shiftwidth=4: |
