diff --git a/ozhera-all/ozhera-log/log-agent/pom.xml b/ozhera-all/ozhera-log/log-agent/pom.xml
index c0da5f15e..6c1f8356f 100644
--- a/ozhera-all/ozhera-log/log-agent/pom.xml
+++ b/ozhera-all/ozhera-log/log-agent/pom.xml
@@ -85,7 +85,11 @@
-
+
+ org.apache.kafka
+ kafka-clients
+ 3.5.1
+
org.projectlombok
lombok
diff --git a/ozhera-all/ozhera-log/log-agent/src/main/java/com/xiaomi/mone/log/agent/extension/KafkaExporter.java b/ozhera-all/ozhera-log/log-agent/src/main/java/com/xiaomi/mone/log/agent/extension/KafkaExporter.java
new file mode 100644
index 000000000..2f191f9cc
--- /dev/null
+++ b/ozhera-all/ozhera-log/log-agent/src/main/java/com/xiaomi/mone/log/agent/extension/KafkaExporter.java
@@ -0,0 +1,62 @@
+package com.xiaomi.mone.log.agent.extension;
+
+import com.google.common.collect.Lists;
+import com.xiaomi.mone.log.agent.export.MsgExporter;
+import com.xiaomi.mone.log.api.model.msg.LineMessage;
+import org.apache.kafka.clients.producer.Producer;
+import org.apache.kafka.clients.producer.ProducerRecord;
+
+import java.util.List;
+
+
+public class KafkaExporter implements MsgExporter {
+
+ private Producer mqProducer;
+
+ private String rmqTopic;
+
+ private Integer batchSize;
+
+ public KafkaExporter(Producer mqProducer) {
+ this.mqProducer = mqProducer;
+ }
+
+ @Override
+ public void close() {
+
+ }
+
+ @Override
+ public void export(LineMessage message) {
+ this.export(Lists.newArrayList(message));
+ }
+
+ @Override
+ public void export(List messageList) {
+ if (messageList.isEmpty()) {
+ return;
+ }
+
+ for (LineMessage lineMessage : messageList) {
+ ProducerRecord record = new ProducerRecord<>(rmqTopic, "message", gson.toJson(lineMessage));
+ mqProducer.send(record);
+ }
+
+ }
+
+ public String getRmqTopic() {
+ return rmqTopic;
+ }
+
+ public void setRmqTopic(String rmqTopic) {
+ this.rmqTopic = rmqTopic;
+ }
+
+ public Integer getBatchSize() {
+ return batchSize;
+ }
+
+ public void setBatchSize(Integer batchSize) {
+ this.batchSize = batchSize;
+ }
+}
diff --git a/ozhera-all/ozhera-log/log-agent/src/main/java/com/xiaomi/mone/log/agent/extension/KafkaMQService.java b/ozhera-all/ozhera-log/log-agent/src/main/java/com/xiaomi/mone/log/agent/extension/KafkaMQService.java
new file mode 100644
index 000000000..0db62462a
--- /dev/null
+++ b/ozhera-all/ozhera-log/log-agent/src/main/java/com/xiaomi/mone/log/agent/extension/KafkaMQService.java
@@ -0,0 +1,99 @@
+package com.xiaomi.mone.log.agent.extension;
+
+import com.google.common.base.Preconditions;
+import com.xiaomi.mone.log.agent.export.MsgExporter;
+import com.xiaomi.mone.log.agent.output.Output;
+import com.xiaomi.mone.log.agent.service.OutPutService;
+import com.xiaomi.mone.log.api.model.meta.LogPattern;
+import com.xiaomi.mone.log.api.model.meta.MQConfig;
+import com.xiaomi.youpin.docean.anno.Service;
+import lombok.extern.slf4j.Slf4j;
+import org.apache.commons.lang3.StringUtils;
+import org.apache.kafka.clients.producer.KafkaProducer;
+import org.apache.kafka.clients.producer.Producer;
+import org.apache.kafka.clients.producer.ProducerConfig;
+
+import java.util.Properties;
+import java.util.concurrent.ConcurrentHashMap;
+
+import static com.xiaomi.mone.log.common.Constant.DEFAULT_CONSUMER_GROUP;
+
+@Service(name = "KafkaMQService")
+@Slf4j
+public class KafkaMQService implements OutPutService {
+
+ private ConcurrentHashMap producerMap;
+
+ public void init() {
+ producerMap = new ConcurrentHashMap<>(128);
+ }
+
+ @Override
+ public boolean compare(Output oldOutPut, Output newOutPut) {
+
+ return false;
+ }
+
+ @Override
+ public void preCheckOutput(Output output) {
+ KafkaOutput rmqOutput = (KafkaOutput) output;
+ Preconditions.checkArgument(null != rmqOutput.getClusterInfo(), "rmqOutput.getClusterInfo can not be null");
+ Preconditions.checkArgument(null != rmqOutput.getTopic(), "rmqOutput.getTopic can not be null");
+ }
+
+ @Override
+ public MsgExporter exporterTrans(Output output) throws Exception {
+ KafkaOutput kafkaOutput = (KafkaOutput) output;
+ String nameSrvAddr = kafkaOutput.getClusterInfo();
+ Producer mqProducer = producerMap.get(nameSrvAddr);
+ if (null == mqProducer) {
+ mqProducer = initMqProducer(kafkaOutput);
+ producerMap.put(String.valueOf(nameSrvAddr), mqProducer);
+ }
+
+ KafkaExporter rmqExporter = new KafkaExporter(mqProducer);
+ rmqExporter.setRmqTopic(kafkaOutput.getTopic());
+ rmqExporter.setBatchSize(kafkaOutput.getBatchExportSize());
+
+ return rmqExporter;
+ }
+
+ private Producer initMqProducer(KafkaOutput output) {
+ Properties properties = new Properties();
+ properties.setProperty(ProducerConfig.BOOTSTRAP_SERVERS_CONFIG, output.getClusterInfo());
+ properties.setProperty(ProducerConfig.KEY_SERIALIZER_CLASS_CONFIG, "org.apache.kafka.common.serialization.StringSerializer");
+ properties.setProperty(ProducerConfig.VALUE_SERIALIZER_CLASS_CONFIG, "org.apache.kafka.common.serialization.StringSerializer");
+ if (StringUtils.isNotEmpty(output.getAk()) && StringUtils.isNotEmpty(output.getSk())) {
+ properties.setProperty("security.protocol", "SASL_SSL");
+ properties.setProperty("sasl.mechanism", "PLAIN");
+ properties.setProperty("sasl.jaas.config", "org.apache.kafka.common.security.plain.PlainLoginModule required username=\"" + output.getAk() + "\" password=\"" + output.getSk() + "\";");
+ }
+ Producer producer = new KafkaProducer<>(properties);
+ return producer;
+ }
+
+ @Override
+ public void removeMQ(Output output) {
+ KafkaOutput kafkaOutput = (KafkaOutput) output;
+ if (null != producerMap.get(kafkaOutput.getClusterInfo())) {
+ producerMap.get(kafkaOutput.getClusterInfo()).close();
+ }
+ }
+
+ @Override
+ public Output configOutPut(LogPattern logPattern) {
+
+ MQConfig mqConfig = logPattern.getMQConfig();
+ KafkaOutput output = new KafkaOutput();
+ output.setOutputType(KafkaOutput.OUTPUT_KAFKAMQ);
+ output.setClusterInfo(mqConfig.getClusterInfo());
+ output.setProducerGroup(mqConfig.getProducerGroup());
+ output.setAk(mqConfig.getAk());
+ output.setSk(mqConfig.getSk());
+ output.setTopic(mqConfig.getTopic());
+ output.setPartitionCnt(mqConfig.getPartitionCnt());
+ output.setTag(mqConfig.getTag());
+ output.setProducerGroup(DEFAULT_CONSUMER_GROUP + (null == logPattern.getPatternCode() ? "" : logPattern.getPatternCode()));
+ return output;
+ }
+}
diff --git a/ozhera-all/ozhera-log/log-agent/src/main/java/com/xiaomi/mone/log/agent/extension/KafkaOutput.java b/ozhera-all/ozhera-log/log-agent/src/main/java/com/xiaomi/mone/log/agent/extension/KafkaOutput.java
new file mode 100644
index 000000000..58f9128bc
--- /dev/null
+++ b/ozhera-all/ozhera-log/log-agent/src/main/java/com/xiaomi/mone/log/agent/extension/KafkaOutput.java
@@ -0,0 +1,45 @@
+package com.xiaomi.mone.log.agent.extension;
+
+import com.xiaomi.mone.log.agent.output.Output;
+import lombok.Data;
+import lombok.EqualsAndHashCode;
+
+import java.io.Serializable;
+
+@Data
+@EqualsAndHashCode
+public class KafkaOutput extends Output implements Serializable {
+
+ public static final String OUTPUT_KAFKAMQ = "kafkamq";
+
+ private String serviceName = "KafkaMQService";
+
+ /**
+ * mq fill:namesrv_addr
+ */
+ private String clusterInfo;
+
+ private String producerGroup;
+
+ private String orgId;
+
+ private String ak;
+
+ private String sk;
+
+ private String topic;
+
+ private Integer partitionCnt;
+
+ private Integer batchExportSize;
+
+ @Override
+ public String getEndpoint() {
+ return clusterInfo;
+ }
+
+ @Override
+ public String getServiceName() {
+ return serviceName;
+ }
+}