spring-kafka ReactiveKafkaProducerTemplate 源码
spring-kafka ReactiveKafkaProducerTemplate 代码
文件路径:/spring-kafka/src/main/java/org/springframework/kafka/core/reactive/ReactiveKafkaProducerTemplate.java
/*
* Copyright 2019-2022 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.core.reactive;
import java.util.List;
import java.util.Map;
import java.util.function.Function;
import org.apache.kafka.clients.producer.Producer;
import org.apache.kafka.clients.producer.ProducerRecord;
import org.apache.kafka.common.Metric;
import org.apache.kafka.common.MetricName;
import org.apache.kafka.common.PartitionInfo;
import org.reactivestreams.Publisher;
import org.springframework.beans.factory.DisposableBean;
import org.springframework.kafka.support.KafkaHeaders;
import org.springframework.kafka.support.converter.MessagingMessageConverter;
import org.springframework.kafka.support.converter.RecordMessageConverter;
import org.springframework.messaging.Message;
import org.springframework.util.Assert;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import reactor.kafka.sender.KafkaSender;
import reactor.kafka.sender.SenderOptions;
import reactor.kafka.sender.SenderRecord;
import reactor.kafka.sender.SenderResult;
import reactor.kafka.sender.TransactionManager;
import reactor.util.function.Tuple2;
import reactor.util.function.Tuples;
/**
* Reactive kafka producer operations implementation.
*
* @param <K> the key type.
* @param <V> the value type.
*
* @author Mark Norkin
*
* @since 2.3.0
*/
public class ReactiveKafkaProducerTemplate<K, V> implements AutoCloseable, DisposableBean {
private final KafkaSender<K, V> sender;
private final RecordMessageConverter messageConverter;
public ReactiveKafkaProducerTemplate(SenderOptions<K, V> senderOptions) {
this(senderOptions, new MessagingMessageConverter());
}
public ReactiveKafkaProducerTemplate(SenderOptions<K, V> senderOptions, RecordMessageConverter messageConverter) {
Assert.notNull(senderOptions, "Sender options can not be null");
Assert.notNull(messageConverter, "Message converter can not be null");
this.sender = KafkaSender.create(senderOptions);
this.messageConverter = messageConverter;
}
public <T> Flux<SenderResult<T>> sendTransactionally(Publisher<? extends SenderRecord<K, V, T>> records) {
Flux<Flux<SenderResult<T>>> sendTransactionally = this.sender.sendTransactionally(Flux.just(records));
return sendTransactionally.flatMap(Function.identity());
}
public <T> Mono<SenderResult<T>> sendTransactionally(SenderRecord<K, V, T> record) {
Flux<SenderResult<T>> sendTransactionally = sendTransactionally(Mono.just(record));
return sendTransactionally.single();
}
public Mono<SenderResult<Void>> send(String topic, V value) {
return send(new ProducerRecord<>(topic, value));
}
public Mono<SenderResult<Void>> send(String topic, K key, V value) {
return send(new ProducerRecord<>(topic, key, value));
}
public Mono<SenderResult<Void>> send(String topic, int partition, K key, V value) {
return send(new ProducerRecord<>(topic, partition, key, value));
}
public Mono<SenderResult<Void>> send(String topic, int partition, long timestamp, K key, V value) {
return send(new ProducerRecord<>(topic, partition, timestamp, key, value));
}
public Mono<SenderResult<Void>> send(String topic, Message<?> message) {
@SuppressWarnings("unchecked")
ProducerRecord<K, V> producerRecord = (ProducerRecord<K, V>) this.messageConverter.fromMessage(message, topic);
if (!producerRecord.headers().iterator().hasNext()) { // possibly no Jackson
byte[] correlationId = message.getHeaders().get(KafkaHeaders.CORRELATION_ID, byte[].class);
if (correlationId != null) {
producerRecord.headers().add(KafkaHeaders.CORRELATION_ID, correlationId);
}
}
return send(producerRecord);
}
public Mono<SenderResult<Void>> send(ProducerRecord<K, V> record) {
return send(SenderRecord.create(record, null));
}
public <T> Mono<SenderResult<T>> send(SenderRecord<K, V, T> record) {
return send(Mono.just(record)).single();
}
public <T> Flux<SenderResult<T>> send(Publisher<? extends SenderRecord<K, V, T>> records) {
return this.sender.send(records);
}
public Flux<PartitionInfo> partitionsFromProducerFor(String topic) {
Mono<List<PartitionInfo>> partitionsInfo = doOnProducer(producer -> producer.partitionsFor(topic));
return partitionsInfo.flatMapIterable(Function.identity());
}
public Flux<Tuple2<MetricName, ? extends Metric>> metricsFromProducer() {
return doOnProducer(Producer::metrics)
.flatMapIterable(Map::entrySet)
.map(m -> Tuples.of(m.getKey(), m.getValue()));
}
public <T> Mono<T> doOnProducer(Function<Producer<K, V>, ? extends T> action) {
return this.sender.doOnProducer(action);
}
public TransactionManager transactionManager() {
return this.sender.transactionManager();
}
@Override
public void destroy() {
doClose();
}
@Override
public void close() {
doClose();
}
private void doClose() {
this.sender.close();
}
}
相关信息
相关文章
0
赞
热门推荐
-
2、 - 优质文章
-
3、 gate.io
-
8、 golang
-
9、 openharmony
-
10、 Vue中input框自动聚焦