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

org.springframework.kafka.requestreply.ReplyingKafkaTemplate Maven / Gradle / Ivy

/*
 * Copyright 2018-2019 the original author or authors.
 *
 * 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
 *
 *      https://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.springframework.kafka.requestreply;

import java.nio.ByteBuffer;
import java.nio.charset.StandardCharsets;
import java.time.Instant;
import java.util.Collection;
import java.util.Iterator;
import java.util.List;
import java.util.UUID;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;

import org.apache.kafka.clients.consumer.ConsumerRecord;
import org.apache.kafka.clients.producer.ProducerRecord;
import org.apache.kafka.common.TopicPartition;
import org.apache.kafka.common.header.Header;
import org.apache.kafka.common.header.Headers;
import org.apache.kafka.common.header.internals.RecordHeader;

import org.springframework.beans.factory.DisposableBean;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.context.SmartLifecycle;
import org.springframework.kafka.KafkaException;
import org.springframework.kafka.core.KafkaTemplate;
import org.springframework.kafka.core.ProducerFactory;
import org.springframework.kafka.listener.BatchMessageListener;
import org.springframework.kafka.listener.ContainerProperties;
import org.springframework.kafka.listener.GenericMessageListenerContainer;
import org.springframework.kafka.support.KafkaHeaders;
import org.springframework.scheduling.TaskScheduler;
import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler;
import org.springframework.util.Assert;

/**
 * A KafkaTemplate that implements request/reply semantics.
 *
 * @param  the key type.
 * @param  the outbound data type.
 * @param  the reply data type.
 *
 * @author Gary Russell
 * @since 2.1.3
 *
 */
