diff --git a/common/source/java/ch/systemsx/cisd/common/utilities/CmdLineHelper.java b/common/source/java/ch/systemsx/cisd/common/utilities/CmdLineHelper.java
deleted file mode 100644
index 01815d50930dd7259f59c4cff8425885c8fd50c8..0000000000000000000000000000000000000000
--- a/common/source/java/ch/systemsx/cisd/common/utilities/CmdLineHelper.java
+++ /dev/null
@@ -1,154 +0,0 @@
-/*
- * Copyright 2007 ETH Zuerich, CISD
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- *      http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package ch.systemsx.cisd.common.utilities;
-
-import java.io.BufferedReader;
-import java.io.IOException;
-import java.io.InputStreamReader;
-import java.util.List;
-
-import org.apache.commons.io.IOUtils;
-import org.apache.log4j.Level;
-import org.apache.log4j.Logger;
-
-/**
- * Utility to execute a command from a command line and log all events.
- * 
- * @author Tomasz Pylak
- */
-public class CmdLineHelper
-{
-    /**
-     * The exit value returned by {@link Process#waitFor()} if the process was terminated by {@link Process#destroy()}
-     * on a UNIX machine.
-     */
-    private static final int EXIT_VALUE_FOR_TERMINATION_UNIX = 143;
-
-    /**
-     * The exit value returned by {@link Process#waitFor()} if the process was terminated by {@link Process#destroy()}
-     * on a MS Windows machine.
-     */
-    private static final int EXIT_VALUE_FOR_TERMINATION_WINDOWS = 1;
-
-    private final Logger operationLog;
-
-    private final Logger machineLog;
-
-    private CmdLineHelper(Logger operationLog, Logger machineLog)
-    {
-        this.operationLog = operationLog;
-        this.machineLog = machineLog;
-    }
-
-    public static boolean run(List<String> cmd, Logger operationLog, Logger machineLog)
-    {
-        return new CmdLineHelper(operationLog, machineLog).run(cmd);
-    }
-
-    public static void logProcessExitValue(final int exitValue, final Process process, String name,
-            Logger operationLog, Logger machineLog)
-    {
-        new CmdLineHelper(operationLog, machineLog).logProcessExecution(exitValue, process, name);
-    }
-
-    private boolean run(List<String> cmd)
-    {
-        final ProcessBuilder processBuilder = new ProcessBuilder(cmd);
-        processBuilder.redirectErrorStream(true);
-        if (operationLog.isDebugEnabled())
-        {
-            operationLog.debug("Executing command: " + cmd);
-        }
-        final Process process;
-        try
-        {
-            process = processBuilder.start();
-        } catch (IOException ex)
-        {
-            machineLog.error(String.format("Cannot execute binary %s", cmd), ex);
-            return false;
-        }
-
-        int exitValue;
-        try
-        {
-            exitValue = process.waitFor();
-        } catch (InterruptedException ex)
-        {
-            machineLog.error(String.format("Execution of %s interupted", process), ex);
-            return false;
-        }
-        logProcessExecution(exitValue, process, cmd.get(0).toString());
-        return (exitValue == 0);
-    }
-
-    private void logProcessExecution(final int exitValue, final Process process, String command)
-    {
-        if (exitValue != 0)
-        {
-            logProcessExitValue(Level.WARN, exitValue, command);
-            logProcessOutput(Level.WARN, process, command);
-        } else if (operationLog.isDebugEnabled())
-        {
-            logProcessExitValue(Level.DEBUG, exitValue, command);
-            logProcessOutput(Level.DEBUG, process, command);
-        }
-    }
-
-    private void logProcessExitValue(final Level logLevel, final int exitValue, final String command)
-    {
-        if (processTerminated(exitValue))
-        {
-            operationLog.log(logLevel, String.format("[%s] process was destroyed.", command));
-        } else
-        {
-            operationLog.log(logLevel, String.format("[%s] process returned with exit value %d.", command, exitValue));
-        }
-    }
-
-    private void logProcessOutput(final Level logLevel, final Process process, final String command)
-    {
-        final BufferedReader reader = new BufferedReader(new InputStreamReader(process.getInputStream()));
-        try
-        {
-            machineLog.log(logLevel, String.format("[%s] output:", command));
-            String ln;
-            while ((ln = reader.readLine()) != null)
-            {
-                if (ln.trim().length() > 0)
-                    machineLog.log(logLevel, String.format("\"%s\"", ln));
-            }
-        } catch (IOException e)
-        {
-            operationLog.warn(String.format("IOException when reading stderr, msg='%s'.", e.getMessage()));
-        } finally
-        {
-            IOUtils.closeQuietly(reader);
-        }
-    }
-
-    public static boolean processTerminated(final int exitValue)
-    {
-        if (OSUtilities.isWindows())
-        {
-            return exitValue == EXIT_VALUE_FOR_TERMINATION_WINDOWS;
-        } else
-        {
-            return exitValue == EXIT_VALUE_FOR_TERMINATION_UNIX;
-        }
-    }
-}
diff --git a/common/source/java/ch/systemsx/cisd/common/utilities/ProcessExecutionHelper.java b/common/source/java/ch/systemsx/cisd/common/utilities/ProcessExecutionHelper.java
new file mode 100644
index 0000000000000000000000000000000000000000..49df956358a0f100159758d4c8832ffef7a0c0e0
--- /dev/null
+++ b/common/source/java/ch/systemsx/cisd/common/utilities/ProcessExecutionHelper.java
@@ -0,0 +1,434 @@
+/*
+ * Copyright 2007 ETH Zuerich, CISD
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *      http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package ch.systemsx.cisd.common.utilities;
+
+import java.io.BufferedReader;
+import java.io.File;
+import java.io.IOException;
+import java.io.InputStreamReader;
+import java.util.ArrayList;
+import java.util.List;
+import java.util.Timer;
+import java.util.TimerTask;
+
+import org.apache.commons.io.IOUtils;
+import org.apache.log4j.Level;
+import org.apache.log4j.Logger;
+
+/**
+ * Utility to execute a command from a command line and log all events.
+ * 
+ * @author Tomasz Pylak
+ * @author Bernd Rinn
+ */
+public class ProcessExecutionHelper
+{
+    /**
+     * The value indicating that there is no exit value available for a process execution.
+     */
+    public static final int NO_EXIT_VALUE = -1;
+
+    /**
+     * The value indicating the process execution went OK.
+     */
+    public static final int EXIT_VALUE_OK = 0;
+
+    /**
+     * The exit value returned by {@link Process#waitFor()} if the process was terminated by {@link Process#destroy()}
+     * on a UNIX machine.
+     */
+    private static final int EXIT_VALUE_FOR_TERMINATION_UNIX = 143;
+
+    /**
+     * The exit value returned by {@link Process#waitFor()} if the process was terminated by {@link Process#destroy()}
+     * on a MS Windows machine.
+     */
+    private static final int EXIT_VALUE_FOR_TERMINATION_WINDOWS = 1;
+
+    private final Logger operationLog;
+
+    private final Logger machineLog;
+
+    /**
+     * Class that keeps around the result of running an Operating System process.
+     * <p>
+     * Since the process output can only ever be read once from a process, it need to be kept around if it is needed
+     * more than once. This is what this class is good for.
+     */
+    public static class ProcessResult
+    {
+        private final Process processOrNull;
+
+        private final List<String> commandLine;
+
+        private final String commandName;
+
+        private final Logger operationLog;
+
+        private final Logger machineLog;
+
+        private List<String> outputLines;
+
+        private ProcessResult(Process processOrNull, List<String> commandLine, Logger operationLog, Logger machineLog)
+        {
+            this.commandLine = commandLine;
+            this.commandName = new File(commandLine.get(0)).getName();
+            this.processOrNull = processOrNull;
+            this.operationLog = operationLog;
+            this.machineLog = machineLog;
+            this.outputLines = null;
+        }
+
+        /**
+         * Returns the command line that belongs to this process.
+         */
+        public List<String> getCommandLine()
+        {
+            return commandLine;
+        }
+
+        /**
+         * Returns the name of the command that belongs to this process.
+         */
+        public String getCommandName()
+        {
+            return commandName;
+        }
+
+        /**
+         * Returns the lines of the process output.
+         */
+        public List<String> getProcessOutput()
+        {
+            if (outputLines == null)
+            {
+                outputLines = readProcessOutputLines(processOrNull, machineLog);
+            }
+            return outputLines;
+        }
+
+        public int exitValue()
+        {
+            if (processOrNull != null)
+            {
+                return processOrNull.exitValue();
+            } else
+            {
+                return NO_EXIT_VALUE;
+            }
+        }
+
+        public boolean isOK()
+        {
+            return exitValue() == EXIT_VALUE_OK;
+        }
+
+        /**
+         * Returns <code>true</code> if the process has been run at all.
+         */
+        public boolean isRun()
+        {
+            return exitValue() != NO_EXIT_VALUE;
+        }
+
+        /**
+         * Returns <code>true</code> if the process has been terminated on the Operating System level.
+         */
+        public boolean isTerminated()
+        {
+            return isProcessTerminated(exitValue());
+        }
+
+        public void log()
+        {
+
+            if (isOK() == false)
+            {
+                logProcessExitValue(Level.WARN);
+                logProcessOutput(Level.WARN);
+            } else if (operationLog.isDebugEnabled())
+            {
+                logProcessExitValue(Level.DEBUG);
+                logProcessOutput(Level.DEBUG);
+            }
+        }
+
+        private void logProcessExitValue(final Level logLevel)
+        {
+            if (isRun() == false)
+            {
+                operationLog.log(logLevel, String.format("[%s] process could not be run.", commandName));
+            } else if (isTerminated())
+            {
+                operationLog.log(logLevel, String.format("[%s] process was destroyed.", commandName));
+            } else
+            {
+                operationLog.log(logLevel, String.format("[%s] process returned with exit value %d.", commandName,
+                        exitValue()));
+            }
+        }
+
+        private void logProcessOutput(final Level logLevel)
+        {
+            assert logLevel != null;
+
+            final List<String> processOutputLines = getProcessOutput();
+            if (processOutputLines.size() == 0)
+            {
+                return;
+            }
+            machineLog.log(logLevel, String.format("[%s] output:", commandName));
+            for (String ln : processOutputLines)
+            {
+                if (ln.trim().length() > 0)
+                {
+                    machineLog.log(logLevel, String.format("\"%s\"", ln));
+                }
+            }
+        }
+
+    }
+
+    /**
+     * Runs an Operating System process, specified by <var>cmd</var>.
+     * 
+     * @param commandLine The command line to run.
+     * @param operationLog The {@link Logger} to use for all message on the higher level.
+     * @param machineLog The {@link Logger} to use for all message on the lower (machine) level.
+     * @return <code>true</code>, if the process did complete successfully, <code>false</code> otherwise.
+     */
+    public static boolean runAndLog(List<String> commandLine, Logger operationLog, Logger machineLog)
+    {
+        return new ProcessExecutionHelper(operationLog, machineLog).run(commandLine, 0L);
+    }
+
+    /**
+     * Runs an Operating System process, specified by <var>cmd</var>.
+     * 
+     * @param commandLine The command line to run.
+     * @param operationLog The {@link Logger} to use for all message on the higher level.
+     * @param machineLog The {@link Logger} to use for all message on the lower (machine) level.
+     * @return The process result.
+     */
+    public static ProcessResult run(List<String> commandLine, Logger operationLog, Logger machineLog)
+    {
+        return new ProcessExecutionHelper(operationLog, machineLog).tryRun(commandLine, 0L);
+    }
+
+    /**
+     * Runs an Operating System process, specified by <var>cmd</var>.
+     * 
+     * @param cmd The command line to run.
+     * @param millisToWaitForCompletion The time to wait for the process to complete in milli seconds. If the process is
+     *            not finished after that time, it will be terminated by a watch dog.
+     * @param operationLog The {@link Logger} to use for all message on the higher level.
+     * @param machineLog The {@link Logger} to use for all message on the lower (machine) level.
+     * @return <code>true</code>, if the process did complete successfully, <code>false</code> otherwise.
+     */
+    public static boolean runAndLog(List<String> cmd, long millisToWaitForCompletion, Logger operationLog, Logger machineLog)
+    {
+        return new ProcessExecutionHelper(operationLog, machineLog).run(cmd, millisToWaitForCompletion);
+    }
+
+    /**
+     * Runs an Operating System process, specified by <var>cmd</var>.
+     * 
+     * @param cmd The command line to run.
+     * @param millisToWaitForCompletion The time to wait for the process to complete in milli seconds. If the process is
+     *            not finished after that time, it will be terminated by a watch dog.
+     * @param operationLog The {@link Logger} to use for all message on the higher level.
+     * @param machineLog The {@link Logger} to use for all message on the lower (machine) level.
+     * @return The process result.
+     */
+    public static ProcessResult run(List<String> cmd, long millisToWaitForCompletion, Logger operationLog,
+            Logger machineLog)
+    {
+        return new ProcessExecutionHelper(operationLog, machineLog).tryRun(cmd, millisToWaitForCompletion);
+    }
+
+    /**
+     * Returns <code>true</code> if the <var>exitValue</var> indicates that the process has been terminated on the
+     * Operating System level.
+     */
+    public static boolean isProcessTerminated(final int exitValue)
+    {
+        if (OSUtilities.isWindows())
+        {
+            return exitValue == EXIT_VALUE_FOR_TERMINATION_WINDOWS;
+        } else
+        {
+            return exitValue == EXIT_VALUE_FOR_TERMINATION_UNIX;
+        }
+    }
+
+    /**
+     * Returns the stdout (and stderr if {@link ProcessBuilder#redirectErrorStream(boolean)} has been called with
+     * <code>true</code>).
+     */
+    public static List<String> readProcessOutputLines(Process processOrNull, Logger machineLog)
+    {
+        final List<String> processOutput = new ArrayList<String>();
+        if (processOrNull == null)
+        {
+            return processOutput;
+        }
+        final BufferedReader reader = new BufferedReader(new InputStreamReader(processOrNull.getInputStream()));
+        try
+        {
+            String ln;
+            while ((ln = reader.readLine()) != null)
+            {
+                processOutput.add(ln);
+            }
+        } catch (IOException e)
+        {
+            machineLog.warn(String.format("IOException when reading stdout, msg='%s'.", e.getMessage()));
+        } finally
+        {
+            IOUtils.closeQuietly(reader);
+        }
+        return processOutput;
+    }
+
+    //
+    // Implementation
+    //
+
+    private ProcessExecutionHelper(Logger operationLog, Logger machineLog)
+    {
+        this.operationLog = operationLog;
+        this.machineLog = machineLog;
+    }
+
+    private ProcessResult tryRun(List<String> commandLine, long millisoWaitForCompletion)
+    {
+        final ProcessBuilder processBuilder = new ProcessBuilder(commandLine);
+        processBuilder.redirectErrorStream(true);
+        if (operationLog.isDebugEnabled())
+        {
+            operationLog.debug("Executing command: " + commandLine);
+        }
+        final Process process;
+        try
+        {
+            process = processBuilder.start();
+        } catch (IOException ex)
+        {
+            machineLog.error(String.format("Cannot execute executable %s", commandLine), ex);
+            return new ProcessResult(null, commandLine, operationLog, machineLog);
+        }
+
+        final Timer watchDogOrNull = tryCreateWatchDog(process, millisoWaitForCompletion, commandLine.get(0));
+
+        try
+        {
+            process.waitFor();
+        } catch (InterruptedException ex)
+        {
+            machineLog.error(String.format("Execution of %s interupted", process), ex);
+        } finally
+        {
+            if (watchDogOrNull != null)
+            {
+                watchDogOrNull.cancel();
+            }
+        }
+        return new ProcessResult(process, commandLine, operationLog, machineLog);
+    }
+
+    private Timer tryCreateWatchDog(final Process process, final long millisToWaitForCompletion, String commandForLog)
+    {
+        final Timer watchDogOrNull;
+        if (millisToWaitForCompletion > 0L)
+        {
+            watchDogOrNull = new Timer(String.format("Watch Dog [%s]", commandForLog));
+            watchDogOrNull.schedule(new TimerTask()
+                {
+                    @Override
+                    public void run()
+                    {
+                        operationLog.warn(String.format("Destroy process since it didn't finish in %d milli seconds",
+                                millisToWaitForCompletion));
+                        process.destroy();
+                    }
+                }, millisToWaitForCompletion);
+        } else
+        {
+            watchDogOrNull = null;
+        }
+        return watchDogOrNull;
+    }
+
+    private boolean run(List<String> cmd, long millisToWaitForCompletion)
+    {
+        final ProcessResult result = tryRun(cmd, millisToWaitForCompletion);
+        result.log();
+        return result.isOK();
+    }
+
+    public static void logProcessExecution(String commandName, int exitValue, List<String> processOutput,
+            Logger operationLog, Logger machineLog)
+    {
+        if (exitValue != EXIT_VALUE_OK)
+        {
+            logProcessExitValue(Level.WARN, operationLog, commandName, exitValue);
+            logProcessOutput(Level.WARN, machineLog, commandName, processOutput);
+        } else if (operationLog.isDebugEnabled())
+        {
+            logProcessExitValue(Level.DEBUG, operationLog, commandName, exitValue);
+            logProcessOutput(Level.DEBUG, machineLog, commandName, processOutput);
+        }
+    }
+
+    private static void logProcessExitValue(final Level logLevel, Logger operationLog, String commandName, int exitValue)
+    {
+        assert logLevel != null;
+        assert operationLog != null;
+        assert commandName != null;
+
+        if (isProcessTerminated(exitValue))
+        {
+            operationLog.log(logLevel, String.format("[%s] process was destroyed.", commandName));
+        } else
+        {
+            operationLog.log(logLevel, String.format("[%s] process returned with exit value %d.", commandName,
+                    exitValue));
+        }
+    }
+
+    private static void logProcessOutput(final Level logLevel, Logger machineLog, String commandName,
+            List<String> processOutputLines)
+    {
+        assert logLevel != null;
+        assert machineLog != null;
+        assert commandName != null;
+        assert processOutputLines != null;
+
+        if (processOutputLines.size() == 0)
+        {
+            return;
+        }
+        machineLog.log(logLevel, String.format("[%s] output:", commandName));
+        for (String ln : processOutputLines)
+        {
+            if (ln.trim().length() > 0)
+            {
+                machineLog.log(logLevel, String.format("\"%s\"", ln));
+            }
+        }
+    }
+}
diff --git a/common/sourceTest/java/ch/systemsx/cisd/common/utilities/ProcessExecutionHelperTest.java b/common/sourceTest/java/ch/systemsx/cisd/common/utilities/ProcessExecutionHelperTest.java
new file mode 100644
index 0000000000000000000000000000000000000000..a76c26b07a3ecff7cf1590cc8e5804b104dc986e
--- /dev/null
+++ b/common/sourceTest/java/ch/systemsx/cisd/common/utilities/ProcessExecutionHelperTest.java
@@ -0,0 +1,190 @@
+/*
+ * Copyright 2007 ETH Zuerich, CISD
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *      http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package ch.systemsx.cisd.common.utilities;
+
+import static org.testng.AssertJUnit.*;
+
+import java.io.File;
+import java.io.IOException;
+import java.util.Arrays;
+import org.apache.log4j.Logger;
+import org.testng.annotations.BeforeClass;
+import org.testng.annotations.BeforeMethod;
+import org.testng.annotations.Test;
+
+import ch.systemsx.cisd.common.logging.LogCategory;
+import ch.systemsx.cisd.common.logging.LogFactory;
+import ch.systemsx.cisd.common.logging.LogInitializer;
+
+/**
+ * Test cases for the {@link ProcessExecutionHelper}.
+ * 
+ * @author Bernd Rinn
+ */
+public class ProcessExecutionHelperTest
+{
+
+    private static final long WATCHDOG_WAIT_MILLIS = 1000L;
+
+    private static final Logger machineLog =
+            LogFactory.getLogger(LogCategory.MACHINE, ProcessExecutionHelperTest.class);
+
+    private static final Logger operationLog =
+            LogFactory.getLogger(LogCategory.OPERATION, ProcessExecutionHelperTest.class);
+
+    private static final File unitTestRootDirectory = new File("targets" + File.separator + "unit-test-wd");
+
+    private static final File workingDirectory = new File(unitTestRootDirectory, "ProcessExecutionHelperTest");
+
+    private final StoringUncaughtExceptionHandler exceptionHandler = new StoringUncaughtExceptionHandler();
+
+    private File createExecutable(String name, String... lines) throws IOException, InterruptedException
+    {
+        final File executable = new File(workingDirectory, name);
+        executable.delete();
+        CollectionIO.writeIterable(executable, Arrays.asList(lines));
+        Runtime.getRuntime().exec(String.format("/bin/chmod +x %s", executable.getPath())).waitFor();
+        executable.deleteOnExit();
+        return executable;
+    }
+
+    private File createExecutable(String name, int exitValue) throws IOException, InterruptedException
+    {
+        return createExecutable(name, "#! /bin/sh", "exit " + exitValue);
+    }
+
+    private File createSleepingExecutable(String name, long millisToSleep) throws IOException, InterruptedException
+    {
+        return createExecutable(name, "#! /bin/sh", "sleep " + (millisToSleep / 1000.0f), "exit 0");
+    }
+
+    @BeforeClass
+    public void init()
+    {
+        LogInitializer.init();
+        unitTestRootDirectory.mkdirs();
+        assert unitTestRootDirectory.isDirectory();
+        Thread.setDefaultUncaughtExceptionHandler(exceptionHandler);
+    }
+
+    @BeforeMethod
+    public void setUp() throws IOException
+    {
+        workingDirectory.delete();
+        workingDirectory.mkdirs();
+        workingDirectory.deleteOnExit();
+        exceptionHandler.reset();
+    }
+
+    @Test(groups =
+        { "requires_unix" })
+    public void testExecutionOKWithoutWatchDog() throws Exception
+    {
+        final File dummyExec = createExecutable("dummy.sh", 0);
+        final boolean ok =
+                ProcessExecutionHelper.runAndLog(Arrays.asList(dummyExec.getAbsolutePath()), operationLog, machineLog);
+        assertTrue(ok);
+    }
+
+    @Test(groups =
+        { "requires_unix" })
+    public void testExecutionFailedWithoutWatchDog() throws Exception
+    {
+        final File dummyExec = createExecutable("dummy.sh", 1);
+        final boolean ok =
+                ProcessExecutionHelper.runAndLog(Arrays.asList(dummyExec.getAbsolutePath()), operationLog, machineLog);
+        assertFalse(ok);
+    }
+
+    @Test(groups =
+        { "requires_unix" })
+    public void testExecutionOKWithWatchDog() throws Exception
+    {
+        final File dummyExec = createExecutable("dummy.sh", 0);
+        final boolean ok =
+                ProcessExecutionHelper.runAndLog(Arrays.asList(dummyExec.getAbsolutePath()), WATCHDOG_WAIT_MILLIS,
+                        operationLog, machineLog);
+        assertTrue(ok);
+    }
+
+    @Test(groups =
+        { "requires_unix" })
+    public void testExecutionFailedWithWatchDog() throws Exception
+    {
+        final File dummyExec = createExecutable("dummy.sh", 1);
+        final boolean ok =
+                ProcessExecutionHelper.runAndLog(Arrays.asList(dummyExec.getAbsolutePath()), WATCHDOG_WAIT_MILLIS,
+                        operationLog, machineLog);
+        assertFalse(ok);
+    }
+
+    @Test(groups =
+        { "requires_unix", "slow" })
+    public void testExecutionOKWithWatchDogWaiting() throws Exception
+    {
+        final File dummyExec = createSleepingExecutable("dummy.sh", WATCHDOG_WAIT_MILLIS / 2);
+        final boolean ok =
+                ProcessExecutionHelper.runAndLog(Arrays.asList(dummyExec.getAbsolutePath()), WATCHDOG_WAIT_MILLIS,
+                        operationLog, machineLog);
+        assertTrue(ok);
+    }
+
+    @Test(groups =
+        { "requires_unix", "slow" })
+    public void testExecutionFailedWithWatchDogHitting() throws Exception
+    {
+        final File dummyExec = createSleepingExecutable("dummy.sh", 2 * WATCHDOG_WAIT_MILLIS);
+        final boolean ok =
+                ProcessExecutionHelper.runAndLog(Arrays.asList(dummyExec.getAbsolutePath()), WATCHDOG_WAIT_MILLIS,
+                        operationLog, machineLog);
+        assertFalse(ok);
+    }
+
+    @Test(groups =
+        { "requires_unix", "slow" })
+    public void testTryExecutionFailedWithWatchDogHitting() throws Exception
+    {
+        final File dummyExec = createSleepingExecutable("dummy.sh", 2 * WATCHDOG_WAIT_MILLIS);
+        final ProcessExecutionHelper.ProcessResult result =
+                ProcessExecutionHelper.run(Arrays.asList(dummyExec.getAbsolutePath()), WATCHDOG_WAIT_MILLIS,
+                        operationLog, machineLog);
+        assertTrue(ProcessExecutionHelper.isProcessTerminated(result.exitValue()));
+    }
+
+    @Test(groups =
+        { "requires_unix", "slow" })
+    public void testTryExecutionReadProcessOutput() throws Exception
+    {
+        final String stdout1 = "This goes to stdout, 1";
+        final String stdout2 = "This goes to stdout, 2";
+        final String stderr1 = "This goes to stderr, 1";
+        final String stderr2 = "This goes to stderr, 2";
+        final File dummyExec =
+                createExecutable("dummy.sh", "echo " + stdout1, "echo " + stderr1, "echo " + stdout2, "echo " + stderr2);
+        final ProcessExecutionHelper.ProcessResult result =
+                ProcessExecutionHelper.run(Arrays.asList(dummyExec.getAbsolutePath()), operationLog, machineLog);
+        final int exitValue = result.exitValue();
+        assertEquals(0, exitValue);
+        result.log();
+        assertEquals(4, result.getProcessOutput().size());
+        assertEquals(stdout1, result.getProcessOutput().get(0));
+        assertEquals(stderr1, result.getProcessOutput().get(1));
+        assertEquals(stdout2, result.getProcessOutput().get(2));
+        assertEquals(stderr2, result.getProcessOutput().get(3));
+    }
+
+}