/*
* Copyright 2012 Atteo.
*
* 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 org.atteo.moonshine.jpa;
import javax.inject.Inject;
import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.EntityTransaction;
import javax.persistence.TransactionRequiredException;
import javax.persistence.criteria.CriteriaBuilder;
import javax.persistence.metamodel.Metamodel;
import javax.transaction.RollbackException;
import javax.transaction.Synchronization;
import javax.transaction.SystemException;
import javax.transaction.Transaction;
import javax.transaction.TransactionManager;
public class TransactionScopedEntityManager extends DelegatingEntityManager {
@Inject
private EntityManagerFactory factory;
@Inject
private TransactionManager transactionManager;
private final ThreadLocal<EntityManager> entityManagerHolder = new ThreadLocal<>();
@Override
protected EntityManager getEntityManager() {
EntityManager entityManager = entityManagerHolder.get();
if (entityManager == null) {
try {
Transaction transaction = transactionManager.getTransaction();
if (transaction == null) {
throw new TransactionRequiredException("Not in transaction. Initiate transaction in JTA.");
}
entityManager = factory.createEntityManager();
final EntityManager entityManagerToClose = entityManager;
entityManagerHolder.set(entityManager);
transaction.registerSynchronization(new Synchronization() {
@Override
public void beforeCompletion() {
}
@Override
public void afterCompletion(int status) {
if (entityManagerHolder.get() != entityManagerToClose) {
throw new RuntimeException("Synchronization called from different thread");
}
if (entityManagerToClose != null) {
entityManagerToClose.close();
}
entityManagerHolder.set(null);
}
});
} catch (SystemException | RollbackException e) {
throw new RuntimeException(e);
}
}
return entityManager;
}
@Override
public boolean isOpen() {
try {
return super.isOpen();
} catch (TransactionRequiredException e) {
return false;
}
}
@Override
public void close() {
throw new IllegalStateException("Cannot close container managed entity manager");
}
@Override
public EntityTransaction getTransaction() {
throw new IllegalStateException("Not allowed to create transaction on shared EntityManager");
}
@Override
public void joinTransaction() {
throw new IllegalStateException("Not allowed to create transaction on shared EntityManager");
}
@Override
public EntityManagerFactory getEntityManagerFactory() {
return factory;
}
@Override
public Metamodel getMetamodel() {
return factory.getMetamodel();
}
@Override
public CriteriaBuilder getCriteriaBuilder() {
return factory.getCriteriaBuilder();
}
@Override
public String toString() {
return "Shared EntityManager proxy for target factory [" + factory + "]";
}
}