/** * Copyright (C) 2012 Stephan Classen * Based on guice-perist (Copyright (C) 2010 Google, Inc.) * * 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 com.github.sclassen.guicejpa; import static com.google.common.base.Preconditions.checkNotNull; import java.lang.annotation.Annotation; import javax.persistence.EntityManager; import org.aopalliance.intercept.MethodInterceptor; import com.google.inject.PrivateModule; import com.google.inject.TypeLiteral; import com.google.inject.binder.LinkedBindingBuilder; /** * Abstract super class of {@link ApplicationManagedPersistenceUnitModule} and * {@link ContainerManagedPersistenceUnitModule}. * * @author Stephan Classen */ abstract class AbstractPersistenceUnitModule extends PrivateModule { // ---- Members /** The provider for {@link EntityManager}. */ private final EntityManagerProviderImpl emProvider; /** The annotation for this persistence unit. May be {@code null}. */ private Class<? extends Annotation> annotation; /** The method interceptor for transactional methods. */ private MethodInterceptor transactionInterceptor; /** This defines if the PU uses resource local or jta transactions. */ private TransactionType transactionType = TransactionType.RESOURCE_LOCAL; // ---- Constructors /** * Constructor. * * @param emProvider the provider for {@link EntityManager}. Must not be {@code null}. */ AbstractPersistenceUnitModule(EntityManagerProviderImpl emProvider) { checkNotNull(emProvider); this.emProvider = emProvider; } // ---- Methods /** * @return the persistence service for the persistence unit. */ abstract PersistenceService getPersistenceService(); /** * @return the unit of work for the persistence unit. */ final UnitOfWork getUnitOfWork() { return emProvider; } /** * @return the type of transaction used for the persistence unit. */ final TransactionType getTransactionType() { return transactionType; } /** * Sets the type of transaction to use for the persistence unit. * * @param transactionType the type of transaction. Must not be {@code null}. */ final void setTransactionType(TransactionType transactionType) { checkNotNull(transactionType); this.transactionType = transactionType; } /** * The method interceptor for intercepting transactional methods. * * @param utFacade the {@link UserTransactionFacade}. * May be {@code null} if {@link #transactionType} is {@link TransactionType#RESOURCE_LOCAL}. * @param peTranslator the {@link PersistenceExceptionTranslator}. * May be {@code null} and then exception translation will be not enabled. * @return the interceptor for intercepting transactional methods. */ final MethodInterceptor getTransactionInterceptor(UserTransactionFacade utFacade, PersistenceExceptionTranslator<?> peTranslator) { if (null == transactionInterceptor) { transactionInterceptor = getTxnInterceptor(utFacade, peTranslator); } return transactionInterceptor; } /** * Returns the appropriate interceptor depending on the value of {@link #transactionType}. * * @param utFacade the {@link UserTransactionFacade}. * May be {@code null} if {@link #transactionType} is {@link TransactionType#RESOURCE_LOCAL}. * @param peTranslator the {@link PersistenceExceptionTranslator}. * May be {@code null} and then exception translation will be not enabled. * @return the interceptor for intercepting transactional methods. Never {@code null}. */ private MethodInterceptor getTxnInterceptor(UserTransactionFacade utFacade, PersistenceExceptionTranslator<?> peTranslator) { if (TransactionType.RESOURCE_LOCAL == transactionType) { return new ResourceLocalTxnInterceptor(emProvider, getAnnotation(), peTranslator); } if (TransactionType.JTA == transactionType) { checkNotNull(utFacade, "the JNDI name of the user transaction must be specified if a " + "persistence unit wants to use JTA transactions"); return new JtaTxnInterceptor(emProvider, getAnnotation(), utFacade, peTranslator); } throw new IllegalStateException("invalid transaction type: " + transactionType); } /** * Binds the given type annotated with the annotation of this persistence unit and * exposes it at the same time. * * @param type the type to bind and expose. * @return the bindingBuilder to define what to bind the given type to. */ protected final <T> LinkedBindingBuilder<T> bindAndExpose(TypeLiteral<T> type) { if (null != annotation) { expose(type).annotatedWith(annotation); return bind(type).annotatedWith(annotation); } else { expose(type); return bind(type); } } /** * Binds the given type annotated with the annotation of this persistence unit and exposes * it at the same time. * * @param type the type to bind and expose. * @return the bindingBuilder to define what to bind the given type to. */ protected final <T> LinkedBindingBuilder<T> bindAndExpose(Class<T> type) { if (null != annotation) { expose(type).annotatedWith(annotation); return bind(type).annotatedWith(annotation); } else { expose(type); return bind(type); } } /** * {@inheritDoc} */ @Override protected final void configure() { bind(UnitOfWork.class).toInstance(emProvider); bind(EntityManagerProvider.class).toInstance(emProvider); bind(PersistenceService.class).toInstance(getPersistenceService()); if (null == annotation) { expose(UnitOfWork.class); expose(EntityManagerProvider.class); expose(PersistenceService.class); } else { bind(UnitOfWork.class).annotatedWith(annotation).toInstance(emProvider); bind(EntityManagerProvider.class).annotatedWith(annotation).toInstance(emProvider); bind(PersistenceService.class).annotatedWith(annotation).toInstance(getPersistenceService()); expose(UnitOfWork.class).annotatedWith(annotation); expose(EntityManagerProvider.class).annotatedWith(annotation); expose(PersistenceService.class).annotatedWith(annotation); } configurePersistence(); } /** * Subclasses can overwrite this method to bind (and expose) classes within the context of the * private module which defines the current persistence unit. */ protected void configurePersistence() { // do nothing } /** * Setter for the annotation of the current persistence unit. The annotation is used to expose * the {@link UnitOfWork}, the {@link EntityManagerProvider} and the {@link PersistenceService}. * If the passed in annotation is {@code null} the above classes will be exposed without an annotation. * This does not work if more than one persistence unit is configured. * * @param annotation the annotation to use for binding the current persistence unit. */ final void annotatedWith(Class<? extends Annotation> annotation) { this.annotation = annotation; } /** * @return the annotation used for binding the current persistence unit. */ final Class<? extends Annotation> getAnnotation() { return annotation; } }