All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.crnk.jpa.JpaModule Maven / Gradle / Ivy

package io.crnk.jpa;

import io.crnk.core.engine.dispatcher.Response;
import io.crnk.core.engine.error.ExceptionMapper;
import io.crnk.core.engine.filter.AbstractDocumentFilter;
import io.crnk.core.engine.filter.DocumentFilterChain;
import io.crnk.core.engine.filter.DocumentFilterContext;
import io.crnk.core.engine.information.resource.ResourceField;
import io.crnk.core.engine.information.resource.ResourceFieldType;
import io.crnk.core.engine.information.resource.ResourceInformation;
import io.crnk.core.engine.information.resource.ResourceInformationProvider;
import io.crnk.core.engine.internal.utils.ClassUtils;
import io.crnk.core.engine.internal.utils.ExceptionUtil;
import io.crnk.core.engine.internal.utils.PreconditionUtil;
import io.crnk.core.engine.properties.PropertiesProvider;
import io.crnk.core.engine.transaction.TransactionRunner;
import io.crnk.core.module.InitializingModule;
import io.crnk.core.queryspec.QuerySpec;
import io.crnk.core.repository.RelationshipRepositoryV2;
import io.crnk.core.repository.ResourceRepositoryV2;
import io.crnk.core.repository.decorate.RelationshipRepositoryDecorator;
import io.crnk.core.repository.decorate.RepositoryDecoratorFactory;
import io.crnk.core.repository.decorate.ResourceRepositoryDecorator;
import io.crnk.core.resource.meta.DefaultHasMoreResourcesMetaInformation;
import io.crnk.core.resource.meta.DefaultPagedMetaInformation;
import io.crnk.jpa.internal.*;
import io.crnk.jpa.internal.query.backend.querydsl.QuerydslQueryImpl;
import io.crnk.jpa.meta.JpaMetaProvider;
import io.crnk.jpa.meta.MetaEntity;
import io.crnk.jpa.meta.internal.JpaMetaEnricher;
import io.crnk.jpa.query.JpaQueryFactory;
import io.crnk.jpa.query.JpaQueryFactoryContext;
import io.crnk.jpa.query.querydsl.QuerydslQueryFactory;
import io.crnk.jpa.query.querydsl.QuerydslRepositoryFilter;
import io.crnk.jpa.query.querydsl.QuerydslTranslationContext;
import io.crnk.jpa.query.querydsl.QuerydslTranslationInterceptor;
import io.crnk.meta.MetaLookup;
import io.crnk.meta.MetaModuleExtension;
import io.crnk.meta.provider.MetaPartition;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import javax.persistence.Entity;
import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import java.io.Serializable;
import java.lang.reflect.Constructor;
import java.util.*;
import java.util.concurrent.Callable;

