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

net.nemerosa.ontrack.extension.influxdb.DefaultInfluxDBConnection.kt Maven / Gradle / Ivy

package net.nemerosa.ontrack.extension.influxdb

import net.nemerosa.ontrack.common.Time
import net.nemerosa.ontrack.model.security.ApplicationManagement
import net.nemerosa.ontrack.model.security.SecurityService
import net.nemerosa.ontrack.model.support.StartupService
import okhttp3.OkHttpClient
import org.influxdb.BatchOptions
import org.influxdb.InfluxDB
import org.influxdb.InfluxDBFactory
import org.slf4j.Logger
import org.slf4j.LoggerFactory
import java.time.Duration
import java.time.LocalDateTime
import java.util.concurrent.locks.ReentrantReadWriteLock
import kotlin.concurrent.write

class DefaultInfluxDBConnection(
    private val influxDBExtensionProperties: InfluxDBExtensionProperties,
    private val securityService: SecurityService
) : InfluxDBConnection, StartupService {

    private val logger: Logger = LoggerFactory.getLogger(DefaultInfluxDBConnection::class.java)

    private var internalConnection: InfluxDB? = null
    private var lastCheck: LocalDateTime? = null

    private val internalConnectionLock = ReentrantReadWriteLock()

    override fun getName(): String = "InfluxDB connection"

    override fun startupOrder(): Int = StartupService.SYSTEM_REGISTRATION

    override fun start() {
        if (current != null && isValid(true)) {
            logger.info("InfluxDB connection to ${influxDBExtensionProperties.uri} is OK")
        } else {
            logger.error("InfluxDB connection to ${influxDBExtensionProperties.uri} is not OK")
        }
    }

    override fun safe(code: InfluxDB.() -> Unit) {
        current?.apply {
            try {
                code()
            } catch (ex: Exception) {
                logger.error("Issue with running against InfluxDB", ex)
            }
        } ?: logger.error("InfluxDB connection is not available")
    }

    override val current: InfluxDB?
        get() =
            internalConnection?.let { connection ->
                val ok = check(connection)
                if (ok) {
                    connection
                } else {
                    logger.info("Connection to InfluxDB is not valid any longer, attempting to renew it.")
                    renew()
                }
            } ?: create()

    override fun isValid(immediate: Boolean): Boolean =
        internalConnection?.run { check(this, immediate = immediate) } ?: false

    override fun reset() {
        securityService.checkGlobalFunction(ApplicationManagement::class.java)
        renew()
    }

    private fun check(connection: InfluxDB, immediate: Boolean = false): Boolean {
        return if (
            immediate ||
            lastCheck == null ||
            Duration.between(lastCheck, Time.now()) > influxDBExtensionProperties.validity
        ) {
            val pong = try {
                connection.ping()
            } catch (_: Exception) {
                return false
            }
            val connectionOK = pong?.isGood ?: false
            if (!connectionOK) {
                false
            } else {
                lastCheck = Time.now()
                true
            }
        } else {
            true
        }
    }

    private fun renew(): InfluxDB? = create()

    private fun create(): InfluxDB? {
        return internalConnectionLock.write {
            doCreate()?.apply {
                if (!check(this)) {
                    logger.warn("InfluxDB connection was created/renewed but is not valid. Recreation will be attempted in ${influxDBExtensionProperties.validity}.")
                }
                lastCheck = Time.now()
                internalConnection = this
            }
        }
    }

    private fun doCreate(): InfluxDB? {
        try {
            logger.info("InfluxDB URI = ${influxDBExtensionProperties.uri}")
            logger.info("InfluxDB database = ${influxDBExtensionProperties.db}")

            var builder = OkHttpClient.Builder()
            if (!influxDBExtensionProperties.ssl.hostCheck) {
                builder = builder.hostnameVerifier { _, _ -> true }
            }

            val influxDB = InfluxDBFactory.connect(
                influxDBExtensionProperties.uri,
                influxDBExtensionProperties.username,
                influxDBExtensionProperties.password,
                builder
            )
            influxDB.setDatabase(influxDBExtensionProperties.db)
            if (influxDBExtensionProperties.create) {
                influxDB.createDatabase(influxDBExtensionProperties.db)
            }
            influxDB.setLogLevel(influxDBExtensionProperties.log)
            influxDB.enableBatch(BatchOptions.DEFAULTS)

            return influxDB
        } catch (ex: Exception) {
            logger.error("Cannot create the InfluxDB connection", ex)
            return null
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy