5052cc5c91ace865b78f6c4546d096d0c066d3c4
[dcaegen2/collectors/hv-ves.git] /
1 /*
2  * ============LICENSE_START=======================================================
3  * dcaegen2-collectors-veshv
4  * ================================================================================
5  * Copyright (C) 2018 NOKIA
6  * ================================================================================
7  * Licensed under the Apache License, Version 2.0 (the "License");
8  * you may not use this file except in compliance with the License.
9  * You may obtain a copy of the License at
10  *
11  *      http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing, software
14  * distributed under the License is distributed on an "AS IS" BASIS,
15  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16  * See the License for the specific language governing permissions and
17  * limitations under the License.
18  * ============LICENSE_END=========================================================
19  */
20 package org.onap.dcae.collectors.veshv.impl.adapters.kafka
21
22 import org.onap.dcae.collectors.veshv.boundary.Sink
23 import org.onap.dcae.collectors.veshv.impl.adapters.ClientContextLogging.withDebug
24 import org.onap.dcae.collectors.veshv.model.ClientContext
25 import org.onap.dcae.collectors.veshv.model.ConsumedMessage
26 import org.onap.dcae.collectors.veshv.model.FailedToConsumeMessage
27 import org.onap.dcae.collectors.veshv.model.MessageDropCause
28 import org.onap.dcae.collectors.veshv.domain.RoutedMessage
29 import org.onap.dcae.collectors.veshv.model.SuccessfullyConsumedMessage
30 import org.onap.dcae.collectors.veshv.domain.VesMessage
31 import org.onap.dcae.collectors.veshv.utils.logging.Logger
32 import org.onap.dcae.collectors.veshv.utils.logging.Marker
33 import org.onap.ves.VesEventOuterClass.CommonEventHeader
34 import reactor.core.publisher.Flux
35 import reactor.kafka.sender.KafkaSender
36 import reactor.kafka.sender.SenderRecord
37
38 /**
39  * @author Piotr Jaszczyk <piotr.jaszczyk@nokia.com>
40  * @since May 2018
41  */
42 internal class KafkaSink(private val sender: KafkaSender<CommonEventHeader, VesMessage>,
43                          private val ctx: ClientContext) : Sink {
44
45     override fun send(messages: Flux<RoutedMessage>): Flux<ConsumedMessage> =
46             messages.map(::vesToKafkaRecord).let { records ->
47                 sender.send(records).map {
48                     val msg = it.correlationMetadata()
49                     if (it.exception() == null) {
50                         logger.trace(ctx::fullMdc, Marker.Invoke()) {
51                             "Message sent to Kafka with metadata: ${it.recordMetadata()}"
52                         }
53                         SuccessfullyConsumedMessage(msg)
54                     } else {
55                         logger.warn(ctx::fullMdc, Marker.Invoke()) {
56                             "Failed to send message to Kafka. Reason: ${it.exception().message}"
57                         }
58                         logger.withDebug(ctx) { log("Kafka send failure details", it.exception()) }
59                         FailedToConsumeMessage(msg, it.exception(), MessageDropCause.KAFKA_FAILURE)
60                     }
61                 }
62             }
63
64     private fun vesToKafkaRecord(routed: RoutedMessage): SenderRecord<CommonEventHeader, VesMessage, RoutedMessage> =
65             SenderRecord.create(
66                     routed.topic,
67                     routed.partition,
68                     FILL_TIMESTAMP_LATER,
69                     routed.message.header,
70                     routed.message,
71                     routed)
72
73     internal fun usesSameSenderAs(other: KafkaSink) = sender === other.sender
74
75     companion object {
76         private val FILL_TIMESTAMP_LATER: Long? = null
77         private val logger = Logger(KafkaSink::class)
78     }
79 }