/**
 * Crnk module that adds support to expose JPA entities as repositories. It
 * supports:
 * 

*

    *
  • Sorting
  • *
  • Filtering
  • *
  • Access to relationships for any operation (sorting, filtering, etc.)
  • *
  • Includes for relationships
  • *
  • Paging
  • *
  • Mapping to DTOs
  • *
  • Criteria API and QueryDSL support
  • *
  • Computed attributes that map JPA Criteria/QueryDSL expressions to DTO * attributes
  • *
  • JpaRepositoryFilter to customize the repositories
  • *
  • Client and server support
  • *
  • No need for crnk annotations by default. Reads the entity * annotations.
  • *
*

*

* Not supported so far: *

*

    *
  • Selection of fields, always all fields are returned.
  • *
  • Sorting and filtering on related resources. Consider doing separate * requests on the relations where necessary.
  • *
*/ public class JpaModule implements InitializingModule { private static final String MODULE_NAME = "jpa"; private Logger logger = LoggerFactory.getLogger(JpaModule.class); private EntityManagerFactory emFactory; private EntityManager em; private ResourceInformationProvider resourceInformationProvider; private TransactionRunner transactionRunner; private ModuleContext context; private JpaModuleConfig config; private JpaMetaEnricher metaEnricher; private MetaLookup jpaMetaLookup; private JpaMetaProvider jpaMetaProvider; /** * Constructor used on client side. */ // protected for CDI protected JpaModule() { } /** * Constructor used on server side. */ private JpaModule(JpaModuleConfig config, EntityManagerFactory emFactory, EntityManager em, TransactionRunner transactionRunner) { this(); this.config = config; this.emFactory = emFactory; this.em = em; this.transactionRunner = transactionRunner; } /** * Creates a new JpaModule for a Crnk client. * * @return module */ public static JpaModule newClientModule() { return new JpaModule(); } /** * Creates a new JpaModule for a Crnk server. No entities are by * default exposed as JSON API resources. Make use of * {@link #addRepository(JpaRepositoryConfig)} to add resources. * * @param em to use * @param transactionRunner to use * @return created module * @deprecated use with JpaModuleConfig */ @Deprecated public static JpaModule newServerModule(EntityManager em, TransactionRunner transactionRunner) { return new JpaModule(new JpaModuleConfig(), null, em, transactionRunner); } /** * Creates a new JpaModule for a Crnk server. All entities managed by * the provided EntityManagerFactory are registered to the module and * exposed as JSON API resources if not later configured otherwise. * * @param emFactory to retrieve the managed entities. * @param em to use * @param transactionRunner to use * @return created module * @deprecated use with JpaModuleConfig */ @Deprecated public static JpaModule newServerModule(EntityManagerFactory emFactory, EntityManager em, TransactionRunner transactionRunner) { JpaModuleConfig config = new JpaModuleConfig(); config.exposeAllEntities(emFactory); return new JpaModule(config, emFactory, em, transactionRunner); } /** * Creates a new JpaModule for a Crnk server. No entities are by * default exposed as JSON API resources. Make use of * {@link #addRepository(JpaRepositoryConfig)} to add resources. * * @param em to use * @param transactionRunner to use * @return created module */ public static JpaModule createServerModule(JpaModuleConfig config, EntityManager em, TransactionRunner transactionRunner) { return new JpaModule(config, null, em, transactionRunner); } /** * Adds the given filter to this module. Filter will be used by all * repositories managed by this module. * * @param filter to add * @deprecated use {@link JpaModuleConfig} */ @Deprecated public void addFilter(JpaRepositoryFilter filter) { config.addFilter(filter); } /** * @deprecated use {@link JpaModuleConfig} */ @Deprecated public void removeFilter(JpaRepositoryFilter filter) { config.removeFilter(filter); } /** * @deprecated use {@link JpaModuleConfig} */ @Deprecated public List getFilters() { return config.getFilters(); } /** * @deprecated use {@link JpaModuleConfig} */ @Deprecated public void setRepositoryFactory(JpaRepositoryFactory repositoryFactory) { checkNotInitialized(); this.config.setRepositoryFactory(repositoryFactory); } /** * @deprecated use {@link JpaModuleConfig} */ @Deprecated public Set> getResourceClasses() { return config.getResourceClasses(); } /** * @deprecated use {@link JpaModuleConfig} */ @Deprecated public void addRepository(JpaRepositoryConfig config) { checkNotInitialized(); this.config.addRepository(config); } /** * Removes the resource with the given type from this module. * * @param resourse class (entity or mapped dto) * @param resourceClass to remove */ public void removeRepository(Class resourceClass) { checkNotInitialized(); config.removeRepository(resourceClass); } /** * Removes all entity classes registered by default. Use * {@link #addRepository(JpaRepositoryConfig)} (Class)} or * classes manually. */ public void removeRepositories() { checkNotInitialized(); config.removeRepositories(); } @Override public String getModuleName() { return MODULE_NAME; } private void checkNotInitialized() { PreconditionUtil.assertNull("module is already initialized, no further changes can be performed", context); } @Override public void setupModule(ModuleContext context) { this.context = context; Set jpaTypes = new HashSet<>(); if (config != null) { for (JpaRepositoryConfig config : config.getRepositories()) { jpaTypes.add(config.getEntityClass()); } } jpaMetaProvider = new JpaMetaProvider(jpaTypes); jpaMetaLookup = new MetaLookup(); jpaMetaLookup.addProvider(jpaMetaProvider); jpaMetaLookup.setModuleContext(context); jpaMetaLookup.initialize(); if (config != null) { initQueryFactory(); } context.addResourceInformationBuilder(getResourceInformationProvider(context.getPropertiesProvider())); context.addExceptionMapper(new OptimisticLockExceptionMapper()); context.addExceptionMapper(new PersistenceExceptionMapper(context)); context.addExceptionMapper(new PersistenceRollbackExceptionMapper(context)); addHibernateConstraintViolationExceptionMapper(); addTransactionRollbackExceptionMapper(); context.addRepositoryDecoratorFactory(new JpaRepositoryDecoratorFactory()); if (em != null) { metaEnricher = new JpaMetaEnricher(); // enrich resource meta model with JPA information where incomplete MetaModuleExtension metaModuleExtension = new MetaModuleExtension(); metaModuleExtension.addProvider(metaEnricher.getProvider()); context.addExtension(metaModuleExtension); setupTransactionMgmt(); } } private void initQueryFactory() { JpaQueryFactory queryFactory = config.getQueryFactory(); queryFactory.initalize(new JpaQueryFactoryContext() { @Override public EntityManager getEntityManager() { return em; } @Override public MetaPartition getMetaPartition() { return jpaMetaProvider.getPartition(); } }); if (queryFactory instanceof QuerydslQueryFactory) { QuerydslQueryFactory querydslFactory = (QuerydslQueryFactory) queryFactory; querydslFactory.addInterceptor(new JpaQuerydslTranslationInterceptor()); } } @Override public void init() { if (em != null) { setupServerRepositories(); } } private void addHibernateConstraintViolationExceptionMapper() { // may not be available depending on environment if (ClassUtils.existsClass("org.hibernate.exception.ConstraintViolationException")) { ExceptionUtil.wrapCatchedExceptions(new Callable() { @Override public Object call() throws Exception { Class mapperClass = Class.forName("io.crnk.jpa.internal.HibernateConstraintViolationExceptionMapper"); Constructor constructor = mapperClass.getConstructor(); ExceptionMapper mapper = (ExceptionMapper) constructor.newInstance(); context.addExceptionMapper(mapper); return null; } }); } } private void addTransactionRollbackExceptionMapper() { // may not be available depending on environment if (ClassUtils.existsClass("javax.transaction.RollbackException")) { ExceptionUtil.wrapCatchedExceptions(new Callable() { @Override public Object call() throws Exception { Class mapperClass = Class.forName("io.crnk.jpa.internal.TransactionRollbackExceptionMapper"); Constructor constructor = mapperClass.getConstructor(ModuleContext.class); ExceptionMapper mapper = (ExceptionMapper) constructor.newInstance(context); context.addExceptionMapper(mapper); return null; } }); } } protected void setupTransactionMgmt() { context.addFilter(new AbstractDocumentFilter() { @Override public Response filter(final DocumentFilterContext context, final DocumentFilterChain chain) { return transactionRunner.doInTransaction(new Callable() { @Override public Response call() throws Exception { return chain.doFilter(context); } }); } }); } private void setupServerRepositories() { metaEnricher.setMetaProvider(jpaMetaProvider); for (JpaRepositoryConfig config : config.getRepositories()) { setupRepository(config); } } private void setupRepository(JpaRepositoryConfig repositoryConfig) { if (repositoryConfig.getListMetaClass() == DefaultPagedMetaInformation.class && !isTotalResourceCountUsed()) { // TODO not that nice... repositoryConfig.setListMetaClass(DefaultHasMoreResourcesMetaInformation.class); } Class resourceClass = repositoryConfig.getResourceClass(); MetaEntity metaEntity = jpaMetaProvider.getMeta(repositoryConfig.getEntityClass()); if (isValidEntity(metaEntity)) { JpaRepositoryFactory repositoryFactory = config.getRepositoryFactory(); JpaEntityRepository jpaRepository = repositoryFactory.createEntityRepository(this, repositoryConfig); ResourceRepositoryV2 repository = filterResourceCreation(resourceClass, jpaRepository); context.addRepository(repository); setupRelationshipRepositories(resourceClass, repositoryConfig.getResourceClass() != repositoryConfig.getEntityClass()); } } private ResourceRepositoryV2 filterResourceCreation(Class resourceClass, JpaEntityRepository repository) { JpaEntityRepository filteredRepository = repository; for (JpaRepositoryFilter filter : config.getFilters()) { if (filter.accept(resourceClass)) { filteredRepository = filter.filterCreation(filteredRepository); } } return filteredRepository; } private RelationshipRepositoryV2 filterRelationshipCreation(Class resourceClass, JpaRelationshipRepository repository) { JpaRelationshipRepository filteredRepository = repository; for (JpaRepositoryFilter filter : config.getFilters()) { if (filter.accept(resourceClass)) { filteredRepository = filter.filterCreation(filteredRepository); } } return filteredRepository; } /** * Sets up relationship repositories for the given document class. In case * of a mapper the resource class might not correspond to the entity class. */ private void setupRelationshipRepositories(Class resourceClass, boolean mapped) { if (context.getResourceInformationBuilder().accept(resourceClass)) { ResourceInformation information = context.getResourceInformationBuilder().build(resourceClass); for (ResourceField field : information.getFields()) { if (field.getResourceFieldType() != ResourceFieldType.RELATIONSHIP) { continue; } Class attrType = field.getElementType(); boolean isEntity = attrType.getAnnotation(Entity.class) != null; if (isEntity) { setupRelationshipRepositoryForEntity(resourceClass, field); } else { setupRelationshipRepositoryForResource(resourceClass, field); } } } } private void setupRelationshipRepositoryForEntity(Class resourceClass, ResourceField field) { // normal entity association Class attrType = field.getElementType(); JpaRepositoryConfig attrConfig = getRepositoryConfig(attrType); // only include relations that are exposed as repositories if (attrConfig != null) { JpaRepositoryFactory repositoryFactory = config.getRepositoryFactory(); RelationshipRepositoryV2 relationshipRepository = filterRelationshipCreation(attrType, repositoryFactory.createRelationshipRepository(this, resourceClass, attrConfig)); context.addRepository(relationshipRepository); } } private void setupRelationshipRepositoryForResource(Class resourceClass, ResourceField field) { Class attrImplClass = field.getElementType(); JpaRepositoryConfig attrConfig = getRepositoryConfig(attrImplClass); PreconditionUtil.verify(attrConfig != null && attrConfig.getMapper() != null, "no mapped entity for %s reference from %s.%s registered", field.getOppositeResourceType(), field.getParentResourceInformation().getResourceType(), field.getUnderlyingName()); JpaRepositoryConfig targetConfig = getRepositoryConfig(attrImplClass); Class targetResourceClass = targetConfig.getResourceClass(); JpaRepositoryFactory repositoryFactory = config.getRepositoryFactory(); RelationshipRepositoryV2 relationshipRepository = filterRelationshipCreation(targetResourceClass, repositoryFactory.createRelationshipRepository(this, resourceClass, attrConfig)); context.addRepository(relationshipRepository); } private boolean isValidEntity(MetaEntity metaEntity) { if (metaEntity.getPrimaryKey() == null) { logger.warn("{} has no primary key and will be ignored", metaEntity.getName()); return false; } if (metaEntity.getPrimaryKey().getElements().size() > 1) { logger.warn("{} has a compound primary key and will be ignored", metaEntity.getName()); return false; } return true; } /** * @return ResourceInformationProvider used to describe JPA classes. */ public ResourceInformationProvider getResourceInformationProvider(PropertiesProvider propertiesProvider) { if (resourceInformationProvider == null) { resourceInformationProvider = new JpaResourceInformationProvider(propertiesProvider); } return resourceInformationProvider; } /** * Sets the information builder to use to read JPA classes. See * {@link JpaResourceInformationProvider}} */ public void setResourceInformationProvider(ResourceInformationProvider resourceInformationProvider) { PreconditionUtil.verify(this.resourceInformationProvider == null, "already set"); this.resourceInformationProvider = resourceInformationProvider; } /** * @return {@link JpaQueryFactory}} implementation used to create JPA * queries. * @deprecated use JpaModuleConfig */ @Deprecated public JpaQueryFactory getQueryFactory() { return config.getQueryFactory(); } /** * @deprecated use JpaModuleConfig */ @Deprecated public void setQueryFactory(JpaQueryFactory queryFactory) { config.setQueryFactory(queryFactory); if (context != null) { initQueryFactory(); } } /** * @return {@link EntityManager}} in use. */ public EntityManager getEntityManager() { return em; } /** * @return {@link EntityManagerFactory}} in use. */ public EntityManagerFactory getEntityManagerFactory() { return emFactory; } /** * @return config * @deprecated use {@link JpaModuleConfig} */ @Deprecated public JpaRepositoryConfig getRepositoryConfig(Class resourceClass) { return config.getRepository(resourceClass); } public MetaLookup getJpaMetaLookup() { return jpaMetaLookup; } /** * @deprecated use {@link JpaModuleConfig} */ @Deprecated public boolean isTotalResourceCountUsed() { return config.isTotalResourceCountUsed(); } /** * @deprecated use {@link JpaModuleConfig} */ public void setTotalResourceCountUsed(boolean totalResourceCountUsed) { config.setTotalResourceCountUsed(totalResourceCountUsed); } /** * @return true if a resource for the given resourceClass is managed by * this module. */ public boolean hasRepository(Class resourceClass) { return config.hasRepository(resourceClass); } public JpaMetaProvider getJpaMetaProvider() { return jpaMetaProvider; } private final class JpaQuerydslTranslationInterceptor implements QuerydslTranslationInterceptor { @Override public void intercept(QuerydslQueryImpl query, QuerydslTranslationContext translationContext) { JpaRequestContext requestContext = (JpaRequestContext) query.getPrivateData(); if (requestContext != null) { for (JpaRepositoryFilter filter : config.getFilters()) { invokeFilter(filter, requestContext, translationContext); } } } private void invokeFilter(JpaRepositoryFilter filter, JpaRequestContext requestContext, QuerydslTranslationContext translationContext) { if (filter instanceof QuerydslRepositoryFilter) { Object repository = requestContext.getRepository(); QuerySpec querySpec = requestContext.getQuerySpec(); ((QuerydslRepositoryFilter) filter).filterQueryTranslation(repository, querySpec, translationContext); } } } class JpaRepositoryDecoratorFactory implements RepositoryDecoratorFactory { @Override public ResourceRepositoryDecorator decorateRepository( ResourceRepositoryV2 repository) { JpaRepositoryConfig config = getRepositoryConfig(repository.getResourceClass()); if (config != null) { return config.getRepositoryDecorator(); } return null; } @Override public RelationshipRepositoryDecorator decorateRepository( RelationshipRepositoryV2 repository) { JpaRepositoryConfig config = getRepositoryConfig(repository.getSourceResourceClass()); if (config != null) { return config.getRepositoryDecorator(repository.getTargetResourceClass()); } return null; } } }