aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/libmain/shared.cc2
-rw-r--r--src/libstore/build.cc19
-rw-r--r--src/libutil/types.hh5
-rw-r--r--src/libutil/util.cc3
4 files changed, 22 insertions, 7 deletions
diff --git a/src/libmain/shared.cc b/src/libmain/shared.cc
index 7c2d92030..68f145820 100644
--- a/src/libmain/shared.cc
+++ b/src/libmain/shared.cc
@@ -393,7 +393,7 @@ int main(int argc, char * * argv)
printMsg(lvlError, format("error: %1%%2%") % (showTrace ? e.prefix() : "") % e.msg());
if (e.prefix() != "" && !showTrace)
printMsg(lvlError, "(use `--show-trace' to show detailed location information)");
- return 1;
+ return e.status;
} catch (std::exception & e) {
printMsg(lvlError, format("error: %1%") % e.what());
return 1;
diff --git a/src/libstore/build.cc b/src/libstore/build.cc
index 0be9c42e0..83bd6754a 100644
--- a/src/libstore/build.cc
+++ b/src/libstore/build.cc
@@ -214,6 +214,10 @@ public:
bool cacheFailure;
+ /* Set if at least one derivation had a BuildError (i.e. permanent
+ failure). */
+ bool permanentFailure;
+
LocalStore & store;
boost::shared_ptr<HookInstance> hook;
@@ -266,7 +270,8 @@ public:
/* Wait for input to become available. */
void waitForInput();
-
+
+ unsigned int exitStatus();
};
@@ -1185,6 +1190,7 @@ void DerivationGoal::tryToBuild()
if (printBuildTrace)
printMsg(lvlError, format("@ build-failed %1% %2% %3% %4%")
% drvPath % drv.outputs["out"].path % 0 % e.msg());
+ worker.permanentFailure = true;
amDone(ecFailed);
return;
}
@@ -1321,6 +1327,7 @@ void DerivationGoal::buildDone()
foreach (DerivationOutputs::iterator, i, drv.outputs)
worker.store.registerFailedPath(i->second.path);
+ worker.permanentFailure = !hookError && !fixedOutput;
amDone(ecFailed);
return;
}
@@ -2444,6 +2451,7 @@ Worker::Worker(LocalStore & store)
nrLocalBuilds = 0;
lastWokenUp = 0;
cacheFailure = queryBoolSetting("build-cache-failure", false);
+ permanentFailure = false;
}
@@ -2770,6 +2778,11 @@ void Worker::waitForInput()
}
+unsigned int Worker::exitStatus()
+{
+ return permanentFailure ? 100 : 1;
+}
+
//////////////////////////////////////////////////////////////////////
@@ -2796,7 +2809,7 @@ void LocalStore::buildDerivations(const PathSet & drvPaths)
}
if (!failed.empty())
- throw Error(format("build of %1% failed") % showPaths(failed));
+ throw Error(format("build of %1% failed") % showPaths(failed), worker.exitStatus());
}
@@ -2812,7 +2825,7 @@ void LocalStore::ensurePath(const Path & path)
worker.run(goals);
if (goal->getExitCode() != Goal::ecSuccess)
- throw Error(format("path `%1%' does not exist and cannot be created") % path);
+ throw Error(format("path `%1%' does not exist and cannot be created") % path, worker.exitStatus());
}
diff --git a/src/libutil/types.hh b/src/libutil/types.hh
index 854a0f689..533fcca22 100644
--- a/src/libutil/types.hh
+++ b/src/libutil/types.hh
@@ -29,7 +29,8 @@ protected:
string prefix_; // used for location traces etc.
string err;
public:
- BaseError(const format & f);
+ unsigned int status; // exit status
+ BaseError(const format & f, unsigned int status = 1);
~BaseError() throw () { };
const char * what() const throw () { return err.c_str(); }
const string & msg() const throw () { return err; }
@@ -41,7 +42,7 @@ public:
class newClass : public superClass \
{ \
public: \
- newClass(const format & f) : superClass(f) { }; \
+ newClass(const format & f, unsigned int status = 1) : superClass(f, status) { }; \
};
MakeError(Error, BaseError)
diff --git a/src/libutil/util.cc b/src/libutil/util.cc
index 990962763..9adaac40d 100644
--- a/src/libutil/util.cc
+++ b/src/libutil/util.cc
@@ -20,7 +20,8 @@ extern char * * environ;
namespace nix {
-BaseError::BaseError(const format & f)
+BaseError::BaseError(const format & f, unsigned int status)
+ : status(status)
{
err = f.str();
}