diff --git a/datastore_server/source/java/ch/systemsx/cisd/etlserver/registrator/AbstractOmniscientTopLevelDataSetRegistrator.java b/datastore_server/source/java/ch/systemsx/cisd/etlserver/registrator/AbstractOmniscientTopLevelDataSetRegistrator.java
index cf720ab6c2050e17edc67e1d89b47d5ed04c644e..14ee954b79ae8d1b1ad5334054bf6afb3d3d56ff 100644
--- a/datastore_server/source/java/ch/systemsx/cisd/etlserver/registrator/AbstractOmniscientTopLevelDataSetRegistrator.java
+++ b/datastore_server/source/java/ch/systemsx/cisd/etlserver/registrator/AbstractOmniscientTopLevelDataSetRegistrator.java
@@ -322,55 +322,54 @@ public abstract class AbstractOmniscientTopLevelDataSetRegistrator<T extends Dat
      * Setup necessary for data set handling is done, then the handleDataSet method (a subclass
      * responsibility) is invoked.
      */
-    public final void handle(File incomingDataSetFileOrIsFinishedFile)
+    public final void handle(final File incomingDataSetFileOrIsFinishedFile)
     {
         if (stopped)
         {
             return;
         }
 
-        if (isRecoveryMarkerFile(incomingDataSetFileOrIsFinishedFile))
+        // get the original incoming dataset file
+        final File incomingDataSetFile =
+                getGlobalState().isUseIsFinishedMarkerFile() ? state.getMarkerFileUtility()
+                        .getIncomingDataSetPathFromMarker(incomingDataSetFileOrIsFinishedFile)
+                        : incomingDataSetFileOrIsFinishedFile;
+
+        if (isRecoveryMarkerFile(incomingDataSetFile))
         {
-            handleRecovery(incomingDataSetFileOrIsFinishedFile);
+            handleRecovery(incomingDataSetFile);
             return;
-        } else if (hasRecoveryMarkerFile(incomingDataSetFileOrIsFinishedFile))
+        } else if (hasRecoveryMarkerFile(incomingDataSetFile))
         {
             operationLog.info("Ignore file, as the recovery marker exists for it: "
-                    + incomingDataSetFileOrIsFinishedFile.getAbsolutePath());
+                    + incomingDataSetFile.getAbsolutePath());
             // will handle only the recovery file - don't do anything
             return;
-        } else if (false == incomingDataSetFileOrIsFinishedFile.exists())
+        } else if (false == incomingDataSetFile.exists())
         {
-            operationLog.info("The file doesn't exist: "
-                    + incomingDataSetFileOrIsFinishedFile.getAbsolutePath());
+            operationLog.info("The file doesn't exist: " + incomingDataSetFile.getAbsolutePath());
             // it can mean that the recovery has already cleaned this file
             return;
         }
 
-        final File isFinishedFile = incomingDataSetFileOrIsFinishedFile;
-        final File incomingDataSetFile;
         final IDelegatedActionWithResult<Boolean> markerFileCleanupAction;
 
         // Figure out what the real incoming data is -- if we use a marker file, it will tell us the
         // name
         if (getGlobalState().isUseIsFinishedMarkerFile())
         {
-            incomingDataSetFile =
-                    state.getMarkerFileUtility().getIncomingDataSetPathFromMarker(isFinishedFile);
-
             markerFileCleanupAction = new IDelegatedActionWithResult<Boolean>()
                 {
                     public Boolean execute(boolean didOperationSucceed)
                     {
                         boolean markerDeleteSucceeded =
                                 state.getMarkerFileUtility().deleteAndLogIsFinishedMarkerFile(
-                                        isFinishedFile);
+                                        incomingDataSetFileOrIsFinishedFile);
                         return markerDeleteSucceeded;
                     }
                 };
         } else
         {
-            incomingDataSetFile = incomingDataSetFileOrIsFinishedFile;
             markerFileCleanupAction = new DoNothingDelegatedAction();
         }
 
@@ -492,8 +491,9 @@ public abstract class AbstractOmniscientTopLevelDataSetRegistrator<T extends Dat
                 state.getGlobalState().getStorageRecoveryManager()
                         .extractPrecommittedCheckpoint(recoveryMarkerFile);
 
