/**
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.apache.camel.component.jpa;
import java.util.Map;
import java.util.concurrent.ExecutorService;
import javax.persistence.EntityManagerFactory;
import org.apache.camel.Endpoint;
import org.apache.camel.impl.UriEndpointComponent;
import org.apache.camel.spi.Metadata;
import org.apache.camel.util.ObjectHelper;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.transaction.PlatformTransactionManager;
import org.springframework.transaction.support.TransactionTemplate;
/**
* A JPA Component
*
* @version
*/
public class JpaComponent extends UriEndpointComponent {
private static final Logger LOG = LoggerFactory.getLogger(JpaComponent.class);
private ExecutorService pollingConsumerExecutorService;
private EntityManagerFactory entityManagerFactory;
private PlatformTransactionManager transactionManager;
@Metadata(defaultValue = "true")
private boolean joinTransaction = true;
private boolean sharedEntityManager;
public JpaComponent() {
super(JpaEndpoint.class);
}
// Properties
//-------------------------------------------------------------------------
public EntityManagerFactory getEntityManagerFactory() {
return entityManagerFactory;
}
/**
* To use the {@link EntityManagerFactory}. This is strongly recommended to configure.
*/
public void setEntityManagerFactory(EntityManagerFactory entityManagerFactory) {
this.entityManagerFactory = entityManagerFactory;
}
public PlatformTransactionManager getTransactionManager() {
return transactionManager;
}
/**
* To use the {@link PlatformTransactionManager} for managing transactions.
*/
public void setTransactionManager(PlatformTransactionManager transactionManager) {
this.transactionManager = transactionManager;
}
public boolean isJoinTransaction() {
return joinTransaction;
}
/**
* The camel-jpa component will join transaction by default.
* You can use this option to turn this off, for example if you use LOCAL_RESOURCE and join transaction
* doesn't work with your JPA provider. This option can also be set globally on the JpaComponent,
* instead of having to set it on all endpoints.
*/
public void setJoinTransaction(boolean joinTransaction) {
this.joinTransaction = joinTransaction;
}
public boolean isSharedEntityManager() {
return sharedEntityManager;
}
/**
* Whether to use Spring's SharedEntityManager for the consumer/producer.
* Note in most cases joinTransaction should be set to false as this is not an EXTENDED EntityManager.
*/
public void setSharedEntityManager(boolean sharedEntityManager) {
this.sharedEntityManager = sharedEntityManager;
}
synchronized ExecutorService getOrCreatePollingConsumerExecutorService() {
if (pollingConsumerExecutorService == null) {
LOG.debug("Creating thread pool for JpaPollingConsumer to support polling using timeout");
pollingConsumerExecutorService = getCamelContext().getExecutorServiceManager().newDefaultThreadPool(this, "JpaPollingConsumer");
}
return pollingConsumerExecutorService;
}
// Implementation methods
//-------------------------------------------------------------------------
@Override
protected Endpoint createEndpoint(String uri, String path, Map<String, Object> options) throws Exception {
JpaEndpoint endpoint = new JpaEndpoint(uri, this);
endpoint.setJoinTransaction(isJoinTransaction());
endpoint.setSharedEntityManager(isSharedEntityManager());
// lets interpret the next string as a class
if (ObjectHelper.isNotEmpty(path)) {
// provide the class loader of this component to work in OSGi environments as camel-jpa must be able
// to resolve the entity classes
Class<?> type = getCamelContext().getClassResolver().resolveClass(path, JpaComponent.class.getClassLoader());
if (type != null) {
endpoint.setEntityType(type);
}
}
return endpoint;
}
@Override
protected void doStart() throws Exception {
super.doStart();
// lookup entity manager factory and use it if only one provided
if (entityManagerFactory == null) {
Map<String, EntityManagerFactory> map = getCamelContext().getRegistry().findByTypeWithName(EntityManagerFactory.class);
if (map != null) {
if (map.size() == 1) {
entityManagerFactory = map.values().iterator().next();
LOG.info("Using EntityManagerFactory found in registry with id ["
+ map.keySet().iterator().next() + "] " + entityManagerFactory);
} else {
LOG.debug("Could not find a single EntityManagerFactory in registry as there was " + map.size() + " instances.");
}
}
} else {
LOG.info("Using EntityManagerFactory configured: " + entityManagerFactory);
}
// lookup transaction manager and use it if only one provided
if (transactionManager == null) {
Map<String, PlatformTransactionManager> map = getCamelContext().getRegistry().findByTypeWithName(PlatformTransactionManager.class);
if (map != null) {
if (map.size() == 1) {
transactionManager = map.values().iterator().next();
LOG.info("Using TransactionManager found in registry with id ["
+ map.keySet().iterator().next() + "] " + transactionManager);
} else {
LOG.debug("Could not find a single TransactionManager in registry as there was " + map.size() + " instances.");
}
}
} else {
LOG.info("Using TransactionManager configured on this component: " + transactionManager);
}
// transaction manager could also be hidden in a template
if (transactionManager == null) {
Map<String, TransactionTemplate> map = getCamelContext().getRegistry().findByTypeWithName(TransactionTemplate.class);
if (map != null) {
if (map.size() == 1) {
transactionManager = map.values().iterator().next().getTransactionManager();
LOG.info("Using TransactionManager found in registry with id ["
+ map.keySet().iterator().next() + "] " + transactionManager);
} else {
LOG.debug("Could not find a single TransactionTemplate in registry as there was " + map.size() + " instances.");
}
}
}
// warn about missing configuration
if (entityManagerFactory == null) {
LOG.warn("No EntityManagerFactory has been configured on this JpaComponent. Each JpaEndpoint will auto create their own EntityManagerFactory.");
}
if (transactionManager == null) {
LOG.warn("No TransactionManager has been configured on this JpaComponent. Each JpaEndpoint will auto create their own JpaTransactionManager.");
}
}
@Override
protected void doStop() throws Exception {
super.doStop();
if (pollingConsumerExecutorService != null) {
getCamelContext().getExecutorServiceManager().shutdown(pollingConsumerExecutorService);
pollingConsumerExecutorService = null;
}
}
}