From 87074b5ab10857123de64d6d535fd24a4120c8a3 Mon Sep 17 00:00:00 2001 From: stevenknight Date: Wed, 15 Sep 2004 01:19:55 +0000 Subject: [PATCH] Handle exceptions in FunctionActions. (Steve Christensen) git-svn-id: http://scons.tigris.org/svn/scons/trunk@1068 fdb21ef1-2011-0410-befe-b5e4ea1792b1 --- etc/TestCmd.py | 16 ++++++++------ etc/TestCommon.py | 4 ++-- src/engine/SCons/Action.py | 7 +++++- src/engine/SCons/Node/FS.py | 12 ++++------- src/engine/SCons/Node/FSTests.py | 2 +- src/engine/SCons/Script/__init__.py | 5 ++++- test/build-errors.py | 33 +++++++++++++++++++++++++++++ www/index.html | 15 +++++++++---- www/project_highlights.html | 4 ++-- 9 files changed, 73 insertions(+), 25 deletions(-) diff --git a/etc/TestCmd.py b/etc/TestCmd.py index 95333981..cc73dd1e 100644 --- a/etc/TestCmd.py +++ b/etc/TestCmd.py @@ -175,8 +175,8 @@ version. # SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS. __author__ = "Steven Knight " -__revision__ = "TestCmd.py 0.8.D001 2004/07/15 06:24:14 knight" -__version__ = "0.8" +__revision__ = "TestCmd.py 0.11.D001 2004/09/13 13:22:19 knight" +__version__ = "0.11" import os import os.path @@ -860,10 +860,14 @@ class TestCmd: f = _mode_writable else: f = _mode_non_writable - try: - os.path.walk(top, _walk_chmod, f) - except: - pass # ignore any problems changing modes + if os.path.isfile(top): + st = os.stat(top) + os.chmod(top, f(st[stat.ST_MODE])) + else: + try: + os.path.walk(top, _walk_chmod, f) + except: + pass # ignore any problems changing modes def write(self, file, content, mode = 'wb'): """Writes the specified content text (second argument) to the diff --git a/etc/TestCommon.py b/etc/TestCommon.py index 43cfdddf..924f8f8b 100644 --- a/etc/TestCommon.py +++ b/etc/TestCommon.py @@ -76,8 +76,8 @@ The TestCommon module also provides the following variables # SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS. __author__ = "Steven Knight " -__revision__ = "TestCommon.py 0.10.D001 2004/08/17 17:39:41 knight" -__version__ = "0.10" +__revision__ = "TestCommon.py 0.11.D001 2004/09/13 13:22:19 knight" +__version__ = "0.11" import os import os.path diff --git a/src/engine/SCons/Action.py b/src/engine/SCons/Action.py index 63d8c987..e110d609 100644 --- a/src/engine/SCons/Action.py +++ b/src/engine/SCons/Action.py @@ -497,7 +497,12 @@ class FunctionAction(ActionBase): def execute(self, target, source, env): rsources = map(rfile, source) - return self.execfunction(target=target, source=rsources, env=env) + try: + result = self.execfunction(target=target, source=rsources, env=env) + except EnvironmentError, e: + # If an IOError/OSError happens, raise a BuildError. + raise SCons.Errors.BuildError(node=target, errstr=e.strerror) + return result def get_contents(self, target, source, env, dict=None): """Return the signature contents of this callable action. diff --git a/src/engine/SCons/Node/FS.py b/src/engine/SCons/Node/FS.py index 7f0ca629..6d2dd8b4 100644 --- a/src/engine/SCons/Node/FS.py +++ b/src/engine/SCons/Node/FS.py @@ -1639,11 +1639,7 @@ class File(Base): if self.get_state() != SCons.Node.up_to_date: if self.exists(): if self.is_derived() and not self.precious: - try: - Unlink(self, [], None) - except OSError, e: - raise SCons.Errors.BuildError(node = self, - errstr = e.strerror) + Unlink(self, [], None) try: delattr(self, '_exists') except AttributeError: @@ -1670,12 +1666,12 @@ class File(Base): self._createDir() try: Unlink(self, None, None) - except OSError: + except SCons.Errors.BuildError: pass try: Link(self, src, None) - except IOError, e: - desc = "Cannot duplicate `%s' in `%s': %s." % (src, self.dir, e.strerror) + except SCons.Errors.BuildError, e: + desc = "Cannot duplicate `%s' in `%s': %s." % (src, self.dir, e.errstr) raise SCons.Errors.StopError, desc self.linked = 1 # The Link() action may or may not have actually diff --git a/src/engine/SCons/Node/FSTests.py b/src/engine/SCons/Node/FSTests.py index c213d265..3e803096 100644 --- a/src/engine/SCons/Node/FSTests.py +++ b/src/engine/SCons/Node/FSTests.py @@ -344,7 +344,7 @@ class BuildDirTestCase(unittest.TestCase): save_Link = SCons.Node.FS.Link def Link_IOError(target, source, env): raise IOError, "Link_IOError" - SCons.Node.FS.Link = Link_IOError + SCons.Node.FS.Link = SCons.Action.Action(Link_IOError, None) test.write(['work', 'src', 'IOError'], "work/src/IOError\n") diff --git a/src/engine/SCons/Script/__init__.py b/src/engine/SCons/Script/__init__.py index 0ad91021..6ae76bf6 100644 --- a/src/engine/SCons/Script/__init__.py +++ b/src/engine/SCons/Script/__init__.py @@ -161,7 +161,10 @@ class BuildTask(SCons.Taskmaster.Task): t, e = sys.exc_info()[:2] if t == SCons.Errors.BuildError: - sys.stderr.write("scons: *** [%s] %s\n" % (e.node, e.errstr)) + fname = e.node + if SCons.Util.is_List(e.node): + fname = string.join(map(str, e.node), ', ') + sys.stderr.write("scons: *** [%s] %s\n" % (fname, e.errstr)) if e.errstr == 'Exception': traceback.print_exception(e.args[0], e.args[1], e.args[2]) elif t == SCons.Errors.ExplicitExit: diff --git a/test/build-errors.py b/test/build-errors.py index 1ea2d7a9..f64786fc 100644 --- a/test/build-errors.py +++ b/test/build-errors.py @@ -187,4 +187,37 @@ else: break test.fail_test(error_message_not_found) +test.write('SConstruct4', r""" +env = Environment() +env.Command('test.out', 'test.in', 'cp $SOURCE $TARGET') +env.InstallAs('test2.out', 'test.out') +# Mark test2.out as precious so we'll handle the exception in +# FunctionAction() rather than when the target is cleaned before building. +env.Precious('test2.out') +env.Default('test2.out') +""") + +test.write('test.in', "test.in 1\n") + +test.run(arguments = '-f SConstruct4 .') + +test.write('test.in', "test.in 2\n") + +test.writable('test2.out', 0) +f = open(test.workpath('test2.out')) + +test.run(arguments = '-f SConstruct4 .', + stderr = None, + status = 2) + +f.close() +test.writable('test2.out', 1) + +test.description_set("Incorrect STDERR:\n%s" % test.stderr()) +errs = [ + "scons: *** [test2.out] Permission denied\n", + "scons: *** [test2.out] permission denied\n", +] +test.fail_test(test.stderr() not in errs) + test.pass_test() diff --git a/www/index.html b/www/index.html index 52ee1f53..eb349787 100644 --- a/www/index.html +++ b/www/index.html @@ -4,7 +4,7 @@

SCons is a next-generation, -cross-platform build tool. +cross-platform, build tool. Think of SCons as an improved substitute for the classic Make utility @@ -19,9 +19,10 @@ or wedge a scripting language onto some other configuration file syntax, SCons configuration files are actually Python scripts. -This gives you a tremendous amount of flexibility +The ability to script your build +gives you a tremendous amount of flexibility to solve complicated build problems -in surprisingly small amounts of code. +in surprisingly small amounts of maintainable code.

@@ -56,7 +57,13 @@ project.

Mission

-

What is the goal of this project? +

The goal of The SCons Project +is to become the premiere build tool for +cross-platform, multi-language software projects +by offering unparalleled +reliability, +flexibility +and ease of use.

What is the scope of this project? diff --git a/www/project_highlights.html b/www/project_highlights.html index 93078bc7..0a153875 100644 --- a/www/project_highlights.html +++ b/www/project_highlights.html @@ -4,13 +4,13 @@

-23 August 2004: +23 August 2004: Bugfix release 0.96.1 fixes a handful of critical problems in 0.96.

-18 August 2004: +18 August 2004: Beta release 0.96 adds Fortran 90/95 support, better Qt support, platform-independent file manipulation actions, -- 2.26.2