-        // TODO: cleanup also with the incoming marker file
-
+        // TODO: cleanup also with the incoming marker file, or are we guaranteed that the marker
+        // file is gone at this step.
+        // then we should ensure that the recovery will actually take place itself!
         final File recoveryFile =
                 state.getGlobalState().getStorageRecoveryManager()
                         .getRecoveryFileFromMarker(recoveryMarkerFile);
@@ -596,7 +596,7 @@ public abstract class AbstractOmniscientTopLevelDataSetRegistrator<T extends Dat
                                 .getGlobalState().getStagingDir());
 
                 recoveryState.getRollbackStack().setLockedState(false);
-                
+
                 recoveryState.getRollbackStack().rollbackAll(rollbackStackDelegate);
                 UnstoreDataAction action =
                         state.getOnErrorActionDecision().computeUndoAction(
@@ -619,6 +619,11 @@ public abstract class AbstractOmniscientTopLevelDataSetRegistrator<T extends Dat
                 logger.registerSuccess();
                 registrationSuccessful = true;
             }
+        } catch (org.jmock.api.ExpectationError r)
+        {
+            // this exception can by only thrown by tests.
+            // propagation of the exception is essential to test some functionalities
+            throw r;
         } catch (Throwable error)
         {
             System.err.println("Caught an error! " + error);
diff --git a/datastore_server/sourceTest/java/ch/systemsx/cisd/etlserver/registrator/AbstractJythonDataSetHandlerTest.java b/datastore_server/sourceTest/java/ch/systemsx/cisd/etlserver/registrator/AbstractJythonDataSetHandlerTest.java
index 59ff7c3cb2f945cf9f4d5c58f104970f0602759a..1d5da8f7cbc76ff8532e7297b71f95ae56a03adb 100644
--- a/datastore_server/sourceTest/java/ch/systemsx/cisd/etlserver/registrator/AbstractJythonDataSetHandlerTest.java
+++ b/datastore_server/sourceTest/java/ch/systemsx/cisd/etlserver/registrator/AbstractJythonDataSetHandlerTest.java
@@ -48,6 +48,7 @@ import ch.systemsx.cisd.etlserver.TopLevelDataSetRegistratorGlobalState;
 import ch.systemsx.cisd.etlserver.validation.IDataSetValidator;
 import ch.systemsx.cisd.openbis.dss.generic.shared.IEncapsulatedOpenBISService;
 import ch.systemsx.cisd.openbis.dss.generic.shared.api.internal.v1.IDataSourceQueryService;
+import ch.systemsx.cisd.openbis.generic.shared.basic.dto.DataSetType;
 import ch.systemsx.cisd.openbis.generic.shared.basic.dto.DatabaseInstance;
 import ch.systemsx.cisd.openbis.generic.shared.dto.StorageFormat;
 
@@ -96,6 +97,21 @@ public abstract class AbstractJythonDataSetHandlerTest extends AbstractFileSyste
 
     protected File subDataSet2;
 
+    protected static final String DATA_SET_CODE = "data-set-code";
+
+    protected static final String DATA_SET_CODE_1 = "data-set-code-1";
+
+    protected static final String CONTAINER_DATA_SET_CODE = "container-data-set-code";
+
+    protected static final DataSetType DATA_SET_TYPE = new DataSetType("O1");
+
+    protected static final String EXPERIMENT_PERM_ID = "experiment-perm-id";
+
+    protected static final String EXPERIMENT_IDENTIFIER = "/SPACE/PROJECT/EXP";
+
+    protected static final String SAMPLE_PERM_ID = "sample-perm-id";
+
+    
     @BeforeTest
     public void init()
     {
diff --git a/datastore_server/sourceTest/java/ch/systemsx/cisd/etlserver/registrator/JythonDropboxRecoveryTest.java b/datastore_server/sourceTest/java/ch/systemsx/cisd/etlserver/registrator/JythonDropboxRecoveryTest.java
new file mode 100644
index 0000000000000000000000000000000000000000..76613f6745465b35bcafc2588af9119c7c39049b
--- /dev/null
+++ b/datastore_server/sourceTest/java/ch/systemsx/cisd/etlserver/registrator/JythonDropboxRecoveryTest.java
@@ -0,0 +1,294 @@
+/*
+ * Copyright 2012 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.etlserver.registrator;
+
+import static ch.systemsx.cisd.common.Constants.IS_FINISHED_PREFIX;
+
+import java.io.File;
+import java.util.Arrays;
+import java.util.HashMap;
+import java.util.LinkedList;
+import java.util.List;
+import java.util.Properties;
+
+import org.jmock.Expectations;
+import org.testng.annotations.DataProvider;
+import org.testng.annotations.Test;
+
+import ch.systemsx.cisd.common.filesystem.FileUtilities;
+import ch.systemsx.cisd.common.test.RecordingMatcher;
+import ch.systemsx.cisd.openbis.generic.shared.basic.dto.Experiment;
+import ch.systemsx.cisd.openbis.generic.shared.basic.dto.ExternalData;
+import ch.systemsx.cisd.openbis.generic.shared.basic.dto.builders.ExperimentBuilder;
+import ch.systemsx.cisd.openbis.generic.shared.dto.AtomicEntityOperationDetails;
+import ch.systemsx.cisd.openbis.generic.shared.dto.identifier.ExperimentIdentifierFactory;
+
+/**
+ * @author jakubs
+ */
+public class JythonDropboxRecoveryTest extends AbstractJythonDataSetHandlerTest
+{
+    private static final String SCRIPTS_FOLDER =
+            "sourceTest/java/ch/systemsx/cisd/etlserver/registrator/";
+
+    @Override
+    protected String getRegistrationScriptsFolderPath()
+    {
+        return SCRIPTS_FOLDER;
+    }
+
+    @DataProvider(name = "recoveryTestCaseProvider")
+    public Object[][] recoveryTestCases()
+    {
+        LinkedList<RecoveryTestCase> testCases = new LinkedList<RecoveryTestCase>();
+        RecoveryTestCase testCase;
+
+        testCase = new RecoveryTestCase("Basic recovery testcase");
+        testCases.add(testCase);
+
+        // result value
+        Object[][] resultsList = new Object[testCases.size()][];
+        int index = 0;
+        for (RecoveryTestCase t : testCases)
+        {
+            resultsList[index++] = new Object[]
+                { t };
+        }
+
+        return resultsList;
+    }
+
+    private static class RecoveryTestCase
+    {
+
+        /**
+         * short description of the test. Will be presented in the test results view
+         */
+        protected String title;
+
+        /**
+         * The dropbox script file that should be used for this test case
+         */
+        protected String dropboxScriptPath = "simple-testcase.py";
+
+        /**
+         * Specifies what properties should be overriden for this test case.
+         */
+        protected HashMap<String, String> overrideProperties;
+
+        /**
+         * If true, than autorecovery should take place. If not, then transaction should be rolled
+         * back, and recovery artifacts removed.
+         */
+        protected boolean canRecoverFromError = true;
+
+        /**
+         * If true than this registration has been succesfull. Which means that the recovery should
+         * continue registration rather rollback.
+         */
+        protected boolean registrationSuccesfull = true;
+
+        /**
+         * True if the registration should throw exception to the top level. With this setting the
+         * handler is said to throw all exception to the top level, so that we can catch them. To
+         * check how the system reacts to error's itself (like rollback mechanism) this should be
+         * set to false.
+         */
+        protected boolean shouldThrowExceptionDuringRegistration = true;
+
+        private RecoveryTestCase(String title)
+        {
+            this.title = title;
+            this.overrideProperties = new HashMap<String, String>();
+            this.overrideProperties.put("TEST_V2_API", "");
+        }
+
+        @Override
+        public String toString()
+        {
+            return title;
+        }
+    }
+
+    @Test(dataProvider = "recoveryTestCaseProvider")
+    public void testRecovery(final RecoveryTestCase testCase)
+    {
+        setUpHomeDataBaseExpectations();
+
+        createData();
+
+        Properties properties =
+                createThreadPropertiesRelativeToScriptsFolder(testCase.dropboxScriptPath,
+                        testCase.overrideProperties);
+
+        createHandler(properties, true, false);
+
+        final RecordingMatcher<ch.systemsx.cisd.openbis.generic.shared.dto.AtomicEntityOperationDetails> atomicatOperationDetails =
+                new RecordingMatcher<ch.systemsx.cisd.openbis.generic.shared.dto.AtomicEntityOperationDetails>();
+
+        context.checking(new RecoveryTestExpectations(testCase, atomicatOperationDetails));
+
+        // create expectations:
+        // the expectations are up to the point when the registration in openbis fails
+
+        // run the actual code
+        handler.handle(markerFile);
+        if (testCase.canRecoverFromError)
+        {
+            File dir =
+                    new File(
+                            "/Users/jakubs/Documents/git/datastore_server/targets/unit-test-wd/ch.systemsx.cisd.etlserver.registrator.JythonDropboxRecoveryTest/");
+            for (Object f : dir.list())
+            {
+                System.out.println("XX:" + f);
+            }
+            File recoveryMarkerFile = getCreatedRecoveryMarkerFile();
+
+            handler.handle(recoveryMarkerFile);
+        } else
+        {
+            // rolllback requirementes
+        }
+
+        // now! we know that the error has happened
+
+        // then we assert there exists a recovery file
+        // assert there is a recovery marker file
+
+        // we continue with the recovery
+
+    }
+
+    private File getCreatedRecoveryMarkerFile()
+    {
+        File originalIncoming =
+                FileUtilities.removePrefixFromFileName(markerFile, IS_FINISHED_PREFIX);
+        File recoveryMarkerFile =
+                new File(originalIncoming.getAbsolutePath()
+                        + IDataSetStorageRecoveryManager.PROCESSING_MARKER);
+        assertTrue("The recovery marker file does not exist! " + recoveryMarkerFile,
+                recoveryMarkerFile.exists());
+        return recoveryMarkerFile;
+    }
+
+    class RecoveryTestExpectations extends Expectations
+    {
+        final RecoveryTestCase testCase;
+
+        final Experiment experiment;
+
+        final RecordingMatcher<AtomicEntityOperationDetails> atomicatOperationDetails;
+
+        public RecoveryTestExpectations(final RecoveryTestCase testCase,
+                final RecordingMatcher<AtomicEntityOperationDetails> atomicatOperationDetails)
+        {
+            this.testCase = testCase;
+            ExperimentBuilder builder = new ExperimentBuilder().identifier(EXPERIMENT_IDENTIFIER);
+            this.experiment = builder.getExperiment();
+
+            this.atomicatOperationDetails = atomicatOperationDetails;
+
+            prepareExpecatations();
+        }
+
+        private void prepareExpecatations()
+        {
+            initialExpectations();
+
+            registerDataSets();
+
+            // now registration has failed with the exception. we continue depending on where
+
+            if (testCase.canRecoverFromError)
+            {
+                checkRegistrationSucceeded();
+
+                if (testCase.registrationSuccesfull)
+                {
+                    setStorageConfirmed();
+                }
+            } else
+            {
+                // rollback
+            }
+        }
+
+        protected void initialExpectations()
+        {
+
+            // create dataset
+            one(openBisService).createDataSetCode();
+            will(returnValue(DATA_SET_CODE));
+
+            // get experiment
+            atLeast(1).of(openBisService).tryToGetExperiment(
+                    new ExperimentIdentifierFactory(experiment.getIdentifier()).createIdentifier());
+            will(returnValue(experiment));
+
+            // validate dataset
+            one(dataSetValidator).assertValidDataSet(DATA_SET_TYPE,
+                    new File(new File(stagingDirectory, DATA_SET_CODE), "sub_data_set_1"));
+        }
+
+        protected void registerDataSets()
+        {
+            one(openBisService).performEntityOperations(with(atomicatOperationDetails));
+
+            Exception e = new IllegalArgumentException("Failure in atomicOperationDetails");
+            will(throwException(e));
+
+        }
+
+        protected void checkRegistrationSucceeded()
+        {
+            one(openBisService).listDataSetsByCode(Arrays.asList(DATA_SET_CODE));
+            if (testCase.registrationSuccesfull)
+            {
+                // with the current implemntation returning the non-empty list should be enough
+                List<ExternalData> externalDatas = (List) Arrays.asList(new Object());
+                will(returnValue(externalDatas));
+            } else
+            {
+                will(returnValue(new LinkedList<ExternalData>()));
+            }
+        }
+
+        protected void setStorageConfirmed()
+        {
+            one(openBisService).setStorageConfirmed(DATA_SET_CODE);
+        }
+
+    }
+
+    /**
+     * creates a very simple dataset to import
+     */
+    private void createData()
+    {
+        incomingDataSetFile = createDirectory(workingDirectory, "data_set");
+
+        assertTrue(incomingDataSetFile.isDirectory());
+
+        subDataSet1 = createDirectory(incomingDataSetFile, "sub_data_set_1");
+
+        FileUtilities.writeToFile(new File(subDataSet1, "read1.me"), "hello world1");
+
+        markerFile = new File(workingDirectory, IS_FINISHED_PREFIX + "data_set");
+        FileUtilities.writeToFile(markerFile, "");
+    }
+
+}
diff --git a/datastore_server/sourceTest/java/ch/systemsx/cisd/etlserver/registrator/JythonTopLevelDataSetRegistratorRollbackTest.java b/datastore_server/sourceTest/java/ch/systemsx/cisd/etlserver/registrator/JythonTopLevelDataSetRegistratorRollbackTest.java
index 867b511cc395c293782700cbe4425a07ee76e80b..5816e2b423b79cd89b630342875104a9354b4059 100644
--- a/datastore_server/sourceTest/java/ch/systemsx/cisd/etlserver/registrator/JythonTopLevelDataSetRegistratorRollbackTest.java
+++ b/datastore_server/sourceTest/java/ch/systemsx/cisd/etlserver/registrator/JythonTopLevelDataSetRegistratorRollbackTest.java
@@ -48,12 +48,6 @@ public class JythonTopLevelDataSetRegistratorRollbackTest extends AbstractJython
     private static final String SCRIPTS_FOLDER =
             "sourceTest/java/ch/systemsx/cisd/etlserver/registrator/";
 
-    private static final String DATA_SET_CODE = "data-set-code";
-
-    private static final DataSetType DATA_SET_TYPE = new DataSetType("O1");
-
-    private static final String EXPERIMENT_IDENTIFIER = "/SPACE/PROJECT/EXP";
-
     @BeforeMethod
     @Override
     public void setUp() throws IOException
diff --git a/datastore_server/sourceTest/java/ch/systemsx/cisd/etlserver/registrator/JythonTopLevelDataSetRegistratorTest.java b/datastore_server/sourceTest/java/ch/systemsx/cisd/etlserver/registrator/JythonTopLevelDataSetRegistratorTest.java
index 40405dcd8cd550bcfaee311e89f13f1f6834d75a..754d620584cf786e5d3d6a048d8807b0af03f905 100644
--- a/datastore_server/sourceTest/java/ch/systemsx/cisd/etlserver/registrator/JythonTopLevelDataSetRegistratorTest.java
+++ b/datastore_server/sourceTest/java/ch/systemsx/cisd/etlserver/registrator/JythonTopLevelDataSetRegistratorTest.java
@@ -31,7 +31,6 @@ import java.util.Map;
 import java.util.Properties;
 
 import org.jmock.Expectations;
-import org.jmock.api.ExpectationError;
 import org.jmock.api.Invocation;
 import org.jmock.lib.action.CustomAction;
 import org.python.core.PyException;
@@ -56,7 +55,6 @@ import ch.systemsx.cisd.openbis.generic.shared.api.v1.dto.SearchCriteria;
 import ch.systemsx.cisd.openbis.generic.shared.api.v1.dto.SearchCriteria.MatchClause;
 import ch.systemsx.cisd.openbis.generic.shared.api.v1.dto.SearchCriteria.MatchClauseAttribute;
 import ch.systemsx.cisd.openbis.generic.shared.basic.dto.ContainerDataSet;
-import ch.systemsx.cisd.openbis.generic.shared.basic.dto.DataSetType;
 import ch.systemsx.cisd.openbis.generic.shared.basic.dto.EntityProperty;
 import ch.systemsx.cisd.openbis.generic.shared.basic.dto.Experiment;
 import ch.systemsx.cisd.openbis.generic.shared.basic.dto.NewExperiment;
@@ -82,20 +80,6 @@ public class JythonTopLevelDataSetRegistratorTest extends AbstractJythonDataSetH
     private static final String SCRIPTS_FOLDER =
             "sourceTest/java/ch/systemsx/cisd/etlserver/registrator/";
 
-    private static final String DATA_SET_CODE = "data-set-code";
-
-    private static final String DATA_SET_CODE_1 = "data-set-code-1";
-
-    private static final String CONTAINER_DATA_SET_CODE = "container-data-set-code";
-
-    private static final DataSetType DATA_SET_TYPE = new DataSetType("O1");
-
-    private static final String EXPERIMENT_PERM_ID = "experiment-perm-id";
-
-    private static final String EXPERIMENT_IDENTIFIER = "/SPACE/PROJECT/EXP";
-
-    private static final String SAMPLE_PERM_ID = "sample-perm-id";
-
     @BeforeMethod
     @Override
     public void setUp() throws IOException
@@ -212,7 +196,6 @@ public class JythonTopLevelDataSetRegistratorTest extends AbstractJythonDataSetH
         testCase.failurePoint = TestCaseParameters.FailurePoint.DURING_OPENBIS_REGISTRATION;
         testCases.addAll(multipleVersionsOfTestCase(testCase));
 
-        // TODO: In this case should it be "invalid dataset error" or what?
         testCase = new TestCaseParameters("The validation error with DELETE on error.");
         for (String error : allErrors)
         {
@@ -336,6 +319,8 @@ public class JythonTopLevelDataSetRegistratorTest extends AbstractJythonDataSetH
         return resultsList;
     }
 
+    //INFO: testCase parameters
+    
     /**
      * Parameters for the single run of the testSimpleTransaction
      * 
@@ -443,6 +428,7 @@ public class JythonTopLevelDataSetRegistratorTest extends AbstractJythonDataSetH
         }
     }
 
+    //INFO: test simple transaction
     @Test(dataProvider = "simpleTransactionTestCaseProvider")
     public void testSimpleTransaction(final TestCaseParameters testCase)
     {
@@ -463,12 +449,10 @@ public class JythonTopLevelDataSetRegistratorTest extends AbstractJythonDataSetH
             createDataWithOneSubDataSet();
         }
 
-        ExperimentBuilder builder = new ExperimentBuilder().identifier(EXPERIMENT_IDENTIFIER);
-        final Experiment experiment = builder.getExperiment();
         final RecordingMatcher<ch.systemsx.cisd.openbis.generic.shared.dto.AtomicEntityOperationDetails> atomicatOperationDetails =
                 new RecordingMatcher<ch.systemsx.cisd.openbis.generic.shared.dto.AtomicEntityOperationDetails>();
 
-        context.checking(getSimpleTransactionExpectations(testCase, experiment, atomicatOperationDetails));
+        context.checking(getSimpleTransactionExpectations(testCase, atomicatOperationDetails));
 
         if (testCase.shouldThrowExceptionDuringRegistration)
         {
@@ -533,9 +517,9 @@ public class JythonTopLevelDataSetRegistratorTest extends AbstractJythonDataSetH
 
     public Expectations getSimpleTransactionExpectations(
             final TestCaseParameters testCase,
-            final Experiment experiment,
-            final RecordingMatcher<ch.systemsx.cisd.openbis.generic.shared.dto.AtomicEntityOperationDetails> atomicatOperationDetails)
+            final RecordingMatcher<AtomicEntityOperationDetails> atomicatOperationDetails)
     {
+        final Experiment experiment = new ExperimentBuilder().identifier(EXPERIMENT_IDENTIFIER).getExperiment();
         return new Expectations()
             {
                 {