public class ReplyingKafkaTemplate extends KafkaTemplate implements BatchMessageListener,
		InitializingBean, SmartLifecycle, DisposableBean, ReplyingKafkaOperations {

	private static final String WITH_CORRELATION_ID = " with correlationId: ";

	private static final long DEFAULT_REPLY_TIMEOUT = 5000L;

	private final GenericMessageListenerContainer replyContainer;

	private final ConcurrentMap> futures = new ConcurrentHashMap<>();

	private final byte[] replyTopic;

	private final byte[] replyPartition;

	private TaskScheduler scheduler = new ThreadPoolTaskScheduler();

	private int phase;

	private boolean autoStartup = true;

	private long replyTimeout = DEFAULT_REPLY_TIMEOUT;

	private boolean schedulerSet;

	private boolean sharedReplyTopic;

	private volatile boolean running;

	public ReplyingKafkaTemplate(ProducerFactory producerFactory,
			GenericMessageListenerContainer replyContainer) {
		this(producerFactory, replyContainer, false);
	}

	public ReplyingKafkaTemplate(ProducerFactory producerFactory,
			GenericMessageListenerContainer replyContainer, boolean autoFlush) {
		super(producerFactory, autoFlush);
		Assert.notNull(replyContainer, "'replyContainer' cannot be null");
		this.replyContainer = replyContainer;
		this.replyContainer.setupMessageListener(this);
		ContainerProperties properties = this.replyContainer.getContainerProperties();
		String tempReplyTopic = null;
		byte[] tempReplyPartition = null;
		if (properties.getTopics() != null && properties.getTopics().length == 1) {
			tempReplyTopic = properties.getTopics()[0];
		}
		else if (properties.getTopicPartitions() != null && properties.getTopicPartitions().length == 1) {
			tempReplyTopic = properties.getTopicPartitions()[0].topic();
			ByteBuffer buffer = ByteBuffer.allocate(4); // NOSONAR magic #
			buffer.putInt(properties.getTopicPartitions()[0].partition());
			tempReplyPartition = buffer.array();
		}
		if (tempReplyTopic == null) {
			this.replyTopic = null;
			this.replyPartition = null;
			this.logger.debug("Could not determine container's reply topic/partition; senders must populate "
					+ "at least the " + KafkaHeaders.REPLY_TOPIC + " header, and optionally the "
					+ KafkaHeaders.REPLY_PARTITION + " header");
		}
		else {
			this.replyTopic = tempReplyTopic.getBytes(StandardCharsets.UTF_8);
			this.replyPartition = tempReplyPartition;
		}
	}

	public void setTaskScheduler(TaskScheduler scheduler) {
		Assert.notNull(scheduler, "'scheduler' cannot be null");
		this.scheduler = scheduler;
		this.schedulerSet = true;
	}

	public void setReplyTimeout(long replyTimeout) {
		Assert.isTrue(replyTimeout >= 0, "'replyTimeout' must be >= 0");
		this.replyTimeout = replyTimeout;
	}

	@Override
	public boolean isRunning() {
		return this.running;
	}

	@Override
	public int getPhase() {
		return this.phase;
	}

	public void setPhase(int phase) {
		this.phase = phase;
	}

	@Override
	public boolean isAutoStartup() {
		return this.autoStartup;
	}

	public void setAutoStartup(boolean autoStartup) {
		this.autoStartup = autoStartup;
	}

	/**
	 * Return the topics/partitions assigned to the replying listener container.
	 * @return the topics/partitions.
	 */
	public Collection getAssignedReplyTopicPartitions() {
		return this.replyContainer.getAssignedPartitions();
	}

	/**
	 * Set to true when multiple templates are using the same topic for replies.
	 * This simply changes logs for unexpected replies to debug instead of error.
	 * @param sharedReplyTopic true if using a shared topic.
	 * @since 2.2
	 */
	public void setSharedReplyTopic(boolean sharedReplyTopic) {
		this.sharedReplyTopic = sharedReplyTopic;
	}

	@Override
	public void afterPropertiesSet() {
		if (!this.schedulerSet) {
			((ThreadPoolTaskScheduler) this.scheduler).initialize();
		}
	}

	@Override
	public synchronized void start() {
		if (!this.running) {
			try {
				afterPropertiesSet();
			}
			catch (Exception e) {
				throw new KafkaException("Failed to initialize", e);
			}
			this.replyContainer.start();
			this.running = true;
		}
	}

	@Override
	public synchronized void stop() {
		if (this.running) {
			this.running = false;
			this.replyContainer.stop();
			this.futures.clear();
		}
	}

	@Override
	public void stop(Runnable callback) {
		stop();
		callback.run();
	}

	@Override
	public RequestReplyFuture sendAndReceive(ProducerRecord record) {
		Assert.state(this.running, "Template has not been start()ed"); // NOSONAR (sync)
		CorrelationKey correlationId = createCorrelationId(record);
		Assert.notNull(correlationId, "the created 'correlationId' cannot be null");
		boolean hasReplyTopic = false;
		Headers headers = record.headers();
		Iterator
iterator = headers.iterator(); while (iterator.hasNext() && !hasReplyTopic) { if (iterator.next().key().equals(KafkaHeaders.REPLY_TOPIC)) { hasReplyTopic = true; } } if (!hasReplyTopic && this.replyTopic != null) { headers.add(new RecordHeader(KafkaHeaders.REPLY_TOPIC, this.replyTopic)); if (this.replyPartition != null) { headers.add(new RecordHeader(KafkaHeaders.REPLY_PARTITION, this.replyPartition)); } } headers.add(new RecordHeader(KafkaHeaders.CORRELATION_ID, correlationId.getCorrelationId())); if (this.logger.isDebugEnabled()) { this.logger.debug("Sending: " + record + WITH_CORRELATION_ID + correlationId); } TemplateRequestReplyFuture future = new TemplateRequestReplyFuture<>(); this.futures.put(correlationId, future); try { future.setSendFuture(send(record)); } catch (Exception e) { this.futures.remove(correlationId); throw new KafkaException("Send failed", e); } scheduleTimeout(record, correlationId); return future; } private void scheduleTimeout(ProducerRecord record, CorrelationKey correlationId) { this.scheduler.schedule(() -> { RequestReplyFuture removed = this.futures.remove(correlationId); if (removed != null) { if (this.logger.isWarnEnabled()) { this.logger.warn("Reply timed out for: " + record + WITH_CORRELATION_ID + correlationId); } removed.setException(new KafkaException("Reply timed out")); } }, Instant.now().plusMillis(this.replyTimeout)); } @Override public void destroy() { if (!this.schedulerSet) { ((ThreadPoolTaskScheduler) this.scheduler).destroy(); } } /** * Subclasses can override this to generate custom correlation ids. * The default implementation is a 16 byte representation of a UUID. * @param record the record. * @return the key. */ protected CorrelationKey createCorrelationId(ProducerRecord record) { UUID uuid = UUID.randomUUID(); byte[] bytes = new byte[16]; // NOSONAR magic # ByteBuffer bb = ByteBuffer.wrap(bytes); bb.putLong(uuid.getMostSignificantBits()); bb.putLong(uuid.getLeastSignificantBits()); return new CorrelationKey(bytes); } @Override public void onMessage(List> data) { data.forEach(record -> { Iterator
iterator = record.headers().iterator(); CorrelationKey correlationId = null; while (correlationId == null && iterator.hasNext()) { Header next = iterator.next(); if (next.key().equals(KafkaHeaders.CORRELATION_ID)) { correlationId = new CorrelationKey(next.value()); } } if (correlationId == null) { this.logger.error("No correlationId found in reply: " + record + " - to use request/reply semantics, the responding server must return the correlation id " + " in the '" + KafkaHeaders.CORRELATION_ID + "' header"); } else { RequestReplyFuture future = this.futures.remove(correlationId); if (future == null) { if (this.sharedReplyTopic) { if (this.logger.isDebugEnabled()) { this.logger.debug(missingCorrelationLogMessage(record, correlationId)); } } else if (this.logger.isErrorEnabled()) { this.logger.error(missingCorrelationLogMessage(record, correlationId)); } } else { if (this.logger.isDebugEnabled()) { this.logger.debug("Received: " + record + WITH_CORRELATION_ID + correlationId); } future.set(record); } } }); } private String missingCorrelationLogMessage(ConsumerRecord record, CorrelationKey correlationId) { return "No pending reply: " + record + WITH_CORRELATION_ID + correlationId + ", perhaps timed out, or using a shared reply topic"; } /** * A listenable future for requests/replies. * * @param the key type. * @param the outbound data type. * @param the reply data type. * TODO: Remove this in 2.3 - adds no value to the super class */ public static class TemplateRequestReplyFuture extends RequestReplyFuture { TemplateRequestReplyFuture() { super(); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy