summaryrefslogtreecommitdiff
path: root/src/main/kotlin/dev/dnpm/etl/processor/config/AppKafkaConfiguration.kt
blob: aa3e6cb02cb7dee3942659b953615512fe15efc8 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
/*
 * This file is part of ETL-Processor
 *
 * Copyright (c) 2023       Comprehensive Cancer Center Mainfranken
 * Copyright (c) 2025-2026  Paul-Christian Volkmer, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Affero General Public License as published
 * by the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU Affero General Public License for more details.
 *
 * You should have received a copy of the GNU Affero General Public License
 * along with this program.  If not, see <https://www.gnu.org/licenses/>.
 */

package dev.dnpm.etl.processor.config

import dev.dnpm.etl.processor.consent.ConsentEvaluator
import dev.dnpm.etl.processor.input.KafkaInputListener
import dev.dnpm.etl.processor.monitoring.ConnectionCheckResult
import dev.dnpm.etl.processor.monitoring.ConnectionCheckService
import dev.dnpm.etl.processor.monitoring.KafkaConnectionCheckService
import dev.dnpm.etl.processor.output.KafkaMtbFileSender
import dev.dnpm.etl.processor.output.MtbFileSender
import dev.dnpm.etl.processor.services.RequestProcessor
import dev.dnpm.etl.processor.services.kafka.KafkaResponseProcessor
import org.slf4j.LoggerFactory
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty
import org.springframework.boot.context.properties.EnableConfigurationProperties
import org.springframework.context.ApplicationEventPublisher
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.core.annotation.Order
import org.springframework.kafka.core.ConsumerFactory
import org.springframework.kafka.core.KafkaTemplate
import org.springframework.kafka.listener.ContainerProperties
import org.springframework.kafka.listener.KafkaMessageListenerContainer
import org.springframework.retry.support.RetryTemplate
import reactor.core.publisher.Sinks
import tools.jackson.databind.json.JsonMapper

@Configuration
@EnableConfigurationProperties(value = [KafkaProperties::class])
@ConditionalOnProperty(value = ["app.kafka.servers"])
@ConditionalOnMissingBean(MtbFileSender::class)
@Order(-5)
class AppKafkaConfiguration {
    private val logger = LoggerFactory.getLogger(AppKafkaConfiguration::class.java)

    @Bean
    fun kafkaMtbFileSender(
        kafkaTemplate: KafkaTemplate<String, String>,
        kafkaProperties: KafkaProperties,
        retryTemplate: RetryTemplate,
        jsonMapper: JsonMapper,
    ): MtbFileSender {
        logger.info("Selected 'KafkaMtbFileSender'")
        return KafkaMtbFileSender(kafkaTemplate, kafkaProperties, retryTemplate, jsonMapper)
    }

    @Bean
    fun kafkaResponseListenerContainer(
        consumerFactory: ConsumerFactory<String, String>,
        kafkaProperties: KafkaProperties,
        kafkaResponseProcessor: KafkaResponseProcessor,
    ): KafkaMessageListenerContainer<String, String> {
        val containerProperties = ContainerProperties(kafkaProperties.outputResponseTopic)
        containerProperties.setMessageListener(kafkaResponseProcessor)
        return KafkaMessageListenerContainer(consumerFactory, containerProperties)
    }

    @Bean
    fun kafkaResponseProcessor(
        applicationEventPublisher: ApplicationEventPublisher,
        jsonMapper: JsonMapper,
    ): KafkaResponseProcessor = KafkaResponseProcessor(applicationEventPublisher, jsonMapper)

    @Bean
    @ConditionalOnProperty(value = ["app.kafka.input-topic"])
    fun kafkaInputListenerContainer(
        consumerFactory: ConsumerFactory<String, String>,
        kafkaProperties: KafkaProperties,
        kafkaInputListener: KafkaInputListener,
    ): KafkaMessageListenerContainer<String, String> {
        if (null == kafkaProperties.inputTopic) {
            throw IllegalArgumentException("Kafka input topic cannot be null")
        }
        val containerProperties = ContainerProperties(kafkaProperties.inputTopic)
        containerProperties.setMessageListener(kafkaInputListener)
        return KafkaMessageListenerContainer(consumerFactory, containerProperties)
    }

    @Bean
    @ConditionalOnProperty(value = ["app.kafka.input-topic"])
    fun kafkaInputListener(
        requestProcessor: RequestProcessor,
        jsonMapper: JsonMapper,
        consentEvaluator: ConsentEvaluator,
    ): KafkaInputListener = KafkaInputListener(requestProcessor, consentEvaluator, jsonMapper)

    @Bean
    fun kafkaConnectionCheckService(
        consumerFactory: ConsumerFactory<String, String>,
        connectionCheckUpdateProducer: Sinks.Many<ConnectionCheckResult>,
    ): ConnectionCheckService =
        KafkaConnectionCheckService(
            consumerFactory.createConsumer(),
            connectionCheckUpdateProducer,
        )
}