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

engine.nlp.NlpProxyBotService.kt Maven / Gradle / Ivy

/*
 * Copyright (C) 2017/2021 e-voyageurs technologies
 *
 * 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
 *
 * http://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 ai.tock.bot.engine.nlp

import ai.tock.shared.error
import ai.tock.shared.property
import io.vertx.core.Vertx
import io.vertx.core.http.HttpClient
import io.vertx.core.http.HttpClientOptions
import io.vertx.core.http.HttpMethod
import io.vertx.core.http.HttpMethod.GET
import io.vertx.core.http.HttpMethod.POST
import io.vertx.core.http.RequestOptions
import io.vertx.ext.web.Router
import io.vertx.ext.web.RoutingContext
import java.net.URL
import mu.KLogger
import mu.KotlinLogging

/**
 * Expose NLP API to BOT app
 *
 */
internal object NlpProxyBotService {

    private val logger: KLogger = KotlinLogging.logger {}
    private val tockNlpProxyOnBotPath = property("tock_nlp_proxy_on_bot_path", "/_proxy_nlp")
    private val tockNlpServiceHost: String
    private val tockNlpServicePort: Int
    private val tockNlpServiceSsl: Boolean
    private val nonForwardedHeaders = setOf(
        "Accept-Encoding",
        "Host",
        "Via",
        "X-Forwarded-For",
        "X-Forwarded-Host",
        "X-Forwarded-Port",
        "X-Forwarded-Proto",
        "X-Forwarded-Server",
    )

    init {
        val tockNlpServiceUrl = URL(System.getenv("tock_nlp_service_url") ?: "http://localhost:8888")
        tockNlpServiceHost = System.getenv("tock_nlp_service_host") ?: tockNlpServiceUrl.host
        tockNlpServicePort = System.getenv("tock_nlp_service_port")?.toInt() ?: tockNlpServiceUrl.port
        tockNlpServiceSsl = (System.getenv("tock_nlp_service_SSL") ?: tockNlpServiceUrl.protocol) == "https"
    }

    fun configure(vertx: Vertx): (Router) -> Unit {
        return { router ->
            router.post("$tockNlpProxyOnBotPath*").handler { context ->
                httpProxyToNlp(context, vertx, POST)
            }
            router.get("$tockNlpProxyOnBotPath*").handler { context ->
                httpProxyToNlp(context, vertx, GET)
            }
        }
    }

    private fun httpProxyToNlp(
        context: RoutingContext,
        vertx: Vertx,
        httpMethod: HttpMethod
    ) {
        try {
            val uri = context.request().uri().substringAfter(tockNlpProxyOnBotPath)
            val client: HttpClient = vertx.createHttpClient(HttpClientOptions().apply { isKeepAlive = false })
            val options = RequestOptions()
                .setHost(tockNlpServiceHost)
                .setPort(tockNlpServicePort)
                .setSsl(tockNlpServiceSsl)
                .setURI(uri)
                .setMethod(httpMethod)
                .apply {
                    context.response().headers().forEach { (key, value) ->
                        if (key !in nonForwardedHeaders) addHeader(key, value)
                    }
                }
            client.request(options).flatMap {
                if (httpMethod == POST) {
                    it.send(context.body().buffer())
                } else {
                    it.send()
                }
            }.flatMap { nlpResponse ->
                context.response().isChunked = true
                val resStatusCode = nlpResponse.statusCode()
                if (resStatusCode != 200 && resStatusCode != 201) {
                    logger.warn { "target server status code error : $resStatusCode" }
                }
                context.response().statusCode = resStatusCode
                context.response().headers().setAll(nlpResponse.headers())
                nlpResponse.pipeTo(context.response())
            }.onFailure { t -> logger.error(t) }
        } catch (e: Exception) {
            logger.error(e)
            context.fail(500)
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy