Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
org.apache.camel.component.jpa.JpaConsumer Maven / Gradle / Ivy
/*
* 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.lang.reflect.Method;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Queue;
import jakarta.persistence.Entity;
import jakarta.persistence.EntityManager;
import jakarta.persistence.EntityManagerFactory;
import jakarta.persistence.LockModeType;
import jakarta.persistence.OptimisticLockException;
import jakarta.persistence.PersistenceException;
import jakarta.persistence.PessimisticLockException;
import jakarta.persistence.Query;
import org.apache.camel.Exchange;
import org.apache.camel.ExchangePropertyKey;
import org.apache.camel.Processor;
import org.apache.camel.support.ObjectHelper;
import org.apache.camel.support.ScheduledBatchPollingConsumer;
import org.apache.camel.util.CastUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.orm.jpa.SharedEntityManagerCreator;
public class JpaConsumer extends ScheduledBatchPollingConsumer {
private static final Logger LOG = LoggerFactory.getLogger(JpaConsumer.class);
private static final Map NOWAIT;
private final EntityManagerFactory entityManagerFactory;
private final TransactionStrategy transactionStrategy;
private EntityManager entityManager;
private QueryFactory queryFactory;
private DeleteHandler deleteHandler;
private DeleteHandler preDeleteHandler;
private String query;
private String namedQuery;
private String nativeQuery;
private LockModeType lockModeType = LockModeType.PESSIMISTIC_WRITE;
private Map parameters;
private Class> resultClass;
private boolean transacted;
private boolean skipLockedEntity;
static {
NOWAIT = new HashMap<>();
NOWAIT.put("jakarta.persistence.lock.timeout", 0L);
}
private static final class DataHolder {
private Exchange exchange;
private Object result;
private EntityManager manager;
private DataHolder() {
}
}
public JpaConsumer(JpaEndpoint endpoint, Processor processor) {
super(endpoint, processor);
this.entityManagerFactory = endpoint.getEntityManagerFactory();
this.transactionStrategy = endpoint.getTransactionStrategy();
}
private void recreateEntityManagerIfNeeded() {
// Recreate EntityManager in case it is disposed due to transaction rollback
if (entityManager == null) {
if (getEndpoint().isSharedEntityManager()) {
this.entityManager = SharedEntityManagerCreator.createSharedEntityManager(entityManagerFactory);
} else {
this.entityManager = entityManagerFactory.createEntityManager();
}
LOG.trace("Recreated EntityManager {} on {}", entityManager, this);
}
}
@Override
protected int poll() throws Exception {
// must reset for each poll
shutdownRunningTask = null;
pendingExchanges = 0;
recreateEntityManagerIfNeeded();
final int[] messagePolled = { 0 };
try {
transactionStrategy.executeInTransaction(() -> {
if (getEndpoint().isJoinTransaction()) {
entityManager.joinTransaction();
}
Queue answer = new LinkedList<>();
Query toExecute = getQueryFactory().createQuery(entityManager);
configureParameters(toExecute);
LOG.trace("Created query {}", toExecute);
List> results = toExecute.getResultList();
LOG.trace("Got result list from query {}", results);
// okay we have some response from jpa so lets mark the consumer as ready
forceConsumerAsReady();
for (Object result : results) {
DataHolder holder = new DataHolder();
holder.manager = entityManager;
holder.result = result;
holder.exchange = createExchange(result, entityManager);
answer.add(holder);
}
PersistenceException cause = null;
try {
messagePolled[0] = processBatch(CastUtils.cast(answer));
} catch (PersistenceException e) {
cause = e;
} catch (Exception e) {
cause = new PersistenceException(e);
}
if (cause != null) {
if (!isTransacted()) {
LOG.warn(
"Error processing last message due: {}. Will commit all previous successful processed message, and ignore this last failure.",
cause.getMessage(), cause);
} else {
// rollback all by throwing exception
throw cause;
}
}
// commit
LOG.debug("Flushing EntityManager");
entityManager.flush();
// must clear after flush
entityManager.clear();
});
} catch (Exception e) {
// Potentially EntityManager could be in an inconsistent state after transaction rollback,
// so disposing it to have it recreated in next poll. cf. Java Persistence API 3.3.2 Transaction Rollback
LOG.debug("Disposing EntityManager {} on {} due to coming transaction rollback", entityManager, this);
entityManager.close();
entityManager = null;
throw new PersistenceException(e);
}
return getEndpoint().getCamelContext().getTypeConverter().convertTo(int.class, messagePolled[0]);
}
@Override
public int processBatch(Queue exchanges) throws Exception {
int total = exchanges.size();
// limit if needed
if (maxMessagesPerPoll > 0 && total > maxMessagesPerPoll) {
LOG.debug("Limiting to maximum messages to poll {} as there were {} messages in this poll.",
maxMessagesPerPoll, total);
total = maxMessagesPerPoll;
}
for (int index = 0; index < total && isBatchAllowed(); index++) {
// only loop if we are started (allowed to run)
DataHolder holder = org.apache.camel.util.ObjectHelper.cast(DataHolder.class, exchanges.poll());
EntityManager batchEntityManager = holder.manager;
Exchange exchange = holder.exchange;
Object result = holder.result;
// add current index and total as properties
exchange.setProperty(ExchangePropertyKey.BATCH_INDEX, index);
exchange.setProperty(ExchangePropertyKey.BATCH_SIZE, total);
exchange.setProperty(ExchangePropertyKey.BATCH_COMPLETE, index == total - 1);
// update pending number of exchanges
pendingExchanges = total - index - 1;
if (lockEntity(result, batchEntityManager)) {
// Run the @PreConsumed callback
createPreDeleteHandler().deleteObject(batchEntityManager, result, exchange);
// process the current exchange
LOG.debug("Processing exchange: {}", exchange);
try {
getProcessor().process(exchange);
} catch (Exception e) {
exchange.setException(e);
}
try {
if (exchange.getException() != null) {
// if we failed then throw exception
throw exchange.getException();
} else {
// Run the @Consumed callback
getDeleteHandler().deleteObject(batchEntityManager, result, exchange);
}
} finally {
releaseExchange(exchange, false);
}
}
}
return total;
}
// Properties
// -------------------------------------------------------------------------
@Override
public JpaEndpoint getEndpoint() {
return (JpaEndpoint) super.getEndpoint();
}
public QueryFactory getQueryFactory() {
if (queryFactory == null) {
queryFactory = createQueryFactory();
if (queryFactory == null) {
throw new IllegalArgumentException(
"No queryType property configured on this consumer, nor an entityType configured on the endpoint so cannot consume");
}
}
return queryFactory;
}
public void setQueryFactory(QueryFactory queryFactory) {
this.queryFactory = queryFactory;
}
public DeleteHandler getDeleteHandler() {
if (deleteHandler == null) {
deleteHandler = createDeleteHandler();
}
return deleteHandler;
}
public void setDeleteHandler(DeleteHandler deleteHandler) {
this.deleteHandler = deleteHandler;
}
public DeleteHandler getPreDeleteHandler() {
if (preDeleteHandler == null) {
preDeleteHandler = createPreDeleteHandler();
}
return preDeleteHandler;
}
public void setPreDeleteHandler(DeleteHandler preDeleteHandler) {
this.preDeleteHandler = preDeleteHandler;
}
public void setParameters(Map params) {
this.parameters = params;
}
public Map getParameters() {
return parameters;
}
public String getNamedQuery() {
return namedQuery;
}
public void setNamedQuery(String namedQuery) {
this.namedQuery = namedQuery;
}
public LockModeType getLockModeType() {
return lockModeType;
}
public void setLockModeType(LockModeType lockModeType) {
this.lockModeType = lockModeType;
}
public String getNativeQuery() {
return nativeQuery;
}
public void setNativeQuery(String nativeQuery) {
this.nativeQuery = nativeQuery;
}
public String getQuery() {
return query;
}
public void setQuery(String query) {
this.query = query;
}
public Class> getResultClass() {
return resultClass;
}
public void setResultClass(Class> resultClass) {
this.resultClass = resultClass;
}
public boolean isTransacted() {
return transacted;
}
/**
* Sets whether to run in transacted mode or not.
*
* This option is default false . When false then all the good messages will commit, and the first
* failed message will rollback. However when true , then all messages will rollback, if just one message
* failed.
*/
public void setTransacted(boolean transacted) {
this.transacted = transacted;
}
/**
* Sets whether to use NOWAIT on lock and silently skip the entity. This allows different instances to process
* entities at the same time but not processing the same entity.
*/
public void setSkipLockedEntity(boolean skipLockedEntity) {
this.skipLockedEntity = skipLockedEntity;
}
public boolean isSkipLockedEntity() {
return skipLockedEntity;
}
// Implementation methods
// -------------------------------------------------------------------------
/**
* A strategy method to lock an object with an exclusive lock so that it can be processed
*
* @param entity the entity to be locked
* @param entityManager entity manager
* @return true if the entity was locked
*/
protected boolean lockEntity(Object entity, EntityManager entityManager) {
if (!getEndpoint().isConsumeLockEntity()) {
return true;
}
try {
LOG.debug("Acquiring exclusive lock on entity: {}", entity);
if (isSkipLockedEntity()) {
entityManager.lock(entity, lockModeType, NOWAIT);
} else {
entityManager.lock(entity, lockModeType);
}
return true;
} catch (Exception e) {
LOG.debug("Failed to achieve lock on entity: {}. Reason: {}", entity, e.getMessage(), e);
if (e instanceof PessimisticLockException || e instanceof OptimisticLockException) {
//transaction marked as rollback can't continue gracefully
throw (PersistenceException) e;
}
//TODO: Find if possible an alternative way to handle results of native queries.
//Result of native queries are Arrays and cannot be locked by all JPA Providers.
return entity.getClass().isArray();
}
}
protected QueryFactory createQueryFactory() {
if (query != null) {
return QueryBuilder.query(query);
} else if (namedQuery != null) {
return QueryBuilder.namedQuery(namedQuery);
} else if (nativeQuery != null) {
if (resultClass != null) {
return QueryBuilder.nativeQuery(nativeQuery, resultClass);
} else {
return QueryBuilder.nativeQuery(nativeQuery);
}
} else {
Class> entityType = getEndpoint().getEntityType();
if (entityType == null) {
return null;
} else {
// Check if we have a property name on the @Entity annotation
String name = getEntityName(entityType);
if (name != null) {
return QueryBuilder.query("select x from " + name + " x");
} else {
// Remove package name of the entity to be conform with JPA 1.0 spec
return QueryBuilder.query("select x from " + entityType.getSimpleName() + " x");
}
}
}
}
protected String getEntityName(Class> clazz) {
Entity entity = clazz.getAnnotation(Entity.class);
// Check if the property name has been defined for Entity annotation
if (entity != null && !entity.name().isEmpty()) {
return entity.name();
} else {
return null;
}
}
protected DeleteHandler createPreDeleteHandler() {
// Look for @PreConsumed to allow custom callback before the Entity has been consumed
final Class> entityType = getEndpoint().getEntityType();
if (entityType != null) {
// Inspect the method(s) annotated with @PreConsumed
List methods = org.apache.camel.util.ObjectHelper.findMethodsWithAnnotation(entityType, PreConsumed.class);
if (methods.size() > 1) {
throw new IllegalStateException(
"Only one method can be annotated with the @PreConsumed annotation but found: " + methods);
} else if (methods.size() == 1) {
// Inspect the parameters of the @PreConsumed method
final Method method = methods.get(0);
final boolean useExchangeParameter = checkParameters(method);
return (EntityManager em, Object entityBean, Exchange exchange) -> {
// The entityBean could be an Object array
if (entityType.isInstance(entityBean)) {
if (useExchangeParameter) {
ObjectHelper.invokeMethod(method, entityBean, exchange);
} else {
ObjectHelper.invokeMethod(method, entityBean);
}
}
};
}
}
// else do nothing
return (EntityManager em, Object entityBean, Exchange exchange) -> {
};
}
protected DeleteHandler createDeleteHandler() {
// look for @Consumed to allow custom callback when the Entity has been consumed
final Class> entityType = getEndpoint().getEntityType();
if (entityType != null) {
List methods = org.apache.camel.util.ObjectHelper.findMethodsWithAnnotation(entityType, Consumed.class);
if (methods.size() > 1) {
throw new IllegalArgumentException(
"Only one method can be annotated with the @Consumed annotation but found: " + methods);
} else if (methods.size() == 1) {
final Method method = methods.get(0);
final boolean useExchangeParameter = checkParameters(method);
return (EntityManager em, Object entityBean, Exchange exchange) -> {
if (entityType.isInstance(entityBean)) {
if (useExchangeParameter) {
ObjectHelper.invokeMethod(method, entityBean, exchange);
} else {
ObjectHelper.invokeMethod(method, entityBean);
}
}
};
}
}
if (getEndpoint().isConsumeDelete()) {
return (EntityManager em, Object entityBean, Exchange exchange) -> em.remove(entityBean);
}
return (EntityManager em, Object entityBean, Exchange exchange) -> {
};
}
protected boolean checkParameters(Method method) {
boolean result = false;
Class>[] receivedParameters = method.getParameterTypes();
if (receivedParameters.length == 1 && receivedParameters[0].isAssignableFrom(Exchange.class)) {
result = true;
}
if (receivedParameters.length > 0 && !result) {
throw new IllegalStateException("@PreConsumed annotated method cannot have parameter other than Exchange");
}
return result;
}
protected void configureParameters(Query query) {
int maxResults = getEndpoint().getMaximumResults();
if (maxResults > 0) {
query.setMaxResults(maxResults);
}
// setup the parameter
if (parameters != null) {
for (Entry entry : parameters.entrySet()) {
query.setParameter(entry.getKey(), entry.getValue());
}
}
}
protected Exchange createExchange(Object result, EntityManager entityManager) {
Exchange exchange = createExchange(false);
exchange.getIn().setBody(result);
exchange.getIn().setHeader(JpaConstants.ENTITY_MANAGER, entityManager);
return exchange;
}
@Override
protected void doInit() throws Exception {
super.doInit();
// need to setup entity manager first
if (getEndpoint().isSharedEntityManager()) {
this.entityManager = SharedEntityManagerCreator.createSharedEntityManager(entityManagerFactory);
} else {
this.entityManager = entityManagerFactory.createEntityManager();
}
LOG.trace("Created EntityManager {} on {}", entityManager, this);
}
@Override
protected void doStop() throws Exception {
// noop
}
@Override
protected void doShutdown() throws Exception {
if (entityManager != null) {
this.entityManager.close();
LOG.trace("Closed EntityManager {} on {}", entityManager, this);
}
super.doShutdown();
}
}