/*
* (C) Copyright 2014 Nuxeo SA (http://nuxeo.com/) and others.
*
* 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.
*
* Contributors:
* Florent Guillaume
*/
package org.nuxeo.ecm.core.event.test;
import static org.junit.Assert.assertEquals;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.concurrent.CountDownLatch;
import javax.inject.Inject;
import org.apache.commons.logging.LogFactory;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.nuxeo.ecm.core.api.ConcurrentUpdateException;
import org.nuxeo.ecm.core.api.CoreSession;
import org.nuxeo.ecm.core.api.DocumentModel;
import org.nuxeo.ecm.core.api.DocumentRef;
import org.nuxeo.ecm.core.api.IdRef;
import org.nuxeo.ecm.core.event.EventService;
import org.nuxeo.ecm.core.storage.sql.IgnoreNonPostgresql;
import org.nuxeo.ecm.core.test.CoreFeature;
import org.nuxeo.ecm.core.test.annotations.Granularity;
import org.nuxeo.ecm.core.test.annotations.RepositoryConfig;
import org.nuxeo.ecm.core.work.AbstractWork;
import org.nuxeo.ecm.core.work.api.WorkManager;
import org.nuxeo.runtime.api.Framework;
import org.nuxeo.runtime.test.runner.ConditionalIgnoreRule;
import org.nuxeo.runtime.test.runner.Features;
import org.nuxeo.runtime.test.runner.FeaturesRunner;
import org.nuxeo.runtime.transaction.TransactionHelper;
@RunWith(FeaturesRunner.class)
@Features(CoreFeature.class)
@RepositoryConfig(cleanup = Granularity.METHOD)
@ConditionalIgnoreRule.Ignore(condition = IgnoreNonPostgresql.class)
public class WorkTest {
@Inject
protected EventService eventService;
@Inject
protected CoreSession session;
protected void waitForAsyncCompletion() {
nextTransaction();
eventService.waitForAsyncCompletion();
}
protected void nextTransaction() {
if (TransactionHelper.isTransactionActiveOrMarkedRollback()) {
TransactionHelper.commitOrRollbackTransaction();
TransactionHelper.startTransaction();
}
}
static Monitor monitor;
public void doTestWorkConcurrencyException(boolean explicitSave) throws Exception {
DocumentModel folder = session.createDocumentModel("/", "folder", "Folder");
folder = session.createDocument(folder);
session.save();
waitForAsyncCompletion();
WorkManager workManager = Framework.getLocalService(WorkManager.class);
monitor = new Monitor();
try {
RemoveFolderWork removeFolderWork = new RemoveFolderWork();
AddChildWork addChildWork = new AddChildWork();
removeFolderWork.init(folder, explicitSave);
addChildWork.init(folder, explicitSave);
workManager.schedule(removeFolderWork);
workManager.schedule(addChildWork);
waitForAsyncCompletion();
assertEquals(Arrays.asList(Boolean.TRUE, Boolean.FALSE), monitor.existList);
} finally {
monitor = null;
}
}
@Test
public void testWorkConcurrencyExceptionExplicitSave() throws Exception {
doTestWorkConcurrencyException(true);
}
@Test
public void testWorkConcurrencyExceptionImplicitSave() throws Exception {
doTestWorkConcurrencyException(false);
}
class Monitor {
final CountDownLatch ready = new CountDownLatch(2);
final CountDownLatch proceed = new CountDownLatch(2);
List<Boolean> existList = new ArrayList<Boolean>();
void ready() {
countDownAndAwait(ready);
}
void proceed() {
countDownAndAwait(proceed);
}
void countDownAndAwait(CountDownLatch latch) {
latch.countDown();
try {
latch.await();
} catch (InterruptedException e) {
// restore interrupted status
Thread.currentThread().interrupt();
throw new RuntimeException(e);
}
}
};
public static abstract class BaseWork extends AbstractWork {
private static final long serialVersionUID = 1L;
protected boolean explicitSave;
public void init(DocumentModel folder, boolean explicitSave) {
setDocument(folder.getRepositoryName(), folder.getId());
this.explicitSave = explicitSave;
}
@Override
public String getTitle() {
return getClass().getName();
}
}
/*
* The following 2 work instance are synced with a latch in order to add a child after the folder is deleted.
*/
/**
* Removes the folder.
*/
public static class RemoveFolderWork extends BaseWork {
private static final long serialVersionUID = 1L;
@Override
public void work() {
openSystemSession();
monitor.ready();
try {
DocumentRef ref = new IdRef(docId);
session.removeDocument(ref);
monitor.proceed();
if (explicitSave) {
session.save();
}
} finally {
closeSession();
}
}
}
/**
* Adds a document in the folder. Retries once.
*/
public static class AddChildWork extends BaseWork {
private static final long serialVersionUID = 1L;
@Override
public int getRetryCount() {
return 1;
}
@Override
public void work() {
openSystemSession();
monitor.ready();
try {
boolean exists = session.exists(new IdRef(docId));
monitor.existList.add(Boolean.valueOf(exists));
if (!exists) {
// after a retry, the folder is really gone
return;
}
monitor.proceed();
DocumentModel doc = session.createDocumentModel("/folder", "doc", "File");
doc = session.createDocument(doc);
if (explicitSave) {
session.save();
}
} catch (Exception cause) {
if (!(cause instanceof ConcurrentUpdateException)) {
LogFactory.getLog(WorkTest.class).error("non concurrent error caught (no retry)", cause);
} else {
LogFactory.getLog(WorkTest.class).info("concurrent error caught (should retry)", cause);
}
throw cause;
} finally {
closeSession();
}
}
}
}