I made it work using wrapped EntityManager, to get thread local entity
manager from factory. See attached files as example. It looks like a bit
overcomplicated, but I think that Spring is doing something similar deep
inside.
On Sunday, August 26, 2012 2:40:43 AM UTC+3, Markus Fischboeck wrote:
>
> Hi!
>
> I've been integrating Guice 3.0 into a desktop application.
> To access my database I use JPA together with EclipseLink. Everything is
> configured straight forward, using the JpaPersistModule guice provides.
> The application runs multiple threads and each thread is using the same
> data access object to read and write data in which the EntityManager is
> injected using constructor injection.
> However it appears that using the DAO within a thread, transactions are
> not commited and entities do not appear in the database.
>
> After inspecting that issue, I've seen that the EntityManager get's
> injected as a singleton, which is not suitable as it is not thread-safe. I
> then tried to inject the EntityManagerFactory instead, which bypasses the
> transaction interception somehow.
>
> Is there a general approach to get this working? Unfortunately I haven't
> found much about this topic especially in conjunction with guice on the
> web. Any help is greatly appreciated!
>
>
>
>
--
You received this message because you are subscribed to the Google Groups
"google-guice" group.
To view this discussion on the web visit
https://groups.google.com/d/msg/google-guice/-/ReK0bCfXsJ8J.
To post to this group, send email to [email protected].
To unsubscribe from this group, send email to
[email protected].
For more options, visit this group at
http://groups.google.com/group/google-guice?hl=en.
import com.google.inject.Inject;
import javax.persistence.EntityManager;
import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation;
public class TransactionalInterceptor implements MethodInterceptor {
@Inject
private EntityManagerProvider entityManagerProvider;
@Override
public Object invoke(MethodInvocation mi) throws Throwable {
EntityManager entityManager = entityManagerProvider.getEntityManager();
entityManager.getTransaction().begin();
try {
Object result = mi.proceed();
entityManager.getTransaction().commit();
return result;
}
catch (Exception e) {
entityManager.getTransaction().rollback();
throw e;
}
}
}
import com.google.inject.Inject;
import com.google.inject.Singleton;
import java.io.IOException;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
@Singleton
public class EntityManagerFlowFilter implements Filter {
private final EntityManagerProvider entityManagerProvider;
@Inject
public EntityManagerFlowFilter(EntityManagerProvider entityManagerProvider) {
this.entityManagerProvider = entityManagerProvider;
}
@Override
public void init(FilterConfig filterConfig) throws ServletException {
entityManagerProvider.init();
}
@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
entityManagerProvider.create();
try {
chain.doFilter(request, response);
}
finally {
entityManagerProvider.close();
}
}
@Override
public void destroy() {
entityManagerProvider.destroy();
}
}
import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.Persistence;
public class EntityManagerProvider {
private final String unitName;
private EntityManagerFactory entityManagerFactory;
private final ThreadLocal<EntityManager> threadLocalEntityManager;
public EntityManagerProvider(String unitName) {
this.unitName = unitName;
threadLocalEntityManager = new ThreadLocal<EntityManager>();
}
public EntityManager getEntityManager() {
EntityManager entityManager = threadLocalEntityManager.get();
if (entityManager == null) {
throw new IllegalStateException("Thread local entityManager is empty");
}
return entityManager;
}
public void init() {
entityManagerFactory = Persistence.createEntityManagerFactory(unitName);
// TODO find better place for this preparation block:
new UserPreparator().prepareUsers(entityManagerFactory.createEntityManager());
}
public void destroy() {
entityManagerFactory.close();
}
public void create() {
EntityManager entityManager = threadLocalEntityManager.get();
if (entityManager != null && entityManager.isOpen()) {
entityManager.close();
}
entityManager = entityManagerFactory.createEntityManager();
threadLocalEntityManager.set(entityManager);
}
public void close() {
EntityManager entityManager = threadLocalEntityManager.get();
if (entityManager != null) {
entityManager.close();
threadLocalEntityManager.remove();
}
}
}
import com.google.inject.Inject;
import com.google.inject.Singleton;
import java.util.Map;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.EntityTransaction;
import javax.persistence.FlushModeType;
import javax.persistence.LockModeType;
import javax.persistence.Query;
import javax.persistence.TypedQuery;
import javax.persistence.criteria.CriteriaBuilder;
import javax.persistence.criteria.CriteriaQuery;
import javax.persistence.metamodel.Metamodel;
@Singleton
public class EntityManagerWrapper implements EntityManager {
private final EntityManagerProvider entityManagerProvider;
@Inject
public EntityManagerWrapper(EntityManagerProvider entityManagerProvider) {
this.entityManagerProvider = entityManagerProvider;
}
@Override
public void persist(Object entity) {
entityManagerProvider.getEntityManager().persist(entity);
}
@Override
public <T> T merge(T entity) {
return entityManagerProvider.getEntityManager().merge(entity);
}
@Override
public void remove(Object entity) {
entityManagerProvider.getEntityManager().remove(entity);
}
@Override
public <T> T find(Class<T> entityClass, Object primaryKey) {
return entityManagerProvider.getEntityManager().find(entityClass, primaryKey);
}
@Override
public <T> T find(Class<T> entityClass, Object primaryKey, Map<String, Object> properties) {
return entityManagerProvider.getEntityManager().find(entityClass, primaryKey, properties);
}
@Override
public <T> T find(Class<T> entityClass, Object primaryKey, LockModeType lockMode) {
return entityManagerProvider.getEntityManager().find(entityClass, primaryKey, lockMode);
}
@Override
public <T> T find(Class<T> entityClass, Object primaryKey, LockModeType lockMode, Map<String, Object> properties) {
return entityManagerProvider.getEntityManager().find(entityClass, primaryKey, lockMode, properties);
}
@Override
public <T> T getReference(Class<T> entityClass, Object primaryKey) {
return entityManagerProvider.getEntityManager().getReference(entityClass, primaryKey);
}
@Override
public void flush() {
entityManagerProvider.getEntityManager().flush();
}
@Override
public void setFlushMode(FlushModeType flushMode) {
entityManagerProvider.getEntityManager().setFlushMode(flushMode);
}
@Override
public FlushModeType getFlushMode() {
return entityManagerProvider.getEntityManager().getFlushMode();
}
@Override
public void lock(Object entity, LockModeType lockMode) {
entityManagerProvider.getEntityManager().lock(entity, lockMode);
}
@Override
public void lock(Object entity, LockModeType lockMode, Map<String, Object> properties) {
entityManagerProvider.getEntityManager().lock(entity, lockMode, properties);
}
@Override
public void refresh(Object entity) {
entityManagerProvider.getEntityManager().refresh(entity);
}
@Override
public void refresh(Object entity, Map<String, Object> properties) {
entityManagerProvider.getEntityManager().refresh(entity, properties);
}
@Override
public void refresh(Object entity, LockModeType lockMode) {
entityManagerProvider.getEntityManager().refresh(entity, lockMode);
}
@Override
public void refresh(Object entity, LockModeType lockMode, Map<String, Object> properties) {
entityManagerProvider.getEntityManager().refresh(entity, lockMode, properties);
}
@Override
public void clear() {
entityManagerProvider.getEntityManager().clear();
}
@Override
public void detach(Object entity) {
entityManagerProvider.getEntityManager().detach(entity);
}
@Override
public boolean contains(Object entity) {
return entityManagerProvider.getEntityManager().contains(entity);
}
@Override
public LockModeType getLockMode(Object entity) {
return entityManagerProvider.getEntityManager().getLockMode(entity);
}
@Override
public void setProperty(String propertyName, Object value) {
entityManagerProvider.getEntityManager().setProperty(propertyName, value);
}
@Override
public Map<String, Object> getProperties() {
return entityManagerProvider.getEntityManager().getProperties();
}
@Override
public Query createQuery(String qlString) {
return entityManagerProvider.getEntityManager().createQuery(qlString);
}
@Override
public <T> TypedQuery<T> createQuery(CriteriaQuery<T> criteriaQuery) {
return entityManagerProvider.getEntityManager().createQuery(criteriaQuery);
}
@Override
public <T> TypedQuery<T> createQuery(String qlString, Class<T> resultClass) {
return entityManagerProvider.getEntityManager().createQuery(qlString, resultClass);
}
@Override
public Query createNamedQuery(String name) {
return entityManagerProvider.getEntityManager().createNamedQuery(name);
}
@Override
public <T> TypedQuery<T> createNamedQuery(String name, Class<T> resultClass) {
return entityManagerProvider.getEntityManager().createNamedQuery(name, resultClass);
}
@Override
public Query createNativeQuery(String sqlString) {
return entityManagerProvider.getEntityManager().createNativeQuery(sqlString);
}
@Override
public Query createNativeQuery(String sqlString, Class resultClass) {
return entityManagerProvider.getEntityManager().createNativeQuery(sqlString, resultClass);
}
@Override
public Query createNativeQuery(String sqlString, String resultSetMapping) {
return entityManagerProvider.getEntityManager().createNativeQuery(sqlString, resultSetMapping);
}
@Override
public void joinTransaction() {
entityManagerProvider.getEntityManager().joinTransaction();
}
@Override
public <T> T unwrap(Class<T> cls) {
return entityManagerProvider.getEntityManager().unwrap(cls);
}
@Override
public Object getDelegate() {
return entityManagerProvider.getEntityManager().getDelegate();
}
@Override
public void close() {
//entityManagerProvider.getEntityManager().close();
// reporting usage of close methods:
Logger.getLogger(getClass().getName()).log(Level.WARNING, "EntityManager.close() method is used", new Exception());
}
@Override
public boolean isOpen() {
return entityManagerProvider.getEntityManager().isOpen();
}
@Override
public EntityTransaction getTransaction() {
//Logger.getLogger(getClass().getName()).log(Level.WARNING, "EntityManager.getTransaction() method is used", new Exception());
return entityManagerProvider.getEntityManager().getTransaction();
}
@Override
public EntityManagerFactory getEntityManagerFactory() {
return entityManagerProvider.getEntityManager().getEntityManagerFactory();
}
@Override
public CriteriaBuilder getCriteriaBuilder() {
return entityManagerProvider.getEntityManager().getCriteriaBuilder();
}
@Override
public Metamodel getMetamodel() {
return entityManagerProvider.getEntityManager().getMetamodel();
}
}