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

org.apache.camel.processor.idempotent.IdempotentConsumer Maven / Gradle / Ivy

There is a newer version: 4.9.0
Show newest version
/*
 * 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.processor.idempotent;

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.atomic.AtomicLong;

import org.apache.camel.AsyncCallback;
import org.apache.camel.AsyncProcessor;
import org.apache.camel.CamelContext;
import org.apache.camel.CamelContextAware;
import org.apache.camel.Exchange;
import org.apache.camel.Expression;
import org.apache.camel.ExtendedExchange;
import org.apache.camel.Navigate;
import org.apache.camel.Processor;
import org.apache.camel.spi.IdAware;
import org.apache.camel.spi.IdempotentRepository;
import org.apache.camel.spi.RouteIdAware;
import org.apache.camel.spi.Synchronization;
import org.apache.camel.support.AsyncProcessorConverterHelper;
import org.apache.camel.support.AsyncProcessorSupport;
import org.apache.camel.support.service.ServiceHelper;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * An implementation of the Idempotent Consumer pattern.
 * 

* This implementation supports idempotent repositories implemented as {@link org.apache.camel.spi.IdempotentRepository}. * * @see org.apache.camel.spi.IdempotentRepository */ public class IdempotentConsumer extends AsyncProcessorSupport implements CamelContextAware, Navigate, IdAware, RouteIdAware { private static final Logger LOG = LoggerFactory.getLogger(IdempotentConsumer.class); private CamelContext camelContext; private String id; private String routeId; private final Expression messageIdExpression; private final AsyncProcessor processor; private final IdempotentRepository idempotentRepository; private final boolean eager; private final boolean completionEager; private final boolean skipDuplicate; private final boolean removeOnFailure; private final AtomicLong duplicateMessageCount = new AtomicLong(); public IdempotentConsumer(Expression messageIdExpression, IdempotentRepository idempotentRepository, boolean eager, boolean completionEager, boolean skipDuplicate, boolean removeOnFailure, Processor processor) { this.messageIdExpression = messageIdExpression; this.idempotentRepository = idempotentRepository; this.eager = eager; this.completionEager = completionEager; this.skipDuplicate = skipDuplicate; this.removeOnFailure = removeOnFailure; this.processor = AsyncProcessorConverterHelper.convert(processor); } @Override public String toString() { return id; } @Override public CamelContext getCamelContext() { return camelContext; } @Override public void setCamelContext(CamelContext camelContext) { this.camelContext = camelContext; } @Override public String getId() { return id; } @Override public void setId(String id) { this.id = id; } public String getRouteId() { return routeId; } public void setRouteId(String routeId) { this.routeId = routeId; } @Override public boolean process(final Exchange exchange, final AsyncCallback callback) { final AsyncCallback target; final String messageId; try { messageId = messageIdExpression.evaluate(exchange, String.class); if (messageId == null) { exchange.setException(new NoMessageIdException(exchange, messageIdExpression)); callback.done(true); return true; } } catch (Exception e) { exchange.setException(e); callback.done(true); return true; } try { boolean newKey; if (eager) { // add the key to the repository newKey = idempotentRepository.add(exchange, messageId); } else { // check if we already have the key newKey = !idempotentRepository.contains(exchange, messageId); } if (!newKey) { // mark the exchange as duplicate exchange.setProperty(Exchange.DUPLICATE_MESSAGE, Boolean.TRUE); // we already have this key so its a duplicate message onDuplicate(exchange, messageId); if (skipDuplicate) { // if we should skip duplicate then we are done LOG.debug("Ignoring duplicate message with id: {} for exchange: {}", messageId, exchange); callback.done(true); return true; } } final Synchronization onCompletion = new IdempotentOnCompletion(idempotentRepository, messageId, eager, removeOnFailure); target = new IdempotentConsumerCallback(exchange, onCompletion, callback, completionEager); if (!completionEager) { // the scope is to do the idempotent completion work as an unit of work on the exchange when its done being routed exchange.adapt(ExtendedExchange.class).addOnCompletion(onCompletion); } } catch (Exception e) { exchange.setException(e); callback.done(true); return true; } // process the exchange return processor.process(exchange, target); } @Override public List next() { if (!hasNext()) { return null; } List answer = new ArrayList<>(1); answer.add(processor); return answer; } @Override public boolean hasNext() { return processor != null; } // Properties // ------------------------------------------------------------------------- public Expression getMessageIdExpression() { return messageIdExpression; } public IdempotentRepository getIdempotentRepository() { return idempotentRepository; } public Processor getProcessor() { return processor; } public long getDuplicateMessageCount() { return duplicateMessageCount.get(); } // Implementation methods // ------------------------------------------------------------------------- @Override protected void doStart() throws Exception { // must add before start so it will have CamelContext injected first if (!camelContext.hasService(idempotentRepository)) { camelContext.addService(idempotentRepository); } ServiceHelper.startService(processor, idempotentRepository); } @Override protected void doStop() throws Exception { ServiceHelper.stopService(processor, idempotentRepository); } @Override protected void doShutdown() throws Exception { ServiceHelper.stopAndShutdownServices(processor, idempotentRepository); camelContext.removeService(idempotentRepository); } public boolean isEager() { return eager; } public boolean isCompletionEager() { return completionEager; } public boolean isSkipDuplicate() { return skipDuplicate; } public boolean isRemoveOnFailure() { return removeOnFailure; } /** * Resets the duplicate message counter to 0L. */ public void resetDuplicateMessageCount() { duplicateMessageCount.set(0L); } private void onDuplicate(Exchange exchange, String messageId) { duplicateMessageCount.incrementAndGet(); onDuplicateMessage(exchange, messageId); } /** * Clear the idempotent repository */ public void clear() { idempotentRepository.clear(); } /** * A strategy method to allow derived classes to overload the behaviour of * processing a duplicate message * * @param exchange the exchange * @param messageId the message ID of this exchange */ protected void onDuplicateMessage(Exchange exchange, String messageId) { // noop } /** * {@link org.apache.camel.AsyncCallback} that is invoked when the idempotent consumer block ends */ private static class IdempotentConsumerCallback implements AsyncCallback { private final Exchange exchange; private final Synchronization onCompletion; private final AsyncCallback callback; private final boolean completionEager; IdempotentConsumerCallback(Exchange exchange, Synchronization onCompletion, AsyncCallback callback, boolean completionEager) { this.exchange = exchange; this.onCompletion = onCompletion; this.callback = callback; this.completionEager = completionEager; } @Override public void done(boolean doneSync) { try { if (completionEager) { if (exchange.isFailed()) { onCompletion.onFailure(exchange); } else { onCompletion.onComplete(exchange); } } // if completion is not eager then the onCompletion is invoked as part of the UoW of the Exchange } finally { callback.done(doneSync); } } @Override public String toString() { return "IdempotentConsumerCallback"; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy